X-Git-Url: http://erislabs.net/gitweb/?a=blobdiff_plain;f=lib%2Fsprintf.c;h=113d880833557a3d61d7ca287e3bf729577691fe;hb=5191b3546cfb6c163228c23f214e325ddf60d46f;hp=ee716895d3b45a22db1fb613eed0399e658e5e0f;hpb=e0a39772b0b68a716a423c697be786d64bd07b2a;p=gnulib.git diff --git a/lib/sprintf.c b/lib/sprintf.c index ee716895d..113d88083 100644 --- a/lib/sprintf.c +++ b/lib/sprintf.c @@ -1,19 +1,18 @@ /* Formatted output to strings. - Copyright (C) 2004, 2006-2007 Free Software Foundation, Inc. + Copyright (C) 2004, 2006-2013 Free Software Foundation, Inc. - This program is free software; you can redistribute it and/or modify + 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 - the Free Software Foundation; either version 2, or (at your option) - any later version. + the Free Software Foundation; either version 3 of the License, or + (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ + You should have received a copy of the GNU General Public License + along with this program. If not, see . */ #ifdef HAVE_CONFIG_H # include @@ -25,15 +24,11 @@ #include #include #include +#include #include #include "vasnprintf.h" -/* Some systems, like OSF/1 4.0 and Woe32, don't have EOVERFLOW. */ -#ifndef EOVERFLOW -# define EOVERFLOW E2BIG -#endif - #ifndef SIZE_MAX # define SIZE_MAX ((size_t) -1) #endif @@ -46,9 +41,19 @@ sprintf (char *str, const char *format, ...) { char *output; size_t len; - size_t lenbuf = SIZE_MAX; + size_t lenbuf; va_list args; + /* vasnprintf fails with EOVERFLOW when the buffer size argument is larger + than INT_MAX (if that fits into a 'size_t' at all). + Also note that glibc's iconv fails with E2BIG when we pass a length that + is so large that str + lenbuf wraps around, i.e. + (uintptr_t) (str + lenbuf) < (uintptr_t) str. + Therefore set lenbuf = min (SIZE_MAX, INT_MAX, - (uintptr_t) str - 1). */ + lenbuf = (SIZE_MAX < INT_MAX ? SIZE_MAX : INT_MAX); + if (lenbuf > ~ (uintptr_t) str) + lenbuf = ~ (uintptr_t) str; + va_start (args, format); output = vasnprintf (str, &lenbuf, format, args); len = lenbuf;