1c9aca921SXin LI /*- 2*f8a6c905SPedro F. Giffuni * Copyright 2014 Garrett D'Amore <garrett@damore.org> 3b0620803SPedro F. Giffuni * Copyright 2010 Nexenta Systems, Inc. All rights reserved. 49b50d902SRodney W. Grimes * Copyright (c) 1989, 1993 59b50d902SRodney W. Grimes * The Regents of the University of California. All rights reserved. 69b50d902SRodney W. Grimes * 79b50d902SRodney W. Grimes * Redistribution and use in source and binary forms, with or without 89b50d902SRodney W. Grimes * modification, are permitted provided that the following conditions 99b50d902SRodney W. Grimes * are met: 109b50d902SRodney W. Grimes * 1. Redistributions of source code must retain the above copyright 119b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer. 129b50d902SRodney W. Grimes * 2. Redistributions in binary form must reproduce the above copyright 139b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer in the 149b50d902SRodney W. Grimes * documentation and/or other materials provided with the distribution. 159b50d902SRodney W. Grimes * 4. Neither the name of the University nor the names of its contributors 169b50d902SRodney W. Grimes * may be used to endorse or promote products derived from this software 179b50d902SRodney W. Grimes * without specific prior written permission. 189b50d902SRodney W. Grimes * 199b50d902SRodney W. Grimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 209b50d902SRodney W. Grimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 219b50d902SRodney W. Grimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 229b50d902SRodney W. Grimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 239b50d902SRodney W. Grimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 249b50d902SRodney W. Grimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 259b50d902SRodney W. Grimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 269b50d902SRodney W. Grimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 279b50d902SRodney W. Grimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 289b50d902SRodney W. Grimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 299b50d902SRodney W. Grimes * SUCH DAMAGE. 309b50d902SRodney W. Grimes */ 3177fae5c1SJilles Tjoelker /* 3277fae5c1SJilles Tjoelker * Important: This file is used both as a standalone program /usr/bin/printf 3377fae5c1SJilles Tjoelker * and as a builtin for /bin/sh (#define SHELL). 3477fae5c1SJilles Tjoelker */ 359b50d902SRodney W. Grimes 361866e8abSJilles Tjoelker #ifndef SHELL 379b50d902SRodney W. Grimes #ifndef lint 383ec30b79SSteve Price static char const copyright[] = 399b50d902SRodney W. Grimes "@(#) Copyright (c) 1989, 1993\n\ 409b50d902SRodney W. Grimes The Regents of the University of California. All rights reserved.\n"; 419b50d902SRodney W. Grimes #endif /* not lint */ 429b50d902SRodney W. Grimes #endif 439b50d902SRodney W. Grimes 449b50d902SRodney W. Grimes #ifndef lint 45c9e05349SWarner Losh #if 0 463ec30b79SSteve Price static char const sccsid[] = "@(#)printf.c 8.1 (Berkeley) 7/20/93"; 47c9e05349SWarner Losh #endif 481ea7321bSMartin Cracauer static const char rcsid[] = 491ea7321bSMartin Cracauer "$FreeBSD$"; 509b50d902SRodney W. Grimes #endif /* not lint */ 519b50d902SRodney W. Grimes 529b50d902SRodney W. Grimes #include <sys/types.h> 539b50d902SRodney W. Grimes 54*f8a6c905SPedro F. Giffuni #include <ctype.h> 559b50d902SRodney W. Grimes #include <err.h> 569b50d902SRodney W. Grimes #include <errno.h> 57d41d23e1SStefan Farfeleder #include <inttypes.h> 589b50d902SRodney W. Grimes #include <limits.h> 59c9aca921SXin LI #include <locale.h> 609b50d902SRodney W. Grimes #include <stdio.h> 619b50d902SRodney W. Grimes #include <stdlib.h> 629b50d902SRodney W. Grimes #include <string.h> 639d19feb5SSteve Price #include <unistd.h> 6498102dabSJilles Tjoelker #include <wchar.h> 659b50d902SRodney W. Grimes 669b50d902SRodney W. Grimes #ifdef SHELL 679b50d902SRodney W. Grimes #define main printfcmd 68d9f93710SJoerg Wunsch #include "bltin/bltin.h" 699897c45fSJilles Tjoelker #include "error.h" 707cbda738SJilles Tjoelker #include "options.h" 719b50d902SRodney W. Grimes #endif 729b50d902SRodney W. Grimes 73bacab7d6STim J. Robbins #define PF(f, func) do { \ 74d72f654cSPeter Wemm char *b = NULL; \ 7598dd6386STim J. Robbins if (havewidth) \ 7698dd6386STim J. Robbins if (haveprec) \ 77d72f654cSPeter Wemm (void)asprintf(&b, f, fieldwidth, precision, func); \ 789b50d902SRodney W. Grimes else \ 79d72f654cSPeter Wemm (void)asprintf(&b, f, fieldwidth, func); \ 8098dd6386STim J. Robbins else if (haveprec) \ 81d72f654cSPeter Wemm (void)asprintf(&b, f, precision, func); \ 829b50d902SRodney W. Grimes else \ 83d72f654cSPeter Wemm (void)asprintf(&b, f, func); \ 84d72f654cSPeter Wemm if (b) { \ 85d72f654cSPeter Wemm (void)fputs(b, stdout); \ 86d72f654cSPeter Wemm free(b); \ 87d72f654cSPeter Wemm } \ 88bacab7d6STim J. Robbins } while (0) 899b50d902SRodney W. Grimes 90d3cb5dedSWarner Losh static int asciicode(void); 919897c45fSJilles Tjoelker static char *printf_doformat(char *, int *); 923ec96cafSStefan Farfeleder static int escape(char *, int, size_t *); 93d3cb5dedSWarner Losh static int getchr(void); 94fd757c50SDavid Schultz static int getfloating(long double *, int); 95d3cb5dedSWarner Losh static int getint(int *); 96d41d23e1SStefan Farfeleder static int getnum(intmax_t *, uintmax_t *, int); 97bacab7d6STim J. Robbins static const char 98bacab7d6STim J. Robbins *getstr(void); 999180853bSXin LI static char *mknum(char *, char); 100d3cb5dedSWarner Losh static void usage(void); 1019b50d902SRodney W. Grimes 102*f8a6c905SPedro F. Giffuni static const char digits[] = "0123456789"; 103*f8a6c905SPedro F. Giffuni 104b0620803SPedro F. Giffuni static int myargc; 105b0620803SPedro F. Giffuni static char **myargv; 1069b50d902SRodney W. Grimes static char **gargv; 107*f8a6c905SPedro F. Giffuni static char **maxargv; 1089b50d902SRodney W. Grimes 1099b50d902SRodney W. Grimes int 110f4ac32deSDavid Malone main(int argc, char *argv[]) 1119b50d902SRodney W. Grimes { 1123ec96cafSStefan Farfeleder size_t len; 1137cbda738SJilles Tjoelker int chopped, end, rval; 114fbd08684SStefan Farfeleder char *format, *fmt, *start; 1151866e8abSJilles Tjoelker #ifndef SHELL 1167cbda738SJilles Tjoelker int ch; 1177cbda738SJilles Tjoelker 11869be0c5eSXin LI (void) setlocale(LC_ALL, ""); 119dc7d8c99SAndrey A. Chernov #endif 1207cbda738SJilles Tjoelker 1219897c45fSJilles Tjoelker #ifdef SHELL 1227cbda738SJilles Tjoelker nextopt(""); 1237cbda738SJilles Tjoelker argc -= argptr - argv; 1247cbda738SJilles Tjoelker argv = argptr; 1257cbda738SJilles Tjoelker #else 126de46de4dSXin LI while ((ch = getopt(argc, argv, "")) != -1) 127de46de4dSXin LI switch (ch) { 128de46de4dSXin LI case '?': 129de46de4dSXin LI default: 130de46de4dSXin LI usage(); 131de46de4dSXin LI return (1); 1329b50d902SRodney W. Grimes } 133de46de4dSXin LI argc -= optind; 134de46de4dSXin LI argv += optind; 1357cbda738SJilles Tjoelker #endif 1369b50d902SRodney W. Grimes 1379b50d902SRodney W. Grimes if (argc < 1) { 1389b50d902SRodney W. Grimes usage(); 1395eccc000SXin LI return (1); 1409b50d902SRodney W. Grimes } 1419b50d902SRodney W. Grimes 1429897c45fSJilles Tjoelker #ifdef SHELL 1439897c45fSJilles Tjoelker INTOFF; 1449897c45fSJilles Tjoelker #endif 1459b50d902SRodney W. Grimes /* 1469b50d902SRodney W. Grimes * Basic algorithm is to scan the format string for conversion 1479b50d902SRodney W. Grimes * specifications -- once one is found, find out if the field 1489b50d902SRodney W. Grimes * width or precision is a '*'; if it is, gather up value. Note, 1499b50d902SRodney W. Grimes * format strings are reused as necessary to use up the provided 1509b50d902SRodney W. Grimes * arguments, arguments of zero/null string are provided to use 1519b50d902SRodney W. Grimes * up the format string. 1529b50d902SRodney W. Grimes */ 1533ec96cafSStefan Farfeleder fmt = format = *argv; 1543ec96cafSStefan Farfeleder chopped = escape(fmt, 1, &len); /* backslash interpretation */ 155fbd08684SStefan Farfeleder rval = end = 0; 1569b50d902SRodney W. Grimes gargv = ++argv; 157b0620803SPedro F. Giffuni 1589b50d902SRodney W. Grimes for (;;) { 159*f8a6c905SPedro F. Giffuni maxargv = gargv; 160b0620803SPedro F. Giffuni 161b0620803SPedro F. Giffuni myargv = gargv; 162b0620803SPedro F. Giffuni for (myargc = 0; gargv[myargc]; myargc++) 163b0620803SPedro F. Giffuni /* nop */; 164fbd08684SStefan Farfeleder start = fmt; 1653ec96cafSStefan Farfeleder while (fmt < format + len) { 166fbd08684SStefan Farfeleder if (fmt[0] == '%') { 167fbd08684SStefan Farfeleder fwrite(start, 1, fmt - start, stdout); 168fbd08684SStefan Farfeleder if (fmt[1] == '%') { 1699b50d902SRodney W. Grimes /* %% prints a % */ 170fbd08684SStefan Farfeleder putchar('%'); 171fbd08684SStefan Farfeleder fmt += 2; 172fbd08684SStefan Farfeleder } else { 1739897c45fSJilles Tjoelker fmt = printf_doformat(fmt, &rval); 1749897c45fSJilles Tjoelker if (fmt == NULL) { 1759897c45fSJilles Tjoelker #ifdef SHELL 1769897c45fSJilles Tjoelker INTON; 1779897c45fSJilles Tjoelker #endif 178fbd08684SStefan Farfeleder return (1); 1799897c45fSJilles Tjoelker } 180fbd08684SStefan Farfeleder end = 0; 1819b50d902SRodney W. Grimes } 182fbd08684SStefan Farfeleder start = fmt; 183fbd08684SStefan Farfeleder } else 184fbd08684SStefan Farfeleder fmt++; 185b0620803SPedro F. Giffuni if (gargv > maxargv) 186b0620803SPedro F. Giffuni maxargv = gargv; 1879b50d902SRodney W. Grimes } 188b0620803SPedro F. Giffuni gargv = maxargv; 1899b50d902SRodney W. Grimes 190fbd08684SStefan Farfeleder if (end == 1) { 1916a6760dbSJilles Tjoelker warnx("missing format character"); 1929897c45fSJilles Tjoelker #ifdef SHELL 1939897c45fSJilles Tjoelker INTON; 1949897c45fSJilles Tjoelker #endif 195fbd08684SStefan Farfeleder return (1); 196fbd08684SStefan Farfeleder } 197fbd08684SStefan Farfeleder fwrite(start, 1, fmt - start, stdout); 1989897c45fSJilles Tjoelker if (chopped || !*gargv) { 1999897c45fSJilles Tjoelker #ifdef SHELL 2009897c45fSJilles Tjoelker INTON; 2019897c45fSJilles Tjoelker #endif 202fbd08684SStefan Farfeleder return (rval); 2039897c45fSJilles Tjoelker } 204fbd08684SStefan Farfeleder /* Restart at the beginning of the format string. */ 205fbd08684SStefan Farfeleder fmt = format; 206fbd08684SStefan Farfeleder end = 1; 207fbd08684SStefan Farfeleder } 208fbd08684SStefan Farfeleder /* NOTREACHED */ 209fbd08684SStefan Farfeleder } 210fbd08684SStefan Farfeleder 211fbd08684SStefan Farfeleder 212fbd08684SStefan Farfeleder static char * 213*f8a6c905SPedro F. Giffuni printf_doformat(char *fmt, int *rval) 214fbd08684SStefan Farfeleder { 215fbd08684SStefan Farfeleder static const char skip1[] = "#'-+ 0"; 216fbd08684SStefan Farfeleder int fieldwidth, haveprec, havewidth, mod_ldbl, precision; 217fbd08684SStefan Farfeleder char convch, nextch; 218*f8a6c905SPedro F. Giffuni char *start; 219*f8a6c905SPedro F. Giffuni char **fargv; 220*f8a6c905SPedro F. Giffuni char *dptr; 221*f8a6c905SPedro F. Giffuni int l; 222fbd08684SStefan Farfeleder 223*f8a6c905SPedro F. Giffuni start = alloca(strlen(fmt) + 1); 224*f8a6c905SPedro F. Giffuni 225*f8a6c905SPedro F. Giffuni dptr = start; 226*f8a6c905SPedro F. Giffuni *dptr++ = '%'; 227*f8a6c905SPedro F. Giffuni *dptr = 0; 228*f8a6c905SPedro F. Giffuni 229*f8a6c905SPedro F. Giffuni fmt++; 230b0620803SPedro F. Giffuni 231b0620803SPedro F. Giffuni /* look for "n$" field index specifier */ 232*f8a6c905SPedro F. Giffuni l = strspn(fmt, digits); 233*f8a6c905SPedro F. Giffuni if ((l > 0) && (fmt[l] == '$')) { 234*f8a6c905SPedro F. Giffuni int idx = atoi(fmt); 235b0620803SPedro F. Giffuni if (idx <= myargc) { 236b0620803SPedro F. Giffuni gargv = &myargv[idx - 1]; 237b0620803SPedro F. Giffuni } else { 238b0620803SPedro F. Giffuni gargv = &myargv[myargc]; 239b0620803SPedro F. Giffuni } 240*f8a6c905SPedro F. Giffuni if (gargv > maxargv) 241*f8a6c905SPedro F. Giffuni maxargv = gargv; 242*f8a6c905SPedro F. Giffuni fmt += l + 1; 243*f8a6c905SPedro F. Giffuni 244*f8a6c905SPedro F. Giffuni /* save format argument */ 245*f8a6c905SPedro F. Giffuni fargv = gargv; 246b0620803SPedro F. Giffuni } else { 247*f8a6c905SPedro F. Giffuni fargv = NULL; 248b0620803SPedro F. Giffuni } 249b0620803SPedro F. Giffuni 2509b50d902SRodney W. Grimes /* skip to field width */ 251*f8a6c905SPedro F. Giffuni while (strchr(skip1, *fmt) != NULL) { 252*f8a6c905SPedro F. Giffuni *dptr++ = *fmt++; 253*f8a6c905SPedro F. Giffuni *dptr = 0; 254*f8a6c905SPedro F. Giffuni } 255*f8a6c905SPedro F. Giffuni 2569b50d902SRodney W. Grimes if (*fmt == '*') { 257*f8a6c905SPedro F. Giffuni 258*f8a6c905SPedro F. Giffuni fmt++; 259*f8a6c905SPedro F. Giffuni l = strspn(fmt, digits); 260*f8a6c905SPedro F. Giffuni if ((l > 0) && (fmt[l] == '$')) { 261*f8a6c905SPedro F. Giffuni int idx = atoi(fmt); 262*f8a6c905SPedro F. Giffuni if (idx <= myargc) { 263*f8a6c905SPedro F. Giffuni gargv = &myargv[idx - 1]; 264*f8a6c905SPedro F. Giffuni } else { 265*f8a6c905SPedro F. Giffuni gargv = &myargv[myargc]; 266*f8a6c905SPedro F. Giffuni } 267*f8a6c905SPedro F. Giffuni fmt += l + 1; 268*f8a6c905SPedro F. Giffuni } 269*f8a6c905SPedro F. Giffuni 2709b50d902SRodney W. Grimes if (getint(&fieldwidth)) 271fbd08684SStefan Farfeleder return (NULL); 272*f8a6c905SPedro F. Giffuni if (gargv > maxargv) 273*f8a6c905SPedro F. Giffuni maxargv = gargv; 27498dd6386STim J. Robbins havewidth = 1; 275*f8a6c905SPedro F. Giffuni 276*f8a6c905SPedro F. Giffuni *dptr++ = '*'; 277*f8a6c905SPedro F. Giffuni *dptr = 0; 278d867cefdSJoerg Wunsch } else { 27998dd6386STim J. Robbins havewidth = 0; 2809b50d902SRodney W. Grimes 2819b50d902SRodney W. Grimes /* skip to possible '.', get following precision */ 282*f8a6c905SPedro F. Giffuni while (isdigit(*fmt)) { 283*f8a6c905SPedro F. Giffuni *dptr++ = *fmt++; 284*f8a6c905SPedro F. Giffuni *dptr = 0; 285d867cefdSJoerg Wunsch } 286*f8a6c905SPedro F. Giffuni } 287*f8a6c905SPedro F. Giffuni 288d867cefdSJoerg Wunsch if (*fmt == '.') { 289d867cefdSJoerg Wunsch /* precision present? */ 290*f8a6c905SPedro F. Giffuni fmt++; 291*f8a6c905SPedro F. Giffuni *dptr++ = '.'; 292*f8a6c905SPedro F. Giffuni 2939b50d902SRodney W. Grimes if (*fmt == '*') { 294*f8a6c905SPedro F. Giffuni 295*f8a6c905SPedro F. Giffuni fmt++; 296*f8a6c905SPedro F. Giffuni l = strspn(fmt, digits); 297*f8a6c905SPedro F. Giffuni if ((l > 0) && (fmt[l] == '$')) { 298*f8a6c905SPedro F. Giffuni int idx = atoi(fmt); 299*f8a6c905SPedro F. Giffuni if (idx <= myargc) { 300*f8a6c905SPedro F. Giffuni gargv = &myargv[idx - 1]; 301*f8a6c905SPedro F. Giffuni } else { 302*f8a6c905SPedro F. Giffuni gargv = &myargv[myargc]; 303*f8a6c905SPedro F. Giffuni } 304*f8a6c905SPedro F. Giffuni fmt += l + 1; 305*f8a6c905SPedro F. Giffuni } 306*f8a6c905SPedro F. Giffuni 3079b50d902SRodney W. Grimes if (getint(&precision)) 308fbd08684SStefan Farfeleder return (NULL); 309*f8a6c905SPedro F. Giffuni if (gargv > maxargv) 310*f8a6c905SPedro F. Giffuni maxargv = gargv; 31198dd6386STim J. Robbins haveprec = 1; 312*f8a6c905SPedro F. Giffuni *dptr++ = '*'; 313*f8a6c905SPedro F. Giffuni *dptr = 0; 314d867cefdSJoerg Wunsch } else { 31598dd6386STim J. Robbins haveprec = 0; 3169b50d902SRodney W. Grimes 3179b50d902SRodney W. Grimes /* skip to conversion char */ 318*f8a6c905SPedro F. Giffuni while (isdigit(*fmt)) { 319*f8a6c905SPedro F. Giffuni *dptr++ = *fmt++; 320*f8a6c905SPedro F. Giffuni *dptr = 0; 321*f8a6c905SPedro F. Giffuni } 322d867cefdSJoerg Wunsch } 323d867cefdSJoerg Wunsch } else 32498dd6386STim J. Robbins haveprec = 0; 3259b50d902SRodney W. Grimes if (!*fmt) { 3266a6760dbSJilles Tjoelker warnx("missing format character"); 327fbd08684SStefan Farfeleder return (NULL); 3289b50d902SRodney W. Grimes } 329*f8a6c905SPedro F. Giffuni *dptr++ = *fmt; 330*f8a6c905SPedro F. Giffuni *dptr = 0; 3319b50d902SRodney W. Grimes 332fd757c50SDavid Schultz /* 333fd757c50SDavid Schultz * Look for a length modifier. POSIX doesn't have these, so 334fd757c50SDavid Schultz * we only support them for floating-point conversions, which 335fd757c50SDavid Schultz * are extensions. This is useful because the L modifier can 336fd757c50SDavid Schultz * be used to gain extra range and precision, while omitting 337fd757c50SDavid Schultz * it is more likely to produce consistent results on different 338fd757c50SDavid Schultz * architectures. This is not so important for integers 339fd757c50SDavid Schultz * because overflow is the only bad thing that can happen to 340fd757c50SDavid Schultz * them, but consider the command printf %a 1.1 341fd757c50SDavid Schultz */ 342fd757c50SDavid Schultz if (*fmt == 'L') { 343fd757c50SDavid Schultz mod_ldbl = 1; 344fd757c50SDavid Schultz fmt++; 345fd757c50SDavid Schultz if (!strchr("aAeEfFgG", *fmt)) { 3466a6760dbSJilles Tjoelker warnx("bad modifier L for %%%c", *fmt); 347fbd08684SStefan Farfeleder return (NULL); 348fd757c50SDavid Schultz } 349fd757c50SDavid Schultz } else { 350fd757c50SDavid Schultz mod_ldbl = 0; 351fd757c50SDavid Schultz } 352fd757c50SDavid Schultz 353*f8a6c905SPedro F. Giffuni /* save the current arg offset, and set to the format arg */ 354*f8a6c905SPedro F. Giffuni if (fargv != NULL) { 355*f8a6c905SPedro F. Giffuni gargv = fargv; 356*f8a6c905SPedro F. Giffuni } 357*f8a6c905SPedro F. Giffuni 3589b50d902SRodney W. Grimes convch = *fmt; 3599b50d902SRodney W. Grimes nextch = *++fmt; 360*f8a6c905SPedro F. Giffuni 3619b50d902SRodney W. Grimes *fmt = '\0'; 3629b50d902SRodney W. Grimes switch (convch) { 363ab5a295bSJuli Mallett case 'b': { 3643ec96cafSStefan Farfeleder size_t len; 365ab5a295bSJuli Mallett char *p; 366ab5a295bSJuli Mallett int getout; 367ab5a295bSJuli Mallett 368bacab7d6STim J. Robbins p = strdup(getstr()); 369bacab7d6STim J. Robbins if (p == NULL) { 3706a6760dbSJilles Tjoelker warnx("%s", strerror(ENOMEM)); 371fbd08684SStefan Farfeleder return (NULL); 372bacab7d6STim J. Robbins } 3733ec96cafSStefan Farfeleder getout = escape(p, 0, &len); 374ab5a295bSJuli Mallett *(fmt - 1) = 's'; 375bacab7d6STim J. Robbins PF(start, p); 376ab5a295bSJuli Mallett *(fmt - 1) = 'b'; 377ab5a295bSJuli Mallett free(p); 378ab5a295bSJuli Mallett if (getout) 379fbd08684SStefan Farfeleder return (fmt); 380ab5a295bSJuli Mallett break; 381ab5a295bSJuli Mallett } 3829b50d902SRodney W. Grimes case 'c': { 3839b50d902SRodney W. Grimes char p; 3849b50d902SRodney W. Grimes 3859b50d902SRodney W. Grimes p = getchr(); 3869b50d902SRodney W. Grimes PF(start, p); 3879b50d902SRodney W. Grimes break; 3889b50d902SRodney W. Grimes } 3899b50d902SRodney W. Grimes case 's': { 39045af1a4cSDavid Malone const char *p; 3919b50d902SRodney W. Grimes 3929b50d902SRodney W. Grimes p = getstr(); 3939b50d902SRodney W. Grimes PF(start, p); 3949b50d902SRodney W. Grimes break; 3959b50d902SRodney W. Grimes } 3969b50d902SRodney W. Grimes case 'd': case 'i': case 'o': case 'u': case 'x': case 'X': { 3979b50d902SRodney W. Grimes char *f; 398d41d23e1SStefan Farfeleder intmax_t val; 399d41d23e1SStefan Farfeleder uintmax_t uval; 400bacab7d6STim J. Robbins int signedconv; 4019b50d902SRodney W. Grimes 402bacab7d6STim J. Robbins signedconv = (convch == 'd' || convch == 'i'); 403d41d23e1SStefan Farfeleder if ((f = mknum(start, convch)) == NULL) 404fbd08684SStefan Farfeleder return (NULL); 405d41d23e1SStefan Farfeleder if (getnum(&val, &uval, signedconv)) 406fbd08684SStefan Farfeleder *rval = 1; 407bacab7d6STim J. Robbins if (signedconv) 408bacab7d6STim J. Robbins PF(f, val); 409bacab7d6STim J. Robbins else 410bacab7d6STim J. Robbins PF(f, uval); 4119b50d902SRodney W. Grimes break; 4129b50d902SRodney W. Grimes } 41303b2eaacSDavid Schultz case 'e': case 'E': 41403b2eaacSDavid Schultz case 'f': case 'F': 41503b2eaacSDavid Schultz case 'g': case 'G': 41603b2eaacSDavid Schultz case 'a': case 'A': { 417fd757c50SDavid Schultz long double p; 4189b50d902SRodney W. Grimes 419fd757c50SDavid Schultz if (getfloating(&p, mod_ldbl)) 420fbd08684SStefan Farfeleder *rval = 1; 421fd757c50SDavid Schultz if (mod_ldbl) 4229b50d902SRodney W. Grimes PF(start, p); 423fd757c50SDavid Schultz else 424fd757c50SDavid Schultz PF(start, (double)p); 4259b50d902SRodney W. Grimes break; 4269b50d902SRodney W. Grimes } 4279b50d902SRodney W. Grimes default: 4286a6760dbSJilles Tjoelker warnx("illegal format character %c", convch); 429fbd08684SStefan Farfeleder return (NULL); 4309b50d902SRodney W. Grimes } 4319b50d902SRodney W. Grimes *fmt = nextch; 432*f8a6c905SPedro F. Giffuni /* return the gargv to the next element */ 433fbd08684SStefan Farfeleder return (fmt); 4349b50d902SRodney W. Grimes } 4359b50d902SRodney W. Grimes 4369b50d902SRodney W. Grimes static char * 4379180853bSXin LI mknum(char *str, char ch) 4389b50d902SRodney W. Grimes { 4393c6e4a5cSBen Smithurst static char *copy; 4403c6e4a5cSBen Smithurst static size_t copy_size; 4413c6e4a5cSBen Smithurst char *newcopy; 442bacab7d6STim J. Robbins size_t len, newlen; 4439b50d902SRodney W. Grimes 4449b50d902SRodney W. Grimes len = strlen(str) + 2; 4453c6e4a5cSBen Smithurst if (len > copy_size) { 4463c6e4a5cSBen Smithurst newlen = ((len + 1023) >> 10) << 10; 4473c6e4a5cSBen Smithurst if ((newcopy = realloc(copy, newlen)) == NULL) 448bacab7d6STim J. Robbins { 4496a6760dbSJilles Tjoelker warnx("%s", strerror(ENOMEM)); 4503c6e4a5cSBen Smithurst return (NULL); 451bacab7d6STim J. Robbins } 4523c6e4a5cSBen Smithurst copy = newcopy; 4533c6e4a5cSBen Smithurst copy_size = newlen; 4543c6e4a5cSBen Smithurst } 45562a721e7SStefan Eßer 4569b50d902SRodney W. Grimes memmove(copy, str, len - 3); 457d41d23e1SStefan Farfeleder copy[len - 3] = 'j'; 4589b50d902SRodney W. Grimes copy[len - 2] = ch; 4599b50d902SRodney W. Grimes copy[len - 1] = '\0'; 4609b50d902SRodney W. Grimes return (copy); 4619b50d902SRodney W. Grimes } 4629b50d902SRodney W. Grimes 463ab5a295bSJuli Mallett static int 4643ec96cafSStefan Farfeleder escape(char *fmt, int percent, size_t *len) 4659b50d902SRodney W. Grimes { 466ab71f271SPedro F. Giffuni char *save, *store, c; 467ab71f271SPedro F. Giffuni int value; 4689b50d902SRodney W. Grimes 469ab71f271SPedro F. Giffuni for (save = store = fmt; ((c = *fmt) != 0); ++fmt, ++store) { 4709b50d902SRodney W. Grimes if (c != '\\') { 4719b50d902SRodney W. Grimes *store = c; 4729b50d902SRodney W. Grimes continue; 4739b50d902SRodney W. Grimes } 4749b50d902SRodney W. Grimes switch (*++fmt) { 4759b50d902SRodney W. Grimes case '\0': /* EOS, user error */ 4769b50d902SRodney W. Grimes *store = '\\'; 4779b50d902SRodney W. Grimes *++store = '\0'; 4783ec96cafSStefan Farfeleder *len = store - save; 479ab5a295bSJuli Mallett return (0); 4809b50d902SRodney W. Grimes case '\\': /* backslash */ 4819b50d902SRodney W. Grimes case '\'': /* single quote */ 4829b50d902SRodney W. Grimes *store = *fmt; 4839b50d902SRodney W. Grimes break; 4849b50d902SRodney W. Grimes case 'a': /* bell/alert */ 485f3f148d2SStefan Farfeleder *store = '\a'; 4869b50d902SRodney W. Grimes break; 4879b50d902SRodney W. Grimes case 'b': /* backspace */ 4889b50d902SRodney W. Grimes *store = '\b'; 4899b50d902SRodney W. Grimes break; 490ab5a295bSJuli Mallett case 'c': 491ab5a295bSJuli Mallett *store = '\0'; 4923ec96cafSStefan Farfeleder *len = store - save; 493ab5a295bSJuli Mallett return (1); 4949b50d902SRodney W. Grimes case 'f': /* form-feed */ 4959b50d902SRodney W. Grimes *store = '\f'; 4969b50d902SRodney W. Grimes break; 4979b50d902SRodney W. Grimes case 'n': /* newline */ 4989b50d902SRodney W. Grimes *store = '\n'; 4999b50d902SRodney W. Grimes break; 5009b50d902SRodney W. Grimes case 'r': /* carriage-return */ 5019b50d902SRodney W. Grimes *store = '\r'; 5029b50d902SRodney W. Grimes break; 5039b50d902SRodney W. Grimes case 't': /* horizontal tab */ 5049b50d902SRodney W. Grimes *store = '\t'; 5059b50d902SRodney W. Grimes break; 5069b50d902SRodney W. Grimes case 'v': /* vertical tab */ 507f3f148d2SStefan Farfeleder *store = '\v'; 5089b50d902SRodney W. Grimes break; 5099b50d902SRodney W. Grimes /* octal constant */ 5109b50d902SRodney W. Grimes case '0': case '1': case '2': case '3': 5119b50d902SRodney W. Grimes case '4': case '5': case '6': case '7': 5129d65050eSDavid Schultz c = (!percent && *fmt == '0') ? 4 : 3; 5139d65050eSDavid Schultz for (value = 0; 5149b50d902SRodney W. Grimes c-- && *fmt >= '0' && *fmt <= '7'; ++fmt) { 5159b50d902SRodney W. Grimes value <<= 3; 5169b50d902SRodney W. Grimes value += *fmt - '0'; 5179b50d902SRodney W. Grimes } 5189b50d902SRodney W. Grimes --fmt; 51912e8db40STim J. Robbins if (percent && value == '%') { 52037fd4590STim J. Robbins *store++ = '%'; 52137fd4590STim J. Robbins *store = '%'; 52237fd4590STim J. Robbins } else 523ab71f271SPedro F. Giffuni *store = (char)value; 5249b50d902SRodney W. Grimes break; 5259b50d902SRodney W. Grimes default: 5269b50d902SRodney W. Grimes *store = *fmt; 5279b50d902SRodney W. Grimes break; 5289b50d902SRodney W. Grimes } 5299b50d902SRodney W. Grimes } 5309b50d902SRodney W. Grimes *store = '\0'; 5313ec96cafSStefan Farfeleder *len = store - save; 532ab5a295bSJuli Mallett return (0); 5339b50d902SRodney W. Grimes } 5349b50d902SRodney W. Grimes 5359b50d902SRodney W. Grimes static int 536f4ac32deSDavid Malone getchr(void) 5379b50d902SRodney W. Grimes { 5389b50d902SRodney W. Grimes if (!*gargv) 5399b50d902SRodney W. Grimes return ('\0'); 5409b50d902SRodney W. Grimes return ((int)**gargv++); 5419b50d902SRodney W. Grimes } 5429b50d902SRodney W. Grimes 54345af1a4cSDavid Malone static const char * 544f4ac32deSDavid Malone getstr(void) 5459b50d902SRodney W. Grimes { 5469b50d902SRodney W. Grimes if (!*gargv) 5479b50d902SRodney W. Grimes return (""); 5489b50d902SRodney W. Grimes return (*gargv++); 5499b50d902SRodney W. Grimes } 5509b50d902SRodney W. Grimes 5519b50d902SRodney W. Grimes static int 552f4ac32deSDavid Malone getint(int *ip) 5539b50d902SRodney W. Grimes { 554d41d23e1SStefan Farfeleder intmax_t val; 555d41d23e1SStefan Farfeleder uintmax_t uval; 556bacab7d6STim J. Robbins int rval; 5579b50d902SRodney W. Grimes 558d41d23e1SStefan Farfeleder if (getnum(&val, &uval, 1)) 5599b50d902SRodney W. Grimes return (1); 560bacab7d6STim J. Robbins rval = 0; 561bacab7d6STim J. Robbins if (val < INT_MIN || val > INT_MAX) { 5626a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 563bacab7d6STim J. Robbins rval = 1; 564bacab7d6STim J. Robbins } 56562a721e7SStefan Eßer *ip = (int)val; 566bacab7d6STim J. Robbins return (rval); 5679b50d902SRodney W. Grimes } 5689b50d902SRodney W. Grimes 5699b50d902SRodney W. Grimes static int 570d41d23e1SStefan Farfeleder getnum(intmax_t *ip, uintmax_t *uip, int signedconv) 5719b50d902SRodney W. Grimes { 5729b50d902SRodney W. Grimes char *ep; 573bacab7d6STim J. Robbins int rval; 5749b50d902SRodney W. Grimes 5759b50d902SRodney W. Grimes if (!*gargv) { 576*f8a6c905SPedro F. Giffuni *ip = 0; 5779b50d902SRodney W. Grimes return (0); 5789b50d902SRodney W. Grimes } 579ab5a295bSJuli Mallett if (**gargv == '"' || **gargv == '\'') { 580bacab7d6STim J. Robbins if (signedconv) 581d41d23e1SStefan Farfeleder *ip = asciicode(); 582bacab7d6STim J. Robbins else 583d41d23e1SStefan Farfeleder *uip = asciicode(); 5849b50d902SRodney W. Grimes return (0); 5859b50d902SRodney W. Grimes } 586bacab7d6STim J. Robbins rval = 0; 587ab5a295bSJuli Mallett errno = 0; 588bacab7d6STim J. Robbins if (signedconv) 589d41d23e1SStefan Farfeleder *ip = strtoimax(*gargv, &ep, 0); 590bacab7d6STim J. Robbins else 591d41d23e1SStefan Farfeleder *uip = strtoumax(*gargv, &ep, 0); 592bacab7d6STim J. Robbins if (ep == *gargv) { 5936a6760dbSJilles Tjoelker warnx("%s: expected numeric value", *gargv); 594bacab7d6STim J. Robbins rval = 1; 595bacab7d6STim J. Robbins } 596bacab7d6STim J. Robbins else if (*ep != '\0') { 5976a6760dbSJilles Tjoelker warnx("%s: not completely converted", *gargv); 598bacab7d6STim J. Robbins rval = 1; 599bacab7d6STim J. Robbins } 600bacab7d6STim J. Robbins if (errno == ERANGE) { 6016a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 602bacab7d6STim J. Robbins rval = 1; 603bacab7d6STim J. Robbins } 604ab5a295bSJuli Mallett ++gargv; 605bacab7d6STim J. Robbins return (rval); 6069b50d902SRodney W. Grimes } 6079b50d902SRodney W. Grimes 608bacab7d6STim J. Robbins static int 609fd757c50SDavid Schultz getfloating(long double *dp, int mod_ldbl) 6109b50d902SRodney W. Grimes { 611ab5a295bSJuli Mallett char *ep; 612bacab7d6STim J. Robbins int rval; 613ab5a295bSJuli Mallett 6145ec2b8dcSStefan Farfeleder if (!*gargv) { 6155ec2b8dcSStefan Farfeleder *dp = 0.0; 616bacab7d6STim J. Robbins return (0); 6175ec2b8dcSStefan Farfeleder } 618ab5a295bSJuli Mallett if (**gargv == '"' || **gargv == '\'') { 619bacab7d6STim J. Robbins *dp = asciicode(); 620bacab7d6STim J. Robbins return (0); 621ab5a295bSJuli Mallett } 6228c423a99SColin Percival rval = 0; 623ab5a295bSJuli Mallett errno = 0; 624fd757c50SDavid Schultz if (mod_ldbl) 625fd757c50SDavid Schultz *dp = strtold(*gargv, &ep); 626fd757c50SDavid Schultz else 627bacab7d6STim J. Robbins *dp = strtod(*gargv, &ep); 628bacab7d6STim J. Robbins if (ep == *gargv) { 6296a6760dbSJilles Tjoelker warnx("%s: expected numeric value", *gargv); 630bacab7d6STim J. Robbins rval = 1; 631bacab7d6STim J. Robbins } else if (*ep != '\0') { 6326a6760dbSJilles Tjoelker warnx("%s: not completely converted", *gargv); 633bacab7d6STim J. Robbins rval = 1; 634bacab7d6STim J. Robbins } 635bacab7d6STim J. Robbins if (errno == ERANGE) { 6366a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 637bacab7d6STim J. Robbins rval = 1; 638bacab7d6STim J. Robbins } 639ab5a295bSJuli Mallett ++gargv; 640bacab7d6STim J. Robbins return (rval); 6419b50d902SRodney W. Grimes } 6429b50d902SRodney W. Grimes 6439b50d902SRodney W. Grimes static int 644f4ac32deSDavid Malone asciicode(void) 6459b50d902SRodney W. Grimes { 646f4ac32deSDavid Malone int ch; 64798102dabSJilles Tjoelker wchar_t wch; 64898102dabSJilles Tjoelker mbstate_t mbs; 6499b50d902SRodney W. Grimes 65098102dabSJilles Tjoelker ch = (unsigned char)**gargv; 65198102dabSJilles Tjoelker if (ch == '\'' || ch == '"') { 65298102dabSJilles Tjoelker memset(&mbs, 0, sizeof(mbs)); 65398102dabSJilles Tjoelker switch (mbrtowc(&wch, *gargv + 1, MB_LEN_MAX, &mbs)) { 65498102dabSJilles Tjoelker case (size_t)-2: 65598102dabSJilles Tjoelker case (size_t)-1: 65698102dabSJilles Tjoelker wch = (unsigned char)gargv[0][1]; 65798102dabSJilles Tjoelker break; 65898102dabSJilles Tjoelker case 0: 65998102dabSJilles Tjoelker wch = 0; 66098102dabSJilles Tjoelker break; 66198102dabSJilles Tjoelker } 66298102dabSJilles Tjoelker ch = wch; 66398102dabSJilles Tjoelker } 6649b50d902SRodney W. Grimes ++gargv; 6659b50d902SRodney W. Grimes return (ch); 6669b50d902SRodney W. Grimes } 6679b50d902SRodney W. Grimes 6689b50d902SRodney W. Grimes static void 669f4ac32deSDavid Malone usage(void) 6709b50d902SRodney W. Grimes { 671f682f10cSRuslan Ermilov (void)fprintf(stderr, "usage: printf format [arguments ...]\n"); 6729b50d902SRodney W. Grimes } 673