1c9aca921SXin LI /*- 29b50d902SRodney W. Grimes * Copyright (c) 1989, 1993 39b50d902SRodney W. Grimes * The Regents of the University of California. All rights reserved. 49b50d902SRodney W. Grimes * 59b50d902SRodney W. Grimes * Redistribution and use in source and binary forms, with or without 69b50d902SRodney W. Grimes * modification, are permitted provided that the following conditions 79b50d902SRodney W. Grimes * are met: 89b50d902SRodney W. Grimes * 1. Redistributions of source code must retain the above copyright 99b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer. 109b50d902SRodney W. Grimes * 2. Redistributions in binary form must reproduce the above copyright 119b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer in the 129b50d902SRodney W. Grimes * documentation and/or other materials provided with the distribution. 139b50d902SRodney W. Grimes * 4. Neither the name of the University nor the names of its contributors 149b50d902SRodney W. Grimes * may be used to endorse or promote products derived from this software 159b50d902SRodney W. Grimes * without specific prior written permission. 169b50d902SRodney W. Grimes * 179b50d902SRodney W. Grimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 189b50d902SRodney W. Grimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 199b50d902SRodney W. Grimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 209b50d902SRodney W. Grimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 219b50d902SRodney W. Grimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 229b50d902SRodney W. Grimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 239b50d902SRodney W. Grimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 249b50d902SRodney W. Grimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 259b50d902SRodney W. Grimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 269b50d902SRodney W. Grimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 279b50d902SRodney W. Grimes * SUCH DAMAGE. 289b50d902SRodney W. Grimes */ 2977fae5c1SJilles Tjoelker /* 3077fae5c1SJilles Tjoelker * Important: This file is used both as a standalone program /usr/bin/printf 3177fae5c1SJilles Tjoelker * and as a builtin for /bin/sh (#define SHELL). 3277fae5c1SJilles Tjoelker */ 339b50d902SRodney W. Grimes 341866e8abSJilles Tjoelker #ifndef SHELL 359b50d902SRodney W. Grimes #ifndef lint 363ec30b79SSteve Price static char const copyright[] = 379b50d902SRodney W. Grimes "@(#) Copyright (c) 1989, 1993\n\ 389b50d902SRodney W. Grimes The Regents of the University of California. All rights reserved.\n"; 399b50d902SRodney W. Grimes #endif /* not lint */ 409b50d902SRodney W. Grimes #endif 419b50d902SRodney W. Grimes 429b50d902SRodney W. Grimes #ifndef lint 43c9e05349SWarner Losh #if 0 443ec30b79SSteve Price static char const sccsid[] = "@(#)printf.c 8.1 (Berkeley) 7/20/93"; 45c9e05349SWarner Losh #endif 461ea7321bSMartin Cracauer static const char rcsid[] = 471ea7321bSMartin Cracauer "$FreeBSD$"; 489b50d902SRodney W. Grimes #endif /* not lint */ 499b50d902SRodney W. Grimes 509b50d902SRodney W. Grimes #include <sys/types.h> 519b50d902SRodney W. Grimes 529b50d902SRodney W. Grimes #include <err.h> 539b50d902SRodney W. Grimes #include <errno.h> 54d41d23e1SStefan Farfeleder #include <inttypes.h> 559b50d902SRodney W. Grimes #include <limits.h> 56c9aca921SXin LI #include <locale.h> 579b50d902SRodney W. Grimes #include <stdio.h> 589b50d902SRodney W. Grimes #include <stdlib.h> 599b50d902SRodney W. Grimes #include <string.h> 609d19feb5SSteve Price #include <unistd.h> 6198102dabSJilles Tjoelker #include <wchar.h> 629b50d902SRodney W. Grimes 639b50d902SRodney W. Grimes #ifdef SHELL 649b50d902SRodney W. Grimes #define main printfcmd 65d9f93710SJoerg Wunsch #include "bltin/bltin.h" 669897c45fSJilles Tjoelker #include "error.h" 679b50d902SRodney W. Grimes #endif 689b50d902SRodney W. Grimes 69bacab7d6STim J. Robbins #define PF(f, func) do { \ 70d72f654cSPeter Wemm char *b = NULL; \ 7198dd6386STim J. Robbins if (havewidth) \ 7298dd6386STim J. Robbins if (haveprec) \ 73d72f654cSPeter Wemm (void)asprintf(&b, f, fieldwidth, precision, func); \ 749b50d902SRodney W. Grimes else \ 75d72f654cSPeter Wemm (void)asprintf(&b, f, fieldwidth, func); \ 7698dd6386STim J. Robbins else if (haveprec) \ 77d72f654cSPeter Wemm (void)asprintf(&b, f, precision, func); \ 789b50d902SRodney W. Grimes else \ 79d72f654cSPeter Wemm (void)asprintf(&b, f, func); \ 80d72f654cSPeter Wemm if (b) { \ 81d72f654cSPeter Wemm (void)fputs(b, stdout); \ 82d72f654cSPeter Wemm free(b); \ 83d72f654cSPeter Wemm } \ 84bacab7d6STim J. Robbins } while (0) 859b50d902SRodney W. Grimes 86d3cb5dedSWarner Losh static int asciicode(void); 879897c45fSJilles Tjoelker static char *printf_doformat(char *, int *); 883ec96cafSStefan Farfeleder static int escape(char *, int, size_t *); 89d3cb5dedSWarner Losh static int getchr(void); 90fd757c50SDavid Schultz static int getfloating(long double *, int); 91d3cb5dedSWarner Losh static int getint(int *); 92d41d23e1SStefan Farfeleder static int getnum(intmax_t *, uintmax_t *, int); 93bacab7d6STim J. Robbins static const char 94bacab7d6STim J. Robbins *getstr(void); 959180853bSXin LI static char *mknum(char *, char); 96d3cb5dedSWarner Losh static void usage(void); 979b50d902SRodney W. Grimes 989b50d902SRodney W. Grimes static char **gargv; 999b50d902SRodney W. Grimes 1009b50d902SRodney W. Grimes int 101f4ac32deSDavid Malone main(int argc, char *argv[]) 1029b50d902SRodney W. Grimes { 1033ec96cafSStefan Farfeleder size_t len; 104de46de4dSXin LI int ch, chopped, end, rval; 105fbd08684SStefan Farfeleder char *format, *fmt, *start; 1069b50d902SRodney W. Grimes 1071866e8abSJilles Tjoelker #ifndef SHELL 10869be0c5eSXin LI (void) setlocale(LC_ALL, ""); 109dc7d8c99SAndrey A. Chernov #endif 1109897c45fSJilles Tjoelker #ifdef SHELL 1119897c45fSJilles Tjoelker optreset = 1; optind = 1; opterr = 0; /* initialize getopt */ 1129897c45fSJilles Tjoelker #endif 113de46de4dSXin LI while ((ch = getopt(argc, argv, "")) != -1) 114de46de4dSXin LI switch (ch) { 115de46de4dSXin LI case '?': 116de46de4dSXin LI default: 117de46de4dSXin LI usage(); 118de46de4dSXin LI return (1); 1199b50d902SRodney W. Grimes } 120de46de4dSXin LI argc -= optind; 121de46de4dSXin LI argv += optind; 1229b50d902SRodney W. Grimes 1239b50d902SRodney W. Grimes if (argc < 1) { 1249b50d902SRodney W. Grimes usage(); 1255eccc000SXin LI return (1); 1269b50d902SRodney W. Grimes } 1279b50d902SRodney W. Grimes 1289897c45fSJilles Tjoelker #ifdef SHELL 1299897c45fSJilles Tjoelker INTOFF; 1309897c45fSJilles Tjoelker #endif 1319b50d902SRodney W. Grimes /* 1329b50d902SRodney W. Grimes * Basic algorithm is to scan the format string for conversion 1339b50d902SRodney W. Grimes * specifications -- once one is found, find out if the field 1349b50d902SRodney W. Grimes * width or precision is a '*'; if it is, gather up value. Note, 1359b50d902SRodney W. Grimes * format strings are reused as necessary to use up the provided 1369b50d902SRodney W. Grimes * arguments, arguments of zero/null string are provided to use 1379b50d902SRodney W. Grimes * up the format string. 1389b50d902SRodney W. Grimes */ 1393ec96cafSStefan Farfeleder fmt = format = *argv; 1403ec96cafSStefan Farfeleder chopped = escape(fmt, 1, &len); /* backslash interpretation */ 141fbd08684SStefan Farfeleder rval = end = 0; 1429b50d902SRodney W. Grimes gargv = ++argv; 1439b50d902SRodney W. Grimes for (;;) { 144fbd08684SStefan Farfeleder start = fmt; 1453ec96cafSStefan Farfeleder while (fmt < format + len) { 146fbd08684SStefan Farfeleder if (fmt[0] == '%') { 147fbd08684SStefan Farfeleder fwrite(start, 1, fmt - start, stdout); 148fbd08684SStefan Farfeleder if (fmt[1] == '%') { 1499b50d902SRodney W. Grimes /* %% prints a % */ 150fbd08684SStefan Farfeleder putchar('%'); 151fbd08684SStefan Farfeleder fmt += 2; 152fbd08684SStefan Farfeleder } else { 1539897c45fSJilles Tjoelker fmt = printf_doformat(fmt, &rval); 1549897c45fSJilles Tjoelker if (fmt == NULL) { 1559897c45fSJilles Tjoelker #ifdef SHELL 1569897c45fSJilles Tjoelker INTON; 1579897c45fSJilles Tjoelker #endif 158fbd08684SStefan Farfeleder return (1); 1599897c45fSJilles Tjoelker } 160fbd08684SStefan Farfeleder end = 0; 1619b50d902SRodney W. Grimes } 162fbd08684SStefan Farfeleder start = fmt; 163fbd08684SStefan Farfeleder } else 164fbd08684SStefan Farfeleder fmt++; 1659b50d902SRodney W. Grimes } 1669b50d902SRodney W. Grimes 167fbd08684SStefan Farfeleder if (end == 1) { 1686a6760dbSJilles Tjoelker warnx("missing format character"); 1699897c45fSJilles Tjoelker #ifdef SHELL 1709897c45fSJilles Tjoelker INTON; 1719897c45fSJilles Tjoelker #endif 172fbd08684SStefan Farfeleder return (1); 173fbd08684SStefan Farfeleder } 174fbd08684SStefan Farfeleder fwrite(start, 1, fmt - start, stdout); 1759897c45fSJilles Tjoelker if (chopped || !*gargv) { 1769897c45fSJilles Tjoelker #ifdef SHELL 1779897c45fSJilles Tjoelker INTON; 1789897c45fSJilles Tjoelker #endif 179fbd08684SStefan Farfeleder return (rval); 1809897c45fSJilles Tjoelker } 181fbd08684SStefan Farfeleder /* Restart at the beginning of the format string. */ 182fbd08684SStefan Farfeleder fmt = format; 183fbd08684SStefan Farfeleder end = 1; 184fbd08684SStefan Farfeleder } 185fbd08684SStefan Farfeleder /* NOTREACHED */ 186fbd08684SStefan Farfeleder } 187fbd08684SStefan Farfeleder 188fbd08684SStefan Farfeleder 189fbd08684SStefan Farfeleder static char * 1909897c45fSJilles Tjoelker printf_doformat(char *start, int *rval) 191fbd08684SStefan Farfeleder { 192fbd08684SStefan Farfeleder static const char skip1[] = "#'-+ 0"; 193fbd08684SStefan Farfeleder static const char skip2[] = "0123456789"; 194fbd08684SStefan Farfeleder char *fmt; 195fbd08684SStefan Farfeleder int fieldwidth, haveprec, havewidth, mod_ldbl, precision; 196fbd08684SStefan Farfeleder char convch, nextch; 197fbd08684SStefan Farfeleder 198fbd08684SStefan Farfeleder fmt = start + 1; 1999b50d902SRodney W. Grimes /* skip to field width */ 2000ba01198SStefan Farfeleder fmt += strspn(fmt, skip1); 2019b50d902SRodney W. Grimes if (*fmt == '*') { 2029b50d902SRodney W. Grimes if (getint(&fieldwidth)) 203fbd08684SStefan Farfeleder return (NULL); 20498dd6386STim J. Robbins havewidth = 1; 205d867cefdSJoerg Wunsch ++fmt; 206d867cefdSJoerg Wunsch } else { 20798dd6386STim J. Robbins havewidth = 0; 2089b50d902SRodney W. Grimes 2099b50d902SRodney W. Grimes /* skip to possible '.', get following precision */ 2100ba01198SStefan Farfeleder fmt += strspn(fmt, skip2); 211d867cefdSJoerg Wunsch } 212d867cefdSJoerg Wunsch if (*fmt == '.') { 213d867cefdSJoerg Wunsch /* precision present? */ 2149b50d902SRodney W. Grimes ++fmt; 2159b50d902SRodney W. Grimes if (*fmt == '*') { 2169b50d902SRodney W. Grimes if (getint(&precision)) 217fbd08684SStefan Farfeleder return (NULL); 21898dd6386STim J. Robbins haveprec = 1; 219d867cefdSJoerg Wunsch ++fmt; 220d867cefdSJoerg Wunsch } else { 22198dd6386STim J. Robbins haveprec = 0; 2229b50d902SRodney W. Grimes 2239b50d902SRodney W. Grimes /* skip to conversion char */ 2240ba01198SStefan Farfeleder fmt += strspn(fmt, skip2); 225d867cefdSJoerg Wunsch } 226d867cefdSJoerg Wunsch } else 22798dd6386STim J. Robbins haveprec = 0; 2289b50d902SRodney W. Grimes if (!*fmt) { 2296a6760dbSJilles Tjoelker warnx("missing format character"); 230fbd08684SStefan Farfeleder return (NULL); 2319b50d902SRodney W. Grimes } 2329b50d902SRodney W. Grimes 233fd757c50SDavid Schultz /* 234fd757c50SDavid Schultz * Look for a length modifier. POSIX doesn't have these, so 235fd757c50SDavid Schultz * we only support them for floating-point conversions, which 236fd757c50SDavid Schultz * are extensions. This is useful because the L modifier can 237fd757c50SDavid Schultz * be used to gain extra range and precision, while omitting 238fd757c50SDavid Schultz * it is more likely to produce consistent results on different 239fd757c50SDavid Schultz * architectures. This is not so important for integers 240fd757c50SDavid Schultz * because overflow is the only bad thing that can happen to 241fd757c50SDavid Schultz * them, but consider the command printf %a 1.1 242fd757c50SDavid Schultz */ 243fd757c50SDavid Schultz if (*fmt == 'L') { 244fd757c50SDavid Schultz mod_ldbl = 1; 245fd757c50SDavid Schultz fmt++; 246fd757c50SDavid Schultz if (!strchr("aAeEfFgG", *fmt)) { 2476a6760dbSJilles Tjoelker warnx("bad modifier L for %%%c", *fmt); 248fbd08684SStefan Farfeleder return (NULL); 249fd757c50SDavid Schultz } 250fd757c50SDavid Schultz } else { 251fd757c50SDavid Schultz mod_ldbl = 0; 252fd757c50SDavid Schultz } 253fd757c50SDavid Schultz 2549b50d902SRodney W. Grimes convch = *fmt; 2559b50d902SRodney W. Grimes nextch = *++fmt; 2569b50d902SRodney W. Grimes *fmt = '\0'; 2579b50d902SRodney W. Grimes switch (convch) { 258ab5a295bSJuli Mallett case 'b': { 2593ec96cafSStefan Farfeleder size_t len; 260ab5a295bSJuli Mallett char *p; 261ab5a295bSJuli Mallett int getout; 262ab5a295bSJuli Mallett 263bacab7d6STim J. Robbins p = strdup(getstr()); 264bacab7d6STim J. Robbins if (p == NULL) { 2656a6760dbSJilles Tjoelker warnx("%s", strerror(ENOMEM)); 266fbd08684SStefan Farfeleder return (NULL); 267bacab7d6STim J. Robbins } 2683ec96cafSStefan Farfeleder getout = escape(p, 0, &len); 269ab5a295bSJuli Mallett *(fmt - 1) = 's'; 270bacab7d6STim J. Robbins PF(start, p); 271ab5a295bSJuli Mallett *(fmt - 1) = 'b'; 272ab5a295bSJuli Mallett free(p); 273ab5a295bSJuli Mallett if (getout) 274fbd08684SStefan Farfeleder return (fmt); 275ab5a295bSJuli Mallett break; 276ab5a295bSJuli Mallett } 2779b50d902SRodney W. Grimes case 'c': { 2789b50d902SRodney W. Grimes char p; 2799b50d902SRodney W. Grimes 2809b50d902SRodney W. Grimes p = getchr(); 2819b50d902SRodney W. Grimes PF(start, p); 2829b50d902SRodney W. Grimes break; 2839b50d902SRodney W. Grimes } 2849b50d902SRodney W. Grimes case 's': { 28545af1a4cSDavid Malone const char *p; 2869b50d902SRodney W. Grimes 2879b50d902SRodney W. Grimes p = getstr(); 2889b50d902SRodney W. Grimes PF(start, p); 2899b50d902SRodney W. Grimes break; 2909b50d902SRodney W. Grimes } 2919b50d902SRodney W. Grimes case 'd': case 'i': case 'o': case 'u': case 'x': case 'X': { 2929b50d902SRodney W. Grimes char *f; 293d41d23e1SStefan Farfeleder intmax_t val; 294d41d23e1SStefan Farfeleder uintmax_t uval; 295bacab7d6STim J. Robbins int signedconv; 2969b50d902SRodney W. Grimes 297bacab7d6STim J. Robbins signedconv = (convch == 'd' || convch == 'i'); 298d41d23e1SStefan Farfeleder if ((f = mknum(start, convch)) == NULL) 299fbd08684SStefan Farfeleder return (NULL); 300d41d23e1SStefan Farfeleder if (getnum(&val, &uval, signedconv)) 301fbd08684SStefan Farfeleder *rval = 1; 302bacab7d6STim J. Robbins if (signedconv) 303bacab7d6STim J. Robbins PF(f, val); 304bacab7d6STim J. Robbins else 305bacab7d6STim J. Robbins PF(f, uval); 3069b50d902SRodney W. Grimes break; 3079b50d902SRodney W. Grimes } 30803b2eaacSDavid Schultz case 'e': case 'E': 30903b2eaacSDavid Schultz case 'f': case 'F': 31003b2eaacSDavid Schultz case 'g': case 'G': 31103b2eaacSDavid Schultz case 'a': case 'A': { 312fd757c50SDavid Schultz long double p; 3139b50d902SRodney W. Grimes 314fd757c50SDavid Schultz if (getfloating(&p, mod_ldbl)) 315fbd08684SStefan Farfeleder *rval = 1; 316fd757c50SDavid Schultz if (mod_ldbl) 3179b50d902SRodney W. Grimes PF(start, p); 318fd757c50SDavid Schultz else 319fd757c50SDavid Schultz PF(start, (double)p); 3209b50d902SRodney W. Grimes break; 3219b50d902SRodney W. Grimes } 3229b50d902SRodney W. Grimes default: 3236a6760dbSJilles Tjoelker warnx("illegal format character %c", convch); 324fbd08684SStefan Farfeleder return (NULL); 3259b50d902SRodney W. Grimes } 3269b50d902SRodney W. Grimes *fmt = nextch; 327fbd08684SStefan Farfeleder return (fmt); 3289b50d902SRodney W. Grimes } 3299b50d902SRodney W. Grimes 3309b50d902SRodney W. Grimes static char * 3319180853bSXin LI mknum(char *str, char ch) 3329b50d902SRodney W. Grimes { 3333c6e4a5cSBen Smithurst static char *copy; 3343c6e4a5cSBen Smithurst static size_t copy_size; 3353c6e4a5cSBen Smithurst char *newcopy; 336bacab7d6STim J. Robbins size_t len, newlen; 3379b50d902SRodney W. Grimes 3389b50d902SRodney W. Grimes len = strlen(str) + 2; 3393c6e4a5cSBen Smithurst if (len > copy_size) { 3403c6e4a5cSBen Smithurst newlen = ((len + 1023) >> 10) << 10; 3413c6e4a5cSBen Smithurst if ((newcopy = realloc(copy, newlen)) == NULL) 342bacab7d6STim J. Robbins { 3436a6760dbSJilles Tjoelker warnx("%s", strerror(ENOMEM)); 3443c6e4a5cSBen Smithurst return (NULL); 345bacab7d6STim J. Robbins } 3463c6e4a5cSBen Smithurst copy = newcopy; 3473c6e4a5cSBen Smithurst copy_size = newlen; 3483c6e4a5cSBen Smithurst } 34962a721e7SStefan Eßer 3509b50d902SRodney W. Grimes memmove(copy, str, len - 3); 351d41d23e1SStefan Farfeleder copy[len - 3] = 'j'; 3529b50d902SRodney W. Grimes copy[len - 2] = ch; 3539b50d902SRodney W. Grimes copy[len - 1] = '\0'; 3549b50d902SRodney W. Grimes return (copy); 3559b50d902SRodney W. Grimes } 3569b50d902SRodney W. Grimes 357ab5a295bSJuli Mallett static int 3583ec96cafSStefan Farfeleder escape(char *fmt, int percent, size_t *len) 3599b50d902SRodney W. Grimes { 360*ab71f271SPedro F. Giffuni char *save, *store, c; 361*ab71f271SPedro F. Giffuni int value; 3629b50d902SRodney W. Grimes 363*ab71f271SPedro F. Giffuni for (save = store = fmt; ((c = *fmt) != 0); ++fmt, ++store) { 3649b50d902SRodney W. Grimes if (c != '\\') { 3659b50d902SRodney W. Grimes *store = c; 3669b50d902SRodney W. Grimes continue; 3679b50d902SRodney W. Grimes } 3689b50d902SRodney W. Grimes switch (*++fmt) { 3699b50d902SRodney W. Grimes case '\0': /* EOS, user error */ 3709b50d902SRodney W. Grimes *store = '\\'; 3719b50d902SRodney W. Grimes *++store = '\0'; 3723ec96cafSStefan Farfeleder *len = store - save; 373ab5a295bSJuli Mallett return (0); 3749b50d902SRodney W. Grimes case '\\': /* backslash */ 3759b50d902SRodney W. Grimes case '\'': /* single quote */ 3769b50d902SRodney W. Grimes *store = *fmt; 3779b50d902SRodney W. Grimes break; 3789b50d902SRodney W. Grimes case 'a': /* bell/alert */ 379f3f148d2SStefan Farfeleder *store = '\a'; 3809b50d902SRodney W. Grimes break; 3819b50d902SRodney W. Grimes case 'b': /* backspace */ 3829b50d902SRodney W. Grimes *store = '\b'; 3839b50d902SRodney W. Grimes break; 384ab5a295bSJuli Mallett case 'c': 385ab5a295bSJuli Mallett *store = '\0'; 3863ec96cafSStefan Farfeleder *len = store - save; 387ab5a295bSJuli Mallett return (1); 3889b50d902SRodney W. Grimes case 'f': /* form-feed */ 3899b50d902SRodney W. Grimes *store = '\f'; 3909b50d902SRodney W. Grimes break; 3919b50d902SRodney W. Grimes case 'n': /* newline */ 3929b50d902SRodney W. Grimes *store = '\n'; 3939b50d902SRodney W. Grimes break; 3949b50d902SRodney W. Grimes case 'r': /* carriage-return */ 3959b50d902SRodney W. Grimes *store = '\r'; 3969b50d902SRodney W. Grimes break; 3979b50d902SRodney W. Grimes case 't': /* horizontal tab */ 3989b50d902SRodney W. Grimes *store = '\t'; 3999b50d902SRodney W. Grimes break; 4009b50d902SRodney W. Grimes case 'v': /* vertical tab */ 401f3f148d2SStefan Farfeleder *store = '\v'; 4029b50d902SRodney W. Grimes break; 4039b50d902SRodney W. Grimes /* octal constant */ 4049b50d902SRodney W. Grimes case '0': case '1': case '2': case '3': 4059b50d902SRodney W. Grimes case '4': case '5': case '6': case '7': 4069d65050eSDavid Schultz c = (!percent && *fmt == '0') ? 4 : 3; 4079d65050eSDavid Schultz for (value = 0; 4089b50d902SRodney W. Grimes c-- && *fmt >= '0' && *fmt <= '7'; ++fmt) { 4099b50d902SRodney W. Grimes value <<= 3; 4109b50d902SRodney W. Grimes value += *fmt - '0'; 4119b50d902SRodney W. Grimes } 4129b50d902SRodney W. Grimes --fmt; 41312e8db40STim J. Robbins if (percent && value == '%') { 41437fd4590STim J. Robbins *store++ = '%'; 41537fd4590STim J. Robbins *store = '%'; 41637fd4590STim J. Robbins } else 417*ab71f271SPedro F. Giffuni *store = (char)value; 4189b50d902SRodney W. Grimes break; 4199b50d902SRodney W. Grimes default: 4209b50d902SRodney W. Grimes *store = *fmt; 4219b50d902SRodney W. Grimes break; 4229b50d902SRodney W. Grimes } 4239b50d902SRodney W. Grimes } 4249b50d902SRodney W. Grimes *store = '\0'; 4253ec96cafSStefan Farfeleder *len = store - save; 426ab5a295bSJuli Mallett return (0); 4279b50d902SRodney W. Grimes } 4289b50d902SRodney W. Grimes 4299b50d902SRodney W. Grimes static int 430f4ac32deSDavid Malone getchr(void) 4319b50d902SRodney W. Grimes { 4329b50d902SRodney W. Grimes if (!*gargv) 4339b50d902SRodney W. Grimes return ('\0'); 4349b50d902SRodney W. Grimes return ((int)**gargv++); 4359b50d902SRodney W. Grimes } 4369b50d902SRodney W. Grimes 43745af1a4cSDavid Malone static const char * 438f4ac32deSDavid Malone getstr(void) 4399b50d902SRodney W. Grimes { 4409b50d902SRodney W. Grimes if (!*gargv) 4419b50d902SRodney W. Grimes return (""); 4429b50d902SRodney W. Grimes return (*gargv++); 4439b50d902SRodney W. Grimes } 4449b50d902SRodney W. Grimes 4459b50d902SRodney W. Grimes static int 446f4ac32deSDavid Malone getint(int *ip) 4479b50d902SRodney W. Grimes { 448d41d23e1SStefan Farfeleder intmax_t val; 449d41d23e1SStefan Farfeleder uintmax_t uval; 450bacab7d6STim J. Robbins int rval; 4519b50d902SRodney W. Grimes 452d41d23e1SStefan Farfeleder if (getnum(&val, &uval, 1)) 4539b50d902SRodney W. Grimes return (1); 454bacab7d6STim J. Robbins rval = 0; 455bacab7d6STim J. Robbins if (val < INT_MIN || val > INT_MAX) { 4566a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 457bacab7d6STim J. Robbins rval = 1; 458bacab7d6STim J. Robbins } 45962a721e7SStefan Eßer *ip = (int)val; 460bacab7d6STim J. Robbins return (rval); 4619b50d902SRodney W. Grimes } 4629b50d902SRodney W. Grimes 4639b50d902SRodney W. Grimes static int 464d41d23e1SStefan Farfeleder getnum(intmax_t *ip, uintmax_t *uip, int signedconv) 4659b50d902SRodney W. Grimes { 4669b50d902SRodney W. Grimes char *ep; 467bacab7d6STim J. Robbins int rval; 4689b50d902SRodney W. Grimes 4699b50d902SRodney W. Grimes if (!*gargv) { 470d41d23e1SStefan Farfeleder *ip = 0; 4719b50d902SRodney W. Grimes return (0); 4729b50d902SRodney W. Grimes } 473ab5a295bSJuli Mallett if (**gargv == '"' || **gargv == '\'') { 474bacab7d6STim J. Robbins if (signedconv) 475d41d23e1SStefan Farfeleder *ip = asciicode(); 476bacab7d6STim J. Robbins else 477d41d23e1SStefan Farfeleder *uip = asciicode(); 4789b50d902SRodney W. Grimes return (0); 4799b50d902SRodney W. Grimes } 480bacab7d6STim J. Robbins rval = 0; 481ab5a295bSJuli Mallett errno = 0; 482bacab7d6STim J. Robbins if (signedconv) 483d41d23e1SStefan Farfeleder *ip = strtoimax(*gargv, &ep, 0); 484bacab7d6STim J. Robbins else 485d41d23e1SStefan Farfeleder *uip = strtoumax(*gargv, &ep, 0); 486bacab7d6STim J. Robbins if (ep == *gargv) { 4876a6760dbSJilles Tjoelker warnx("%s: expected numeric value", *gargv); 488bacab7d6STim J. Robbins rval = 1; 489bacab7d6STim J. Robbins } 490bacab7d6STim J. Robbins else if (*ep != '\0') { 4916a6760dbSJilles Tjoelker warnx("%s: not completely converted", *gargv); 492bacab7d6STim J. Robbins rval = 1; 493bacab7d6STim J. Robbins } 494bacab7d6STim J. Robbins if (errno == ERANGE) { 4956a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 496bacab7d6STim J. Robbins rval = 1; 497bacab7d6STim J. Robbins } 498ab5a295bSJuli Mallett ++gargv; 499bacab7d6STim J. Robbins return (rval); 5009b50d902SRodney W. Grimes } 5019b50d902SRodney W. Grimes 502bacab7d6STim J. Robbins static int 503fd757c50SDavid Schultz getfloating(long double *dp, int mod_ldbl) 5049b50d902SRodney W. Grimes { 505ab5a295bSJuli Mallett char *ep; 506bacab7d6STim J. Robbins int rval; 507ab5a295bSJuli Mallett 5085ec2b8dcSStefan Farfeleder if (!*gargv) { 5095ec2b8dcSStefan Farfeleder *dp = 0.0; 510bacab7d6STim J. Robbins return (0); 5115ec2b8dcSStefan Farfeleder } 512ab5a295bSJuli Mallett if (**gargv == '"' || **gargv == '\'') { 513bacab7d6STim J. Robbins *dp = asciicode(); 514bacab7d6STim J. Robbins return (0); 515ab5a295bSJuli Mallett } 5168c423a99SColin Percival rval = 0; 517ab5a295bSJuli Mallett errno = 0; 518fd757c50SDavid Schultz if (mod_ldbl) 519fd757c50SDavid Schultz *dp = strtold(*gargv, &ep); 520fd757c50SDavid Schultz else 521bacab7d6STim J. Robbins *dp = strtod(*gargv, &ep); 522bacab7d6STim J. Robbins if (ep == *gargv) { 5236a6760dbSJilles Tjoelker warnx("%s: expected numeric value", *gargv); 524bacab7d6STim J. Robbins rval = 1; 525bacab7d6STim J. Robbins } else if (*ep != '\0') { 5266a6760dbSJilles Tjoelker warnx("%s: not completely converted", *gargv); 527bacab7d6STim J. Robbins rval = 1; 528bacab7d6STim J. Robbins } 529bacab7d6STim J. Robbins if (errno == ERANGE) { 5306a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 531bacab7d6STim J. Robbins rval = 1; 532bacab7d6STim J. Robbins } 533ab5a295bSJuli Mallett ++gargv; 534bacab7d6STim J. Robbins return (rval); 5359b50d902SRodney W. Grimes } 5369b50d902SRodney W. Grimes 5379b50d902SRodney W. Grimes static int 538f4ac32deSDavid Malone asciicode(void) 5399b50d902SRodney W. Grimes { 540f4ac32deSDavid Malone int ch; 54198102dabSJilles Tjoelker wchar_t wch; 54298102dabSJilles Tjoelker mbstate_t mbs; 5439b50d902SRodney W. Grimes 54498102dabSJilles Tjoelker ch = (unsigned char)**gargv; 54598102dabSJilles Tjoelker if (ch == '\'' || ch == '"') { 54698102dabSJilles Tjoelker memset(&mbs, 0, sizeof(mbs)); 54798102dabSJilles Tjoelker switch (mbrtowc(&wch, *gargv + 1, MB_LEN_MAX, &mbs)) { 54898102dabSJilles Tjoelker case (size_t)-2: 54998102dabSJilles Tjoelker case (size_t)-1: 55098102dabSJilles Tjoelker wch = (unsigned char)gargv[0][1]; 55198102dabSJilles Tjoelker break; 55298102dabSJilles Tjoelker case 0: 55398102dabSJilles Tjoelker wch = 0; 55498102dabSJilles Tjoelker break; 55598102dabSJilles Tjoelker } 55698102dabSJilles Tjoelker ch = wch; 55798102dabSJilles Tjoelker } 5589b50d902SRodney W. Grimes ++gargv; 5599b50d902SRodney W. Grimes return (ch); 5609b50d902SRodney W. Grimes } 5619b50d902SRodney W. Grimes 5629b50d902SRodney W. Grimes static void 563f4ac32deSDavid Malone usage(void) 5649b50d902SRodney W. Grimes { 565f682f10cSRuslan Ermilov (void)fprintf(stderr, "usage: printf format [arguments ...]\n"); 5669b50d902SRodney W. Grimes } 567