i-bash/builtins/printf.def

1119 lines
23 KiB
Modula-2
Raw Normal View History

1998-04-17 19:52:44 +00:00
This file is printf.def, from which is created printf.c.
It implements the builtin "printf" in Bash.
2009-01-12 13:36:28 +00:00
Copyright (C) 1997-2009 Free Software Foundation, Inc.
1998-04-17 19:52:44 +00:00
This file is part of GNU Bash, the Bourne Again SHell.
2009-01-12 13:36:28 +00:00
Bash is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
1998-04-17 19:52:44 +00:00
2009-01-12 13:36:28 +00:00
Bash is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
1998-04-17 19:52:44 +00:00
2009-01-12 13:36:28 +00:00
You should have received a copy of the GNU General Public License
along with Bash. If not, see <http://www.gnu.org/licenses/>.
1998-04-17 19:52:44 +00:00
$PRODUCES printf.c
$BUILTIN printf
$FUNCTION printf_builtin
2005-12-07 14:08:12 +00:00
$SHORT_DOC printf [-v var] format [arguments]
2009-01-12 13:36:28 +00:00
Formats and prints ARGUMENTS under control of the FORMAT.
Options:
-v var assign the output to shell variable VAR rather than
display it on the standard output
FORMAT is a character string which contains three types of objects: plain
characters, which are simply copied to standard output; character escape
sequences, which are converted and copied to the standard output; and
1998-04-17 19:52:44 +00:00
format specifications, each of which causes printing of the next successive
2009-01-12 13:36:28 +00:00
argument.
In addition to the standard format specifications described in printf(1)
and printf(3), printf interprets:
%b expand backslash escape sequences in the corresponding argument
%q quote the argument in a way that can be reused as shell input
Exit Status:
Returns success unless an invalid option is given or a write or assignment
error occurs.
1998-04-17 19:52:44 +00:00
$END
#include <config.h>
#include "../bashtypes.h"
#include <errno.h>
#if defined (HAVE_LIMITS_H)
# include <limits.h>
#else
2001-11-13 17:56:06 +00:00
/* Assume 32-bit ints. */
1998-04-17 19:52:44 +00:00
# define INT_MAX 2147483647
# define INT_MIN (-2147483647-1)
#endif
2008-11-18 13:15:12 +00:00
#if defined (PREFER_STDARG)
# include <stdarg.h>
#else
# include <varargs.h>
#endif
1998-04-17 19:52:44 +00:00
#include <stdio.h>
2001-11-13 17:56:06 +00:00
#include <chartypes.h>
#ifdef HAVE_INTTYPES_H
# include <inttypes.h>
#endif
1998-04-17 19:52:44 +00:00
#include "../bashansi.h"
2004-07-27 13:29:18 +00:00
#include "../bashintl.h"
2001-11-13 17:56:06 +00:00
1998-04-17 19:52:44 +00:00
#include "../shell.h"
2009-01-12 13:36:28 +00:00
#include "shmbutil.h"
2000-03-17 21:46:59 +00:00
#include "stdc.h"
1998-04-17 19:52:44 +00:00
#include "bashgetopt.h"
2001-11-13 17:56:06 +00:00
#include "common.h"
2008-11-18 13:15:12 +00:00
#if defined (PRI_MACROS_BROKEN)
# undef PRIdMAX
#endif
2002-07-17 14:10:11 +00:00
#if !defined (PRIdMAX)
# if HAVE_LONG_LONG
# define PRIdMAX "lld"
# else
# define PRIdMAX "ld"
# endif
2001-11-13 17:56:06 +00:00
#endif
1998-04-17 19:52:44 +00:00
#if !defined (errno)
extern int errno;
#endif
2005-12-07 14:08:12 +00:00
#define PC(c) \
do { \
char b[2]; \
tw++; \
b[0] = c; b[1] = '\0'; \
if (vflag) \
vbadd (b, 1); \
else \
putchar (c); \
} while (0)
1998-04-17 19:52:44 +00:00
#define PF(f, func) \
do { \
2005-12-07 14:08:12 +00:00
int nw; \
2006-10-10 14:15:34 +00:00
clearerr (stdout); \
2001-11-13 17:56:06 +00:00
if (have_fieldwidth && have_precision) \
2009-01-12 13:36:28 +00:00
nw = vflag ? vbprintf (f, fieldwidth, precision, func) : printf (f, fieldwidth, precision, func); \
2001-11-13 17:56:06 +00:00
else if (have_fieldwidth) \
2009-01-12 13:36:28 +00:00
nw = vflag ? vbprintf (f, fieldwidth, func) : printf (f, fieldwidth, func); \
2001-11-13 17:56:06 +00:00
else if (have_precision) \
2011-11-21 20:57:16 -05:00
nw = vflag ? vbprintf (f, precision, func) : printf (f, precision, func); \
1998-04-17 19:52:44 +00:00
else \
2009-01-12 13:36:28 +00:00
nw = vflag ? vbprintf (f, func) : printf (f, func); \
2005-12-07 14:08:12 +00:00
tw += nw; \
2009-01-12 13:36:28 +00:00
if (ferror (stdout)) \
2005-12-07 14:08:12 +00:00
{ \
2009-01-12 13:36:28 +00:00
sh_wrerror (); \
clearerr (stdout); \
return (EXECUTION_FAILURE); \
2005-12-07 14:08:12 +00:00
} \
1998-04-17 19:52:44 +00:00
} while (0)
2001-11-13 17:56:06 +00:00
/* We free the buffer used by mklong() if it's `too big'. */
1998-04-17 19:52:44 +00:00
#define PRETURN(value) \
2001-11-13 17:56:06 +00:00
do \
{ \
2005-12-07 14:08:12 +00:00
if (vflag) \
{ \
2011-11-21 20:51:19 -05:00
bind_printf_variable (vname, vbuf, 0); \
2005-12-07 14:08:12 +00:00
stupidly_hack_special_variables (vname); \
} \
2001-11-13 17:56:06 +00:00
if (conv_bufsize > 4096 ) \
{ \
2005-12-07 14:08:12 +00:00
free (conv_buf); \
2001-11-13 17:56:06 +00:00
conv_bufsize = 0; \
conv_buf = 0; \
} \
2005-12-07 14:08:12 +00:00
if (vbsize > 4096) \
{ \
free (vbuf); \
vbsize = 0; \
vbuf = 0; \
} \
2009-01-12 13:36:28 +00:00
else if (vbuf) \
vbuf[0] = 0; \
terminate_immediately--; \
2001-11-13 17:56:06 +00:00
fflush (stdout); \
2006-10-10 14:15:34 +00:00
if (ferror (stdout)) \
{ \
2011-11-21 20:51:19 -05:00
sh_wrerror (); \
2006-10-10 14:15:34 +00:00
clearerr (stdout); \
return (EXECUTION_FAILURE); \
} \
2001-11-13 17:56:06 +00:00
return (value); \
} \
while (0)
#define SKIP1 "#'-+ 0"
#define LENMODS "hjlLtz"
1998-04-17 19:52:44 +00:00
2011-11-21 20:51:19 -05:00
#if !HAVE_ASPRINTF
2008-11-18 13:15:12 +00:00
extern int asprintf __P((char **, const char *, ...)) __attribute__((__format__ (printf, 2, 3)));
#endif
2011-11-21 20:51:19 -05:00
#if !HAVE_VSNPRINTF
2011-11-21 20:57:16 -05:00
extern int vsnprintf __P((char *, size_t, const char *, va_list)) __attribute__((__format__ (printf, 3, 0)));
2009-01-12 13:36:28 +00:00
#endif
2002-07-17 14:10:11 +00:00
static void printf_erange __P((char *));
2005-12-07 14:08:12 +00:00
static int printstr __P((char *, char *, int, int, int));
2004-07-27 13:29:18 +00:00
static int tescape __P((char *, char *, int *));
1998-04-17 19:52:44 +00:00
static char *bexpand __P((char *, int, int *, int *));
2005-12-07 14:08:12 +00:00
static char *vbadd __P((char *, int));
2009-01-12 13:36:28 +00:00
static int vbprintf __P((const char *, ...)) __attribute__((__format__ (printf, 1, 2)));
2002-07-17 14:10:11 +00:00
static char *mklong __P((char *, char *, size_t));
1998-04-17 19:52:44 +00:00
static int getchr __P((void));
static char *getstr __P((void));
static int getint __P((void));
2001-11-13 17:56:06 +00:00
static intmax_t getintmax __P((void));
static uintmax_t getuintmax __P((void));
2011-11-21 20:51:19 -05:00
static SHELL_VAR *bind_printf_variable __P((char *, char *, int));
2002-07-17 14:10:11 +00:00
2004-07-27 13:29:18 +00:00
#if defined (HAVE_LONG_DOUBLE) && HAVE_DECL_STRTOLD && !defined(STRTOLD_BROKEN)
2002-07-17 14:10:11 +00:00
typedef long double floatmax_t;
# define FLOATMAX_CONV "L"
# define strtofltmax strtold
#else
typedef double floatmax_t;
# define FLOATMAX_CONV ""
# define strtofltmax strtod
2001-11-13 17:56:06 +00:00
#endif
2002-07-17 14:10:11 +00:00
static floatmax_t getfloatmax __P((void));
2009-01-12 13:36:28 +00:00
static intmax_t asciicode __P((void));
1998-04-17 19:52:44 +00:00
static WORD_LIST *garglist;
static int retval;
2001-11-13 17:56:06 +00:00
static int conversion_error;
1998-04-17 19:52:44 +00:00
2005-12-07 14:08:12 +00:00
/* printf -v var support */
static int vflag = 0;
static char *vbuf, *vname;
static size_t vbsize;
static int vblen;
static intmax_t tw;
2001-11-13 17:56:06 +00:00
static char *conv_buf;
static size_t conv_bufsize;
1998-04-17 19:52:44 +00:00
int
printf_builtin (list)
WORD_LIST *list;
{
2001-11-13 17:56:06 +00:00
int ch, fieldwidth, precision;
int have_fieldwidth, have_precision;
char convch, thisch, nextch, *format, *modstart, *fmt, *start;
1998-04-17 19:52:44 +00:00
2001-11-13 17:56:06 +00:00
conversion_error = 0;
1998-04-17 19:52:44 +00:00
retval = EXECUTION_SUCCESS;
2002-07-17 14:10:11 +00:00
2005-12-07 14:08:12 +00:00
vflag = 0;
reset_internal_getopt ();
while ((ch = internal_getopt (list, "v:")) != -1)
{
switch (ch)
{
case 'v':
2011-11-21 20:51:19 -05:00
vname = list_optarg;
#if defined (ARRAY_VARS)
if (legal_identifier (vname) || valid_array_reference (vname))
#else
if (legal_identifier (vname))
#endif
2005-12-07 14:08:12 +00:00
{
vflag = 1;
vblen = 0;
2009-01-12 13:36:28 +00:00
if (vbuf)
vbuf[0] = 0;
2005-12-07 14:08:12 +00:00
}
else
{
sh_invalidid (vname);
return (EX_USAGE);
}
break;
default:
builtin_usage ();
return (EX_USAGE);
}
}
2002-07-17 14:10:11 +00:00
list = loptend; /* skip over possible `--' */
1998-04-17 19:52:44 +00:00
if (list == 0)
{
builtin_usage ();
return (EX_USAGE);
}
if (list->word->word == 0 || list->word->word[0] == '\0')
return (EXECUTION_SUCCESS);
2000-03-17 21:46:59 +00:00
format = list->word->word;
2005-12-07 14:08:12 +00:00
tw = 0;
1998-04-17 19:52:44 +00:00
garglist = list->next;
1998-07-23 14:37:54 +00:00
/* If the format string is empty after preprocessing, return immediately. */
2000-03-17 21:46:59 +00:00
if (format == 0 || *format == 0)
1998-07-23 14:37:54 +00:00
return (EXECUTION_SUCCESS);
2009-01-12 13:36:28 +00:00
terminate_immediately++;
1998-07-23 14:37:54 +00:00
1998-04-17 19:52:44 +00:00
/* Basic algorithm is to scan the format string for conversion
specifications -- once one is found, find out if the field
width or precision is a '*'; if it is, gather up value. Note,
format strings are reused as necessary to use up the provided
arguments, arguments of zero/null string are provided to use
up the format string. */
do
{
2001-11-13 17:56:06 +00:00
tw = 0;
1998-04-17 19:52:44 +00:00
/* find next format specification */
2000-03-17 21:46:59 +00:00
for (fmt = format; *fmt; fmt++)
1998-04-17 19:52:44 +00:00
{
2001-11-13 17:56:06 +00:00
precision = fieldwidth = 0;
have_fieldwidth = have_precision = 0;
2000-03-17 21:46:59 +00:00
if (*fmt == '\\')
{
fmt++;
2004-07-27 13:29:18 +00:00
/* A NULL third argument to tescape means to bypass the
special processing for arguments to %b. */
fmt += tescape (fmt, &nextch, (int *)NULL);
2005-12-07 14:08:12 +00:00
PC (nextch);
2000-03-17 21:46:59 +00:00
fmt--; /* for loop will increment it for us again */
continue;
}
1998-04-17 19:52:44 +00:00
if (*fmt != '%')
{
2005-12-07 14:08:12 +00:00
PC (*fmt);
1998-04-17 19:52:44 +00:00
continue;
}
/* ASSERT(*fmt == '%') */
start = fmt++;
if (*fmt == '%') /* %% prints a % */
{
2005-12-07 14:08:12 +00:00
PC ('%');
1998-04-17 19:52:44 +00:00
continue;
}
/* found format specification, skip to field width */
for (; *fmt && strchr(SKIP1, *fmt); ++fmt)
;
2001-11-13 17:56:06 +00:00
/* Skip optional field width. */
if (*fmt == '*')
{
fmt++;
have_fieldwidth = 1;
fieldwidth = getint ();
}
else
while (DIGIT (*fmt))
fmt++;
/* Skip optional '.' and precision */
1998-04-17 19:52:44 +00:00
if (*fmt == '.')
{
++fmt;
2001-11-13 17:56:06 +00:00
if (*fmt == '*')
{
fmt++;
have_precision = 1;
precision = getint ();
}
else
2005-12-07 14:08:12 +00:00
{
/* Negative precisions are allowed but treated as if the
precision were missing; I would like to allow a leading
`+' in the precision number as an extension, but lots
of asprintf/fprintf implementations get this wrong. */
#if 0
if (*fmt == '-' || *fmt == '+')
#else
if (*fmt == '-')
#endif
fmt++;
while (DIGIT (*fmt))
fmt++;
}
1998-04-17 19:52:44 +00:00
}
/* skip possible format modifiers */
2001-11-13 17:56:06 +00:00
modstart = fmt;
while (*fmt && strchr (LENMODS, *fmt))
fmt++;
1998-04-17 19:52:44 +00:00
1999-02-19 17:11:39 +00:00
if (*fmt == 0)
{
2004-07-27 13:29:18 +00:00
builtin_error (_("`%s': missing format character"), start);
1999-02-19 17:11:39 +00:00
PRETURN (EXECUTION_FAILURE);
}
1998-04-17 19:52:44 +00:00
convch = *fmt;
2001-11-13 17:56:06 +00:00
thisch = modstart[0];
nextch = modstart[1];
modstart[0] = convch;
modstart[1] = '\0';
1998-04-17 19:52:44 +00:00
switch(convch)
{
case 'c':
{
char p;
p = getchr ();
PF(start, p);
break;
}
case 's':
{
char *p;
p = getstr ();
PF(start, p);
break;
}
2001-11-13 17:56:06 +00:00
case 'n':
{
char *var;
var = getstr ();
if (var && *var)
{
if (legal_identifier (var))
bind_var_to_int (var, tw);
else
{
2002-07-17 14:10:11 +00:00
sh_invalidid (var);
2001-11-13 17:56:06 +00:00
PRETURN (EXECUTION_FAILURE);
}
}
break;
}
1998-04-17 19:52:44 +00:00
case 'b': /* expand escapes in argument */
{
char *p, *xp;
2005-12-07 14:08:12 +00:00
int rlen, r;
1998-04-17 19:52:44 +00:00
p = getstr ();
2005-12-07 14:08:12 +00:00
ch = rlen = r = 0;
1998-04-17 19:52:44 +00:00
xp = bexpand (p, strlen (p), &ch, &rlen);
if (xp)
{
/* Have to use printstr because of possible NUL bytes
in XP -- printf does not handle that well. */
2005-12-07 14:08:12 +00:00
r = printstr (start, xp, rlen, fieldwidth, precision);
if (r < 0)
{
sh_wrerror ();
clearerr (stdout);
retval = EXECUTION_FAILURE;
}
1998-04-17 19:52:44 +00:00
free (xp);
}
2005-12-07 14:08:12 +00:00
if (ch || r < 0)
1998-04-17 19:52:44 +00:00
PRETURN (retval);
break;
}
case 'q': /* print with shell quoting */
{
char *p, *xp;
2005-12-07 14:08:12 +00:00
int r;
1998-04-17 19:52:44 +00:00
2005-12-07 14:08:12 +00:00
r = 0;
1998-04-17 19:52:44 +00:00
p = getstr ();
2006-10-10 14:15:34 +00:00
if (p && *p == 0) /* XXX - getstr never returns null */
xp = savestring ("''");
else if (ansic_shouldquote (p))
2002-07-17 14:10:11 +00:00
xp = ansic_quote (p, 0, (int *)0);
else
xp = sh_backslash_quote (p);
1998-04-17 19:52:44 +00:00
if (xp)
{
/* Use printstr to get fieldwidth and precision right. */
2005-12-07 14:08:12 +00:00
r = printstr (start, xp, strlen (xp), fieldwidth, precision);
if (r < 0)
{
sh_wrerror ();
clearerr (stdout);
}
1998-04-17 19:52:44 +00:00
free (xp);
}
2005-12-07 14:08:12 +00:00
if (r < 0)
PRETURN (EXECUTION_FAILURE);
1998-04-17 19:52:44 +00:00
break;
}
case 'd':
case 'i':
{
char *f;
2002-07-17 14:10:11 +00:00
long p;
intmax_t pp;
2001-11-13 17:56:06 +00:00
2002-07-17 14:10:11 +00:00
p = pp = getintmax ();
if (p != pp)
2001-11-13 17:56:06 +00:00
{
2002-07-17 14:10:11 +00:00
f = mklong (start, PRIdMAX, sizeof (PRIdMAX) - 2);
PF (f, pp);
2001-11-13 17:56:06 +00:00
}
1998-04-17 19:52:44 +00:00
else
2001-11-13 17:56:06 +00:00
{
2002-07-17 14:10:11 +00:00
/* Optimize the common case where the integer fits
in "long". This also works around some long
long and/or intmax_t library bugs in the common
case, e.g. glibc 2.2 x86. */
f = mklong (start, "l", 1);
PF (f, p);
2001-11-13 17:56:06 +00:00
}
1998-04-17 19:52:44 +00:00
break;
}
case 'o':
case 'u':
case 'x':
case 'X':
{
char *f;
2002-07-17 14:10:11 +00:00
unsigned long p;
uintmax_t pp;
2001-11-13 17:56:06 +00:00
2002-07-17 14:10:11 +00:00
p = pp = getuintmax ();
if (p != pp)
2001-11-13 17:56:06 +00:00
{
2002-07-17 14:10:11 +00:00
f = mklong (start, PRIdMAX, sizeof (PRIdMAX) - 2);
PF (f, pp);
2001-11-13 17:56:06 +00:00
}
1998-04-17 19:52:44 +00:00
else
2001-11-13 17:56:06 +00:00
{
2002-07-17 14:10:11 +00:00
f = mklong (start, "l", 1);
2001-11-13 17:56:06 +00:00
PF (f, p);
}
1998-04-17 19:52:44 +00:00
break;
}
case 'e':
case 'E':
case 'f':
2001-11-13 17:56:06 +00:00
case 'F':
1998-04-17 19:52:44 +00:00
case 'g':
case 'G':
2001-11-13 17:56:06 +00:00
#if defined (HAVE_PRINTF_A_FORMAT)
case 'a':
case 'A':
#endif
1998-04-17 19:52:44 +00:00
{
2001-11-13 17:56:06 +00:00
char *f;
2002-07-17 14:10:11 +00:00
floatmax_t p;
2001-11-13 17:56:06 +00:00
2002-07-17 14:10:11 +00:00
p = getfloatmax ();
f = mklong (start, FLOATMAX_CONV, sizeof(FLOATMAX_CONV) - 1);
PF (f, p);
1998-04-17 19:52:44 +00:00
break;
}
2000-03-17 21:46:59 +00:00
/* We don't output unrecognized format characters; we print an
error message and return a failure exit status. */
1998-04-17 19:52:44 +00:00
default:
2004-07-27 13:29:18 +00:00
builtin_error (_("`%c': invalid format character"), convch);
1998-04-17 19:52:44 +00:00
PRETURN (EXECUTION_FAILURE);
}
2001-11-13 17:56:06 +00:00
modstart[0] = thisch;
modstart[1] = nextch;
1998-04-17 19:52:44 +00:00
}
2005-12-07 14:08:12 +00:00
if (ferror (stdout))
{
sh_wrerror ();
clearerr (stdout);
PRETURN (EXECUTION_FAILURE);
}
1998-04-17 19:52:44 +00:00
}
1998-07-23 14:37:54 +00:00
while (garglist && garglist != list->next);
1998-04-17 19:52:44 +00:00
2001-11-13 17:56:06 +00:00
if (conversion_error)
retval = EXECUTION_FAILURE;
1998-04-17 19:52:44 +00:00
PRETURN (retval);
}
2002-07-17 14:10:11 +00:00
static void
printf_erange (s)
char *s;
{
2009-01-12 13:36:28 +00:00
builtin_error (_("warning: %s: %s"), s, strerror(ERANGE));
2002-07-17 14:10:11 +00:00
}
1998-04-17 19:52:44 +00:00
/* We duplicate a lot of what printf(3) does here. */
2005-12-07 14:08:12 +00:00
static int
1998-04-17 19:52:44 +00:00
printstr (fmt, string, len, fieldwidth, precision)
char *fmt; /* format */
char *string; /* expanded string argument */
int len; /* length of expanded string */
int fieldwidth; /* argument for width of `*' */
int precision; /* argument for precision of `*' */
{
#if 0
char *s;
#endif
int padlen, nc, ljust, i;
int fw, pr; /* fieldwidth and precision */
2005-12-07 14:08:12 +00:00
#if 0
1998-04-17 19:52:44 +00:00
if (string == 0 || *string == '\0')
2005-12-07 14:08:12 +00:00
#else
if (string == 0 || len == 0)
#endif
2009-01-12 13:36:28 +00:00
return 0;
1998-04-17 19:52:44 +00:00
#if 0
s = fmt;
#endif
if (*fmt == '%')
fmt++;
2001-11-13 17:56:06 +00:00
ljust = fw = 0;
pr = -1;
1998-04-17 19:52:44 +00:00
/* skip flags */
2001-11-13 17:56:06 +00:00
while (strchr (SKIP1, *fmt))
1998-04-17 19:52:44 +00:00
{
if (*fmt == '-')
ljust = 1;
fmt++;
}
/* get fieldwidth, if present */
if (*fmt == '*')
{
fmt++;
fw = fieldwidth;
2001-11-13 17:56:06 +00:00
if (fw < 0)
{
fw = -fw;
ljust = 1;
}
1998-04-17 19:52:44 +00:00
}
2001-11-13 17:56:06 +00:00
else if (DIGIT (*fmt))
1998-04-17 19:52:44 +00:00
{
fw = *fmt++ - '0';
2001-11-13 17:56:06 +00:00
while (DIGIT (*fmt))
1998-04-17 19:52:44 +00:00
fw = (fw * 10) + (*fmt++ - '0');
}
/* get precision, if present */
if (*fmt == '.')
{
fmt++;
if (*fmt == '*')
{
fmt++;
pr = precision;
}
2001-11-13 17:56:06 +00:00
else if (DIGIT (*fmt))
1998-04-17 19:52:44 +00:00
{
pr = *fmt++ - '0';
2001-11-13 17:56:06 +00:00
while (DIGIT (*fmt))
1998-04-17 19:52:44 +00:00
pr = (pr * 10) + (*fmt++ - '0');
}
}
#if 0
/* If we remove this, get rid of `s'. */
if (*fmt != 'b' && *fmt != 'q')
{
internal_error ("format parsing problem: %s", s);
fw = pr = 0;
}
#endif
/* chars from string to print */
2001-11-13 17:56:06 +00:00
nc = (pr >= 0 && pr <= len) ? pr : len;
1998-04-17 19:52:44 +00:00
padlen = fw - nc;
if (padlen < 0)
padlen = 0;
if (ljust)
padlen = -padlen;
/* leading pad characters */
for (; padlen > 0; padlen--)
2005-12-07 14:08:12 +00:00
PC (' ');
1998-04-17 19:52:44 +00:00
/* output NC characters from STRING */
for (i = 0; i < nc; i++)
2005-12-07 14:08:12 +00:00
PC (string[i]);
1998-04-17 19:52:44 +00:00
/* output any necessary trailing padding */
for (; padlen < 0; padlen++)
2005-12-07 14:08:12 +00:00
PC (' ');
return (ferror (stdout) ? -1 : 0);
1998-04-17 19:52:44 +00:00
}
/* Convert STRING by expanding the escape sequences specified by the
POSIX standard for printf's `%b' format string. If SAWC is non-null,
2004-07-27 13:29:18 +00:00
perform the processing appropriate for %b arguments. In particular,
1998-04-17 19:52:44 +00:00
recognize `\c' and use that as a string terminator. If we see \c, set
*SAWC to 1 before returning. LEN is the length of STRING. */
2000-03-17 21:46:59 +00:00
/* Translate a single backslash-escape sequence starting at ESTART (the
character after the backslash) and return the number of characters
consumed by the sequence. CP is the place to return the translated
value. *SAWC is set to 1 if the escape sequence was \c, since that means
to short-circuit the rest of the processing. If SAWC is null, we don't
do the \c short-circuiting, and \c is treated as an unrecognized escape
2004-07-27 13:29:18 +00:00
sequence; we also bypass the other processing specific to %b arguments. */
2000-03-17 21:46:59 +00:00
static int
2004-07-27 13:29:18 +00:00
tescape (estart, cp, sawc)
2000-03-17 21:46:59 +00:00
char *estart;
char *cp;
int *sawc;
{
register char *p;
int temp, c, evalue;
p = estart;
switch (c = *p++)
{
#if defined (__STDC__)
case 'a': *cp = '\a'; break;
#else
case 'a': *cp = '\007'; break;
#endif
case 'b': *cp = '\b'; break;
2001-04-06 19:14:31 +00:00
case 'e':
case 'E': *cp = '\033'; break; /* ESC -- non-ANSI */
2000-03-17 21:46:59 +00:00
case 'f': *cp = '\f'; break;
case 'n': *cp = '\n'; break;
case 'r': *cp = '\r'; break;
case 't': *cp = '\t'; break;
case 'v': *cp = '\v'; break;
2004-07-27 13:29:18 +00:00
/* The octal escape sequences are `\0' followed by up to three octal
digits (if SAWC), or `\' followed by up to three octal digits (if
!SAWC). As an extension, we allow the latter form even if SAWC. */
case '0': case '1': case '2': case '3':
case '4': case '5': case '6': case '7':
evalue = OCTVALUE (c);
for (temp = 2 + (!evalue && !!sawc); ISOCTAL (*p) && temp--; p++)
2000-03-17 21:46:59 +00:00
evalue = (evalue * 8) + OCTVALUE (*p);
2001-11-13 17:56:06 +00:00
*cp = evalue & 0xFF;
2000-03-17 21:46:59 +00:00
break;
/* And, as another extension, we allow \xNNN, where each N is a
hex digit. */
case 'x':
2004-07-27 13:29:18 +00:00
#if 0
for (evalue = 0; ISXDIGIT ((unsigned char)*p); p++)
#else
2001-11-13 17:56:06 +00:00
for (temp = 2, evalue = 0; ISXDIGIT ((unsigned char)*p) && temp--; p++)
2004-07-27 13:29:18 +00:00
#endif
2000-03-17 21:46:59 +00:00
evalue = (evalue * 16) + HEXVALUE (*p);
2004-07-27 13:29:18 +00:00
if (p == estart + 1)
2000-03-17 21:46:59 +00:00
{
2004-07-27 13:29:18 +00:00
builtin_error (_("missing hex digit for \\x"));
2000-03-17 21:46:59 +00:00
*cp = '\\';
return 0;
}
2001-11-13 17:56:06 +00:00
*cp = evalue & 0xFF;
2000-03-17 21:46:59 +00:00
break;
case '\\': /* \\ -> \ */
*cp = c;
break;
2004-07-27 13:29:18 +00:00
/* SAWC == 0 means that \', \", and \? are recognized as escape
sequences, though the only processing performed is backslash
removal. */
case '\'': case '"': case '?':
if (!sawc)
2000-03-17 21:46:59 +00:00
*cp = c;
else
{
*cp = '\\';
return 0;
}
break;
case 'c':
if (sawc)
{
*sawc = 1;
break;
}
/* other backslash escapes are passed through unaltered */
default:
*cp = '\\';
return 0;
}
return (p - estart);
}
1998-04-17 19:52:44 +00:00
static char *
bexpand (string, len, sawc, lenp)
char *string;
int len, *sawc, *lenp;
{
2000-03-17 21:46:59 +00:00
int temp;
char *ret, *r, *s, c;
1998-04-17 19:52:44 +00:00
2005-12-07 14:08:12 +00:00
#if 0
1998-04-17 19:52:44 +00:00
if (string == 0 || *string == '\0')
2005-12-07 14:08:12 +00:00
#else
if (string == 0 || len == 0)
#endif
1998-04-17 19:52:44 +00:00
{
if (sawc)
*sawc = 0;
if (lenp)
*lenp = 0;
return ((char *)NULL);
}
2001-11-13 17:56:06 +00:00
ret = (char *)xmalloc (len + 1);
1998-04-17 19:52:44 +00:00
for (r = ret, s = string; s && *s; )
{
c = *s++;
if (c != '\\' || *s == '\0')
{
*r++ = c;
continue;
}
2000-03-17 21:46:59 +00:00
temp = 0;
2004-07-27 13:29:18 +00:00
s += tescape (s, &c, &temp);
2000-03-17 21:46:59 +00:00
if (temp)
1998-04-17 19:52:44 +00:00
{
if (sawc)
*sawc = 1;
2000-03-17 21:46:59 +00:00
break;
1998-04-17 19:52:44 +00:00
}
*r++ = c;
}
*r = '\0';
if (lenp)
*lenp = r - ret;
return ret;
}
2005-12-07 14:08:12 +00:00
static char *
vbadd (buf, blen)
char *buf;
int blen;
{
size_t nlen;
nlen = vblen + blen + 1;
if (nlen >= vbsize)
{
vbsize = ((nlen + 63) >> 6) << 6;
vbuf = (char *)xrealloc (vbuf, vbsize);
}
if (blen == 1)
vbuf[vblen++] = buf[0];
2009-01-12 13:36:28 +00:00
else if (blen > 1)
2005-12-07 14:08:12 +00:00
{
FASTCOPY (buf, vbuf + vblen, blen);
vblen += blen;
}
vbuf[vblen] = '\0';
#ifdef DEBUG
if (strlen (vbuf) != vblen)
2006-10-10 14:15:34 +00:00
internal_error ("printf:vbadd: vblen (%d) != strlen (vbuf) (%d)", vblen, (int)strlen (vbuf));
2005-12-07 14:08:12 +00:00
#endif
return vbuf;
}
2009-01-12 13:36:28 +00:00
static int
#if defined (PREFER_STDARG)
vbprintf (const char *format, ...)
#else
vbprintf (format, va_alist)
const char *format;
va_dcl
#endif
{
va_list args;
size_t nlen;
int blen;
SH_VA_START (args, format);
blen = vsnprintf (vbuf + vblen, vbsize - vblen, format, args);
2009-02-19 22:21:29 +00:00
va_end (args);
2009-01-12 13:36:28 +00:00
nlen = vblen + blen + 1;
if (nlen >= vbsize)
{
vbsize = ((nlen + 63) >> 6) << 6;
vbuf = (char *)xrealloc (vbuf, vbsize);
2009-02-19 22:21:29 +00:00
SH_VA_START (args, format);
2009-01-12 13:36:28 +00:00
blen = vsnprintf (vbuf + vblen, vbsize - vblen, format, args);
2009-02-19 22:21:29 +00:00
va_end (args);
2009-01-12 13:36:28 +00:00
}
vblen += blen;
vbuf[vblen] = '\0';
#ifdef DEBUG
if (strlen (vbuf) != vblen)
internal_error ("printf:vbadd: vblen (%d) != strlen (vbuf) (%d)", vblen, (int)strlen (vbuf));
#endif
return (blen);
}
1998-04-17 19:52:44 +00:00
static char *
2002-07-17 14:10:11 +00:00
mklong (str, modifiers, mlen)
1998-04-17 19:52:44 +00:00
char *str;
2001-11-13 17:56:06 +00:00
char *modifiers;
2002-07-17 14:10:11 +00:00
size_t mlen;
1998-04-17 19:52:44 +00:00
{
2002-07-17 14:10:11 +00:00
size_t len, slen;
2001-11-13 17:56:06 +00:00
slen = strlen (str);
len = slen + mlen + 1;
if (len > conv_bufsize)
{
conv_bufsize = (((len + 1023) >> 10) << 10);
conv_buf = (char *)xrealloc (conv_buf, conv_bufsize);
}
FASTCOPY (str, conv_buf, slen - 1);
FASTCOPY (modifiers, conv_buf + slen - 1, mlen);
conv_buf[len - 2] = str[slen - 1];
conv_buf[len - 1] = '\0';
return (conv_buf);
1998-04-17 19:52:44 +00:00
}
static int
getchr ()
{
int ret;
if (garglist == 0)
return ('\0');
ret = (int)garglist->word->word[0];
garglist = garglist->next;
return ret;
}
static char *
getstr ()
{
char *ret;
if (garglist == 0)
return ("");
ret = garglist->word->word;
garglist = garglist->next;
return ret;
}
static int
getint ()
{
2002-07-17 14:10:11 +00:00
intmax_t ret;
1998-04-17 19:52:44 +00:00
2002-07-17 14:10:11 +00:00
ret = getintmax ();
1998-04-17 19:52:44 +00:00
if (ret > INT_MAX)
{
2002-07-17 14:10:11 +00:00
printf_erange (garglist->word->word);
2001-04-06 19:14:31 +00:00
ret = INT_MAX;
}
else if (ret < INT_MIN)
{
2002-07-17 14:10:11 +00:00
printf_erange (garglist->word->word);
2001-04-06 19:14:31 +00:00
ret = INT_MIN;
1998-04-17 19:52:44 +00:00
}
return ((int)ret);
}
2001-11-13 17:56:06 +00:00
static intmax_t
getintmax ()
1998-04-17 19:52:44 +00:00
{
2001-11-13 17:56:06 +00:00
intmax_t ret;
2001-04-06 19:14:31 +00:00
char *ep;
1998-04-17 19:52:44 +00:00
if (garglist == 0)
2001-11-13 17:56:06 +00:00
return (0);
if (garglist->word->word[0] == '\'' || garglist->word->word[0] == '"')
return asciicode ();
errno = 0;
ret = strtoimax (garglist->word->word, &ep, 0);
if (*ep)
2001-04-06 19:14:31 +00:00
{
2002-07-17 14:10:11 +00:00
sh_invalidnum (garglist->word->word);
2001-11-13 17:56:06 +00:00
/* POSIX.2 says ``...a diagnostic message shall be written to standard
error, and the utility shall not exit with a zero exit status, but
shall continue processing any remaining operands and shall write the
value accumulated at the time the error was detected to standard
output.'' Yecch. */
2011-11-21 20:51:19 -05:00
#if 0
ret = 0; /* return partially-converted value from strtoimax */
#endif
2001-11-13 17:56:06 +00:00
conversion_error = 1;
2001-04-06 19:14:31 +00:00
}
2001-11-13 17:56:06 +00:00
else if (errno == ERANGE)
2002-07-17 14:10:11 +00:00
printf_erange (garglist->word->word);
2001-11-13 17:56:06 +00:00
garglist = garglist->next;
return (ret);
}
static uintmax_t
getuintmax ()
{
uintmax_t ret;
char *ep;
if (garglist == 0)
return (0);
1998-04-17 19:52:44 +00:00
if (garglist->word->word[0] == '\'' || garglist->word->word[0] == '"')
2001-11-13 17:56:06 +00:00
return asciicode ();
errno = 0;
ret = strtoumax (garglist->word->word, &ep, 0);
if (*ep)
2001-04-06 19:14:31 +00:00
{
2002-07-17 14:10:11 +00:00
sh_invalidnum (garglist->word->word);
/* Same POSIX.2 conversion error requirements as getintmax(). */
2001-11-13 17:56:06 +00:00
ret = 0;
conversion_error = 1;
2001-04-06 19:14:31 +00:00
}
2001-11-13 17:56:06 +00:00
else if (errno == ERANGE)
2002-07-17 14:10:11 +00:00
printf_erange (garglist->word->word);
2001-11-13 17:56:06 +00:00
garglist = garglist->next;
return (ret);
}
2002-07-17 14:10:11 +00:00
static floatmax_t
getfloatmax ()
2001-11-13 17:56:06 +00:00
{
2002-07-17 14:10:11 +00:00
floatmax_t ret;
2001-11-13 17:56:06 +00:00
char *ep;
if (garglist == 0)
return (0);
if (garglist->word->word[0] == '\'' || garglist->word->word[0] == '"')
return asciicode ();
1998-04-17 19:52:44 +00:00
2001-04-06 19:14:31 +00:00
errno = 0;
2002-07-17 14:10:11 +00:00
ret = strtofltmax (garglist->word->word, &ep);
2001-11-13 17:56:06 +00:00
2001-04-06 19:14:31 +00:00
if (*ep)
{
2002-07-17 14:10:11 +00:00
sh_invalidnum (garglist->word->word);
2001-11-13 17:56:06 +00:00
/* Same thing about POSIX.2 conversion error requirements. */
ret = 0;
conversion_error = 1;
2001-04-06 19:14:31 +00:00
}
else if (errno == ERANGE)
2002-07-17 14:10:11 +00:00
printf_erange (garglist->word->word);
2001-04-06 19:14:31 +00:00
1998-04-17 19:52:44 +00:00
garglist = garglist->next;
2001-11-13 17:56:06 +00:00
return (ret);
1998-04-17 19:52:44 +00:00
}
/* NO check is needed for garglist here. */
2009-01-12 13:36:28 +00:00
static intmax_t
1998-04-17 19:52:44 +00:00
asciicode ()
{
2009-01-12 13:36:28 +00:00
register intmax_t ch;
#if defined (HANDLE_MULTIBYTE)
wchar_t wc;
size_t mblength, slen;
#endif
DECLARE_MBSTATE;
#if defined (HANDLE_MULTIBYTE)
slen = strlen (garglist->word->word+1);
mblength = MBLEN (garglist->word->word+1, slen);
if (mblength > 1)
{
mblength = mbtowc (&wc, garglist->word->word+1, slen);
ch = wc; /* XXX */
}
else
#endif
ch = (unsigned char)garglist->word->word[1];
1998-04-17 19:52:44 +00:00
garglist = garglist->next;
return (ch);
}
2011-11-21 20:51:19 -05:00
static SHELL_VAR *
bind_printf_variable (name, value, flags)
char *name;
char *value;
int flags;
{
#if defined (ARRAY_VARS)
if (valid_array_reference (name) == 0)
return (bind_variable (name, value, flags));
else
return (assign_array_element (name, value, flags));
#else /* !ARRAY_VARS */
return bind_variable (name, value, flags);
#endif /* !ARRAY_VARS */
}