1c9aca921SXin LI /*- 2*8a16b7a1SPedro F. Giffuni * SPDX-License-Identifier: BSD-3-Clause 3*8a16b7a1SPedro F. Giffuni * 4f8a6c905SPedro F. Giffuni * Copyright 2014 Garrett D'Amore <garrett@damore.org> 5b0620803SPedro F. Giffuni * Copyright 2010 Nexenta Systems, Inc. All rights reserved. 69b50d902SRodney W. Grimes * Copyright (c) 1989, 1993 79b50d902SRodney W. Grimes * The Regents of the University of California. All rights reserved. 89b50d902SRodney W. Grimes * 99b50d902SRodney W. Grimes * Redistribution and use in source and binary forms, with or without 109b50d902SRodney W. Grimes * modification, are permitted provided that the following conditions 119b50d902SRodney W. Grimes * are met: 129b50d902SRodney W. Grimes * 1. Redistributions of source code must retain the above copyright 139b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer. 149b50d902SRodney W. Grimes * 2. Redistributions in binary form must reproduce the above copyright 159b50d902SRodney W. Grimes * notice, this list of conditions and the following disclaimer in the 169b50d902SRodney W. Grimes * documentation and/or other materials provided with the distribution. 17fbbd9655SWarner Losh * 3. Neither the name of the University nor the names of its contributors 189b50d902SRodney W. Grimes * may be used to endorse or promote products derived from this software 199b50d902SRodney W. Grimes * without specific prior written permission. 209b50d902SRodney W. Grimes * 219b50d902SRodney W. Grimes * THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND 229b50d902SRodney W. Grimes * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE 239b50d902SRodney W. Grimes * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE 249b50d902SRodney W. Grimes * ARE DISCLAIMED. IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE 259b50d902SRodney W. Grimes * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL 269b50d902SRodney W. Grimes * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS 279b50d902SRodney W. Grimes * OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) 289b50d902SRodney W. Grimes * HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT 299b50d902SRodney W. Grimes * LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY 309b50d902SRodney W. Grimes * OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF 319b50d902SRodney W. Grimes * SUCH DAMAGE. 329b50d902SRodney W. Grimes */ 3377fae5c1SJilles Tjoelker /* 3477fae5c1SJilles Tjoelker * Important: This file is used both as a standalone program /usr/bin/printf 3577fae5c1SJilles Tjoelker * and as a builtin for /bin/sh (#define SHELL). 3677fae5c1SJilles Tjoelker */ 379b50d902SRodney W. Grimes 381866e8abSJilles Tjoelker #ifndef SHELL 399b50d902SRodney W. Grimes #ifndef lint 403ec30b79SSteve Price static char const copyright[] = 419b50d902SRodney W. Grimes "@(#) Copyright (c) 1989, 1993\n\ 429b50d902SRodney W. Grimes The Regents of the University of California. All rights reserved.\n"; 439b50d902SRodney W. Grimes #endif /* not lint */ 449b50d902SRodney W. Grimes #endif 459b50d902SRodney W. Grimes 469b50d902SRodney W. Grimes #ifndef lint 47c9e05349SWarner Losh #if 0 483ec30b79SSteve Price static char const sccsid[] = "@(#)printf.c 8.1 (Berkeley) 7/20/93"; 49c9e05349SWarner Losh #endif 501ea7321bSMartin Cracauer static const char rcsid[] = 511ea7321bSMartin Cracauer "$FreeBSD$"; 529b50d902SRodney W. Grimes #endif /* not lint */ 539b50d902SRodney W. Grimes 549b50d902SRodney W. Grimes #include <sys/types.h> 559b50d902SRodney W. Grimes 56f8a6c905SPedro F. Giffuni #include <ctype.h> 579b50d902SRodney W. Grimes #include <err.h> 589b50d902SRodney W. Grimes #include <errno.h> 59d41d23e1SStefan Farfeleder #include <inttypes.h> 609b50d902SRodney W. Grimes #include <limits.h> 61c9aca921SXin LI #include <locale.h> 629b50d902SRodney W. Grimes #include <stdio.h> 639b50d902SRodney W. Grimes #include <stdlib.h> 649b50d902SRodney W. Grimes #include <string.h> 659d19feb5SSteve Price #include <unistd.h> 6698102dabSJilles Tjoelker #include <wchar.h> 679b50d902SRodney W. Grimes 689b50d902SRodney W. Grimes #ifdef SHELL 699b50d902SRodney W. Grimes #define main printfcmd 70d9f93710SJoerg Wunsch #include "bltin/bltin.h" 717cbda738SJilles Tjoelker #include "options.h" 729b50d902SRodney W. Grimes #endif 739b50d902SRodney W. Grimes 74bacab7d6STim J. Robbins #define PF(f, func) do { \ 7598dd6386STim J. Robbins if (havewidth) \ 7698dd6386STim J. Robbins if (haveprec) \ 77342b089bSJilles Tjoelker (void)printf(f, fieldwidth, precision, func); \ 789b50d902SRodney W. Grimes else \ 79342b089bSJilles Tjoelker (void)printf(f, fieldwidth, func); \ 8098dd6386STim J. Robbins else if (haveprec) \ 81342b089bSJilles Tjoelker (void)printf(f, precision, func); \ 829b50d902SRodney W. Grimes else \ 83342b089bSJilles Tjoelker (void)printf(f, func); \ 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 98f8a6c905SPedro F. Giffuni static const char digits[] = "0123456789"; 99f8a6c905SPedro F. Giffuni 10030238f49SPedro F. Giffuni static char end_fmt[1]; 10130238f49SPedro F. Giffuni 102b0620803SPedro F. Giffuni static int myargc; 103b0620803SPedro F. Giffuni static char **myargv; 1049b50d902SRodney W. Grimes static char **gargv; 105f8a6c905SPedro F. Giffuni static char **maxargv; 1069b50d902SRodney W. Grimes 1079b50d902SRodney W. Grimes int 108f4ac32deSDavid Malone main(int argc, char *argv[]) 1099b50d902SRodney W. Grimes { 1103ec96cafSStefan Farfeleder size_t len; 111437bce62SPedro F. Giffuni int end, rval; 112fbd08684SStefan Farfeleder char *format, *fmt, *start; 1131866e8abSJilles Tjoelker #ifndef SHELL 1147cbda738SJilles Tjoelker int ch; 1157cbda738SJilles Tjoelker 11669be0c5eSXin LI (void) setlocale(LC_ALL, ""); 117dc7d8c99SAndrey A. Chernov #endif 1187cbda738SJilles Tjoelker 1199897c45fSJilles Tjoelker #ifdef SHELL 1207cbda738SJilles Tjoelker nextopt(""); 1217cbda738SJilles Tjoelker argc -= argptr - argv; 1227cbda738SJilles Tjoelker argv = argptr; 1237cbda738SJilles Tjoelker #else 124de46de4dSXin LI while ((ch = getopt(argc, argv, "")) != -1) 125de46de4dSXin LI switch (ch) { 126de46de4dSXin LI case '?': 127de46de4dSXin LI default: 128de46de4dSXin LI usage(); 129de46de4dSXin LI return (1); 1309b50d902SRodney W. Grimes } 131de46de4dSXin LI argc -= optind; 132de46de4dSXin LI argv += optind; 1337cbda738SJilles Tjoelker #endif 1349b50d902SRodney W. Grimes 1359b50d902SRodney W. Grimes if (argc < 1) { 1369b50d902SRodney W. Grimes usage(); 1375eccc000SXin LI return (1); 1389b50d902SRodney W. Grimes } 1399b50d902SRodney W. Grimes 1409897c45fSJilles Tjoelker #ifdef SHELL 1419897c45fSJilles Tjoelker INTOFF; 1429897c45fSJilles Tjoelker #endif 1439b50d902SRodney W. Grimes /* 1449b50d902SRodney W. Grimes * Basic algorithm is to scan the format string for conversion 1459b50d902SRodney W. Grimes * specifications -- once one is found, find out if the field 1469b50d902SRodney W. Grimes * width or precision is a '*'; if it is, gather up value. Note, 1479b50d902SRodney W. Grimes * format strings are reused as necessary to use up the provided 1489b50d902SRodney W. Grimes * arguments, arguments of zero/null string are provided to use 1499b50d902SRodney W. Grimes * up the format string. 1509b50d902SRodney W. Grimes */ 1513ec96cafSStefan Farfeleder fmt = format = *argv; 152437bce62SPedro F. Giffuni escape(fmt, 1, &len); /* backslash interpretation */ 153fbd08684SStefan Farfeleder rval = end = 0; 1549b50d902SRodney W. Grimes gargv = ++argv; 155b0620803SPedro F. Giffuni 1569b50d902SRodney W. Grimes for (;;) { 157f8a6c905SPedro F. Giffuni maxargv = gargv; 158b0620803SPedro F. Giffuni 159b0620803SPedro F. Giffuni myargv = gargv; 160b0620803SPedro F. Giffuni for (myargc = 0; gargv[myargc]; myargc++) 161b0620803SPedro F. Giffuni /* nop */; 162fbd08684SStefan Farfeleder start = fmt; 1633ec96cafSStefan Farfeleder while (fmt < format + len) { 164fbd08684SStefan Farfeleder if (fmt[0] == '%') { 165fbd08684SStefan Farfeleder fwrite(start, 1, fmt - start, stdout); 166fbd08684SStefan Farfeleder if (fmt[1] == '%') { 1679b50d902SRodney W. Grimes /* %% prints a % */ 168fbd08684SStefan Farfeleder putchar('%'); 169fbd08684SStefan Farfeleder fmt += 2; 170fbd08684SStefan Farfeleder } else { 1719897c45fSJilles Tjoelker fmt = printf_doformat(fmt, &rval); 17230238f49SPedro F. Giffuni if (fmt == NULL || fmt == end_fmt) { 1739897c45fSJilles Tjoelker #ifdef SHELL 1749897c45fSJilles Tjoelker INTON; 1759897c45fSJilles Tjoelker #endif 17630238f49SPedro F. Giffuni return (fmt == NULL ? 1 : rval); 1779897c45fSJilles Tjoelker } 178fbd08684SStefan Farfeleder end = 0; 1799b50d902SRodney W. Grimes } 180fbd08684SStefan Farfeleder start = fmt; 181fbd08684SStefan Farfeleder } else 182fbd08684SStefan Farfeleder fmt++; 183b0620803SPedro F. Giffuni if (gargv > maxargv) 184b0620803SPedro F. Giffuni maxargv = gargv; 1859b50d902SRodney W. Grimes } 186b0620803SPedro F. Giffuni gargv = maxargv; 1879b50d902SRodney W. Grimes 188fbd08684SStefan Farfeleder if (end == 1) { 1896a6760dbSJilles Tjoelker warnx("missing format character"); 1909897c45fSJilles Tjoelker #ifdef SHELL 1919897c45fSJilles Tjoelker INTON; 1929897c45fSJilles Tjoelker #endif 193fbd08684SStefan Farfeleder return (1); 194fbd08684SStefan Farfeleder } 195fbd08684SStefan Farfeleder fwrite(start, 1, fmt - start, stdout); 196437bce62SPedro F. Giffuni if (!*gargv) { 1979897c45fSJilles Tjoelker #ifdef SHELL 1989897c45fSJilles Tjoelker INTON; 1999897c45fSJilles Tjoelker #endif 200fbd08684SStefan Farfeleder return (rval); 2019897c45fSJilles Tjoelker } 202fbd08684SStefan Farfeleder /* Restart at the beginning of the format string. */ 203fbd08684SStefan Farfeleder fmt = format; 204fbd08684SStefan Farfeleder end = 1; 205fbd08684SStefan Farfeleder } 206fbd08684SStefan Farfeleder /* NOTREACHED */ 207fbd08684SStefan Farfeleder } 208fbd08684SStefan Farfeleder 209fbd08684SStefan Farfeleder 210fbd08684SStefan Farfeleder static char * 211f8a6c905SPedro F. Giffuni printf_doformat(char *fmt, int *rval) 212fbd08684SStefan Farfeleder { 213fbd08684SStefan Farfeleder static const char skip1[] = "#'-+ 0"; 214fbd08684SStefan Farfeleder int fieldwidth, haveprec, havewidth, mod_ldbl, precision; 215fbd08684SStefan Farfeleder char convch, nextch; 21676f66be6SPedro F. Giffuni char start[strlen(fmt) + 1]; 217f8a6c905SPedro F. Giffuni char **fargv; 218f8a6c905SPedro F. Giffuni char *dptr; 219f8a6c905SPedro F. Giffuni int l; 220fbd08684SStefan Farfeleder 221f8a6c905SPedro F. Giffuni dptr = start; 222f8a6c905SPedro F. Giffuni *dptr++ = '%'; 223f8a6c905SPedro F. Giffuni *dptr = 0; 224f8a6c905SPedro F. Giffuni 225f8a6c905SPedro F. Giffuni fmt++; 226b0620803SPedro F. Giffuni 227b0620803SPedro F. Giffuni /* look for "n$" field index specifier */ 228f8a6c905SPedro F. Giffuni l = strspn(fmt, digits); 229f8a6c905SPedro F. Giffuni if ((l > 0) && (fmt[l] == '$')) { 230f8a6c905SPedro F. Giffuni int idx = atoi(fmt); 231b0620803SPedro F. Giffuni if (idx <= myargc) { 232b0620803SPedro F. Giffuni gargv = &myargv[idx - 1]; 233b0620803SPedro F. Giffuni } else { 234b0620803SPedro F. Giffuni gargv = &myargv[myargc]; 235b0620803SPedro F. Giffuni } 236f8a6c905SPedro F. Giffuni if (gargv > maxargv) 237f8a6c905SPedro F. Giffuni maxargv = gargv; 238f8a6c905SPedro F. Giffuni fmt += l + 1; 239f8a6c905SPedro F. Giffuni 240f8a6c905SPedro F. Giffuni /* save format argument */ 241f8a6c905SPedro F. Giffuni fargv = gargv; 242b0620803SPedro F. Giffuni } else { 243f8a6c905SPedro F. Giffuni fargv = NULL; 244b0620803SPedro F. Giffuni } 245b0620803SPedro F. Giffuni 2469b50d902SRodney W. Grimes /* skip to field width */ 2474aa17924SPedro F. Giffuni while (*fmt && strchr(skip1, *fmt) != NULL) { 248f8a6c905SPedro F. Giffuni *dptr++ = *fmt++; 249f8a6c905SPedro F. Giffuni *dptr = 0; 250f8a6c905SPedro F. Giffuni } 251f8a6c905SPedro F. Giffuni 2529b50d902SRodney W. Grimes if (*fmt == '*') { 253f8a6c905SPedro F. Giffuni 254f8a6c905SPedro F. Giffuni fmt++; 255f8a6c905SPedro F. Giffuni l = strspn(fmt, digits); 256f8a6c905SPedro F. Giffuni if ((l > 0) && (fmt[l] == '$')) { 257f8a6c905SPedro F. Giffuni int idx = atoi(fmt); 2584aa17924SPedro F. Giffuni if (fargv == NULL) { 2594aa17924SPedro F. Giffuni warnx("incomplete use of n$"); 2604aa17924SPedro F. Giffuni return (NULL); 2614aa17924SPedro F. Giffuni } 262f8a6c905SPedro F. Giffuni if (idx <= myargc) { 263f8a6c905SPedro F. Giffuni gargv = &myargv[idx - 1]; 264f8a6c905SPedro F. Giffuni } else { 265f8a6c905SPedro F. Giffuni gargv = &myargv[myargc]; 266f8a6c905SPedro F. Giffuni } 267f8a6c905SPedro F. Giffuni fmt += l + 1; 2684aa17924SPedro F. Giffuni } else if (fargv != NULL) { 2694aa17924SPedro F. Giffuni warnx("incomplete use of n$"); 2704aa17924SPedro F. Giffuni return (NULL); 271f8a6c905SPedro F. Giffuni } 272f8a6c905SPedro F. Giffuni 2739b50d902SRodney W. Grimes if (getint(&fieldwidth)) 274fbd08684SStefan Farfeleder return (NULL); 275f8a6c905SPedro F. Giffuni if (gargv > maxargv) 276f8a6c905SPedro F. Giffuni maxargv = gargv; 27798dd6386STim J. Robbins havewidth = 1; 278f8a6c905SPedro F. Giffuni 279f8a6c905SPedro F. Giffuni *dptr++ = '*'; 280f8a6c905SPedro F. Giffuni *dptr = 0; 281d867cefdSJoerg Wunsch } else { 28298dd6386STim J. Robbins havewidth = 0; 2839b50d902SRodney W. Grimes 2849b50d902SRodney W. Grimes /* skip to possible '.', get following precision */ 285f8a6c905SPedro F. Giffuni while (isdigit(*fmt)) { 286f8a6c905SPedro F. Giffuni *dptr++ = *fmt++; 287f8a6c905SPedro F. Giffuni *dptr = 0; 288d867cefdSJoerg Wunsch } 289f8a6c905SPedro F. Giffuni } 290f8a6c905SPedro F. Giffuni 291d867cefdSJoerg Wunsch if (*fmt == '.') { 292d867cefdSJoerg Wunsch /* precision present? */ 293f8a6c905SPedro F. Giffuni fmt++; 294f8a6c905SPedro F. Giffuni *dptr++ = '.'; 295f8a6c905SPedro F. Giffuni 2969b50d902SRodney W. Grimes if (*fmt == '*') { 297f8a6c905SPedro F. Giffuni 298f8a6c905SPedro F. Giffuni fmt++; 299f8a6c905SPedro F. Giffuni l = strspn(fmt, digits); 300f8a6c905SPedro F. Giffuni if ((l > 0) && (fmt[l] == '$')) { 301f8a6c905SPedro F. Giffuni int idx = atoi(fmt); 3024aa17924SPedro F. Giffuni if (fargv == NULL) { 3034aa17924SPedro F. Giffuni warnx("incomplete use of n$"); 3044aa17924SPedro F. Giffuni return (NULL); 3054aa17924SPedro F. Giffuni } 306f8a6c905SPedro F. Giffuni if (idx <= myargc) { 307f8a6c905SPedro F. Giffuni gargv = &myargv[idx - 1]; 308f8a6c905SPedro F. Giffuni } else { 309f8a6c905SPedro F. Giffuni gargv = &myargv[myargc]; 310f8a6c905SPedro F. Giffuni } 311f8a6c905SPedro F. Giffuni fmt += l + 1; 3124aa17924SPedro F. Giffuni } else if (fargv != NULL) { 3134aa17924SPedro F. Giffuni warnx("incomplete use of n$"); 3144aa17924SPedro F. Giffuni return (NULL); 315f8a6c905SPedro F. Giffuni } 316f8a6c905SPedro F. Giffuni 3179b50d902SRodney W. Grimes if (getint(&precision)) 318fbd08684SStefan Farfeleder return (NULL); 319f8a6c905SPedro F. Giffuni if (gargv > maxargv) 320f8a6c905SPedro F. Giffuni maxargv = gargv; 32198dd6386STim J. Robbins haveprec = 1; 322f8a6c905SPedro F. Giffuni *dptr++ = '*'; 323f8a6c905SPedro F. Giffuni *dptr = 0; 324d867cefdSJoerg Wunsch } else { 32598dd6386STim J. Robbins haveprec = 0; 3269b50d902SRodney W. Grimes 3279b50d902SRodney W. Grimes /* skip to conversion char */ 328f8a6c905SPedro F. Giffuni while (isdigit(*fmt)) { 329f8a6c905SPedro F. Giffuni *dptr++ = *fmt++; 330f8a6c905SPedro F. Giffuni *dptr = 0; 331f8a6c905SPedro F. Giffuni } 332d867cefdSJoerg Wunsch } 333d867cefdSJoerg Wunsch } else 33498dd6386STim J. Robbins haveprec = 0; 3359b50d902SRodney W. Grimes if (!*fmt) { 3366a6760dbSJilles Tjoelker warnx("missing format character"); 337fbd08684SStefan Farfeleder return (NULL); 3389b50d902SRodney W. Grimes } 339f8a6c905SPedro F. Giffuni *dptr++ = *fmt; 340f8a6c905SPedro F. Giffuni *dptr = 0; 3419b50d902SRodney W. Grimes 342fd757c50SDavid Schultz /* 343fd757c50SDavid Schultz * Look for a length modifier. POSIX doesn't have these, so 344fd757c50SDavid Schultz * we only support them for floating-point conversions, which 345fd757c50SDavid Schultz * are extensions. This is useful because the L modifier can 346fd757c50SDavid Schultz * be used to gain extra range and precision, while omitting 347fd757c50SDavid Schultz * it is more likely to produce consistent results on different 348fd757c50SDavid Schultz * architectures. This is not so important for integers 349fd757c50SDavid Schultz * because overflow is the only bad thing that can happen to 350fd757c50SDavid Schultz * them, but consider the command printf %a 1.1 351fd757c50SDavid Schultz */ 352fd757c50SDavid Schultz if (*fmt == 'L') { 353fd757c50SDavid Schultz mod_ldbl = 1; 354fd757c50SDavid Schultz fmt++; 355fd757c50SDavid Schultz if (!strchr("aAeEfFgG", *fmt)) { 3566a6760dbSJilles Tjoelker warnx("bad modifier L for %%%c", *fmt); 357fbd08684SStefan Farfeleder return (NULL); 358fd757c50SDavid Schultz } 359fd757c50SDavid Schultz } else { 360fd757c50SDavid Schultz mod_ldbl = 0; 361fd757c50SDavid Schultz } 362fd757c50SDavid Schultz 363f8a6c905SPedro F. Giffuni /* save the current arg offset, and set to the format arg */ 364f8a6c905SPedro F. Giffuni if (fargv != NULL) { 365f8a6c905SPedro F. Giffuni gargv = fargv; 366f8a6c905SPedro F. Giffuni } 367f8a6c905SPedro F. Giffuni 3689b50d902SRodney W. Grimes convch = *fmt; 3699b50d902SRodney W. Grimes nextch = *++fmt; 370f8a6c905SPedro F. Giffuni 3719b50d902SRodney W. Grimes *fmt = '\0'; 3729b50d902SRodney W. Grimes switch (convch) { 373ab5a295bSJuli Mallett case 'b': { 3743ec96cafSStefan Farfeleder size_t len; 375ab5a295bSJuli Mallett char *p; 376ab5a295bSJuli Mallett int getout; 377ab5a295bSJuli Mallett 378bacab7d6STim J. Robbins p = strdup(getstr()); 379bacab7d6STim J. Robbins if (p == NULL) { 3806a6760dbSJilles Tjoelker warnx("%s", strerror(ENOMEM)); 381fbd08684SStefan Farfeleder return (NULL); 382bacab7d6STim J. Robbins } 3833ec96cafSStefan Farfeleder getout = escape(p, 0, &len); 384437bce62SPedro F. Giffuni fputs(p, stdout); 385ab5a295bSJuli Mallett free(p); 386ab5a295bSJuli Mallett if (getout) 38730238f49SPedro F. Giffuni return (end_fmt); 388ab5a295bSJuli Mallett break; 389ab5a295bSJuli Mallett } 3909b50d902SRodney W. Grimes case 'c': { 3919b50d902SRodney W. Grimes char p; 3929b50d902SRodney W. Grimes 3939b50d902SRodney W. Grimes p = getchr(); 394342b089bSJilles Tjoelker if (p != '\0') 3959b50d902SRodney W. Grimes PF(start, p); 3969b50d902SRodney W. Grimes break; 3979b50d902SRodney W. Grimes } 3989b50d902SRodney W. Grimes case 's': { 39945af1a4cSDavid Malone const char *p; 4009b50d902SRodney W. Grimes 4019b50d902SRodney W. Grimes p = getstr(); 4029b50d902SRodney W. Grimes PF(start, p); 4039b50d902SRodney W. Grimes break; 4049b50d902SRodney W. Grimes } 4059b50d902SRodney W. Grimes case 'd': case 'i': case 'o': case 'u': case 'x': case 'X': { 4069b50d902SRodney W. Grimes char *f; 407d41d23e1SStefan Farfeleder intmax_t val; 408d41d23e1SStefan Farfeleder uintmax_t uval; 409bacab7d6STim J. Robbins int signedconv; 4109b50d902SRodney W. Grimes 411bacab7d6STim J. Robbins signedconv = (convch == 'd' || convch == 'i'); 412d41d23e1SStefan Farfeleder if ((f = mknum(start, convch)) == NULL) 413fbd08684SStefan Farfeleder return (NULL); 414d41d23e1SStefan Farfeleder if (getnum(&val, &uval, signedconv)) 415fbd08684SStefan Farfeleder *rval = 1; 416bacab7d6STim J. Robbins if (signedconv) 417bacab7d6STim J. Robbins PF(f, val); 418bacab7d6STim J. Robbins else 419bacab7d6STim J. Robbins PF(f, uval); 4209b50d902SRodney W. Grimes break; 4219b50d902SRodney W. Grimes } 42203b2eaacSDavid Schultz case 'e': case 'E': 42303b2eaacSDavid Schultz case 'f': case 'F': 42403b2eaacSDavid Schultz case 'g': case 'G': 42503b2eaacSDavid Schultz case 'a': case 'A': { 426fd757c50SDavid Schultz long double p; 4279b50d902SRodney W. Grimes 428fd757c50SDavid Schultz if (getfloating(&p, mod_ldbl)) 429fbd08684SStefan Farfeleder *rval = 1; 430fd757c50SDavid Schultz if (mod_ldbl) 4319b50d902SRodney W. Grimes PF(start, p); 432fd757c50SDavid Schultz else 433fd757c50SDavid Schultz PF(start, (double)p); 4349b50d902SRodney W. Grimes break; 4359b50d902SRodney W. Grimes } 4369b50d902SRodney W. Grimes default: 4376a6760dbSJilles Tjoelker warnx("illegal format character %c", convch); 438fbd08684SStefan Farfeleder return (NULL); 4399b50d902SRodney W. Grimes } 4409b50d902SRodney W. Grimes *fmt = nextch; 441f8a6c905SPedro F. Giffuni /* return the gargv to the next element */ 442fbd08684SStefan Farfeleder return (fmt); 4439b50d902SRodney W. Grimes } 4449b50d902SRodney W. Grimes 4459b50d902SRodney W. Grimes static char * 4469180853bSXin LI mknum(char *str, char ch) 4479b50d902SRodney W. Grimes { 4483c6e4a5cSBen Smithurst static char *copy; 4493c6e4a5cSBen Smithurst static size_t copy_size; 4503c6e4a5cSBen Smithurst char *newcopy; 451bacab7d6STim J. Robbins size_t len, newlen; 4529b50d902SRodney W. Grimes 4539b50d902SRodney W. Grimes len = strlen(str) + 2; 4543c6e4a5cSBen Smithurst if (len > copy_size) { 4553c6e4a5cSBen Smithurst newlen = ((len + 1023) >> 10) << 10; 456ea1a630aSPedro F. Giffuni if ((newcopy = realloc(copy, newlen)) == NULL) { 4576a6760dbSJilles Tjoelker warnx("%s", strerror(ENOMEM)); 4583c6e4a5cSBen Smithurst return (NULL); 459bacab7d6STim J. Robbins } 4603c6e4a5cSBen Smithurst copy = newcopy; 4613c6e4a5cSBen Smithurst copy_size = newlen; 4623c6e4a5cSBen Smithurst } 46362a721e7SStefan Eßer 4649b50d902SRodney W. Grimes memmove(copy, str, len - 3); 465d41d23e1SStefan Farfeleder copy[len - 3] = 'j'; 4669b50d902SRodney W. Grimes copy[len - 2] = ch; 4679b50d902SRodney W. Grimes copy[len - 1] = '\0'; 4689b50d902SRodney W. Grimes return (copy); 4699b50d902SRodney W. Grimes } 4709b50d902SRodney W. Grimes 471ab5a295bSJuli Mallett static int 4723ec96cafSStefan Farfeleder escape(char *fmt, int percent, size_t *len) 4739b50d902SRodney W. Grimes { 474ab71f271SPedro F. Giffuni char *save, *store, c; 475ab71f271SPedro F. Giffuni int value; 4769b50d902SRodney W. Grimes 477ab71f271SPedro F. Giffuni for (save = store = fmt; ((c = *fmt) != 0); ++fmt, ++store) { 4789b50d902SRodney W. Grimes if (c != '\\') { 4799b50d902SRodney W. Grimes *store = c; 4809b50d902SRodney W. Grimes continue; 4819b50d902SRodney W. Grimes } 4829b50d902SRodney W. Grimes switch (*++fmt) { 4839b50d902SRodney W. Grimes case '\0': /* EOS, user error */ 4849b50d902SRodney W. Grimes *store = '\\'; 4859b50d902SRodney W. Grimes *++store = '\0'; 4863ec96cafSStefan Farfeleder *len = store - save; 487ab5a295bSJuli Mallett return (0); 4889b50d902SRodney W. Grimes case '\\': /* backslash */ 4899b50d902SRodney W. Grimes case '\'': /* single quote */ 4909b50d902SRodney W. Grimes *store = *fmt; 4919b50d902SRodney W. Grimes break; 4929b50d902SRodney W. Grimes case 'a': /* bell/alert */ 493f3f148d2SStefan Farfeleder *store = '\a'; 4949b50d902SRodney W. Grimes break; 4959b50d902SRodney W. Grimes case 'b': /* backspace */ 4969b50d902SRodney W. Grimes *store = '\b'; 4979b50d902SRodney W. Grimes break; 498ab5a295bSJuli Mallett case 'c': 499437bce62SPedro F. Giffuni if (!percent) { 500ab5a295bSJuli Mallett *store = '\0'; 5013ec96cafSStefan Farfeleder *len = store - save; 502ab5a295bSJuli Mallett return (1); 503437bce62SPedro F. Giffuni } 504437bce62SPedro F. Giffuni *store = 'c'; 505437bce62SPedro F. Giffuni break; 5069b50d902SRodney W. Grimes case 'f': /* form-feed */ 5079b50d902SRodney W. Grimes *store = '\f'; 5089b50d902SRodney W. Grimes break; 5099b50d902SRodney W. Grimes case 'n': /* newline */ 5109b50d902SRodney W. Grimes *store = '\n'; 5119b50d902SRodney W. Grimes break; 5129b50d902SRodney W. Grimes case 'r': /* carriage-return */ 5139b50d902SRodney W. Grimes *store = '\r'; 5149b50d902SRodney W. Grimes break; 5159b50d902SRodney W. Grimes case 't': /* horizontal tab */ 5169b50d902SRodney W. Grimes *store = '\t'; 5179b50d902SRodney W. Grimes break; 5189b50d902SRodney W. Grimes case 'v': /* vertical tab */ 519f3f148d2SStefan Farfeleder *store = '\v'; 5209b50d902SRodney W. Grimes break; 5219b50d902SRodney W. Grimes /* octal constant */ 5229b50d902SRodney W. Grimes case '0': case '1': case '2': case '3': 5239b50d902SRodney W. Grimes case '4': case '5': case '6': case '7': 5249d65050eSDavid Schultz c = (!percent && *fmt == '0') ? 4 : 3; 5259d65050eSDavid Schultz for (value = 0; 5269b50d902SRodney W. Grimes c-- && *fmt >= '0' && *fmt <= '7'; ++fmt) { 5279b50d902SRodney W. Grimes value <<= 3; 5289b50d902SRodney W. Grimes value += *fmt - '0'; 5299b50d902SRodney W. Grimes } 5309b50d902SRodney W. Grimes --fmt; 53112e8db40STim J. Robbins if (percent && value == '%') { 53237fd4590STim J. Robbins *store++ = '%'; 53337fd4590STim J. Robbins *store = '%'; 53437fd4590STim J. Robbins } else 535ab71f271SPedro F. Giffuni *store = (char)value; 5369b50d902SRodney W. Grimes break; 5379b50d902SRodney W. Grimes default: 5389b50d902SRodney W. Grimes *store = *fmt; 5399b50d902SRodney W. Grimes break; 5409b50d902SRodney W. Grimes } 5419b50d902SRodney W. Grimes } 5429b50d902SRodney W. Grimes *store = '\0'; 5433ec96cafSStefan Farfeleder *len = store - save; 544ab5a295bSJuli Mallett return (0); 5459b50d902SRodney W. Grimes } 5469b50d902SRodney W. Grimes 5479b50d902SRodney W. Grimes static int 548f4ac32deSDavid Malone getchr(void) 5499b50d902SRodney W. Grimes { 5509b50d902SRodney W. Grimes if (!*gargv) 5519b50d902SRodney W. Grimes return ('\0'); 5529b50d902SRodney W. Grimes return ((int)**gargv++); 5539b50d902SRodney W. Grimes } 5549b50d902SRodney W. Grimes 55545af1a4cSDavid Malone static const char * 556f4ac32deSDavid Malone getstr(void) 5579b50d902SRodney W. Grimes { 5589b50d902SRodney W. Grimes if (!*gargv) 5599b50d902SRodney W. Grimes return (""); 5609b50d902SRodney W. Grimes return (*gargv++); 5619b50d902SRodney W. Grimes } 5629b50d902SRodney W. Grimes 5639b50d902SRodney W. Grimes static int 564f4ac32deSDavid Malone getint(int *ip) 5659b50d902SRodney W. Grimes { 566d41d23e1SStefan Farfeleder intmax_t val; 567d41d23e1SStefan Farfeleder uintmax_t uval; 568bacab7d6STim J. Robbins int rval; 5699b50d902SRodney W. Grimes 570d41d23e1SStefan Farfeleder if (getnum(&val, &uval, 1)) 5719b50d902SRodney W. Grimes return (1); 572bacab7d6STim J. Robbins rval = 0; 573bacab7d6STim J. Robbins if (val < INT_MIN || val > INT_MAX) { 5746a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 575bacab7d6STim J. Robbins rval = 1; 576bacab7d6STim J. Robbins } 57762a721e7SStefan Eßer *ip = (int)val; 578bacab7d6STim J. Robbins return (rval); 5799b50d902SRodney W. Grimes } 5809b50d902SRodney W. Grimes 5819b50d902SRodney W. Grimes static int 582d41d23e1SStefan Farfeleder getnum(intmax_t *ip, uintmax_t *uip, int signedconv) 5839b50d902SRodney W. Grimes { 5849b50d902SRodney W. Grimes char *ep; 585bacab7d6STim J. Robbins int rval; 5869b50d902SRodney W. Grimes 5879b50d902SRodney W. Grimes if (!*gargv) { 5884e4d9802SJilles Tjoelker *ip = *uip = 0; 5899b50d902SRodney W. Grimes return (0); 5909b50d902SRodney W. Grimes } 591ab5a295bSJuli Mallett if (**gargv == '"' || **gargv == '\'') { 592bacab7d6STim J. Robbins if (signedconv) 593d41d23e1SStefan Farfeleder *ip = asciicode(); 594bacab7d6STim J. Robbins else 595d41d23e1SStefan Farfeleder *uip = asciicode(); 5969b50d902SRodney W. Grimes return (0); 5979b50d902SRodney W. Grimes } 598bacab7d6STim J. Robbins rval = 0; 599ab5a295bSJuli Mallett errno = 0; 600bacab7d6STim J. Robbins if (signedconv) 601d41d23e1SStefan Farfeleder *ip = strtoimax(*gargv, &ep, 0); 602bacab7d6STim J. Robbins else 603d41d23e1SStefan Farfeleder *uip = strtoumax(*gargv, &ep, 0); 604bacab7d6STim J. Robbins if (ep == *gargv) { 6056a6760dbSJilles Tjoelker warnx("%s: expected numeric value", *gargv); 606bacab7d6STim J. Robbins rval = 1; 607bacab7d6STim J. Robbins } 608bacab7d6STim J. Robbins else if (*ep != '\0') { 6096a6760dbSJilles Tjoelker warnx("%s: not completely converted", *gargv); 610bacab7d6STim J. Robbins rval = 1; 611bacab7d6STim J. Robbins } 612bacab7d6STim J. Robbins if (errno == ERANGE) { 6136a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 614bacab7d6STim J. Robbins rval = 1; 615bacab7d6STim J. Robbins } 616ab5a295bSJuli Mallett ++gargv; 617bacab7d6STim J. Robbins return (rval); 6189b50d902SRodney W. Grimes } 6199b50d902SRodney W. Grimes 620bacab7d6STim J. Robbins static int 621fd757c50SDavid Schultz getfloating(long double *dp, int mod_ldbl) 6229b50d902SRodney W. Grimes { 623ab5a295bSJuli Mallett char *ep; 624bacab7d6STim J. Robbins int rval; 625ab5a295bSJuli Mallett 6265ec2b8dcSStefan Farfeleder if (!*gargv) { 6275ec2b8dcSStefan Farfeleder *dp = 0.0; 628bacab7d6STim J. Robbins return (0); 6295ec2b8dcSStefan Farfeleder } 630ab5a295bSJuli Mallett if (**gargv == '"' || **gargv == '\'') { 631bacab7d6STim J. Robbins *dp = asciicode(); 632bacab7d6STim J. Robbins return (0); 633ab5a295bSJuli Mallett } 6348c423a99SColin Percival rval = 0; 635ab5a295bSJuli Mallett errno = 0; 636fd757c50SDavid Schultz if (mod_ldbl) 637fd757c50SDavid Schultz *dp = strtold(*gargv, &ep); 638fd757c50SDavid Schultz else 639bacab7d6STim J. Robbins *dp = strtod(*gargv, &ep); 640bacab7d6STim J. Robbins if (ep == *gargv) { 6416a6760dbSJilles Tjoelker warnx("%s: expected numeric value", *gargv); 642bacab7d6STim J. Robbins rval = 1; 643bacab7d6STim J. Robbins } else if (*ep != '\0') { 6446a6760dbSJilles Tjoelker warnx("%s: not completely converted", *gargv); 645bacab7d6STim J. Robbins rval = 1; 646bacab7d6STim J. Robbins } 647bacab7d6STim J. Robbins if (errno == ERANGE) { 6486a6760dbSJilles Tjoelker warnx("%s: %s", *gargv, strerror(ERANGE)); 649bacab7d6STim J. Robbins rval = 1; 650bacab7d6STim J. Robbins } 651ab5a295bSJuli Mallett ++gargv; 652bacab7d6STim J. Robbins return (rval); 6539b50d902SRodney W. Grimes } 6549b50d902SRodney W. Grimes 6559b50d902SRodney W. Grimes static int 656f4ac32deSDavid Malone asciicode(void) 6579b50d902SRodney W. Grimes { 658f4ac32deSDavid Malone int ch; 65998102dabSJilles Tjoelker wchar_t wch; 66098102dabSJilles Tjoelker mbstate_t mbs; 6619b50d902SRodney W. Grimes 66298102dabSJilles Tjoelker ch = (unsigned char)**gargv; 66398102dabSJilles Tjoelker if (ch == '\'' || ch == '"') { 66498102dabSJilles Tjoelker memset(&mbs, 0, sizeof(mbs)); 66598102dabSJilles Tjoelker switch (mbrtowc(&wch, *gargv + 1, MB_LEN_MAX, &mbs)) { 66698102dabSJilles Tjoelker case (size_t)-2: 66798102dabSJilles Tjoelker case (size_t)-1: 66898102dabSJilles Tjoelker wch = (unsigned char)gargv[0][1]; 66998102dabSJilles Tjoelker break; 67098102dabSJilles Tjoelker case 0: 67198102dabSJilles Tjoelker wch = 0; 67298102dabSJilles Tjoelker break; 67398102dabSJilles Tjoelker } 67498102dabSJilles Tjoelker ch = wch; 67598102dabSJilles Tjoelker } 6769b50d902SRodney W. Grimes ++gargv; 6779b50d902SRodney W. Grimes return (ch); 6789b50d902SRodney W. Grimes } 6799b50d902SRodney W. Grimes 6809b50d902SRodney W. Grimes static void 681f4ac32deSDavid Malone usage(void) 6829b50d902SRodney W. Grimes { 683f682f10cSRuslan Ermilov (void)fprintf(stderr, "usage: printf format [arguments ...]\n"); 6849b50d902SRodney W. Grimes } 685