printf.c 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422
  1. /* Formatted output to strings, using POSIX/XSI format strings with positions.
  2. Copyright (C) 2003, 2006 Free Software Foundation, Inc.
  3. Written by Bruno Haible <bruno@clisp.org>, 2003.
  4. This program is free software; you can redistribute it and/or modify it
  5. under the terms of the GNU Library General Public License as published
  6. by the Free Software Foundation; either version 2, or (at your option)
  7. any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  11. Library General Public License for more details.
  12. You should have received a copy of the GNU Library General Public
  13. License along with this program; if not, write to the Free Software
  14. Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
  15. USA. */
  16. #ifdef HAVE_CONFIG_H
  17. # include <config.h>
  18. #endif
  19. #ifdef __GNUC__
  20. # define alloca __builtin_alloca
  21. # define HAVE_ALLOCA 1
  22. #else
  23. # ifdef _MSC_VER
  24. # include <malloc.h>
  25. # define alloca _alloca
  26. # else
  27. # if defined HAVE_ALLOCA_H || defined _LIBC
  28. # include <alloca.h>
  29. # else
  30. # ifdef _AIX
  31. #pragma alloca
  32. # else
  33. # ifndef alloca
  34. char *alloca ();
  35. # endif
  36. # endif
  37. # endif
  38. # endif
  39. #endif
  40. #include <stdio.h>
  41. #if !HAVE_POSIX_PRINTF
  42. #include <errno.h>
  43. #include <limits.h>
  44. #include <stdlib.h>
  45. #include <string.h>
  46. /* Some systems, like OSF/1 4.0 and Woe32, don't have EOVERFLOW. */
  47. #ifndef EOVERFLOW
  48. # define EOVERFLOW E2BIG
  49. #endif
  50. /* When building a DLL, we must export some functions. Note that because
  51. the functions are only defined for binary backward compatibility, we
  52. don't need to use __declspec(dllimport) in any case. */
  53. #if defined _MSC_VER && BUILDING_DLL
  54. # define DLL_EXPORTED __declspec(dllexport)
  55. #else
  56. # define DLL_EXPORTED
  57. #endif
  58. #define STATIC static
  59. /* This needs to be consistent with libgnuintl.h.in. */
  60. #if defined __NetBSD__ || defined __CYGWIN__ || defined __MINGW32__
  61. /* Don't break __attribute__((format(printf,M,N))).
  62. This redefinition is only possible because the libc in NetBSD, Cygwin,
  63. mingw does not have a function __printf__. */
  64. # define libintl_printf __printf__
  65. #endif
  66. /* Define auxiliary functions declared in "printf-args.h". */
  67. #include "printf-args.c"
  68. /* Define auxiliary functions declared in "printf-parse.h". */
  69. #include "printf-parse.c"
  70. /* Define functions declared in "vasnprintf.h". */
  71. #define vasnprintf libintl_vasnprintf
  72. #include "vasnprintf.c"
  73. #if 0 /* not needed */
  74. #define asnprintf libintl_asnprintf
  75. #include "asnprintf.c"
  76. #endif
  77. DLL_EXPORTED
  78. int
  79. libintl_vfprintf (FILE *stream, const char *format, va_list args)
  80. {
  81. if (strchr (format, '$') == NULL)
  82. return vfprintf (stream, format, args);
  83. else
  84. {
  85. size_t length;
  86. char *result = libintl_vasnprintf (NULL, &length, format, args);
  87. int retval = -1;
  88. if (result != NULL)
  89. {
  90. size_t written = fwrite (result, 1, length, stream);
  91. free (result);
  92. if (written == length)
  93. {
  94. if (length > INT_MAX)
  95. errno = EOVERFLOW;
  96. else
  97. retval = length;
  98. }
  99. }
  100. return retval;
  101. }
  102. }
  103. DLL_EXPORTED
  104. int
  105. libintl_fprintf (FILE *stream, const char *format, ...)
  106. {
  107. va_list args;
  108. int retval;
  109. va_start (args, format);
  110. retval = libintl_vfprintf (stream, format, args);
  111. va_end (args);
  112. return retval;
  113. }
  114. DLL_EXPORTED
  115. int
  116. libintl_vprintf (const char *format, va_list args)
  117. {
  118. return libintl_vfprintf (stdout, format, args);
  119. }
  120. DLL_EXPORTED
  121. int
  122. libintl_printf (const char *format, ...)
  123. {
  124. va_list args;
  125. int retval;
  126. va_start (args, format);
  127. retval = libintl_vprintf (format, args);
  128. va_end (args);
  129. return retval;
  130. }
  131. DLL_EXPORTED
  132. int
  133. libintl_vsprintf (char *resultbuf, const char *format, va_list args)
  134. {
  135. if (strchr (format, '$') == NULL)
  136. return vsprintf (resultbuf, format, args);
  137. else
  138. {
  139. size_t length = (size_t) ~0 / (4 * sizeof (char));
  140. char *result = libintl_vasnprintf (resultbuf, &length, format, args);
  141. if (result != resultbuf)
  142. {
  143. free (result);
  144. return -1;
  145. }
  146. if (length > INT_MAX)
  147. {
  148. errno = EOVERFLOW;
  149. return -1;
  150. }
  151. else
  152. return length;
  153. }
  154. }
  155. DLL_EXPORTED
  156. int
  157. libintl_sprintf (char *resultbuf, const char *format, ...)
  158. {
  159. va_list args;
  160. int retval;
  161. va_start (args, format);
  162. retval = libintl_vsprintf (resultbuf, format, args);
  163. va_end (args);
  164. return retval;
  165. }
  166. #if HAVE_SNPRINTF
  167. # if HAVE_DECL__SNPRINTF
  168. /* Windows. */
  169. # define system_vsnprintf _vsnprintf
  170. # else
  171. /* Unix. */
  172. # define system_vsnprintf vsnprintf
  173. # endif
  174. DLL_EXPORTED
  175. int
  176. libintl_vsnprintf (char *resultbuf, size_t length, const char *format, va_list args)
  177. {
  178. if (strchr (format, '$') == NULL)
  179. return system_vsnprintf (resultbuf, length, format, args);
  180. else
  181. {
  182. size_t maxlength = length;
  183. char *result = libintl_vasnprintf (resultbuf, &length, format, args);
  184. if (result != resultbuf)
  185. {
  186. if (maxlength > 0)
  187. {
  188. size_t pruned_length =
  189. (length < maxlength ? length : maxlength - 1);
  190. memcpy (resultbuf, result, pruned_length);
  191. resultbuf[pruned_length] = '\0';
  192. }
  193. free (result);
  194. }
  195. if (length > INT_MAX)
  196. {
  197. errno = EOVERFLOW;
  198. return -1;
  199. }
  200. else
  201. return length;
  202. }
  203. }
  204. DLL_EXPORTED
  205. int
  206. libintl_snprintf (char *resultbuf, size_t length, const char *format, ...)
  207. {
  208. va_list args;
  209. int retval;
  210. va_start (args, format);
  211. retval = libintl_vsnprintf (resultbuf, length, format, args);
  212. va_end (args);
  213. return retval;
  214. }
  215. #endif
  216. #if HAVE_ASPRINTF
  217. DLL_EXPORTED
  218. int
  219. libintl_vasprintf (char **resultp, const char *format, va_list args)
  220. {
  221. size_t length;
  222. char *result = libintl_vasnprintf (NULL, &length, format, args);
  223. if (result == NULL)
  224. return -1;
  225. if (length > INT_MAX)
  226. {
  227. free (result);
  228. errno = EOVERFLOW;
  229. return -1;
  230. }
  231. *resultp = result;
  232. return length;
  233. }
  234. DLL_EXPORTED
  235. int
  236. libintl_asprintf (char **resultp, const char *format, ...)
  237. {
  238. va_list args;
  239. int retval;
  240. va_start (args, format);
  241. retval = libintl_vasprintf (resultp, format, args);
  242. va_end (args);
  243. return retval;
  244. }
  245. #endif
  246. #if HAVE_FWPRINTF
  247. #include <wchar.h>
  248. #define WIDE_CHAR_VERSION 1
  249. /* Define auxiliary functions declared in "wprintf-parse.h". */
  250. #include "printf-parse.c"
  251. /* Define functions declared in "vasnprintf.h". */
  252. #define vasnwprintf libintl_vasnwprintf
  253. #include "vasnprintf.c"
  254. #if 0 /* not needed */
  255. #define asnwprintf libintl_asnwprintf
  256. #include "asnprintf.c"
  257. #endif
  258. # if HAVE_DECL__SNWPRINTF
  259. /* Windows. */
  260. # define system_vswprintf _vsnwprintf
  261. # else
  262. /* Unix. */
  263. # define system_vswprintf vswprintf
  264. # endif
  265. DLL_EXPORTED
  266. int
  267. libintl_vfwprintf (FILE *stream, const wchar_t *format, va_list args)
  268. {
  269. if (wcschr (format, '$') == NULL)
  270. return vfwprintf (stream, format, args);
  271. else
  272. {
  273. size_t length;
  274. wchar_t *result = libintl_vasnwprintf (NULL, &length, format, args);
  275. int retval = -1;
  276. if (result != NULL)
  277. {
  278. size_t i;
  279. for (i = 0; i < length; i++)
  280. if (fputwc (result[i], stream) == WEOF)
  281. break;
  282. free (result);
  283. if (i == length)
  284. {
  285. if (length > INT_MAX)
  286. errno = EOVERFLOW;
  287. else
  288. retval = length;
  289. }
  290. }
  291. return retval;
  292. }
  293. }
  294. DLL_EXPORTED
  295. int
  296. libintl_fwprintf (FILE *stream, const wchar_t *format, ...)
  297. {
  298. va_list args;
  299. int retval;
  300. va_start (args, format);
  301. retval = libintl_vfwprintf (stream, format, args);
  302. va_end (args);
  303. return retval;
  304. }
  305. DLL_EXPORTED
  306. int
  307. libintl_vwprintf (const wchar_t *format, va_list args)
  308. {
  309. return libintl_vfwprintf (stdout, format, args);
  310. }
  311. DLL_EXPORTED
  312. int
  313. libintl_wprintf (const wchar_t *format, ...)
  314. {
  315. va_list args;
  316. int retval;
  317. va_start (args, format);
  318. retval = libintl_vwprintf (format, args);
  319. va_end (args);
  320. return retval;
  321. }
  322. DLL_EXPORTED
  323. int
  324. libintl_vswprintf (wchar_t *resultbuf, size_t length, const wchar_t *format, va_list args)
  325. {
  326. if (wcschr (format, '$') == NULL)
  327. return system_vswprintf (resultbuf, length, format, args);
  328. else
  329. {
  330. size_t maxlength = length;
  331. wchar_t *result = libintl_vasnwprintf (resultbuf, &length, format, args);
  332. if (result != resultbuf)
  333. {
  334. if (maxlength > 0)
  335. {
  336. size_t pruned_length =
  337. (length < maxlength ? length : maxlength - 1);
  338. memcpy (resultbuf, result, pruned_length * sizeof (wchar_t));
  339. resultbuf[pruned_length] = 0;
  340. }
  341. free (result);
  342. /* Unlike vsnprintf, which has to return the number of character that
  343. would have been produced if the resultbuf had been sufficiently
  344. large, the vswprintf function has to return a negative value if
  345. the resultbuf was not sufficiently large. */
  346. if (length >= maxlength)
  347. return -1;
  348. }
  349. if (length > INT_MAX)
  350. {
  351. errno = EOVERFLOW;
  352. return -1;
  353. }
  354. else
  355. return length;
  356. }
  357. }
  358. DLL_EXPORTED
  359. int
  360. libintl_swprintf (wchar_t *resultbuf, size_t length, const wchar_t *format, ...)
  361. {
  362. va_list args;
  363. int retval;
  364. va_start (args, format);
  365. retval = libintl_vswprintf (resultbuf, length, format, args);
  366. va_end (args);
  367. return retval;
  368. }
  369. #endif
  370. #endif