vasnprintf.c 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890
  1. /* vsprintf with automatic memory allocation.
  2. Copyright (C) 1999, 2002-2006 Free Software Foundation, Inc.
  3. This program is free software; you can redistribute it and/or modify it
  4. under the terms of the GNU Library General Public License as published
  5. by the Free Software Foundation; either version 2, or (at your option)
  6. any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
  10. Library General Public License for more details.
  11. You should have received a copy of the GNU Library General Public
  12. License along with this program; if not, write to the Free Software
  13. Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301,
  14. USA. */
  15. /* Tell glibc's <stdio.h> to provide a prototype for snprintf().
  16. This must come before <config.h> because <config.h> may include
  17. <features.h>, and once <features.h> has been included, it's too late. */
  18. #ifndef _GNU_SOURCE
  19. # define _GNU_SOURCE 1
  20. #endif
  21. #include <config.h>
  22. #ifndef IN_LIBINTL
  23. # include <alloca.h>
  24. #endif
  25. /* Specification. */
  26. #if WIDE_CHAR_VERSION
  27. # include "vasnwprintf.h"
  28. #else
  29. # include "vasnprintf.h"
  30. #endif
  31. #include <stdio.h> /* snprintf(), sprintf() */
  32. #include <stdlib.h> /* abort(), malloc(), realloc(), free() */
  33. #include <string.h> /* memcpy(), strlen() */
  34. #include <errno.h> /* errno */
  35. #include <limits.h> /* CHAR_BIT */
  36. #include <float.h> /* DBL_MAX_EXP, LDBL_MAX_EXP */
  37. #if WIDE_CHAR_VERSION
  38. # include "wprintf-parse.h"
  39. #else
  40. # include "printf-parse.h"
  41. #endif
  42. /* Checked size_t computations. */
  43. #include "xsize.h"
  44. #ifdef HAVE_WCHAR_T
  45. # ifdef HAVE_WCSLEN
  46. # define local_wcslen wcslen
  47. # else
  48. /* Solaris 2.5.1 has wcslen() in a separate library libw.so. To avoid
  49. a dependency towards this library, here is a local substitute.
  50. Define this substitute only once, even if this file is included
  51. twice in the same compilation unit. */
  52. # ifndef local_wcslen_defined
  53. # define local_wcslen_defined 1
  54. static size_t
  55. local_wcslen (const wchar_t *s)
  56. {
  57. const wchar_t *ptr;
  58. for (ptr = s; *ptr != (wchar_t) 0; ptr++)
  59. ;
  60. return ptr - s;
  61. }
  62. # endif
  63. # endif
  64. #endif
  65. #if WIDE_CHAR_VERSION
  66. # define VASNPRINTF vasnwprintf
  67. # define CHAR_T wchar_t
  68. # define DIRECTIVE wchar_t_directive
  69. # define DIRECTIVES wchar_t_directives
  70. # define PRINTF_PARSE wprintf_parse
  71. # define USE_SNPRINTF 1
  72. # if HAVE_DECL__SNWPRINTF
  73. /* On Windows, the function swprintf() has a different signature than
  74. on Unix; we use the _snwprintf() function instead. */
  75. # define SNPRINTF _snwprintf
  76. # else
  77. /* Unix. */
  78. # define SNPRINTF swprintf
  79. # endif
  80. #else
  81. # define VASNPRINTF vasnprintf
  82. # define CHAR_T char
  83. # define DIRECTIVE char_directive
  84. # define DIRECTIVES char_directives
  85. # define PRINTF_PARSE printf_parse
  86. # define USE_SNPRINTF (HAVE_DECL__SNPRINTF || HAVE_SNPRINTF)
  87. # if HAVE_DECL__SNPRINTF
  88. /* Windows. */
  89. # define SNPRINTF _snprintf
  90. # else
  91. /* Unix. */
  92. # define SNPRINTF snprintf
  93. # endif
  94. #endif
  95. CHAR_T *
  96. VASNPRINTF (CHAR_T *resultbuf, size_t *lengthp, const CHAR_T *format, va_list args)
  97. {
  98. DIRECTIVES d;
  99. arguments a;
  100. if (PRINTF_PARSE (format, &d, &a) < 0)
  101. {
  102. errno = EINVAL;
  103. return NULL;
  104. }
  105. #define CLEANUP() \
  106. free (d.dir); \
  107. if (a.arg) \
  108. free (a.arg);
  109. if (printf_fetchargs (args, &a) < 0)
  110. {
  111. CLEANUP ();
  112. errno = EINVAL;
  113. return NULL;
  114. }
  115. {
  116. size_t buf_neededlength;
  117. CHAR_T *buf;
  118. CHAR_T *buf_malloced;
  119. const CHAR_T *cp;
  120. size_t i;
  121. DIRECTIVE *dp;
  122. /* Output string accumulator. */
  123. CHAR_T *result;
  124. size_t allocated;
  125. size_t length;
  126. /* Allocate a small buffer that will hold a directive passed to
  127. sprintf or snprintf. */
  128. buf_neededlength =
  129. xsum4 (7, d.max_width_length, d.max_precision_length, 6);
  130. #if HAVE_ALLOCA
  131. if (buf_neededlength < 4000 / sizeof (CHAR_T))
  132. {
  133. buf = (CHAR_T *) alloca (buf_neededlength * sizeof (CHAR_T));
  134. buf_malloced = NULL;
  135. }
  136. else
  137. #endif
  138. {
  139. size_t buf_memsize = xtimes (buf_neededlength, sizeof (CHAR_T));
  140. if (size_overflow_p (buf_memsize))
  141. goto out_of_memory_1;
  142. buf = (CHAR_T *) malloc (buf_memsize);
  143. if (buf == NULL)
  144. goto out_of_memory_1;
  145. buf_malloced = buf;
  146. }
  147. if (resultbuf != NULL)
  148. {
  149. result = resultbuf;
  150. allocated = *lengthp;
  151. }
  152. else
  153. {
  154. result = NULL;
  155. allocated = 0;
  156. }
  157. length = 0;
  158. /* Invariants:
  159. result is either == resultbuf or == NULL or malloc-allocated.
  160. If length > 0, then result != NULL. */
  161. /* Ensures that allocated >= needed. Aborts through a jump to
  162. out_of_memory if needed is SIZE_MAX or otherwise too big. */
  163. #define ENSURE_ALLOCATION(needed) \
  164. if ((needed) > allocated) \
  165. { \
  166. size_t memory_size; \
  167. CHAR_T *memory; \
  168. \
  169. allocated = (allocated > 0 ? xtimes (allocated, 2) : 12); \
  170. if ((needed) > allocated) \
  171. allocated = (needed); \
  172. memory_size = xtimes (allocated, sizeof (CHAR_T)); \
  173. if (size_overflow_p (memory_size)) \
  174. goto out_of_memory; \
  175. if (result == resultbuf || result == NULL) \
  176. memory = (CHAR_T *) malloc (memory_size); \
  177. else \
  178. memory = (CHAR_T *) realloc (result, memory_size); \
  179. if (memory == NULL) \
  180. goto out_of_memory; \
  181. if (result == resultbuf && length > 0) \
  182. memcpy (memory, result, length * sizeof (CHAR_T)); \
  183. result = memory; \
  184. }
  185. for (cp = format, i = 0, dp = &d.dir[0]; ; cp = dp->dir_end, i++, dp++)
  186. {
  187. if (cp != dp->dir_start)
  188. {
  189. size_t n = dp->dir_start - cp;
  190. size_t augmented_length = xsum (length, n);
  191. ENSURE_ALLOCATION (augmented_length);
  192. memcpy (result + length, cp, n * sizeof (CHAR_T));
  193. length = augmented_length;
  194. }
  195. if (i == d.count)
  196. break;
  197. /* Execute a single directive. */
  198. if (dp->conversion == '%')
  199. {
  200. size_t augmented_length;
  201. if (!(dp->arg_index == ARG_NONE))
  202. abort ();
  203. augmented_length = xsum (length, 1);
  204. ENSURE_ALLOCATION (augmented_length);
  205. result[length] = '%';
  206. length = augmented_length;
  207. }
  208. else
  209. {
  210. if (!(dp->arg_index != ARG_NONE))
  211. abort ();
  212. if (dp->conversion == 'n')
  213. {
  214. switch (a.arg[dp->arg_index].type)
  215. {
  216. case TYPE_COUNT_SCHAR_POINTER:
  217. *a.arg[dp->arg_index].a.a_count_schar_pointer = length;
  218. break;
  219. case TYPE_COUNT_SHORT_POINTER:
  220. *a.arg[dp->arg_index].a.a_count_short_pointer = length;
  221. break;
  222. case TYPE_COUNT_INT_POINTER:
  223. *a.arg[dp->arg_index].a.a_count_int_pointer = length;
  224. break;
  225. case TYPE_COUNT_LONGINT_POINTER:
  226. *a.arg[dp->arg_index].a.a_count_longint_pointer = length;
  227. break;
  228. #ifdef HAVE_LONG_LONG_INT
  229. case TYPE_COUNT_LONGLONGINT_POINTER:
  230. *a.arg[dp->arg_index].a.a_count_longlongint_pointer = length;
  231. break;
  232. #endif
  233. default:
  234. abort ();
  235. }
  236. }
  237. else
  238. {
  239. arg_type type = a.arg[dp->arg_index].type;
  240. CHAR_T *p;
  241. unsigned int prefix_count;
  242. int prefixes[2];
  243. #if !USE_SNPRINTF
  244. size_t tmp_length;
  245. CHAR_T tmpbuf[700];
  246. CHAR_T *tmp;
  247. /* Allocate a temporary buffer of sufficient size for calling
  248. sprintf. */
  249. {
  250. size_t width;
  251. size_t precision;
  252. width = 0;
  253. if (dp->width_start != dp->width_end)
  254. {
  255. if (dp->width_arg_index != ARG_NONE)
  256. {
  257. int arg;
  258. if (!(a.arg[dp->width_arg_index].type == TYPE_INT))
  259. abort ();
  260. arg = a.arg[dp->width_arg_index].a.a_int;
  261. width = (arg < 0 ? (unsigned int) (-arg) : arg);
  262. }
  263. else
  264. {
  265. const CHAR_T *digitp = dp->width_start;
  266. do
  267. width = xsum (xtimes (width, 10), *digitp++ - '0');
  268. while (digitp != dp->width_end);
  269. }
  270. }
  271. precision = 6;
  272. if (dp->precision_start != dp->precision_end)
  273. {
  274. if (dp->precision_arg_index != ARG_NONE)
  275. {
  276. int arg;
  277. if (!(a.arg[dp->precision_arg_index].type == TYPE_INT))
  278. abort ();
  279. arg = a.arg[dp->precision_arg_index].a.a_int;
  280. precision = (arg < 0 ? 0 : arg);
  281. }
  282. else
  283. {
  284. const CHAR_T *digitp = dp->precision_start + 1;
  285. precision = 0;
  286. while (digitp != dp->precision_end)
  287. precision = xsum (xtimes (precision, 10), *digitp++ - '0');
  288. }
  289. }
  290. switch (dp->conversion)
  291. {
  292. case 'd': case 'i': case 'u':
  293. # ifdef HAVE_LONG_LONG_INT
  294. if (type == TYPE_LONGLONGINT || type == TYPE_ULONGLONGINT)
  295. tmp_length =
  296. (unsigned int) (sizeof (unsigned long long) * CHAR_BIT
  297. * 0.30103 /* binary -> decimal */
  298. )
  299. + 1; /* turn floor into ceil */
  300. else
  301. # endif
  302. if (type == TYPE_LONGINT || type == TYPE_ULONGINT)
  303. tmp_length =
  304. (unsigned int) (sizeof (unsigned long) * CHAR_BIT
  305. * 0.30103 /* binary -> decimal */
  306. )
  307. + 1; /* turn floor into ceil */
  308. else
  309. tmp_length =
  310. (unsigned int) (sizeof (unsigned int) * CHAR_BIT
  311. * 0.30103 /* binary -> decimal */
  312. )
  313. + 1; /* turn floor into ceil */
  314. if (tmp_length < precision)
  315. tmp_length = precision;
  316. /* Multiply by 2, as an estimate for FLAG_GROUP. */
  317. tmp_length = xsum (tmp_length, tmp_length);
  318. /* Add 1, to account for a leading sign. */
  319. tmp_length = xsum (tmp_length, 1);
  320. break;
  321. case 'o':
  322. # ifdef HAVE_LONG_LONG_INT
  323. if (type == TYPE_LONGLONGINT || type == TYPE_ULONGLONGINT)
  324. tmp_length =
  325. (unsigned int) (sizeof (unsigned long long) * CHAR_BIT
  326. * 0.333334 /* binary -> octal */
  327. )
  328. + 1; /* turn floor into ceil */
  329. else
  330. # endif
  331. if (type == TYPE_LONGINT || type == TYPE_ULONGINT)
  332. tmp_length =
  333. (unsigned int) (sizeof (unsigned long) * CHAR_BIT
  334. * 0.333334 /* binary -> octal */
  335. )
  336. + 1; /* turn floor into ceil */
  337. else
  338. tmp_length =
  339. (unsigned int) (sizeof (unsigned int) * CHAR_BIT
  340. * 0.333334 /* binary -> octal */
  341. )
  342. + 1; /* turn floor into ceil */
  343. if (tmp_length < precision)
  344. tmp_length = precision;
  345. /* Add 1, to account for a leading sign. */
  346. tmp_length = xsum (tmp_length, 1);
  347. break;
  348. case 'x': case 'X':
  349. # ifdef HAVE_LONG_LONG_INT
  350. if (type == TYPE_LONGLONGINT || type == TYPE_ULONGLONGINT)
  351. tmp_length =
  352. (unsigned int) (sizeof (unsigned long long) * CHAR_BIT
  353. * 0.25 /* binary -> hexadecimal */
  354. )
  355. + 1; /* turn floor into ceil */
  356. else
  357. # endif
  358. if (type == TYPE_LONGINT || type == TYPE_ULONGINT)
  359. tmp_length =
  360. (unsigned int) (sizeof (unsigned long) * CHAR_BIT
  361. * 0.25 /* binary -> hexadecimal */
  362. )
  363. + 1; /* turn floor into ceil */
  364. else
  365. tmp_length =
  366. (unsigned int) (sizeof (unsigned int) * CHAR_BIT
  367. * 0.25 /* binary -> hexadecimal */
  368. )
  369. + 1; /* turn floor into ceil */
  370. if (tmp_length < precision)
  371. tmp_length = precision;
  372. /* Add 2, to account for a leading sign or alternate form. */
  373. tmp_length = xsum (tmp_length, 2);
  374. break;
  375. case 'f': case 'F':
  376. # ifdef HAVE_LONG_DOUBLE
  377. if (type == TYPE_LONGDOUBLE)
  378. tmp_length =
  379. (unsigned int) (LDBL_MAX_EXP
  380. * 0.30103 /* binary -> decimal */
  381. * 2 /* estimate for FLAG_GROUP */
  382. )
  383. + 1 /* turn floor into ceil */
  384. + 10; /* sign, decimal point etc. */
  385. else
  386. # endif
  387. tmp_length =
  388. (unsigned int) (DBL_MAX_EXP
  389. * 0.30103 /* binary -> decimal */
  390. * 2 /* estimate for FLAG_GROUP */
  391. )
  392. + 1 /* turn floor into ceil */
  393. + 10; /* sign, decimal point etc. */
  394. tmp_length = xsum (tmp_length, precision);
  395. break;
  396. case 'e': case 'E': case 'g': case 'G':
  397. case 'a': case 'A':
  398. tmp_length =
  399. 12; /* sign, decimal point, exponent etc. */
  400. tmp_length = xsum (tmp_length, precision);
  401. break;
  402. case 'c':
  403. # if defined HAVE_WINT_T && !WIDE_CHAR_VERSION
  404. if (type == TYPE_WIDE_CHAR)
  405. tmp_length = MB_CUR_MAX;
  406. else
  407. # endif
  408. tmp_length = 1;
  409. break;
  410. case 's':
  411. # ifdef HAVE_WCHAR_T
  412. if (type == TYPE_WIDE_STRING)
  413. {
  414. tmp_length =
  415. local_wcslen (a.arg[dp->arg_index].a.a_wide_string);
  416. # if !WIDE_CHAR_VERSION
  417. tmp_length = xtimes (tmp_length, MB_CUR_MAX);
  418. # endif
  419. }
  420. else
  421. # endif
  422. tmp_length = strlen (a.arg[dp->arg_index].a.a_string);
  423. break;
  424. case 'p':
  425. tmp_length =
  426. (unsigned int) (sizeof (void *) * CHAR_BIT
  427. * 0.25 /* binary -> hexadecimal */
  428. )
  429. + 1 /* turn floor into ceil */
  430. + 2; /* account for leading 0x */
  431. break;
  432. default:
  433. abort ();
  434. }
  435. if (tmp_length < width)
  436. tmp_length = width;
  437. tmp_length = xsum (tmp_length, 1); /* account for trailing NUL */
  438. }
  439. if (tmp_length <= sizeof (tmpbuf) / sizeof (CHAR_T))
  440. tmp = tmpbuf;
  441. else
  442. {
  443. size_t tmp_memsize = xtimes (tmp_length, sizeof (CHAR_T));
  444. if (size_overflow_p (tmp_memsize))
  445. /* Overflow, would lead to out of memory. */
  446. goto out_of_memory;
  447. tmp = (CHAR_T *) malloc (tmp_memsize);
  448. if (tmp == NULL)
  449. /* Out of memory. */
  450. goto out_of_memory;
  451. }
  452. #endif
  453. /* Construct the format string for calling snprintf or
  454. sprintf. */
  455. p = buf;
  456. *p++ = '%';
  457. if (dp->flags & FLAG_GROUP)
  458. *p++ = '\'';
  459. if (dp->flags & FLAG_LEFT)
  460. *p++ = '-';
  461. if (dp->flags & FLAG_SHOWSIGN)
  462. *p++ = '+';
  463. if (dp->flags & FLAG_SPACE)
  464. *p++ = ' ';
  465. if (dp->flags & FLAG_ALT)
  466. *p++ = '#';
  467. if (dp->flags & FLAG_ZERO)
  468. *p++ = '0';
  469. if (dp->width_start != dp->width_end)
  470. {
  471. size_t n = dp->width_end - dp->width_start;
  472. memcpy (p, dp->width_start, n * sizeof (CHAR_T));
  473. p += n;
  474. }
  475. if (dp->precision_start != dp->precision_end)
  476. {
  477. size_t n = dp->precision_end - dp->precision_start;
  478. memcpy (p, dp->precision_start, n * sizeof (CHAR_T));
  479. p += n;
  480. }
  481. switch (type)
  482. {
  483. #ifdef HAVE_LONG_LONG_INT
  484. case TYPE_LONGLONGINT:
  485. case TYPE_ULONGLONGINT:
  486. *p++ = 'l';
  487. /*FALLTHROUGH*/
  488. #endif
  489. case TYPE_LONGINT:
  490. case TYPE_ULONGINT:
  491. #ifdef HAVE_WINT_T
  492. case TYPE_WIDE_CHAR:
  493. #endif
  494. #ifdef HAVE_WCHAR_T
  495. case TYPE_WIDE_STRING:
  496. #endif
  497. *p++ = 'l';
  498. break;
  499. #ifdef HAVE_LONG_DOUBLE
  500. case TYPE_LONGDOUBLE:
  501. *p++ = 'L';
  502. break;
  503. #endif
  504. default:
  505. break;
  506. }
  507. *p = dp->conversion;
  508. #if USE_SNPRINTF
  509. p[1] = '%';
  510. p[2] = 'n';
  511. p[3] = '\0';
  512. #else
  513. p[1] = '\0';
  514. #endif
  515. /* Construct the arguments for calling snprintf or sprintf. */
  516. prefix_count = 0;
  517. if (dp->width_arg_index != ARG_NONE)
  518. {
  519. if (!(a.arg[dp->width_arg_index].type == TYPE_INT))
  520. abort ();
  521. prefixes[prefix_count++] = a.arg[dp->width_arg_index].a.a_int;
  522. }
  523. if (dp->precision_arg_index != ARG_NONE)
  524. {
  525. if (!(a.arg[dp->precision_arg_index].type == TYPE_INT))
  526. abort ();
  527. prefixes[prefix_count++] = a.arg[dp->precision_arg_index].a.a_int;
  528. }
  529. #if USE_SNPRINTF
  530. /* Prepare checking whether snprintf returns the count
  531. via %n. */
  532. ENSURE_ALLOCATION (xsum (length, 1));
  533. result[length] = '\0';
  534. #endif
  535. for (;;)
  536. {
  537. size_t maxlen;
  538. int count;
  539. int retcount;
  540. maxlen = allocated - length;
  541. count = -1;
  542. retcount = 0;
  543. #if USE_SNPRINTF
  544. # define SNPRINTF_BUF(arg) \
  545. switch (prefix_count) \
  546. { \
  547. case 0: \
  548. retcount = SNPRINTF (result + length, maxlen, buf, \
  549. arg, &count); \
  550. break; \
  551. case 1: \
  552. retcount = SNPRINTF (result + length, maxlen, buf, \
  553. prefixes[0], arg, &count); \
  554. break; \
  555. case 2: \
  556. retcount = SNPRINTF (result + length, maxlen, buf, \
  557. prefixes[0], prefixes[1], arg, \
  558. &count); \
  559. break; \
  560. default: \
  561. abort (); \
  562. }
  563. #else
  564. # define SNPRINTF_BUF(arg) \
  565. switch (prefix_count) \
  566. { \
  567. case 0: \
  568. count = sprintf (tmp, buf, arg); \
  569. break; \
  570. case 1: \
  571. count = sprintf (tmp, buf, prefixes[0], arg); \
  572. break; \
  573. case 2: \
  574. count = sprintf (tmp, buf, prefixes[0], prefixes[1],\
  575. arg); \
  576. break; \
  577. default: \
  578. abort (); \
  579. }
  580. #endif
  581. switch (type)
  582. {
  583. case TYPE_SCHAR:
  584. {
  585. int arg = a.arg[dp->arg_index].a.a_schar;
  586. SNPRINTF_BUF (arg);
  587. }
  588. break;
  589. case TYPE_UCHAR:
  590. {
  591. unsigned int arg = a.arg[dp->arg_index].a.a_uchar;
  592. SNPRINTF_BUF (arg);
  593. }
  594. break;
  595. case TYPE_SHORT:
  596. {
  597. int arg = a.arg[dp->arg_index].a.a_short;
  598. SNPRINTF_BUF (arg);
  599. }
  600. break;
  601. case TYPE_USHORT:
  602. {
  603. unsigned int arg = a.arg[dp->arg_index].a.a_ushort;
  604. SNPRINTF_BUF (arg);
  605. }
  606. break;
  607. case TYPE_INT:
  608. {
  609. int arg = a.arg[dp->arg_index].a.a_int;
  610. SNPRINTF_BUF (arg);
  611. }
  612. break;
  613. case TYPE_UINT:
  614. {
  615. unsigned int arg = a.arg[dp->arg_index].a.a_uint;
  616. SNPRINTF_BUF (arg);
  617. }
  618. break;
  619. case TYPE_LONGINT:
  620. {
  621. long int arg = a.arg[dp->arg_index].a.a_longint;
  622. SNPRINTF_BUF (arg);
  623. }
  624. break;
  625. case TYPE_ULONGINT:
  626. {
  627. unsigned long int arg = a.arg[dp->arg_index].a.a_ulongint;
  628. SNPRINTF_BUF (arg);
  629. }
  630. break;
  631. #ifdef HAVE_LONG_LONG_INT
  632. case TYPE_LONGLONGINT:
  633. {
  634. long long int arg = a.arg[dp->arg_index].a.a_longlongint;
  635. SNPRINTF_BUF (arg);
  636. }
  637. break;
  638. case TYPE_ULONGLONGINT:
  639. {
  640. unsigned long long int arg = a.arg[dp->arg_index].a.a_ulonglongint;
  641. SNPRINTF_BUF (arg);
  642. }
  643. break;
  644. #endif
  645. case TYPE_DOUBLE:
  646. {
  647. double arg = a.arg[dp->arg_index].a.a_double;
  648. SNPRINTF_BUF (arg);
  649. }
  650. break;
  651. #ifdef HAVE_LONG_DOUBLE
  652. case TYPE_LONGDOUBLE:
  653. {
  654. long double arg = a.arg[dp->arg_index].a.a_longdouble;
  655. SNPRINTF_BUF (arg);
  656. }
  657. break;
  658. #endif
  659. case TYPE_CHAR:
  660. {
  661. int arg = a.arg[dp->arg_index].a.a_char;
  662. SNPRINTF_BUF (arg);
  663. }
  664. break;
  665. #ifdef HAVE_WINT_T
  666. case TYPE_WIDE_CHAR:
  667. {
  668. wint_t arg = a.arg[dp->arg_index].a.a_wide_char;
  669. SNPRINTF_BUF (arg);
  670. }
  671. break;
  672. #endif
  673. case TYPE_STRING:
  674. {
  675. const char *arg = a.arg[dp->arg_index].a.a_string;
  676. SNPRINTF_BUF (arg);
  677. }
  678. break;
  679. #ifdef HAVE_WCHAR_T
  680. case TYPE_WIDE_STRING:
  681. {
  682. const wchar_t *arg = a.arg[dp->arg_index].a.a_wide_string;
  683. SNPRINTF_BUF (arg);
  684. }
  685. break;
  686. #endif
  687. case TYPE_POINTER:
  688. {
  689. void *arg = a.arg[dp->arg_index].a.a_pointer;
  690. SNPRINTF_BUF (arg);
  691. }
  692. break;
  693. default:
  694. abort ();
  695. }
  696. #if USE_SNPRINTF
  697. /* Portability: Not all implementations of snprintf()
  698. are ISO C 99 compliant. Determine the number of
  699. bytes that snprintf() has produced or would have
  700. produced. */
  701. if (count >= 0)
  702. {
  703. /* Verify that snprintf() has NUL-terminated its
  704. result. */
  705. if (count < maxlen && result[length + count] != '\0')
  706. abort ();
  707. /* Portability hack. */
  708. if (retcount > count)
  709. count = retcount;
  710. }
  711. else
  712. {
  713. /* snprintf() doesn't understand the '%n'
  714. directive. */
  715. if (p[1] != '\0')
  716. {
  717. /* Don't use the '%n' directive; instead, look
  718. at the snprintf() return value. */
  719. p[1] = '\0';
  720. continue;
  721. }
  722. else
  723. {
  724. /* Look at the snprintf() return value. */
  725. if (retcount < 0)
  726. {
  727. /* HP-UX 10.20 snprintf() is doubly deficient:
  728. It doesn't understand the '%n' directive,
  729. *and* it returns -1 (rather than the length
  730. that would have been required) when the
  731. buffer is too small. */
  732. size_t bigger_need =
  733. xsum (xtimes (allocated, 2), 12);
  734. ENSURE_ALLOCATION (bigger_need);
  735. continue;
  736. }
  737. else
  738. count = retcount;
  739. }
  740. }
  741. #endif
  742. /* Attempt to handle failure. */
  743. if (count < 0)
  744. {
  745. if (!(result == resultbuf || result == NULL))
  746. free (result);
  747. if (buf_malloced != NULL)
  748. free (buf_malloced);
  749. CLEANUP ();
  750. errno = EINVAL;
  751. return NULL;
  752. }
  753. #if !USE_SNPRINTF
  754. if (count >= tmp_length)
  755. /* tmp_length was incorrectly calculated - fix the
  756. code above! */
  757. abort ();
  758. #endif
  759. /* Make room for the result. */
  760. if (count >= maxlen)
  761. {
  762. /* Need at least count bytes. But allocate
  763. proportionally, to avoid looping eternally if
  764. snprintf() reports a too small count. */
  765. size_t n =
  766. xmax (xsum (length, count), xtimes (allocated, 2));
  767. ENSURE_ALLOCATION (n);
  768. #if USE_SNPRINTF
  769. continue;
  770. #endif
  771. }
  772. #if USE_SNPRINTF
  773. /* The snprintf() result did fit. */
  774. #else
  775. /* Append the sprintf() result. */
  776. memcpy (result + length, tmp, count * sizeof (CHAR_T));
  777. if (tmp != tmpbuf)
  778. free (tmp);
  779. #endif
  780. length += count;
  781. break;
  782. }
  783. }
  784. }
  785. }
  786. /* Add the final NUL. */
  787. ENSURE_ALLOCATION (xsum (length, 1));
  788. result[length] = '\0';
  789. if (result != resultbuf && length + 1 < allocated)
  790. {
  791. /* Shrink the allocated memory if possible. */
  792. CHAR_T *memory;
  793. memory = (CHAR_T *) realloc (result, (length + 1) * sizeof (CHAR_T));
  794. if (memory != NULL)
  795. result = memory;
  796. }
  797. if (buf_malloced != NULL)
  798. free (buf_malloced);
  799. CLEANUP ();
  800. *lengthp = length;
  801. /* Note that we can produce a big string of a length > INT_MAX. POSIX
  802. says that snprintf() fails with errno = EOVERFLOW in this case, but
  803. that's only because snprintf() returns an 'int'. This function does
  804. not have this limitation. */
  805. return result;
  806. out_of_memory:
  807. if (!(result == resultbuf || result == NULL))
  808. free (result);
  809. if (buf_malloced != NULL)
  810. free (buf_malloced);
  811. out_of_memory_1:
  812. CLEANUP ();
  813. errno = ENOMEM;
  814. return NULL;
  815. }
  816. }
  817. #undef SNPRINTF
  818. #undef USE_SNPRINTF
  819. #undef PRINTF_PARSE
  820. #undef DIRECTIVES
  821. #undef DIRECTIVE
  822. #undef CHAR_T
  823. #undef VASNPRINTF