14a5d661aSToomas Soome /*-
24a5d661aSToomas Soome * Copyright (c) 1986, 1988, 1991, 1993
34a5d661aSToomas Soome * The Regents of the University of California. All rights reserved.
44a5d661aSToomas Soome * (c) UNIX System Laboratories, Inc.
54a5d661aSToomas Soome * All or some portions of this file are derived from material licensed
64a5d661aSToomas Soome * to the University of California by American Telephone and Telegraph
74a5d661aSToomas Soome * Co. or Unix System Laboratories, Inc. and are reproduced herein with
84a5d661aSToomas Soome * the permission of UNIX System Laboratories, Inc.
94a5d661aSToomas Soome *
104a5d661aSToomas Soome * Redistribution and use in source and binary forms, with or without
114a5d661aSToomas Soome * modification, are permitted provided that the following conditions
124a5d661aSToomas Soome * are met:
134a5d661aSToomas Soome * 1. Redistributions of source code must retain the above copyright
144a5d661aSToomas Soome * notice, this list of conditions and the following disclaimer.
154a5d661aSToomas Soome * 2. Redistributions in binary form must reproduce the above copyright
164a5d661aSToomas Soome * notice, this list of conditions and the following disclaimer in the
174a5d661aSToomas Soome * documentation and/or other materials provided with the distribution.
184a5d661aSToomas Soome * 4. Neither the name of the University nor the names of its contributors
194a5d661aSToomas Soome * may be used to endorse or promote products derived from this software
204a5d661aSToomas Soome * without specific prior written permission.
214a5d661aSToomas Soome *
224a5d661aSToomas Soome * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
234a5d661aSToomas Soome * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
244a5d661aSToomas Soome * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
254a5d661aSToomas Soome * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
264a5d661aSToomas Soome * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
274a5d661aSToomas Soome * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
284a5d661aSToomas Soome * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
294a5d661aSToomas Soome * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
304a5d661aSToomas Soome * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
314a5d661aSToomas Soome * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
324a5d661aSToomas Soome * SUCH DAMAGE.
334a5d661aSToomas Soome *
344a5d661aSToomas Soome * @(#)subr_prf.c 8.3 (Berkeley) 1/21/94
354a5d661aSToomas Soome */
364a5d661aSToomas Soome
374a5d661aSToomas Soome #include <sys/cdefs.h>
384a5d661aSToomas Soome
394a5d661aSToomas Soome /*
404a5d661aSToomas Soome * Standaloneified version of the FreeBSD kernel printf family.
414a5d661aSToomas Soome */
424a5d661aSToomas Soome
434a5d661aSToomas Soome #include <sys/types.h>
444a5d661aSToomas Soome #include <sys/stddef.h>
454a5d661aSToomas Soome #include <sys/stdint.h>
464a5d661aSToomas Soome #include <limits.h>
474a5d661aSToomas Soome #include <string.h>
484a5d661aSToomas Soome #include "stand.h"
494a5d661aSToomas Soome
504a5d661aSToomas Soome /*
514a5d661aSToomas Soome * Note that stdarg.h and the ANSI style va_start macro is used for both
524a5d661aSToomas Soome * ANSI and traditional C compilers.
534a5d661aSToomas Soome */
544a5d661aSToomas Soome #include <machine/stdarg.h>
554a5d661aSToomas Soome
564a5d661aSToomas Soome #define MAXNBUF (sizeof(intmax_t) * CHAR_BIT + 1)
574a5d661aSToomas Soome
584a5d661aSToomas Soome typedef void (kvprintf_fn_t)(int, void *);
594a5d661aSToomas Soome
604a5d661aSToomas Soome static char *ksprintn (char *buf, uintmax_t num, int base, int *len, int upper);
614a5d661aSToomas Soome static int kvprintf(char const *fmt, kvprintf_fn_t *func, void *arg, int radix, va_list ap);
624a5d661aSToomas Soome
634a5d661aSToomas Soome static void
putchar_wrapper(int cc,void * arg)644a5d661aSToomas Soome putchar_wrapper(int cc, void *arg)
654a5d661aSToomas Soome {
664a5d661aSToomas Soome
674a5d661aSToomas Soome putchar(cc);
684a5d661aSToomas Soome }
694a5d661aSToomas Soome
704a5d661aSToomas Soome int
printf(const char * fmt,...)714a5d661aSToomas Soome printf(const char *fmt, ...)
724a5d661aSToomas Soome {
734a5d661aSToomas Soome va_list ap;
744a5d661aSToomas Soome int retval;
754a5d661aSToomas Soome
764a5d661aSToomas Soome va_start(ap, fmt);
774a5d661aSToomas Soome retval = kvprintf(fmt, putchar_wrapper, NULL, 10, ap);
784a5d661aSToomas Soome va_end(ap);
794a5d661aSToomas Soome return retval;
804a5d661aSToomas Soome }
814a5d661aSToomas Soome
824a5d661aSToomas Soome void
vprintf(const char * fmt,va_list ap)834a5d661aSToomas Soome vprintf(const char *fmt, va_list ap)
844a5d661aSToomas Soome {
854a5d661aSToomas Soome
864a5d661aSToomas Soome kvprintf(fmt, putchar_wrapper, NULL, 10, ap);
874a5d661aSToomas Soome }
884a5d661aSToomas Soome
894a5d661aSToomas Soome int
sprintf(char * buf,const char * cfmt,...)904a5d661aSToomas Soome sprintf(char *buf, const char *cfmt, ...)
914a5d661aSToomas Soome {
924a5d661aSToomas Soome int retval;
934a5d661aSToomas Soome va_list ap;
944a5d661aSToomas Soome
954a5d661aSToomas Soome va_start(ap, cfmt);
964a5d661aSToomas Soome retval = kvprintf(cfmt, NULL, (void *)buf, 10, ap);
974a5d661aSToomas Soome buf[retval] = '\0';
984a5d661aSToomas Soome va_end(ap);
994a5d661aSToomas Soome return retval;
1004a5d661aSToomas Soome }
1014a5d661aSToomas Soome
1024a5d661aSToomas Soome struct print_buf {
1034a5d661aSToomas Soome char *buf;
1044a5d661aSToomas Soome size_t size;
1054a5d661aSToomas Soome };
1064a5d661aSToomas Soome
1074a5d661aSToomas Soome static void
snprint_func(int ch,void * arg)1084a5d661aSToomas Soome snprint_func(int ch, void *arg)
1094a5d661aSToomas Soome {
1104a5d661aSToomas Soome struct print_buf *pbuf = arg;
1114a5d661aSToomas Soome
1124a5d661aSToomas Soome if (pbuf->size < 2) {
1134a5d661aSToomas Soome /*
1144a5d661aSToomas Soome * Reserve last buffer position for the terminating
1154a5d661aSToomas Soome * character:
1164a5d661aSToomas Soome */
1174a5d661aSToomas Soome return;
1184a5d661aSToomas Soome }
1194a5d661aSToomas Soome *(pbuf->buf)++ = ch;
1204a5d661aSToomas Soome pbuf->size--;
1214a5d661aSToomas Soome }
1224a5d661aSToomas Soome
1234a5d661aSToomas Soome int
asprintf(char ** buf,const char * cfmt,...)124*45ac0b4bSToomas Soome asprintf(char **buf, const char *cfmt, ...)
125*45ac0b4bSToomas Soome {
126*45ac0b4bSToomas Soome int retval;
127*45ac0b4bSToomas Soome struct print_buf arg;
128*45ac0b4bSToomas Soome va_list ap;
129*45ac0b4bSToomas Soome
130*45ac0b4bSToomas Soome *buf = NULL;
131*45ac0b4bSToomas Soome va_start(ap, cfmt);
132*45ac0b4bSToomas Soome retval = kvprintf(cfmt, NULL, NULL, 10, ap);
133*45ac0b4bSToomas Soome va_end(ap);
134*45ac0b4bSToomas Soome if (retval <= 0)
135*45ac0b4bSToomas Soome return (-1);
136*45ac0b4bSToomas Soome
137*45ac0b4bSToomas Soome arg.size = retval + 1;
138*45ac0b4bSToomas Soome arg.buf = *buf = malloc(arg.size);
139*45ac0b4bSToomas Soome if (*buf == NULL)
140*45ac0b4bSToomas Soome return (-1);
141*45ac0b4bSToomas Soome
142*45ac0b4bSToomas Soome va_start(ap, cfmt);
143*45ac0b4bSToomas Soome retval = kvprintf(cfmt, &snprint_func, &arg, 10, ap);
144*45ac0b4bSToomas Soome va_end(ap);
145*45ac0b4bSToomas Soome
146*45ac0b4bSToomas Soome if (arg.size >= 1)
147*45ac0b4bSToomas Soome *(arg.buf)++ = 0;
148*45ac0b4bSToomas Soome return (retval);
149*45ac0b4bSToomas Soome }
150*45ac0b4bSToomas Soome
151*45ac0b4bSToomas Soome int
snprintf(char * buf,size_t size,const char * cfmt,...)1524a5d661aSToomas Soome snprintf(char *buf, size_t size, const char *cfmt, ...)
1534a5d661aSToomas Soome {
1544a5d661aSToomas Soome int retval;
1554a5d661aSToomas Soome va_list ap;
1564a5d661aSToomas Soome struct print_buf arg;
1574a5d661aSToomas Soome
1584a5d661aSToomas Soome arg.buf = buf;
1594a5d661aSToomas Soome arg.size = size;
1604a5d661aSToomas Soome
1614a5d661aSToomas Soome va_start(ap, cfmt);
1624a5d661aSToomas Soome retval = kvprintf(cfmt, &snprint_func, &arg, 10, ap);
1634a5d661aSToomas Soome va_end(ap);
1644a5d661aSToomas Soome
1654a5d661aSToomas Soome if (arg.size >= 1)
1664a5d661aSToomas Soome *(arg.buf)++ = 0;
1674a5d661aSToomas Soome return retval;
1684a5d661aSToomas Soome }
1694a5d661aSToomas Soome
1704a5d661aSToomas Soome void
vsprintf(char * buf,const char * cfmt,va_list ap)1714a5d661aSToomas Soome vsprintf(char *buf, const char *cfmt, va_list ap)
1724a5d661aSToomas Soome {
1734a5d661aSToomas Soome int retval;
1744a5d661aSToomas Soome
1754a5d661aSToomas Soome retval = kvprintf(cfmt, NULL, (void *)buf, 10, ap);
1764a5d661aSToomas Soome buf[retval] = '\0';
1774a5d661aSToomas Soome }
1784a5d661aSToomas Soome
1794a5d661aSToomas Soome void
vsnprintf(char * buf,size_t size,const char * cfmt,va_list ap)1804a5d661aSToomas Soome vsnprintf(char *buf, size_t size, const char *cfmt, va_list ap)
1814a5d661aSToomas Soome {
1824a5d661aSToomas Soome int retval;
1834a5d661aSToomas Soome struct print_buf arg;
1844a5d661aSToomas Soome
1854a5d661aSToomas Soome arg.buf = buf;
1864a5d661aSToomas Soome arg.size = size;
1874a5d661aSToomas Soome
1884a5d661aSToomas Soome retval = kvprintf(cfmt, &snprint_func, &arg, 10, ap);
1894a5d661aSToomas Soome buf[retval] = '\0';
1904a5d661aSToomas Soome }
1914a5d661aSToomas Soome
1924a5d661aSToomas Soome /*
1934a5d661aSToomas Soome * Put a NUL-terminated ASCII number (base <= 36) in a buffer in reverse
1944a5d661aSToomas Soome * order; return an optional length and a pointer to the last character
1954a5d661aSToomas Soome * written in the buffer (i.e., the first character of the string).
1964a5d661aSToomas Soome * The buffer pointed to by `nbuf' must have length >= MAXNBUF.
1974a5d661aSToomas Soome */
1984a5d661aSToomas Soome static char *
ksprintn(char * nbuf,uintmax_t num,int base,int * lenp,int upper)1994a5d661aSToomas Soome ksprintn(char *nbuf, uintmax_t num, int base, int *lenp, int upper)
2004a5d661aSToomas Soome {
2014a5d661aSToomas Soome char *p, c;
2024a5d661aSToomas Soome
2034a5d661aSToomas Soome p = nbuf;
2044a5d661aSToomas Soome *p = '\0';
2054a5d661aSToomas Soome do {
2064a5d661aSToomas Soome c = hex2ascii(num % base);
2074a5d661aSToomas Soome *++p = upper ? toupper(c) : c;
2084a5d661aSToomas Soome } while (num /= base);
2094a5d661aSToomas Soome if (lenp)
2104a5d661aSToomas Soome *lenp = p - nbuf;
2114a5d661aSToomas Soome return (p);
2124a5d661aSToomas Soome }
2134a5d661aSToomas Soome
2144a5d661aSToomas Soome /*
2154a5d661aSToomas Soome * Scaled down version of printf(3).
2164a5d661aSToomas Soome *
2174a5d661aSToomas Soome * Two additional formats:
2184a5d661aSToomas Soome *
2194a5d661aSToomas Soome * The format %b is supported to decode error registers.
2204a5d661aSToomas Soome * Its usage is:
2214a5d661aSToomas Soome *
2224a5d661aSToomas Soome * printf("reg=%b\n", regval, "<base><arg>*");
2234a5d661aSToomas Soome *
2244a5d661aSToomas Soome * where <base> is the output base expressed as a control character, e.g.
2254a5d661aSToomas Soome * \10 gives octal; \20 gives hex. Each arg is a sequence of characters,
2264a5d661aSToomas Soome * the first of which gives the bit number to be inspected (origin 1), and
2274a5d661aSToomas Soome * the next characters (up to a control character, i.e. a character <= 32),
2284a5d661aSToomas Soome * give the name of the register. Thus:
2294a5d661aSToomas Soome *
2304a5d661aSToomas Soome * kvprintf("reg=%b\n", 3, "\10\2BITTWO\1BITONE");
2314a5d661aSToomas Soome *
2324a5d661aSToomas Soome * would produce output:
2334a5d661aSToomas Soome *
2344a5d661aSToomas Soome * reg=3<BITTWO,BITONE>
2354a5d661aSToomas Soome *
2364a5d661aSToomas Soome * XXX: %D -- Hexdump, takes pointer and separator string:
2374a5d661aSToomas Soome * ("%6D", ptr, ":") -> XX:XX:XX:XX:XX:XX
2384a5d661aSToomas Soome * ("%*D", len, ptr, " " -> XX XX XX XX ...
2394a5d661aSToomas Soome */
2404a5d661aSToomas Soome static int
kvprintf(char const * fmt,kvprintf_fn_t * func,void * arg,int radix,va_list ap)2414a5d661aSToomas Soome kvprintf(char const *fmt, kvprintf_fn_t *func, void *arg, int radix, va_list ap)
2424a5d661aSToomas Soome {
243*45ac0b4bSToomas Soome #define PCHAR(c) { \
244*45ac0b4bSToomas Soome int cc = (c); \
245*45ac0b4bSToomas Soome \
246*45ac0b4bSToomas Soome if (func) { \
247*45ac0b4bSToomas Soome (*func)(cc, arg); \
248*45ac0b4bSToomas Soome } else if (d != NULL) { \
249*45ac0b4bSToomas Soome *d++ = cc; \
250*45ac0b4bSToomas Soome } \
251*45ac0b4bSToomas Soome retval++; \
252*45ac0b4bSToomas Soome }
253*45ac0b4bSToomas Soome
2544a5d661aSToomas Soome char nbuf[MAXNBUF];
2554a5d661aSToomas Soome char *d;
2564a5d661aSToomas Soome const char *p, *percent, *q;
257c5121490SToomas Soome uint16_t *S;
2584a5d661aSToomas Soome u_char *up;
2594a5d661aSToomas Soome int ch, n;
2604a5d661aSToomas Soome uintmax_t num;
2614a5d661aSToomas Soome int base, lflag, qflag, tmp, width, ladjust, sharpflag, neg, sign, dot;
2624a5d661aSToomas Soome int cflag, hflag, jflag, tflag, zflag;
2634a5d661aSToomas Soome int dwidth, upper;
2644a5d661aSToomas Soome char padc;
2654a5d661aSToomas Soome int stop = 0, retval = 0;
2664a5d661aSToomas Soome
2674a5d661aSToomas Soome num = 0;
2684a5d661aSToomas Soome if (!func)
2694a5d661aSToomas Soome d = (char *) arg;
2704a5d661aSToomas Soome else
2714a5d661aSToomas Soome d = NULL;
2724a5d661aSToomas Soome
2734a5d661aSToomas Soome if (fmt == NULL)
2744a5d661aSToomas Soome fmt = "(fmt null)\n";
2754a5d661aSToomas Soome
2764a5d661aSToomas Soome if (radix < 2 || radix > 36)
2774a5d661aSToomas Soome radix = 10;
2784a5d661aSToomas Soome
2794a5d661aSToomas Soome for (;;) {
2804a5d661aSToomas Soome padc = ' ';
2814a5d661aSToomas Soome width = 0;
2824a5d661aSToomas Soome while ((ch = (u_char)*fmt++) != '%' || stop) {
2834a5d661aSToomas Soome if (ch == '\0')
2844a5d661aSToomas Soome return (retval);
2854a5d661aSToomas Soome PCHAR(ch);
2864a5d661aSToomas Soome }
2874a5d661aSToomas Soome percent = fmt - 1;
2884a5d661aSToomas Soome qflag = 0; lflag = 0; ladjust = 0; sharpflag = 0; neg = 0;
2894a5d661aSToomas Soome sign = 0; dot = 0; dwidth = 0; upper = 0;
2904a5d661aSToomas Soome cflag = 0; hflag = 0; jflag = 0; tflag = 0; zflag = 0;
2914a5d661aSToomas Soome reswitch: switch (ch = (u_char)*fmt++) {
2924a5d661aSToomas Soome case '.':
2934a5d661aSToomas Soome dot = 1;
2944a5d661aSToomas Soome goto reswitch;
2954a5d661aSToomas Soome case '#':
2964a5d661aSToomas Soome sharpflag = 1;
2974a5d661aSToomas Soome goto reswitch;
2984a5d661aSToomas Soome case '+':
2994a5d661aSToomas Soome sign = 1;
3004a5d661aSToomas Soome goto reswitch;
3014a5d661aSToomas Soome case '-':
3024a5d661aSToomas Soome ladjust = 1;
3034a5d661aSToomas Soome goto reswitch;
3044a5d661aSToomas Soome case '%':
3054a5d661aSToomas Soome PCHAR(ch);
3064a5d661aSToomas Soome break;
3074a5d661aSToomas Soome case '*':
3084a5d661aSToomas Soome if (!dot) {
3094a5d661aSToomas Soome width = va_arg(ap, int);
3104a5d661aSToomas Soome if (width < 0) {
3114a5d661aSToomas Soome ladjust = !ladjust;
3124a5d661aSToomas Soome width = -width;
3134a5d661aSToomas Soome }
3144a5d661aSToomas Soome } else {
3154a5d661aSToomas Soome dwidth = va_arg(ap, int);
3164a5d661aSToomas Soome }
3174a5d661aSToomas Soome goto reswitch;
3184a5d661aSToomas Soome case '0':
3194a5d661aSToomas Soome if (!dot) {
3204a5d661aSToomas Soome padc = '0';
3214a5d661aSToomas Soome goto reswitch;
3224a5d661aSToomas Soome }
3234a5d661aSToomas Soome case '1': case '2': case '3': case '4':
3244a5d661aSToomas Soome case '5': case '6': case '7': case '8': case '9':
3254a5d661aSToomas Soome for (n = 0;; ++fmt) {
3264a5d661aSToomas Soome n = n * 10 + ch - '0';
3274a5d661aSToomas Soome ch = *fmt;
3284a5d661aSToomas Soome if (ch < '0' || ch > '9')
3294a5d661aSToomas Soome break;
3304a5d661aSToomas Soome }
3314a5d661aSToomas Soome if (dot)
3324a5d661aSToomas Soome dwidth = n;
3334a5d661aSToomas Soome else
3344a5d661aSToomas Soome width = n;
3354a5d661aSToomas Soome goto reswitch;
3364a5d661aSToomas Soome case 'b':
3374a5d661aSToomas Soome num = (u_int)va_arg(ap, int);
3384a5d661aSToomas Soome p = va_arg(ap, char *);
3394a5d661aSToomas Soome for (q = ksprintn(nbuf, num, *p++, NULL, 0); *q;)
3404a5d661aSToomas Soome PCHAR(*q--);
3414a5d661aSToomas Soome
3424a5d661aSToomas Soome if (num == 0)
3434a5d661aSToomas Soome break;
3444a5d661aSToomas Soome
3454a5d661aSToomas Soome for (tmp = 0; *p;) {
3464a5d661aSToomas Soome n = *p++;
3474a5d661aSToomas Soome if (num & (1 << (n - 1))) {
3484a5d661aSToomas Soome PCHAR(tmp ? ',' : '<');
3494a5d661aSToomas Soome for (; (n = *p) > ' '; ++p)
3504a5d661aSToomas Soome PCHAR(n);
3514a5d661aSToomas Soome tmp = 1;
3524a5d661aSToomas Soome } else
3534a5d661aSToomas Soome for (; *p > ' '; ++p)
3544a5d661aSToomas Soome continue;
3554a5d661aSToomas Soome }
3564a5d661aSToomas Soome if (tmp)
3574a5d661aSToomas Soome PCHAR('>');
3584a5d661aSToomas Soome break;
3594a5d661aSToomas Soome case 'c':
3604a5d661aSToomas Soome PCHAR(va_arg(ap, int));
3614a5d661aSToomas Soome break;
3624a5d661aSToomas Soome case 'D':
3634a5d661aSToomas Soome up = va_arg(ap, u_char *);
3644a5d661aSToomas Soome p = va_arg(ap, char *);
3654a5d661aSToomas Soome if (!width)
3664a5d661aSToomas Soome width = 16;
3674a5d661aSToomas Soome while(width--) {
3684a5d661aSToomas Soome PCHAR(hex2ascii(*up >> 4));
3694a5d661aSToomas Soome PCHAR(hex2ascii(*up & 0x0f));
3704a5d661aSToomas Soome up++;
3714a5d661aSToomas Soome if (width)
3724a5d661aSToomas Soome for (q=p;*q;q++)
3734a5d661aSToomas Soome PCHAR(*q);
3744a5d661aSToomas Soome }
3754a5d661aSToomas Soome break;
3764a5d661aSToomas Soome case 'd':
3774a5d661aSToomas Soome case 'i':
3784a5d661aSToomas Soome base = 10;
3794a5d661aSToomas Soome sign = 1;
3804a5d661aSToomas Soome goto handle_sign;
3814a5d661aSToomas Soome case 'h':
3824a5d661aSToomas Soome if (hflag) {
3834a5d661aSToomas Soome hflag = 0;
3844a5d661aSToomas Soome cflag = 1;
3854a5d661aSToomas Soome } else
3864a5d661aSToomas Soome hflag = 1;
3874a5d661aSToomas Soome goto reswitch;
3884a5d661aSToomas Soome case 'j':
3894a5d661aSToomas Soome jflag = 1;
3904a5d661aSToomas Soome goto reswitch;
3914a5d661aSToomas Soome case 'l':
3924a5d661aSToomas Soome if (lflag) {
3934a5d661aSToomas Soome lflag = 0;
3944a5d661aSToomas Soome qflag = 1;
3954a5d661aSToomas Soome } else
3964a5d661aSToomas Soome lflag = 1;
3974a5d661aSToomas Soome goto reswitch;
3984a5d661aSToomas Soome case 'n':
3994a5d661aSToomas Soome if (jflag)
4004a5d661aSToomas Soome *(va_arg(ap, intmax_t *)) = retval;
4014a5d661aSToomas Soome else if (qflag)
4024a5d661aSToomas Soome *(va_arg(ap, quad_t *)) = retval;
4034a5d661aSToomas Soome else if (lflag)
4044a5d661aSToomas Soome *(va_arg(ap, long *)) = retval;
4054a5d661aSToomas Soome else if (zflag)
4064a5d661aSToomas Soome *(va_arg(ap, size_t *)) = retval;
4074a5d661aSToomas Soome else if (hflag)
4084a5d661aSToomas Soome *(va_arg(ap, short *)) = retval;
4094a5d661aSToomas Soome else if (cflag)
4104a5d661aSToomas Soome *(va_arg(ap, char *)) = retval;
4114a5d661aSToomas Soome else
4124a5d661aSToomas Soome *(va_arg(ap, int *)) = retval;
4134a5d661aSToomas Soome break;
4144a5d661aSToomas Soome case 'o':
4154a5d661aSToomas Soome base = 8;
4164a5d661aSToomas Soome goto handle_nosign;
4174a5d661aSToomas Soome case 'p':
4184a5d661aSToomas Soome base = 16;
4194a5d661aSToomas Soome sharpflag = (width == 0);
4204a5d661aSToomas Soome sign = 0;
4214a5d661aSToomas Soome num = (uintptr_t)va_arg(ap, void *);
4224a5d661aSToomas Soome goto number;
4234a5d661aSToomas Soome case 'q':
4244a5d661aSToomas Soome qflag = 1;
4254a5d661aSToomas Soome goto reswitch;
4264a5d661aSToomas Soome case 'r':
4274a5d661aSToomas Soome base = radix;
4284a5d661aSToomas Soome if (sign)
4294a5d661aSToomas Soome goto handle_sign;
4304a5d661aSToomas Soome goto handle_nosign;
4314a5d661aSToomas Soome case 's':
4324a5d661aSToomas Soome p = va_arg(ap, char *);
4334a5d661aSToomas Soome if (p == NULL)
4344a5d661aSToomas Soome p = "(null)";
4354a5d661aSToomas Soome if (!dot)
4364a5d661aSToomas Soome n = strlen (p);
4374a5d661aSToomas Soome else
4384a5d661aSToomas Soome for (n = 0; n < dwidth && p[n]; n++)
4394a5d661aSToomas Soome continue;
4404a5d661aSToomas Soome
4414a5d661aSToomas Soome width -= n;
4424a5d661aSToomas Soome
4434a5d661aSToomas Soome if (!ladjust && width > 0)
4444a5d661aSToomas Soome while (width--)
4454a5d661aSToomas Soome PCHAR(padc);
4464a5d661aSToomas Soome while (n--)
4474a5d661aSToomas Soome PCHAR(*p++);
4484a5d661aSToomas Soome if (ladjust && width > 0)
4494a5d661aSToomas Soome while (width--)
4504a5d661aSToomas Soome PCHAR(padc);
4514a5d661aSToomas Soome break;
452c5121490SToomas Soome case 'S': /* Assume console can cope with wide chars */
453c5121490SToomas Soome S = va_arg(ap, uint16_t *);
454c5121490SToomas Soome if (S == NULL)
455c5121490SToomas Soome S = (uint16_t *)L"(null)";
456c5121490SToomas Soome if (!dot) {
457c5121490SToomas Soome for (n = 0; S[n] != 0; n++)
458c5121490SToomas Soome continue;
459c5121490SToomas Soome } else {
460c5121490SToomas Soome for (n = 0; n < dwidth && S[n]; n++)
461c5121490SToomas Soome continue;
462c5121490SToomas Soome }
463c5121490SToomas Soome
464c5121490SToomas Soome width -= n;
465c5121490SToomas Soome
466c5121490SToomas Soome if (!ladjust && width > 0)
467c5121490SToomas Soome while (width--)
468c5121490SToomas Soome PCHAR(padc);
469c5121490SToomas Soome while (n--)
470c5121490SToomas Soome PCHAR(*S++);
471c5121490SToomas Soome if (ladjust && width > 0)
472c5121490SToomas Soome while (width--)
473c5121490SToomas Soome PCHAR(padc);
474c5121490SToomas Soome break;
4754a5d661aSToomas Soome case 't':
4764a5d661aSToomas Soome tflag = 1;
4774a5d661aSToomas Soome goto reswitch;
4784a5d661aSToomas Soome case 'u':
4794a5d661aSToomas Soome base = 10;
4804a5d661aSToomas Soome goto handle_nosign;
4814a5d661aSToomas Soome case 'X':
4824a5d661aSToomas Soome upper = 1;
4834a5d661aSToomas Soome case 'x':
4844a5d661aSToomas Soome base = 16;
4854a5d661aSToomas Soome goto handle_nosign;
4864a5d661aSToomas Soome case 'y':
4874a5d661aSToomas Soome base = 16;
4884a5d661aSToomas Soome sign = 1;
4894a5d661aSToomas Soome goto handle_sign;
4904a5d661aSToomas Soome case 'z':
4914a5d661aSToomas Soome zflag = 1;
4924a5d661aSToomas Soome goto reswitch;
4934a5d661aSToomas Soome handle_nosign:
4944a5d661aSToomas Soome sign = 0;
4954a5d661aSToomas Soome if (jflag)
4964a5d661aSToomas Soome num = va_arg(ap, uintmax_t);
4974a5d661aSToomas Soome else if (qflag)
4984a5d661aSToomas Soome num = va_arg(ap, u_quad_t);
4994a5d661aSToomas Soome else if (tflag)
5004a5d661aSToomas Soome num = va_arg(ap, ptrdiff_t);
5014a5d661aSToomas Soome else if (lflag)
5024a5d661aSToomas Soome num = va_arg(ap, u_long);
5034a5d661aSToomas Soome else if (zflag)
5044a5d661aSToomas Soome num = va_arg(ap, size_t);
5054a5d661aSToomas Soome else if (hflag)
5064a5d661aSToomas Soome num = (u_short)va_arg(ap, int);
5074a5d661aSToomas Soome else if (cflag)
5084a5d661aSToomas Soome num = (u_char)va_arg(ap, int);
5094a5d661aSToomas Soome else
5104a5d661aSToomas Soome num = va_arg(ap, u_int);
5114a5d661aSToomas Soome goto number;
5124a5d661aSToomas Soome handle_sign:
5134a5d661aSToomas Soome if (jflag)
5144a5d661aSToomas Soome num = va_arg(ap, intmax_t);
5154a5d661aSToomas Soome else if (qflag)
5164a5d661aSToomas Soome num = va_arg(ap, quad_t);
5174a5d661aSToomas Soome else if (tflag)
5184a5d661aSToomas Soome num = va_arg(ap, ptrdiff_t);
5194a5d661aSToomas Soome else if (lflag)
5204a5d661aSToomas Soome num = va_arg(ap, long);
5214a5d661aSToomas Soome else if (zflag)
5224a5d661aSToomas Soome num = va_arg(ap, ssize_t);
5234a5d661aSToomas Soome else if (hflag)
5244a5d661aSToomas Soome num = (short)va_arg(ap, int);
5254a5d661aSToomas Soome else if (cflag)
5264a5d661aSToomas Soome num = (char)va_arg(ap, int);
5274a5d661aSToomas Soome else
5284a5d661aSToomas Soome num = va_arg(ap, int);
5294a5d661aSToomas Soome number:
5304a5d661aSToomas Soome if (sign && (intmax_t)num < 0) {
5314a5d661aSToomas Soome neg = 1;
5324a5d661aSToomas Soome num = -(intmax_t)num;
5334a5d661aSToomas Soome }
5344a5d661aSToomas Soome p = ksprintn(nbuf, num, base, &n, upper);
5354a5d661aSToomas Soome tmp = 0;
5364a5d661aSToomas Soome if (sharpflag && num != 0) {
5374a5d661aSToomas Soome if (base == 8)
5384a5d661aSToomas Soome tmp++;
5394a5d661aSToomas Soome else if (base == 16)
5404a5d661aSToomas Soome tmp += 2;
5414a5d661aSToomas Soome }
5424a5d661aSToomas Soome if (neg)
5434a5d661aSToomas Soome tmp++;
5444a5d661aSToomas Soome
5454a5d661aSToomas Soome if (!ladjust && padc == '0')
5464a5d661aSToomas Soome dwidth = width - tmp;
5474a5d661aSToomas Soome width -= tmp + imax(dwidth, n);
5484a5d661aSToomas Soome dwidth -= n;
5494a5d661aSToomas Soome if (!ladjust)
5504a5d661aSToomas Soome while (width-- > 0)
5514a5d661aSToomas Soome PCHAR(' ');
5524a5d661aSToomas Soome if (neg)
5534a5d661aSToomas Soome PCHAR('-');
5544a5d661aSToomas Soome if (sharpflag && num != 0) {
5554a5d661aSToomas Soome if (base == 8) {
5564a5d661aSToomas Soome PCHAR('0');
5574a5d661aSToomas Soome } else if (base == 16) {
5584a5d661aSToomas Soome PCHAR('0');
5594a5d661aSToomas Soome PCHAR('x');
5604a5d661aSToomas Soome }
5614a5d661aSToomas Soome }
5624a5d661aSToomas Soome while (dwidth-- > 0)
5634a5d661aSToomas Soome PCHAR('0');
5644a5d661aSToomas Soome
5654a5d661aSToomas Soome while (*p)
5664a5d661aSToomas Soome PCHAR(*p--);
5674a5d661aSToomas Soome
5684a5d661aSToomas Soome if (ladjust)
5694a5d661aSToomas Soome while (width-- > 0)
5704a5d661aSToomas Soome PCHAR(' ');
5714a5d661aSToomas Soome
5724a5d661aSToomas Soome break;
5734a5d661aSToomas Soome default:
5744a5d661aSToomas Soome while (percent < fmt)
5754a5d661aSToomas Soome PCHAR(*percent++);
5764a5d661aSToomas Soome /*
5774a5d661aSToomas Soome * Since we ignore a formatting argument it is no
5784a5d661aSToomas Soome * longer safe to obey the remaining formatting
5794a5d661aSToomas Soome * arguments as the arguments will no longer match
5804a5d661aSToomas Soome * the format specs.
5814a5d661aSToomas Soome */
5824a5d661aSToomas Soome stop = 1;
5834a5d661aSToomas Soome break;
5844a5d661aSToomas Soome }
5854a5d661aSToomas Soome }
5864a5d661aSToomas Soome #undef PCHAR
5874a5d661aSToomas Soome }
588