X-Git-Url: http://erislabs.net/gitweb/?a=blobdiff_plain;f=lib%2Fsnprintf.c;h=fbb73c9bed7c7f7c82406aa624b83929514c2f97;hb=23eecb48e39afd0d267d64d40ba6bf97aa865e13;hp=c23d7e14a39d92b42953128c060ef6f6835b9114;hpb=6dc8b58f2a657faf8c1dd2e9438faab2a016c034;p=gnulib.git diff --git a/lib/snprintf.c b/lib/snprintf.c index c23d7e14a..fbb73c9be 100644 --- a/lib/snprintf.c +++ b/lib/snprintf.c @@ -1,6 +1,6 @@ /* Formatted output to strings. - Copyright (C) 2004 Free Software Foundation, Inc. - Written by Simon Josefsson. + Copyright (C) 2004, 2006-2013 Free Software Foundation, Inc. + Written by Simon Josefsson and Paul Eggert. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by @@ -13,55 +13,59 @@ GNU General Public License for more details. You should have received a copy of the GNU General Public License along - with this program; if not, write to the Free Software Foundation, - Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ + with this program; if not, see . */ -#ifdef HAVE_CONFIG_H -# include -#endif +#include /* Specification. */ -#include "snprintf.h" +#include -/* Get va_list, va_start, va_end. */ +#include +#include #include -/* Get free. */ #include -/* Get memcpy, size_t. */ #include -/* Get vasnprintf. */ #include "vasnprintf.h" -/* Get MIN. */ -#include "minmax.h" - /* Print formatted output to string STR. Similar to sprintf, but additional length SIZE limit how much is written into STR. Returns string length of formatted string (which may be larger than SIZE). STR may be NULL, in which case nothing will be written. On error, - return a negative value. */ + return a negative value. */ int snprintf (char *str, size_t size, const char *format, ...) { char *output; size_t len; + size_t lenbuf = size; va_list args; va_start (args, format); - output = vasnprintf (NULL, &len, format, args); + output = vasnprintf (str, &lenbuf, format, args); + len = lenbuf; va_end (args); if (!output) return -1; - if (str && size > 0) + if (output != str) { - memcpy (str, output, MIN (len + 1, size)); - str[size - 1] = '\0'; + if (size) + { + size_t pruned_len = (len < size ? len : size - 1); + memcpy (str, output, pruned_len); + str[pruned_len] = '\0'; + } + + free (output); } - free (output); + if (INT_MAX < len) + { + errno = EOVERFLOW; + return -1; + } return len; }