]> git.saurik.com Git - apple/libc.git/blob - stdio/FreeBSD/vfprintf.c.patch
b3ffc421818d80d318e810181df5668f6d4d65a7
[apple/libc.git] / stdio / FreeBSD / vfprintf.c.patch
1 --- vfprintf.c.orig 2010-11-22 22:37:24.000000000 -0800
2 +++ vfprintf.c 2010-11-22 22:38:38.000000000 -0800
3 @@ -36,6 +36,8 @@ static char sccsid[] = "@(#)vfprintf.c 8
4 #include <sys/cdefs.h>
5 __FBSDID("$FreeBSD: src/lib/libc/stdio/vfprintf.c,v 1.90 2009/02/28 06:06:57 das Exp $");
6
7 +#include "xlocale_private.h"
8 +
9 /*
10 * Actual printf innards.
11 *
12 @@ -54,7 +56,10 @@ __FBSDID("$FreeBSD: src/lib/libc/stdio/v
13 #include <stdlib.h>
14 #include <string.h>
15 #include <wchar.h>
16 +#if 0 // xprintf pending API review
17 #include <printf.h>
18 +#endif
19 +#include <errno.h>
20
21 #include <stdarg.h>
22 #include "un-namespace.h"
23 @@ -64,10 +69,11 @@ __FBSDID("$FreeBSD: src/lib/libc/stdio/v
24 #include "fvwrite.h"
25 #include "printflocal.h"
26
27 -static int __sprint(FILE *, struct __suio *);
28 -static int __sbprintf(FILE *, const char *, va_list) __printflike(2, 0)
29 - __noinline;
30 -static char *__wcsconv(wchar_t *, int);
31 +static int __sprint(FILE *, locale_t, struct __suio *);
32 +static int __sbprintf(FILE *, locale_t, const char *, va_list) __printflike(3, 0);
33 +static char *__wcsconv(wchar_t *, int, locale_t);
34 +
35 +__private_extern__ const char *__fix_nogrouping(const char *);
36
37 #define CHAR char
38 #include "printfcommon.h"
39 @@ -87,12 +93,12 @@ struct grouping_state {
40 * of bytes that will be needed.
41 */
42 static int
43 -grouping_init(struct grouping_state *gs, int ndigits)
44 +grouping_init(struct grouping_state *gs, int ndigits, locale_t loc)
45 {
46 struct lconv *locale;
47
48 - locale = localeconv();
49 - gs->grouping = locale->grouping;
50 + locale = localeconv_l(loc);
51 + gs->grouping = __fix_nogrouping(locale->grouping);
52 gs->thousands_sep = locale->thousands_sep;
53 gs->thousep_len = strlen(gs->thousands_sep);
54
55 @@ -116,11 +122,11 @@ grouping_init(struct grouping_state *gs,
56 */
57 static int
58 grouping_print(struct grouping_state *gs, struct io_state *iop,
59 - const CHAR *cp, const CHAR *ep)
60 + const CHAR *cp, const CHAR *ep, locale_t loc)
61 {
62 const CHAR *cp0 = cp;
63
64 - if (io_printandpad(iop, cp, ep, gs->lead, zeroes))
65 + if (io_printandpad(iop, cp, ep, gs->lead, zeroes, loc))
66 return (-1);
67 cp += gs->lead;
68 while (gs->nseps > 0 || gs->nrepeats > 0) {
69 @@ -130,9 +136,9 @@ grouping_print(struct grouping_state *gs
70 gs->grouping--;
71 gs->nseps--;
72 }
73 - if (io_print(iop, gs->thousands_sep, gs->thousep_len))
74 + if (io_print(iop, gs->thousands_sep, gs->thousep_len, loc))
75 return (-1);
76 - if (io_printandpad(iop, cp, ep, *gs->grouping, zeroes))
77 + if (io_printandpad(iop, cp, ep, *gs->grouping, zeroes, loc))
78 return (-1);
79 cp += *gs->grouping;
80 }
81 @@ -146,7 +152,7 @@ grouping_print(struct grouping_state *gs
82 * then reset it so that it can be reused.
83 */
84 static int
85 -__sprint(FILE *fp, struct __suio *uio)
86 +__sprint(FILE *fp, locale_t loc __unused, struct __suio *uio)
87 {
88 int err;
89
90 @@ -166,11 +172,14 @@ __sprint(FILE *fp, struct __suio *uio)
91 * worries about ungetc buffers and so forth.
92 */
93 static int
94 -__sbprintf(FILE *fp, const char *fmt, va_list ap)
95 +__sbprintf(FILE *fp, locale_t loc, const char *fmt, va_list ap)
96 {
97 int ret;
98 FILE fake;
99 unsigned char buf[BUFSIZ];
100 + struct __sFILEX ext;
101 + fake._extra = &ext;
102 + INITEXTRA(&fake);
103
104 /* XXX This is probably not needed. */
105 if (prepwrite(fp) != 0)
106 @@ -190,7 +199,7 @@ __sbprintf(FILE *fp, const char *fmt, va
107 fake._lbfsize = 0; /* not actually used, but Just In Case */
108
109 /* do the work, then copy any error status */
110 - ret = __vfprintf(&fake, fmt, ap);
111 + ret = __vfprintf(&fake, loc, fmt, ap);
112 if (ret >= 0 && __fflush(&fake))
113 ret = EOF;
114 if (fake._flags & __SERR)
115 @@ -205,7 +214,7 @@ __sbprintf(FILE *fp, const char *fmt, va
116 * string ends is null-terminated.
117 */
118 static char *
119 -__wcsconv(wchar_t *wcsarg, int prec)
120 +__wcsconv(wchar_t *wcsarg, int prec, locale_t loc)
121 {
122 static const mbstate_t initial;
123 mbstate_t mbs;
124 @@ -218,7 +227,7 @@ __wcsconv(wchar_t *wcsarg, int prec)
125 if (prec < 0) {
126 p = wcsarg;
127 mbs = initial;
128 - nbytes = wcsrtombs(NULL, (const wchar_t **)&p, 0, &mbs);
129 + nbytes = wcsrtombs_l(NULL, (const wchar_t **)&p, 0, &mbs, loc);
130 if (nbytes == (size_t)-1)
131 return (NULL);
132 } else {
133 @@ -234,7 +243,7 @@ __wcsconv(wchar_t *wcsarg, int prec)
134 p = wcsarg;
135 mbs = initial;
136 for (;;) {
137 - clen = wcrtomb(buf, *p++, &mbs);
138 + clen = wcrtomb_l(buf, *p++, &mbs, loc);
139 if (clen == 0 || clen == (size_t)-1 ||
140 nbytes + clen > prec)
141 break;
142 @@ -248,8 +257,8 @@ __wcsconv(wchar_t *wcsarg, int prec)
143 /* Fill the output buffer. */
144 p = wcsarg;
145 mbs = initial;
146 - if ((nbytes = wcsrtombs(convbuf, (const wchar_t **)&p,
147 - nbytes, &mbs)) == (size_t)-1) {
148 + if ((nbytes = wcsrtombs_l(convbuf, (const wchar_t **)&p,
149 + nbytes, &mbs, loc)) == (size_t)-1) {
150 free(convbuf);
151 return (NULL);
152 }
153 @@ -261,22 +270,30 @@ __wcsconv(wchar_t *wcsarg, int prec)
154 * MT-safe version
155 */
156 int
157 -vfprintf(FILE * __restrict fp, const char * __restrict fmt0, va_list ap)
158 +vfprintf_l(FILE * __restrict fp, locale_t loc, const char * __restrict fmt0, va_list ap)
159
160 {
161 int ret;
162
163 + NORMALIZE_LOCALE(loc);
164 FLOCKFILE(fp);
165 /* optimise fprintf(stderr) (and other unbuffered Unix files) */
166 if ((fp->_flags & (__SNBF|__SWR|__SRW)) == (__SNBF|__SWR) &&
167 fp->_file >= 0)
168 - ret = __sbprintf(fp, fmt0, ap);
169 + ret = __sbprintf(fp, loc, fmt0, ap);
170 else
171 - ret = __vfprintf(fp, fmt0, ap);
172 + ret = __vfprintf(fp, loc, fmt0, ap);
173 FUNLOCKFILE(fp);
174 return (ret);
175 }
176
177 +int
178 +vfprintf(FILE * __restrict fp, const char * __restrict fmt0, va_list ap)
179 +
180 +{
181 + return vfprintf_l(fp, __current_locale(), fmt0, ap);
182 +}
183 +
184 /*
185 * The size of the buffer we use as scratch space for integer
186 * conversions, among other things. We need enough space to
187 @@ -291,8 +308,8 @@ vfprintf(FILE * __restrict fp, const cha
188 /*
189 * Non-MT-safe version
190 */
191 -int
192 -__vfprintf(FILE *fp, const char *fmt0, va_list ap)
193 +__private_extern__ int
194 +__vfprintf(FILE *fp, locale_t loc, const char *fmt0, va_list ap)
195 {
196 char *fmt; /* format string */
197 int ch; /* character from fmt */
198 @@ -335,6 +352,11 @@ __vfprintf(FILE *fp, const char *fmt0, v
199 char expstr[MAXEXPDIG+2]; /* buffer for exponent string: e+ZZZ */
200 char *dtoaresult; /* buffer allocated by dtoa */
201 #endif
202 +#ifdef VECTORS
203 + union arg vval; /* Vector argument. */
204 + char *pct; /* Pointer to '%' at beginning of specifier. */
205 + char vsep; /* Vector separator character. */
206 +#endif
207 u_long ulval; /* integer arguments %[diouxX] */
208 uintmax_t ujval; /* %j, %ll, %q, %t, %z integers */
209 int base; /* base for [diouxX] conversion */
210 @@ -357,19 +379,19 @@ __vfprintf(FILE *fp, const char *fmt0, v
211
212 /* BEWARE, these `goto error' on error. */
213 #define PRINT(ptr, len) { \
214 - if (io_print(&io, (ptr), (len))) \
215 + if (io_print(&io, (ptr), (len), loc)) \
216 goto error; \
217 }
218 #define PAD(howmany, with) { \
219 - if (io_pad(&io, (howmany), (with))) \
220 + if (io_pad(&io, (howmany), (with), loc)) \
221 goto error; \
222 }
223 #define PRINTANDPAD(p, ep, len, with) { \
224 - if (io_printandpad(&io, (p), (ep), (len), (with))) \
225 + if (io_printandpad(&io, (p), (ep), (len), (with), loc)) \
226 goto error; \
227 }
228 #define FLUSH() { \
229 - if (io_flush(&io)) \
230 + if (io_flush(&io, loc)) \
231 goto error; \
232 }
233
234 @@ -405,7 +427,7 @@ __vfprintf(FILE *fp, const char *fmt0, v
235 #define UJARG() \
236 (flags&INTMAXT ? GETARG(uintmax_t) : \
237 flags&SIZET ? (uintmax_t)GETARG(size_t) : \
238 - flags&PTRDIFFT ? (uintmax_t)GETARG(ptrdiff_t) : \
239 + flags&PTRDIFFT ? (uintmax_t)(unsigned long)GETARG(ptrdiff_t) : \
240 (uintmax_t)GETARG(unsigned long long))
241
242 /*
243 @@ -436,14 +458,19 @@ __vfprintf(FILE *fp, const char *fmt0, v
244 val = GETARG (int); \
245 }
246
247 +#if 0 // xprintf pending API review
248 if (__use_xprintf == 0 && getenv("USE_XPRINTF"))
249 __use_xprintf = 1;
250 if (__use_xprintf > 0)
251 - return (__xvprintf(fp, fmt0, ap));
252 + return (__xvprintf(fp, loc, fmt0, ap));
253 +#endif
254
255 /* sorry, fprintf(read_only_file, "") returns EOF, not 0 */
256 - if (prepwrite(fp) != 0)
257 + if (prepwrite(fp) != 0) {
258 + errno = EBADF;
259 return (EOF);
260 + }
261 + ORIENT(fp, -1);
262
263 convbuf = NULL;
264 fmt = (char *)fmt0;
265 @@ -454,7 +481,7 @@ __vfprintf(FILE *fp, const char *fmt0, v
266 ret = 0;
267 #ifndef NO_FLOATING_POINT
268 dtoaresult = NULL;
269 - decimal_point = localeconv()->decimal_point;
270 + decimal_point = localeconv_l(loc)->decimal_point;
271 /* The overwhelmingly common case is decpt_len == 1. */
272 decpt_len = (decimal_point[1] == '\0' ? 1 : strlen(decimal_point));
273 #endif
274 @@ -475,6 +502,9 @@ __vfprintf(FILE *fp, const char *fmt0, v
275 }
276 if (ch == '\0')
277 goto done;
278 +#ifdef VECTORS
279 + pct = fmt;
280 +#endif /* VECTORS */
281 fmt++; /* skip over '%' */
282
283 flags = 0;
284 @@ -484,6 +514,9 @@ __vfprintf(FILE *fp, const char *fmt0, v
285 gs.grouping = NULL;
286 sign = '\0';
287 ox[1] = '\0';
288 +#ifdef VECTORS
289 + vsep = 'X'; /* Illegal value, changed to defaults later. */
290 +#endif /* VECTORS */
291
292 rflag: ch = *fmt++;
293 reswitch: switch (ch) {
294 @@ -499,6 +532,11 @@ reswitch: switch (ch) {
295 case '#':
296 flags |= ALT;
297 goto rflag;
298 +#ifdef VECTORS
299 + case ',': case ';': case ':': case '_':
300 + vsep = ch;
301 + goto rflag;
302 +#endif /* VECTORS */
303 case '*':
304 /*-
305 * ``A negative field width argument is taken as a
306 @@ -595,14 +633,18 @@ reswitch: switch (ch) {
307 flags |= LONGINT;
308 /*FALLTHROUGH*/
309 case 'c':
310 +#ifdef VECTORS
311 + if (flags & VECTOR)
312 + break;
313 +#endif /* VECTORS */
314 if (flags & LONGINT) {
315 static const mbstate_t initial;
316 mbstate_t mbs;
317 size_t mbseqlen;
318
319 mbs = initial;
320 - mbseqlen = wcrtomb(cp = buf,
321 - (wchar_t)GETARG(wint_t), &mbs);
322 + mbseqlen = wcrtomb_l(cp = buf,
323 + (wchar_t)GETARG(wint_t), &mbs, loc);
324 if (mbseqlen == (size_t)-1) {
325 fp->_flags |= __SERR;
326 goto error;
327 @@ -619,6 +661,10 @@ reswitch: switch (ch) {
328 /*FALLTHROUGH*/
329 case 'd':
330 case 'i':
331 +#ifdef VECTORS
332 + if (flags & VECTOR)
333 + break;
334 +#endif /* VECTORS */
335 if (flags & INTMAX_SIZE) {
336 ujval = SJARG();
337 if ((intmax_t)ujval < 0) {
338 @@ -637,6 +683,12 @@ reswitch: switch (ch) {
339 #ifndef NO_FLOATING_POINT
340 case 'a':
341 case 'A':
342 +#ifdef VECTORS
343 + if (flags & VECTOR) {
344 + flags |= FPT;
345 + break;
346 + }
347 +#endif /* VECTORS */
348 if (ch == 'a') {
349 ox[1] = 'x';
350 xdigs = xdigs_lower;
351 @@ -650,6 +702,12 @@ reswitch: switch (ch) {
352 prec++;
353 if (dtoaresult != NULL)
354 freedtoa(dtoaresult);
355 +#ifdef LDBL_COMPAT
356 + fparg.dbl = GETARG(double);
357 + dtoaresult = cp =
358 + __hdtoa(fparg.dbl, xdigs, prec,
359 + &expt, &signflag, &dtoaend);
360 +#else /* !LDBL_COMPAT */
361 if (flags & LONGDBL) {
362 fparg.ldbl = GETARG(long double);
363 dtoaresult = cp =
364 @@ -661,6 +719,7 @@ reswitch: switch (ch) {
365 __hdtoa(fparg.dbl, xdigs, prec,
366 &expt, &signflag, &dtoaend);
367 }
368 +#endif /* LDBL_COMPAT */
369 if (prec < 0)
370 prec = dtoaend - cp;
371 if (expt == INT_MAX)
372 @@ -668,6 +727,12 @@ reswitch: switch (ch) {
373 goto fp_common;
374 case 'e':
375 case 'E':
376 +#ifdef VECTORS
377 + if (flags & VECTOR) {
378 + flags |= FPT;
379 + break;
380 + }
381 +#endif /* VECTORS */
382 expchar = ch;
383 if (prec < 0) /* account for digit before decpt */
384 prec = DEFPREC + 1;
385 @@ -676,10 +741,22 @@ reswitch: switch (ch) {
386 goto fp_begin;
387 case 'f':
388 case 'F':
389 +#ifdef VECTORS
390 + if (flags & VECTOR) {
391 + flags |= FPT;
392 + break;
393 + }
394 +#endif /* VECTORS */
395 expchar = '\0';
396 goto fp_begin;
397 case 'g':
398 case 'G':
399 +#ifdef VECTORS
400 + if (flags & VECTOR) {
401 + flags |= FPT;
402 + break;
403 + }
404 +#endif /* VECTORS */
405 expchar = ch - ('g' - 'e');
406 if (prec == 0)
407 prec = 1;
408 @@ -688,6 +765,14 @@ fp_begin:
409 prec = DEFPREC;
410 if (dtoaresult != NULL)
411 freedtoa(dtoaresult);
412 +#ifdef LDBL_COMPAT
413 + fparg.dbl = GETARG(double);
414 + dtoaresult = cp =
415 + dtoa(fparg.dbl, expchar ? 2 : 3, prec,
416 + &expt, &signflag, &dtoaend);
417 + if (expt == 9999)
418 + expt = INT_MAX;
419 +#else /* !LDBL_COMPAT */
420 if (flags & LONGDBL) {
421 fparg.ldbl = GETARG(long double);
422 dtoaresult = cp =
423 @@ -701,6 +786,7 @@ fp_begin:
424 if (expt == 9999)
425 expt = INT_MAX;
426 }
427 +#endif /* LDBL_COMPAT */
428 fp_common:
429 if (signflag)
430 sign = '-';
431 @@ -750,37 +836,46 @@ fp_common:
432 if (prec || flags & ALT)
433 size += prec + decpt_len;
434 if ((flags & GROUPING) && expt > 0)
435 - size += grouping_init(&gs, expt);
436 + size += grouping_init(&gs, expt, loc);
437 }
438 break;
439 #endif /* !NO_FLOATING_POINT */
440 case 'n':
441 + {
442 /*
443 * Assignment-like behavior is specified if the
444 * value overflows or is otherwise unrepresentable.
445 * C99 says to use `signed char' for %hhn conversions.
446 */
447 - if (flags & LLONGINT)
448 - *GETARG(long long *) = ret;
449 + void *ptr = GETARG(void *);
450 + if (ptr == NULL)
451 + continue;
452 + else if (flags & LLONGINT)
453 + *(long long *)ptr = ret;
454 else if (flags & SIZET)
455 - *GETARG(ssize_t *) = (ssize_t)ret;
456 + *(ssize_t *)ptr = (ssize_t)ret;
457 else if (flags & PTRDIFFT)
458 - *GETARG(ptrdiff_t *) = ret;
459 + *(ptrdiff_t *)ptr = ret;
460 else if (flags & INTMAXT)
461 - *GETARG(intmax_t *) = ret;
462 + *(intmax_t *)ptr = ret;
463 else if (flags & LONGINT)
464 - *GETARG(long *) = ret;
465 + *(long *)ptr = ret;
466 else if (flags & SHORTINT)
467 - *GETARG(short *) = ret;
468 + *(short *)ptr = ret;
469 else if (flags & CHARINT)
470 - *GETARG(signed char *) = ret;
471 + *(signed char *)ptr = ret;
472 else
473 - *GETARG(int *) = ret;
474 + *(int *)ptr = ret;
475 continue; /* no output */
476 + }
477 case 'O':
478 flags |= LONGINT;
479 /*FALLTHROUGH*/
480 case 'o':
481 +#ifdef VECTORS
482 + if (flags & VECTOR)
483 + break;
484 +#endif /* VECTORS */
485 if (flags & INTMAX_SIZE)
486 ujval = UJARG();
487 else
488 @@ -795,6 +890,10 @@ fp_common:
489 * defined manner.''
490 * -- ANSI X3J11
491 */
492 +#ifdef VECTORS
493 + if (flags & VECTOR)
494 + break;
495 +#endif /* VECTORS */
496 ujval = (uintmax_t)(uintptr_t)GETARG(void *);
497 base = 16;
498 xdigs = xdigs_lower;
499 @@ -813,7 +912,7 @@ fp_common:
500 if ((wcp = GETARG(wchar_t *)) == NULL)
501 cp = "(null)";
502 else {
503 - convbuf = __wcsconv(wcp, prec);
504 + convbuf = __wcsconv(wcp, prec, loc);
505 if (convbuf == NULL) {
506 fp->_flags |= __SERR;
507 goto error;
508 @@ -829,6 +928,10 @@ fp_common:
509 flags |= LONGINT;
510 /*FALLTHROUGH*/
511 case 'u':
512 +#ifdef VECTORS
513 + if (flags & VECTOR)
514 + break;
515 +#endif /* VECTORS */
516 if (flags & INTMAX_SIZE)
517 ujval = UJARG();
518 else
519 @@ -841,6 +944,10 @@ fp_common:
520 case 'x':
521 xdigs = xdigs_lower;
522 hex:
523 +#ifdef VECTORS
524 + if (flags & VECTOR)
525 + break;
526 +#endif /* VECTORS */
527 if (flags & INTMAX_SIZE)
528 ujval = UJARG();
529 else
530 @@ -858,6 +965,7 @@ nosign: sign = '\0';
531 * ``... diouXx conversions ... if a precision is
532 * specified, the 0 flag will be ignored.''
533 * -- ANSI X3J11
534 + * except for %#.0o and zero value
535 */
536 number: if ((dprec = prec) >= 0)
537 flags &= ~ZEROPAD;
538 @@ -885,10 +993,15 @@ number: if ((dprec = prec) >= 0)
539 }
540 size = buf + BUF - cp;
541 if (size > BUF) /* should never happen */
542 - abort();
543 + LIBC_ABORT("size (%d) > BUF (%d)", size, BUF);
544 if ((flags & GROUPING) && size != 0)
545 - size += grouping_init(&gs, size);
546 + size += grouping_init(&gs, size, loc);
547 break;
548 +#ifdef VECTORS
549 + case 'v':
550 + flags |= VECTOR;
551 + goto rflag;
552 +#endif /* VECTORS */
553 default: /* "%?" prints ?, unless ? is NUL */
554 if (ch == '\0')
555 goto done;
556 @@ -900,6 +1013,290 @@ number: if ((dprec = prec) >= 0)
557 break;
558 }
559
560 +#ifdef VECTORS
561 + if (flags & VECTOR) {
562 + /*
563 + * Do the minimum amount of work necessary to construct
564 + * a format specifier that can be used to recursively
565 + * call vfprintf() for each element in the vector.
566 + */
567 + int i, j; /* Counter. */
568 + int vcnt; /* Number of elements in vector. */
569 + char *vfmt; /* Pointer to format specifier. */
570 +#define EXTRAHH 2
571 + char vfmt_buf[32 + EXTRAHH]; /* Static buffer for format spec. */
572 + int vwidth = 0; /* Width specified via '*'. */
573 + int vprec = 0; /* Precision specified via '*'. */
574 + char *vstr; /* Used for asprintf(). */
575 + int vlen; /* Length returned by asprintf(). */
576 + enum {
577 + V_CHAR, V_SHORT, V_INT,
578 + V_PCHAR, V_PSHORT, V_PINT,
579 + V_FLOAT,
580 +#ifdef V64TYPE
581 + V_LONGLONG, V_PLONGLONG,
582 + V_DOUBLE,
583 +#endif /* V64TYPE */
584 + } vtype;
585 +
586 + vval.vectorarg = GETARG(VECTORTYPE);
587 + /*
588 + * Set vfmt. If vfmt_buf may not be big enough,
589 + * malloc() space, taking care to free it later.
590 + * (EXTRAHH is for possible extra "hh")
591 + */
592 + if (&fmt[-1] - pct + EXTRAHH < sizeof(vfmt_buf))
593 + vfmt = vfmt_buf;
594 + else
595 + vfmt = (char *)malloc(&fmt[-1] - pct + EXTRAHH + 1);
596 +
597 + /* Set the separator character, if not specified. */
598 + if (vsep == 'X') {
599 + if (ch == 'c')
600 + vsep = '\0';
601 + else
602 + vsep = ' ';
603 + }
604 +
605 + /* Create the format specifier. */
606 + for (i = j = 0; i < &fmt[-1] - pct; i++) {
607 + switch (pct[i]) {
608 + case ',': case ';': case ':': case '_':
609 + case 'v': case 'h': case 'l':
610 + /* Ignore. */
611 + break;
612 + case '*':
613 + if (pct[i - 1] != '.')
614 + vwidth = 1;
615 + else
616 + vprec = 1;
617 + /* FALLTHROUGH */
618 + default:
619 + vfmt[j++] = pct[i];
620 + }
621 + }
622 +
623 + /*
624 + * Determine the number of elements in the vector and
625 + * finish up the format specifier.
626 + */
627 + if (flags & SHORTINT) {
628 + switch (ch) {
629 + case 'c':
630 + vtype = V_SHORT;
631 + break;
632 + case 'p':
633 + vtype = V_PSHORT;
634 + break;
635 + default:
636 + vfmt[j++] = 'h';
637 + vtype = V_SHORT;
638 + break;
639 + }
640 + vcnt = 8;
641 + } else if (flags & LONGINT) {
642 + vcnt = 4;
643 + vtype = (ch == 'p') ? V_PINT : V_INT;
644 +#ifdef V64TYPE
645 + } else if (flags & LLONGINT) {
646 + switch (ch) {
647 + case 'a':
648 + case 'A':
649 + case 'e':
650 + case 'E':
651 + case 'f':
652 + case 'g':
653 + case 'G':
654 + vcnt = 2;
655 + vtype = V_DOUBLE;
656 + break;
657 + case 'd':
658 + case 'i':
659 + case 'u':
660 + case 'o':
661 + case 'p':
662 + case 'x':
663 + case 'X':
664 + vfmt[j++] = 'l';
665 + vfmt[j++] = 'l';
666 + vcnt = 2;
667 + vtype = (ch == 'p') ? V_PLONGLONG : V_LONGLONG;
668 + break;
669 + default:
670 + /*
671 + * The default case should never
672 + * happen.
673 + */
674 + case 'c':
675 + vcnt = 16;
676 + vtype = V_CHAR;
677 + }
678 +#endif /* V64TYPE */
679 + } else {
680 + switch (ch) {
681 + case 'a':
682 + case 'A':
683 + case 'e':
684 + case 'E':
685 + case 'f':
686 + case 'g':
687 + case 'G':
688 + vcnt = 4;
689 + vtype = V_FLOAT;
690 + break;
691 + default:
692 + /*
693 + * The default case should never
694 + * happen.
695 + */
696 + case 'd':
697 + case 'i':
698 + case 'u':
699 + case 'o':
700 + case 'x':
701 + case 'X':
702 + vfmt[j++] = 'h';
703 + vfmt[j++] = 'h';
704 + /* drop through */
705 + case 'p':
706 + case 'c':
707 + vcnt = 16;
708 + vtype = (ch == 'p') ? V_PCHAR : V_CHAR;
709 + }
710 + }
711 + vfmt[j++] = ch;
712 + vfmt[j++] = '\0';
713 +
714 +/* Get a vector element. */
715 +#ifdef V64TYPE
716 +#define VPRINT(type, ind, args...) do { \
717 + switch (type) { \
718 + case V_CHAR: \
719 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vuchararg[ind]); \
720 + break; \
721 + case V_PCHAR: \
722 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, (void *)(uintptr_t)vval.vuchararg[ind]); \
723 + break; \
724 + case V_SHORT: \
725 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vushortarg[ind]); \
726 + break; \
727 + case V_PSHORT: \
728 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, (void *)(uintptr_t)vval.vushortarg[ind]); \
729 + break; \
730 + case V_INT: \
731 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vuintarg[ind]); \
732 + break; \
733 + case V_PINT: \
734 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, (void *)(uintptr_t)vval.vuintarg[ind]); \
735 + break; \
736 + case V_LONGLONG: \
737 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vulonglongarg[ind]); \
738 + break; \
739 + case V_PLONGLONG: \
740 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, (void *)(uintptr_t)vval.vulonglongarg[ind]); \
741 + break; \
742 + case V_FLOAT: \
743 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vfloatarg[ind]); \
744 + break; \
745 + case V_DOUBLE: \
746 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vdoublearg[ind]); \
747 + break; \
748 + } \
749 + ret += vlen; \
750 + PRINT(vstr, vlen); \
751 + FLUSH(); \
752 + free(vstr); \
753 +} while (0)
754 +#else /* !V64TYPE */
755 +#define VPRINT(type, ind, args...) do { \
756 + switch (type) { \
757 + case V_CHAR: \
758 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vuchararg[ind]); \
759 + break; \
760 + case V_PCHAR: \
761 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, (void *)(uintptr_t)vval.vuchararg[ind]); \
762 + break; \
763 + case V_SHORT: \
764 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vushortarg[ind]); \
765 + break; \
766 + case V_PSHORT: \
767 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, (void *)(uintptr_t)vval.vushortarg[ind]); \
768 + break; \
769 + case V_INT: \
770 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vuintarg[ind]); \
771 + break; \
772 + case V_PINT: \
773 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, (void *)(uintptr_t)vval.vuintarg[ind]); \
774 + break; \
775 + case V_FLOAT: \
776 + vlen = asprintf_l(&vstr, loc, vfmt , ## args, vval.vfloatarg[ind]); \
777 + break; \
778 + } \
779 + ret += vlen; \
780 + PRINT(vstr, vlen); \
781 + FLUSH(); \
782 + free(vstr); \
783 +} while (0)
784 +#endif /* V64TYPE */
785 +
786 + /* Actually print. */
787 + if (vwidth == 0) {
788 + if (vprec == 0) {
789 + /* First element. */
790 + VPRINT(vtype, 0);
791 + for (i = 1; i < vcnt; i++) {
792 + /* Separator. */
793 + if(vsep)
794 + PRINT(&vsep, 1);
795 +
796 + /* Element. */
797 + VPRINT(vtype, i);
798 + }
799 + } else {
800 + /* First element. */
801 + VPRINT(vtype, 0, prec);
802 + for (i = 1; i < vcnt; i++) {
803 + /* Separator. */
804 + if(vsep)
805 + PRINT(&vsep, 1);
806 +
807 + /* Element. */
808 + VPRINT(vtype, i, prec);
809 + }
810 + }
811 + } else {
812 + if (vprec == 0) {
813 + /* First element. */
814 + VPRINT(vtype, 0, width);
815 + for (i = 1; i < vcnt; i++) {
816 + /* Separator. */
817 + if(vsep)
818 + PRINT(&vsep, 1);
819 +
820 + /* Element. */
821 + VPRINT(vtype, i, width);
822 + }
823 + } else {
824 + /* First element. */
825 + VPRINT(vtype, 0, width, prec);
826 + for (i = 1; i < vcnt; i++) {
827 + /* Separator. */
828 + if(vsep)
829 + PRINT(&vsep, 1);
830 +
831 + /* Element. */
832 + VPRINT(vtype, i, width, prec);
833 + }
834 + }
835 + }
836 +#undef VPRINT
837 +
838 + if (vfmt != vfmt_buf)
839 + free(vfmt);
840 +
841 + continue;
842 + }
843 +#endif /* VECTORS */
844 /*
845 * All reasonable formats wind up here. At this point, `cp'
846 * points to a string which (if not flags&LADJUST) should be
847 @@ -950,7 +1347,7 @@ number: if ((dprec = prec) >= 0)
848 /* leading zeroes from decimal precision */
849 PAD(dprec - size, zeroes);
850 if (gs.grouping) {
851 - if (grouping_print(&gs, &io, cp, buf+BUF) < 0)
852 + if (grouping_print(&gs, &io, cp, buf+BUF, loc) < 0)
853 goto error;
854 } else {
855 PRINT(cp, size);
856 @@ -968,7 +1365,7 @@ number: if ((dprec = prec) >= 0)
857 } else {
858 if (gs.grouping) {
859 n = grouping_print(&gs, &io,
860 - cp, dtoaend);
861 + cp, dtoaend, loc);
862 if (n < 0)
863 goto error;
864 cp += n;