Page Menu
Home
GnuPG
Search
Configure Global Search
Log In
Files
F27325952
estream-printf.c
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Size
57 KB
Subscribers
None
estream-printf.c
View Options
/* estream-printf.c - Versatile C-99 compliant printf formatting
* Copyright (C) 2007, 2008 g10 Code GmbH
*
* This file is part of Libestream.
*
* Libestream 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 2 of the License,
* or (at your option) any later version.
*
* Libestream 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.
*
* You should have received a copy of the GNU General Public License
* along with Libestream; if not, see <http://www.gnu.org/licenses/>.
*/
/* Required autoconf tests:
AC_TYPE_LONG_LONG_INT defines HAVE_LONG_LONG_INT
AC_TYPE_LONG_DOUBLE defines HAVE_LONG_DOUBLE
AC_TYPE_INTMAX_T defines HAVE_INTMAX_T
AC_TYPE_UINTMAX_T defines HAVE_UINTMAX_T
AC_CHECK_TYPES([ptrdiff_t]) defines HAVE_PTRDIFF_T
AC_CHECK_SIZEOF([unsigned long]) defines SIZEOF_UNSIGNED_LONG
AC_CHECK_SIZEOF([void *]) defines SIZEOF_VOID_P
HAVE_LANGINFO_THOUSANDS_SEP
Note that the file estream.m4 provides the autoconf macro
ESTREAM_PRINTF_INIT which runs all required checks.
See estream-printf.h for ways to tune this code.
Missing stuff: wchar and wint_t
thousands_sep in pr_float.
*/
#ifdef HAVE_CONFIG_H
# include <config.h>
#endif
#include
<stdio.h>
#include
<stdlib.h>
#include
<string.h>
#include
<unistd.h>
#include
<stdarg.h>
#include
<errno.h>
#include
<stddef.h>
#include
<assert.h>
#if defined(HAVE_INTMAX_T) || defined(HAVE_UINTMAX_T)
# include <stdint.h>
#endif
#ifdef HAVE_LANGINFO_THOUSANDS_SEP
#include
<langinfo.h>
#endif
#ifdef TEST
# include <locale.h>
#else
# ifdef _ESTREAM_PRINTF_EXTRA_INCLUDE
# include _ESTREAM_PRINTF_EXTRA_INCLUDE
# endif
#endif
#include
"estream-printf.h"
/* Allow redefinition of asprintf used malloc functions. */
#if defined(_ESTREAM_PRINTF_MALLOC) && !defined(TEST)
#define my_printf_malloc(a) _ESTREAM_PRINTF_MALLOC((a))
#else
#define my_printf_malloc(a) malloc((a))
#endif
#if defined(_ESTREAM_PRINTF_FREE) && !defined(TEST)
#define my_printf_free(a) _ESTREAM_PRINTF_FREE((a))
#else
#define my_printf_free(a) free((a))
#endif
/* Calculate array dimension. */
#ifndef DIM
#define DIM(array) (sizeof (array) / sizeof (*array))
#endif
/* We allow for that many args without requiring malloced memory. */
#define DEFAULT_MAX_ARGSPECS 5
/* We allow for that many values without requiring malloced memory. */
#define DEFAULT_MAX_VALUES 8
/* We allocate this many new array argspec elements each time. */
#define ARGSPECS_BUMP_VALUE 10
/* Special values for the field width and the precision. */
#define NO_FIELD_VALUE (-1)
#define STAR_FIELD_VALUE (-2)
/* Bit valuues used for the conversion flags. */
#define FLAG_GROUPING 1
#define FLAG_LEFT_JUST 2
#define FLAG_PLUS_SIGN 4
#define FLAG_SPACE_PLUS 8
#define FLAG_ALT_CONV 16
#define FLAG_ZERO_PAD 32
/* Constants used the length modifiers. */
typedef
enum
{
LENMOD_NONE
=
0
,
LENMOD_CHAR
,
/* "hh" */
LENMOD_SHORT
,
/* "h" */
LENMOD_LONG
,
/* "l" */
LENMOD_LONGLONG
,
/* "ll" */
LENMOD_INTMAX
,
/* "j" */
LENMOD_SIZET
,
/* "z" */
LENMOD_PTRDIFF
,
/* "t" */
LENMOD_LONGDBL
/* "L" */
}
lenmod_t
;
/* All the conversion specifiers. */
typedef
enum
{
CONSPEC_UNKNOWN
=
0
,
CONSPEC_DECIMAL
,
CONSPEC_OCTAL
,
CONSPEC_UNSIGNED
,
CONSPEC_HEX
,
CONSPEC_HEX_UP
,
CONSPEC_FLOAT
,
CONSPEC_FLOAT_UP
,
CONSPEC_EXP
,
CONSPEC_EXP_UP
,
CONSPEC_F_OR_G
,
CONSPEC_F_OR_G_UP
,
CONSPEC_HEX_EXP
,
CONSPEC_HEX_EXP_UP
,
CONSPEC_CHAR
,
CONSPEC_STRING
,
CONSPEC_POINTER
,
CONSPEC_STRERROR
,
CONSPEC_BYTES_SO_FAR
}
conspec_t
;
/* Constants describing all the suppoorted types. Note that we list
all the types we know about even if certain types are not available
on this system. */
typedef
enum
{
VALTYPE_UNSUPPORTED
=
0
,
/* Artificial type for error detection. */
VALTYPE_CHAR
,
VALTYPE_SCHAR
,
VALTYPE_UCHAR
,
VALTYPE_SHORT
,
VALTYPE_USHORT
,
VALTYPE_INT
,
VALTYPE_UINT
,
VALTYPE_LONG
,
VALTYPE_ULONG
,
VALTYPE_LONGLONG
,
VALTYPE_ULONGLONG
,
VALTYPE_DOUBLE
,
VALTYPE_LONGDOUBLE
,
VALTYPE_STRING
,
VALTYPE_INTMAX
,
VALTYPE_UINTMAX
,
VALTYPE_SIZE
,
VALTYPE_PTRDIFF
,
VALTYPE_POINTER
,
VALTYPE_CHAR_PTR
,
VALTYPE_SCHAR_PTR
,
VALTYPE_SHORT_PTR
,
VALTYPE_INT_PTR
,
VALTYPE_LONG_PTR
,
VALTYPE_LONGLONG_PTR
,
VALTYPE_INTMAX_PTR
,
VALTYPE_SIZE_PTR
,
VALTYPE_PTRDIFF_PTR
}
valtype_t
;
/* A union used to store the actual values. */
typedef
union
{
char
a_char
;
signed
char
a_schar
;
unsigned
char
a_uchar
;
short
a_short
;
unsigned
short
a_ushort
;
int
a_int
;
unsigned
int
a_uint
;
long
int
a_long
;
unsigned
long
int
a_ulong
;
#ifdef HAVE_LONG_LONG_INT
long
long
int
a_longlong
;
unsigned
long
long
int
a_ulonglong
;
#endif
double
a_double
;
#ifdef HAVE_LONG_DOUBLE
long
double
a_longdouble
;
#endif
const
char
*
a_string
;
#ifdef HAVE_INTMAX_T
intmax_t
a_intmax
;
#endif
#ifdef HAVE_UINTMAX_T
intmax_t
a_uintmax
;
#endif
size_t
a_size
;
#ifdef HAVE_PTRDIFF_T
ptrdiff_t
a_ptrdiff
;
#endif
void
*
a_void_ptr
;
char
*
a_char_ptr
;
signed
char
*
a_schar_ptr
;
short
*
a_short_ptr
;
int
*
a_int_ptr
;
long
*
a_long_ptr
;
#ifdef HAVE_LONG_LONG_INT
long
long
int
*
a_longlong_ptr
;
#endif
#ifdef HAVE_INTMAX_T
intmax_t
*
a_intmax_ptr
;
#endif
size_t
*
a_size_ptr
;
#ifdef HAVE_PTRDIFF_T
ptrdiff_t
*
a_ptrdiff_ptr
;
#endif
}
value_t
;
/* An object used to keep track of a format option and arguments. */
struct
argspec_s
{
size_t
length
;
/* The length of these args including the percent. */
unsigned
int
flags
;
/* The conversion flags (bits defined by FLAG_foo). */
int
width
;
/* The field width. */
int
precision
;
/* The precision. */
lenmod_t
lenmod
;
/* The length modifier. */
conspec_t
conspec
;
/* The conversion specifier. */
int
arg_pos
;
/* The position of the argument. This one may
be -1 to indicate that no value is expected
(e.g. for "%m"). */
int
width_pos
;
/* The position of the argument for a field
width star's value. 0 for not used. */
int
precision_pos
;
/* The position of the argument for the a
precision star's value. 0 for not used. */
valtype_t
vt
;
/* The type of the corresponding argument. */
};
typedef
struct
argspec_s
*
argspec_t
;
/* An object to build up a table of values and their types. */
struct
valueitem_s
{
valtype_t
vt
;
/* The type of the value. */
value_t
value
;
/* The value. */
};
typedef
struct
valueitem_s
*
valueitem_t
;
#ifdef TEST
static
int
verbose
;
static
void
dump_argspecs
(
argspec_t
arg
,
size_t
argcount
)
{
int
idx
;
for
(
idx
=
0
;
argcount
;
argcount
--
,
arg
++
,
idx
++
)
fprintf
(
stderr
,
"%2d: len=%u flags=%u width=%d prec=%d mod=%d "
"con=%d vt=%d pos=%d-%d-%d
\n
"
,
idx
,
(
unsigned
int
)
arg
->
length
,
arg
->
flags
,
arg
->
width
,
arg
->
precision
,
arg
->
lenmod
,
arg
->
conspec
,
arg
->
vt
,
arg
->
arg_pos
,
arg
->
width_pos
,
arg
->
precision_pos
);
}
#endif
/*TEST*/
/* Set the vt field for ARG. */
static
void
compute_type
(
argspec_t
arg
)
{
switch
(
arg
->
conspec
)
{
case
CONSPEC_UNKNOWN
:
arg
->
vt
=
VALTYPE_UNSUPPORTED
;
break
;
case
CONSPEC_DECIMAL
:
switch
(
arg
->
lenmod
)
{
case
LENMOD_CHAR
:
arg
->
vt
=
VALTYPE_SCHAR
;
break
;
case
LENMOD_SHORT
:
arg
->
vt
=
VALTYPE_SHORT
;
break
;
case
LENMOD_LONG
:
arg
->
vt
=
VALTYPE_LONG
;
break
;
case
LENMOD_LONGLONG
:
arg
->
vt
=
VALTYPE_LONGLONG
;
break
;
case
LENMOD_INTMAX
:
arg
->
vt
=
VALTYPE_INTMAX
;
break
;
case
LENMOD_SIZET
:
arg
->
vt
=
VALTYPE_SIZE
;
break
;
case
LENMOD_PTRDIFF
:
arg
->
vt
=
VALTYPE_PTRDIFF
;
break
;
default
:
arg
->
vt
=
VALTYPE_INT
;
break
;
}
break
;
case
CONSPEC_OCTAL
:
case
CONSPEC_UNSIGNED
:
case
CONSPEC_HEX
:
case
CONSPEC_HEX_UP
:
switch
(
arg
->
lenmod
)
{
case
LENMOD_CHAR
:
arg
->
vt
=
VALTYPE_UCHAR
;
break
;
case
LENMOD_SHORT
:
arg
->
vt
=
VALTYPE_USHORT
;
break
;
case
LENMOD_LONG
:
arg
->
vt
=
VALTYPE_ULONG
;
break
;
case
LENMOD_LONGLONG
:
arg
->
vt
=
VALTYPE_ULONGLONG
;
break
;
case
LENMOD_INTMAX
:
arg
->
vt
=
VALTYPE_UINTMAX
;
break
;
case
LENMOD_SIZET
:
arg
->
vt
=
VALTYPE_SIZE
;
break
;
case
LENMOD_PTRDIFF
:
arg
->
vt
=
VALTYPE_PTRDIFF
;
break
;
default
:
arg
->
vt
=
VALTYPE_UINT
;
break
;
}
break
;
case
CONSPEC_FLOAT
:
case
CONSPEC_FLOAT_UP
:
case
CONSPEC_EXP
:
case
CONSPEC_EXP_UP
:
case
CONSPEC_F_OR_G
:
case
CONSPEC_F_OR_G_UP
:
case
CONSPEC_HEX_EXP
:
case
CONSPEC_HEX_EXP_UP
:
switch
(
arg
->
lenmod
)
{
case
LENMOD_LONGDBL
:
arg
->
vt
=
VALTYPE_LONGDOUBLE
;
break
;
case
LENMOD_LONG
:
arg
->
vt
=
VALTYPE_DOUBLE
;
break
;
default
:
arg
->
vt
=
VALTYPE_DOUBLE
;
break
;
}
break
;
case
CONSPEC_CHAR
:
arg
->
vt
=
VALTYPE_INT
;
break
;
case
CONSPEC_STRING
:
arg
->
vt
=
VALTYPE_STRING
;
break
;
case
CONSPEC_POINTER
:
arg
->
vt
=
VALTYPE_POINTER
;
break
;
case
CONSPEC_STRERROR
:
arg
->
vt
=
VALTYPE_STRING
;
break
;
case
CONSPEC_BYTES_SO_FAR
:
switch
(
arg
->
lenmod
)
{
case
LENMOD_CHAR
:
arg
->
vt
=
VALTYPE_SCHAR_PTR
;
break
;
case
LENMOD_SHORT
:
arg
->
vt
=
VALTYPE_SHORT_PTR
;
break
;
case
LENMOD_LONG
:
arg
->
vt
=
VALTYPE_LONG_PTR
;
break
;
case
LENMOD_LONGLONG
:
arg
->
vt
=
VALTYPE_LONGLONG_PTR
;
break
;
case
LENMOD_INTMAX
:
arg
->
vt
=
VALTYPE_INTMAX_PTR
;
break
;
case
LENMOD_SIZET
:
arg
->
vt
=
VALTYPE_SIZE_PTR
;
break
;
case
LENMOD_PTRDIFF
:
arg
->
vt
=
VALTYPE_PTRDIFF_PTR
;
break
;
default
:
arg
->
vt
=
VALTYPE_INT_PTR
;
break
;
}
break
;
}
}
/* Parse the FORMAT string and populate the specification array stored
at the address ARGSPECS_ADDR. The caller has provided enough space
to store up to MAX_ARGSPECS in that buffer. The function may
however ignore the provided buffer and malloc a larger one. On
success the addrrss of that larger buffer will be stored at
ARGSPECS_ADDR. The actual number of specifications will be
returned at R_ARGSPECS_COUNT. */
static
int
parse_format
(
const
char
*
format
,
argspec_t
*
argspecs_addr
,
size_t
max_argspecs
,
size_t
*
r_argspecs_count
)
{
const
char
*
s
;
argspec_t
argspecs
=
*
argspecs_addr
;
argspec_t
arg
;
size_t
argcount
=
0
;
if
(
!
format
)
goto
leave_einval
;
for
(;
*
format
;
format
++
)
{
unsigned
int
flags
;
int
width
,
precision
;
lenmod_t
lenmod
;
conspec_t
conspec
;
int
arg_pos
,
width_pos
,
precision_pos
;
if
(
*
format
!=
'%'
)
continue
;
s
=
++
format
;
if
(
!*
s
)
goto
leave_einval
;
if
(
*
s
==
'%'
)
continue
;
/* Just a quoted percent. */
/* First check whether there is a positional argument. */
arg_pos
=
0
;
/* No positional argument given. */
if
(
*
s
>=
'1'
&&
*
s
<=
'9'
)
{
const
char
*
save_s
=
s
;
arg_pos
=
(
*
s
++
-
'0'
);
for
(;
*
s
>=
'0'
&&
*
s
<=
'9'
;
s
++
)
arg_pos
=
10
*
arg_pos
+
(
*
s
-
'0'
);
if
(
arg_pos
<
0
)
goto
leave_einval
;
/* Overflow during conversion. */
if
(
*
s
==
'$'
)
s
++
;
else
{
arg_pos
=
0
;
s
=
save_s
;
}
}
/* Parse the flags. */
flags
=
0
;
for
(
;
*
s
;
s
++
)
{
switch
(
*
s
)
{
case
'\''
:
flags
|=
FLAG_GROUPING
;
break
;
case
'-'
:
flags
|=
FLAG_LEFT_JUST
;
break
;
case
'+'
:
flags
|=
FLAG_PLUS_SIGN
;
break
;
case
' '
:
flags
|=
FLAG_SPACE_PLUS
;
break
;
case
'#'
:
flags
|=
FLAG_ALT_CONV
;
break
;
case
'0'
:
flags
|=
FLAG_ZERO_PAD
;
break
;
default
:
goto
flags_parsed
;
}
}
flags_parsed
:
/* Parse the field width. */
width_pos
=
0
;
if
(
*
s
==
'*'
)
{
width
=
STAR_FIELD_VALUE
;
s
++
;
/* If we have a positional argument, another one might also
be used to give the position of the star's value. */
if
(
arg_pos
&&
*
s
>=
'1'
&&
*
s
<=
'9'
)
{
width_pos
=
(
*
s
++
-
'0'
);
for
(;
*
s
>=
'0'
&&
*
s
<=
'9'
;
s
++
)
width_pos
=
10
*
width_pos
+
(
*
s
-
'0'
);
if
(
width_pos
<
1
)
goto
leave_einval
;
/* Overflow during conversion. */
if
(
*
s
!=
'$'
)
goto
leave_einval
;
/* Not followed by $. */
s
++
;
}
}
else
if
(
*
s
>=
'0'
&&
*
s
<=
'9'
)
{
width
=
(
*
s
++
-
'0'
);
for
(;
*
s
>=
'0'
&&
*
s
<=
'9'
;
s
++
)
{
if
(
!
width
&&
*
s
==
'0'
)
goto
leave_einval
;
/* Leading zeroes are not allowed.
Fixme: check what other
implementations do. */
width
=
10
*
width
+
(
*
s
-
'0'
);
}
if
(
width
<
0
)
goto
leave_einval
;
/* Overflow during conversion. */
}
else
width
=
NO_FIELD_VALUE
;
/* Parse the precision. */
precision_pos
=
0
;
precision
=
NO_FIELD_VALUE
;
if
(
*
s
==
'.'
)
{
int
ignore_value
=
(
s
[
1
]
==
'-'
);
s
++
;
if
(
*
s
==
'*'
)
{
precision
=
STAR_FIELD_VALUE
;
s
++
;
/* If we have a positional argument, another one might also
be used to give the position of the star's value. */
if
(
arg_pos
&&
*
s
>=
'1'
&&
*
s
<=
'9'
)
{
precision_pos
=
(
*
s
++
-
'0'
);
for
(;
*
s
>=
'0'
&&
*
s
<=
'9'
;
s
++
)
precision_pos
=
10
*
precision_pos
+
(
*
s
-
'0'
);
if
(
precision_pos
<
1
)
goto
leave_einval
;
/* Overflow during conversion. */
if
(
*
s
!=
'$'
)
goto
leave_einval
;
/* Not followed by $. */
s
++
;
}
}
else
if
(
*
s
>=
'0'
&&
*
s
<=
'9'
)
{
precision
=
(
*
s
++
-
'0'
);
for
(;
*
s
>=
'0'
&&
*
s
<=
'9'
;
s
++
)
{
if
(
!
precision
&&
*
s
==
'0'
)
goto
leave_einval
;
/* Leading zeroes are not allowed.
Fixme: check what other
implementations do. */
precision
=
10
*
precision
+
(
*
s
-
'0'
);
}
if
(
precision
<
0
)
goto
leave_einval
;
/* Overflow during conversion. */
}
else
precision
=
0
;
if
(
ignore_value
)
precision
=
NO_FIELD_VALUE
;
}
/* Parse the length modifiers. */
switch
(
*
s
)
{
case
'h'
:
if
(
s
[
1
]
==
'h'
)
{
lenmod
=
LENMOD_CHAR
;
s
++
;
}
else
lenmod
=
LENMOD_SHORT
;
s
++
;
break
;
case
'l'
:
if
(
s
[
1
]
==
'l'
)
{
lenmod
=
LENMOD_LONGLONG
;
s
++
;
}
else
lenmod
=
LENMOD_LONG
;
s
++
;
break
;
case
'j'
:
lenmod
=
LENMOD_INTMAX
;
s
++
;
break
;
case
'z'
:
lenmod
=
LENMOD_SIZET
;
s
++
;
break
;
case
't'
:
lenmod
=
LENMOD_PTRDIFF
;
s
++
;
break
;
case
'L'
:
lenmod
=
LENMOD_LONGDBL
;
s
++
;
break
;
default
:
lenmod
=
LENMOD_NONE
;
break
;
}
/* Parse the conversion specifier. */
switch
(
*
s
)
{
case
'd'
:
case
'i'
:
conspec
=
CONSPEC_DECIMAL
;
break
;
case
'o'
:
conspec
=
CONSPEC_OCTAL
;
break
;
case
'u'
:
conspec
=
CONSPEC_UNSIGNED
;
break
;
case
'x'
:
conspec
=
CONSPEC_HEX
;
break
;
case
'X'
:
conspec
=
CONSPEC_HEX_UP
;
break
;
case
'f'
:
conspec
=
CONSPEC_FLOAT
;
break
;
case
'F'
:
conspec
=
CONSPEC_FLOAT_UP
;
break
;
case
'e'
:
conspec
=
CONSPEC_EXP
;
break
;
case
'E'
:
conspec
=
CONSPEC_EXP_UP
;
break
;
case
'g'
:
conspec
=
CONSPEC_F_OR_G
;
break
;
case
'G'
:
conspec
=
CONSPEC_F_OR_G_UP
;
break
;
case
'a'
:
conspec
=
CONSPEC_HEX_EXP
;
break
;
case
'A'
:
conspec
=
CONSPEC_HEX_EXP_UP
;
break
;
case
'c'
:
conspec
=
CONSPEC_CHAR
;
break
;
case
's'
:
conspec
=
CONSPEC_STRING
;
break
;
case
'p'
:
conspec
=
CONSPEC_POINTER
;
break
;
case
'n'
:
conspec
=
CONSPEC_BYTES_SO_FAR
;
break
;
case
'C'
:
conspec
=
CONSPEC_CHAR
;
lenmod
=
LENMOD_LONG
;
break
;
case
'S'
:
conspec
=
CONSPEC_STRING
;
lenmod
=
LENMOD_LONG
;
break
;
case
'm'
:
conspec
=
CONSPEC_STRERROR
;
arg_pos
=
-1
;
break
;
default
:
conspec
=
CONSPEC_UNKNOWN
;
}
/* Save the args. */
if
(
argcount
>=
max_argspecs
)
{
/* We either need to allocate a new array instead of the
caller provided one or realloc the array. Instead of
using realloc we allocate a new one and release the
original one then. */
size_t
n
,
newmax
;
argspec_t
newarg
;
newmax
=
max_argspecs
+
ARGSPECS_BUMP_VALUE
;
if
(
newmax
<=
max_argspecs
)
goto
leave_einval
;
/* Too many arguments. */
newarg
=
calloc
(
newmax
,
sizeof
*
newarg
);
if
(
!
newarg
)
goto
leave
;
for
(
n
=
0
;
n
<
argcount
;
n
++
)
newarg
[
n
]
=
argspecs
[
n
];
if
(
argspecs
!=
*
argspecs_addr
)
free
(
argspecs
);
argspecs
=
newarg
;
max_argspecs
=
newmax
;
}
arg
=
argspecs
+
argcount
;
arg
->
length
=
s
-
format
+
2
;
arg
->
flags
=
flags
;
arg
->
width
=
width
;
arg
->
precision
=
precision
;
arg
->
lenmod
=
lenmod
;
arg
->
conspec
=
conspec
;
arg
->
arg_pos
=
arg_pos
;
arg
->
width_pos
=
width_pos
;
arg
->
precision_pos
=
precision_pos
;
compute_type
(
arg
);
argcount
++
;
format
=
s
;
}
*
argspecs_addr
=
argspecs
;
*
r_argspecs_count
=
argcount
;
return
0
;
/* Success. */
leave_einval
:
errno
=
EINVAL
;
leave
:
if
(
argspecs
!=
*
argspecs_addr
)
free
(
argspecs
);
*
argspecs_addr
=
NULL
;
return
-1
;
}
/* This function reads all the values as specified by VALUETABLE into
VALUETABLE. The values are expected in VAARGS. The function
returns -1 if a specified type is not supported. */
static
int
read_values
(
valueitem_t
valuetable
,
size_t
valuetable_len
,
va_list
vaargs
)
{
int
validx
;
for
(
validx
=
0
;
validx
<
valuetable_len
;
validx
++
)
{
value_t
*
value
=
&
valuetable
[
validx
].
value
;
valtype_t
vt
=
valuetable
[
validx
].
vt
;
switch
(
vt
)
{
case
VALTYPE_CHAR
:
value
->
a_char
=
va_arg
(
vaargs
,
int
);
break
;
case
VALTYPE_CHAR_PTR
:
value
->
a_char_ptr
=
va_arg
(
vaargs
,
char
*
);
break
;
case
VALTYPE_SCHAR
:
value
->
a_schar
=
va_arg
(
vaargs
,
int
);
break
;
case
VALTYPE_SCHAR_PTR
:
value
->
a_schar_ptr
=
va_arg
(
vaargs
,
signed
char
*
);
break
;
case
VALTYPE_UCHAR
:
value
->
a_uchar
=
va_arg
(
vaargs
,
int
);
break
;
case
VALTYPE_SHORT
:
value
->
a_short
=
va_arg
(
vaargs
,
int
);
break
;
case
VALTYPE_USHORT
:
value
->
a_ushort
=
va_arg
(
vaargs
,
int
);
break
;
case
VALTYPE_SHORT_PTR
:
value
->
a_short_ptr
=
va_arg
(
vaargs
,
short
*
);
break
;
case
VALTYPE_INT
:
value
->
a_int
=
va_arg
(
vaargs
,
int
);
break
;
case
VALTYPE_INT_PTR
:
value
->
a_int_ptr
=
va_arg
(
vaargs
,
int
*
);
break
;
case
VALTYPE_UINT
:
value
->
a_uint
=
va_arg
(
vaargs
,
unsigned
int
);
break
;
case
VALTYPE_LONG
:
value
->
a_long
=
va_arg
(
vaargs
,
long
);
break
;
case
VALTYPE_ULONG
:
value
->
a_ulong
=
va_arg
(
vaargs
,
unsigned
long
);
break
;
case
VALTYPE_LONG_PTR
:
value
->
a_long_ptr
=
va_arg
(
vaargs
,
long
*
);
break
;
#ifdef HAVE_LONG_LONG_INT
case
VALTYPE_LONGLONG
:
value
->
a_longlong
=
va_arg
(
vaargs
,
long
long
int
);
break
;
case
VALTYPE_ULONGLONG
:
value
->
a_ulonglong
=
va_arg
(
vaargs
,
unsigned
long
long
int
);
break
;
case
VALTYPE_LONGLONG_PTR
:
value
->
a_longlong_ptr
=
va_arg
(
vaargs
,
long
long
*
);
break
;
#endif
case
VALTYPE_DOUBLE
:
value
->
a_double
=
va_arg
(
vaargs
,
double
);
break
;
#ifdef HAVE_LONG_DOUBLE
case
VALTYPE_LONGDOUBLE
:
value
->
a_longdouble
=
va_arg
(
vaargs
,
long
double
);
break
;
#endif
case
VALTYPE_STRING
:
value
->
a_string
=
va_arg
(
vaargs
,
const
char
*
);
break
;
case
VALTYPE_POINTER
:
value
->
a_void_ptr
=
va_arg
(
vaargs
,
void
*
);
break
;
#ifdef HAVE_INTMAX_T
case
VALTYPE_INTMAX
:
value
->
a_intmax
=
va_arg
(
vaargs
,
intmax_t
);
break
;
case
VALTYPE_INTMAX_PTR
:
value
->
a_intmax_ptr
=
va_arg
(
vaargs
,
intmax_t
*
);
break
;
#endif
#ifdef HAVE_UINTMAX_T
case
VALTYPE_UINTMAX
:
value
->
a_uintmax
=
va_arg
(
vaargs
,
uintmax_t
);
break
;
#endif
case
VALTYPE_SIZE
:
value
->
a_size
=
va_arg
(
vaargs
,
size_t
);
break
;
case
VALTYPE_SIZE_PTR
:
value
->
a_size_ptr
=
va_arg
(
vaargs
,
size_t
*
);
break
;
#ifdef HAVE_PTRDIFF_T
case
VALTYPE_PTRDIFF
:
value
->
a_ptrdiff
=
va_arg
(
vaargs
,
ptrdiff_t
);
break
;
case
VALTYPE_PTRDIFF_PTR
:
value
->
a_ptrdiff_ptr
=
va_arg
(
vaargs
,
ptrdiff_t
*
);
break
;
#endif
default
:
/* Unsupported type. */
return
-1
;
}
}
return
0
;
}
/* Output COUNT padding characters PADCHAR and update NBYTES by the
number of bytes actually written. */
static
int
pad_out
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
int
padchar
,
int
count
,
size_t
*
nbytes
)
{
char
buf
[
32
];
size_t
n
;
int
rc
;
while
(
count
>
0
)
{
n
=
(
count
<=
sizeof
buf
)
?
count
:
sizeof
buf
;
memset
(
buf
,
padchar
,
n
);
rc
=
outfnc
(
outfncarg
,
buf
,
n
);
if
(
rc
)
return
rc
;
*
nbytes
+=
n
;
count
-=
n
;
}
return
0
;
}
/* "d,i,o,u,x,X" formatting. OUTFNC and OUTFNCARG describes the
output routine, ARG gives the argument description and VALUE the
actual value (its type is available through arg->vt). */
static
int
pr_integer
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
argspec_t
arg
,
value_t
value
,
size_t
*
nbytes
)
{
int
rc
;
#ifdef HAVE_LONG_LONG_INT
unsigned
long
long
aulong
;
#else
unsigned
long
aulong
;
#endif
char
numbuf
[
100
];
char
*
p
,
*
pend
;
size_t
n
;
char
signchar
=
0
;
int
n_prec
;
/* Number of extra precision digits required. */
int
n_extra
;
/* Extra number of prefix or sign characters. */
if
(
arg
->
conspec
==
CONSPEC_DECIMAL
)
{
#ifdef HAVE_LONG_LONG_INT
long
long
along
;
#else
long
along
;
#endif
switch
(
arg
->
vt
)
{
case
VALTYPE_SHORT
:
along
=
value
.
a_short
;
break
;
case
VALTYPE_INT
:
along
=
value
.
a_int
;
break
;
case
VALTYPE_LONG
:
along
=
value
.
a_long
;
break
;
#ifdef HAVE_LONG_LONG_INT
case
VALTYPE_LONGLONG
:
along
=
value
.
a_longlong
;
break
;
case
VALTYPE_SIZE
:
along
=
value
.
a_size
;
break
;
# ifdef HAVE_INTMAX_T
case
VALTYPE_INTMAX
:
along
=
value
.
a_intmax
;
break
;
# endif
# ifdef HAVE_PTRDIFF_T
case
VALTYPE_PTRDIFF
:
along
=
value
.
a_ptrdiff
;
break
;
# endif
#endif
/*HAVE_LONG_LONG_INT*/
default
:
return
-1
;
}
if
(
along
<
0
)
{
aulong
=
-
along
;
signchar
=
'-'
;
}
else
aulong
=
along
;
}
else
{
switch
(
arg
->
vt
)
{
case
VALTYPE_USHORT
:
aulong
=
value
.
a_ushort
;
break
;
case
VALTYPE_UINT
:
aulong
=
value
.
a_uint
;
break
;
case
VALTYPE_ULONG
:
aulong
=
value
.
a_ulong
;
break
;
#ifdef HAVE_LONG_LONG_INT
case
VALTYPE_ULONGLONG
:
aulong
=
value
.
a_ulonglong
;
break
;
case
VALTYPE_SIZE
:
aulong
=
value
.
a_size
;
break
;
# ifdef HAVE_UINTMAX_T
case
VALTYPE_UINTMAX
:
aulong
=
value
.
a_uintmax
;
break
;
# endif
# ifdef HAVE_PTRDIFF_T
case
VALTYPE_PTRDIFF
:
aulong
=
value
.
a_ptrdiff
;
break
;
# endif
#endif
/*HAVE_LONG_LONG_INT*/
default
:
return
-1
;
}
}
if
(
signchar
==
'-'
)
;
else
if
((
arg
->
flags
&
FLAG_PLUS_SIGN
))
signchar
=
'+'
;
else
if
((
arg
->
flags
&
FLAG_SPACE_PLUS
))
signchar
=
' '
;
n_extra
=
!!
signchar
;
/* We build the string up backwards. */
p
=
pend
=
numbuf
+
DIM
(
numbuf
);
if
((
!
aulong
&&
!
arg
->
precision
))
;
else
if
(
arg
->
conspec
==
CONSPEC_DECIMAL
||
arg
->
conspec
==
CONSPEC_UNSIGNED
)
{
int
grouping
=
-1
;
const
char
*
grouping_string
=
#ifdef HAVE_LANGINFO_THOUSANDS_SEP
nl_langinfo
(
THOUSANDS_SEP
);
#else
"'"
;
#endif
do
{
if
((
arg
->
flags
&
FLAG_GROUPING
)
&&
(
++
grouping
==
3
)
&&
*
grouping_string
)
{
*--
p
=
*
grouping_string
;
grouping
=
0
;
}
*--
p
=
'0'
+
(
aulong
%
10
);
aulong
/=
10
;
}
while
(
aulong
);
}
else
if
(
arg
->
conspec
==
CONSPEC_OCTAL
)
{
do
{
*--
p
=
'0'
+
(
aulong
%
8
);
aulong
/=
8
;
}
while
(
aulong
);
if
((
arg
->
flags
&
FLAG_ALT_CONV
)
&&
*
p
!=
'0'
)
*--
p
=
'0'
;
}
else
/* HEX or HEXUP */
{
const
char
*
digits
=
((
arg
->
conspec
==
CONSPEC_HEX
)
?
"0123456789abcdef"
:
"0123456789ABCDEF"
);
do
{
*--
p
=
digits
[(
aulong
%
16
)];
aulong
/=
16
;
}
while
(
aulong
);
if
((
arg
->
flags
&
FLAG_ALT_CONV
))
n_extra
+=
2
;
}
n
=
pend
-
p
;
if
((
arg
->
flags
&
FLAG_ZERO_PAD
)
&&
arg
->
precision
==
NO_FIELD_VALUE
&&
!
(
arg
->
flags
&
FLAG_LEFT_JUST
)
&&
n
&&
arg
->
width
-
n_extra
>
n
)
n_prec
=
arg
->
width
-
n_extra
-
n
;
else
if
(
arg
->
precision
>
0
&&
arg
->
precision
>
n
)
n_prec
=
arg
->
precision
-
n
;
else
n_prec
=
0
;
if
(
!
(
arg
->
flags
&
FLAG_LEFT_JUST
)
&&
arg
->
width
>=
0
&&
arg
->
width
-
n_extra
>
n
&&
arg
->
width
-
n_extra
-
n
>=
n_prec
)
{
rc
=
pad_out
(
outfnc
,
outfncarg
,
' '
,
arg
->
width
-
n_extra
-
n
-
n_prec
,
nbytes
);
if
(
rc
)
return
rc
;
}
if
(
signchar
)
{
rc
=
outfnc
(
outfncarg
,
&
signchar
,
1
);
if
(
rc
)
return
rc
;
*
nbytes
+=
1
;
}
if
((
arg
->
flags
&
FLAG_ALT_CONV
)
&&
(
arg
->
conspec
==
CONSPEC_HEX
||
arg
->
conspec
==
CONSPEC_HEX_UP
))
{
rc
=
outfnc
(
outfncarg
,
arg
->
conspec
==
CONSPEC_HEX
?
"0x"
:
"0X"
,
2
);
if
(
rc
)
return
rc
;
*
nbytes
+=
2
;
}
if
(
n_prec
)
{
rc
=
pad_out
(
outfnc
,
outfncarg
,
'0'
,
n_prec
,
nbytes
);
if
(
rc
)
return
rc
;
}
rc
=
outfnc
(
outfncarg
,
p
,
pend
-
p
);
if
(
rc
)
return
rc
;
*
nbytes
+=
pend
-
p
;
if
((
arg
->
flags
&
FLAG_LEFT_JUST
)
&&
arg
->
width
>=
0
&&
arg
->
width
-
n_extra
-
n_prec
>
n
)
{
rc
=
pad_out
(
outfnc
,
outfncarg
,
' '
,
arg
->
width
-
n_extra
-
n_prec
-
n
,
nbytes
);
if
(
rc
)
return
rc
;
}
return
0
;
}
/* "e,E,f,F,g,G,a,A" formatting. OUTFNC and OUTFNCARG describes the
output routine, ARG gives the argument description and VALUE the
actual value (its type is available through arg->vt). For
portability reasons sprintf is used for the actual formatting.
This is useful because sprint is the only standard function to
convert a floating number into its ascii representation. To avoid
using malloc we just pass the precision to sprintf and do the final
formatting with our own code. */
static
int
pr_float
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
argspec_t
arg
,
value_t
value
,
size_t
*
nbytes
)
{
int
rc
;
#ifdef HAVE_LONG_DOUBLE
long
double
adblfloat
=
0
;
/* Just to please gcc. */
int
use_dbl
=
0
;
#endif
double
afloat
;
char
numbuf
[
200
];
char
formatstr
[
20
];
char
*
p
,
*
pend
;
size_t
n
;
char
signchar
=
0
;
int
n_extra
;
/* Extra number of prefix or sign characters. */
switch
(
arg
->
vt
)
{
case
VALTYPE_DOUBLE
:
afloat
=
value
.
a_double
;
break
;
#ifdef HAVE_LONG_DOUBLE
case
VALTYPE_LONGDOUBLE
:
afloat
=
0
;
/* Just to please gcc. */
adblfloat
=
value
.
a_longdouble
;
use_dbl
=
1
;
break
;
#endif
default
:
return
-1
;
}
/* We build the string using sprint. */
p
=
formatstr
+
sizeof
formatstr
;
*--
p
=
0
;
switch
(
arg
->
conspec
)
{
case
CONSPEC_FLOAT
:
*--
p
=
'f'
;
break
;
case
CONSPEC_FLOAT_UP
:
*--
p
=
'F'
;
break
;
case
CONSPEC_EXP
:
*--
p
=
'e'
;
break
;
case
CONSPEC_EXP_UP
:
*--
p
=
'E'
;
break
;
case
CONSPEC_F_OR_G
:
*--
p
=
'g'
;
break
;
case
CONSPEC_F_OR_G_UP
:
*--
p
=
'G'
;
break
;
case
CONSPEC_HEX_EXP
:
*--
p
=
'a'
;
break
;
case
CONSPEC_HEX_EXP_UP
:
*--
p
=
'A'
;
break
;
default
:
return
-1
;
/* Actually a bug. */
}
#ifdef HAVE_LONG_DOUBLE
if
(
use_dbl
)
*--
p
=
'L'
;
#endif
if
(
arg
->
precision
!=
NO_FIELD_VALUE
)
{
/* Limit it to a meaningful value so that even a stupid sprintf
won't overflow our buffer. */
n
=
arg
->
precision
<=
100
?
arg
->
precision
:
100
;
do
{
*--
p
=
'0'
+
(
n
%
10
);
n
/=
10
;
}
while
(
n
);
*--
p
=
'.'
;
}
if
((
arg
->
flags
&
FLAG_ALT_CONV
))
*--
p
=
'#'
;
*--
p
=
'%'
;
#ifdef HAVE_LONG_DOUBLE
if
(
use_dbl
)
sprintf
(
numbuf
,
p
,
adblfloat
);
else
#endif
/*HAVE_LONG_DOUBLE*/
sprintf
(
numbuf
,
p
,
afloat
);
p
=
numbuf
;
n
=
strlen
(
numbuf
);
pend
=
p
+
n
;
if
(
*
p
==
'-'
)
{
signchar
=
'-'
;
p
++
;
n
--
;
}
else
if
((
arg
->
flags
&
FLAG_PLUS_SIGN
))
signchar
=
'+'
;
else
if
((
arg
->
flags
&
FLAG_SPACE_PLUS
))
signchar
=
' '
;
n_extra
=
!!
signchar
;
if
(
!
(
arg
->
flags
&
FLAG_LEFT_JUST
)
&&
arg
->
width
>=
0
&&
arg
->
width
-
n_extra
>
n
)
{
rc
=
pad_out
(
outfnc
,
outfncarg
,
' '
,
arg
->
width
-
n_extra
-
n
,
nbytes
);
if
(
rc
)
return
rc
;
}
if
(
signchar
)
{
rc
=
outfnc
(
outfncarg
,
&
signchar
,
1
);
if
(
rc
)
return
rc
;
*
nbytes
+=
1
;
}
rc
=
outfnc
(
outfncarg
,
p
,
pend
-
p
);
if
(
rc
)
return
rc
;
*
nbytes
+=
pend
-
p
;
if
((
arg
->
flags
&
FLAG_LEFT_JUST
)
&&
arg
->
width
>=
0
&&
arg
->
width
-
n_extra
>
n
)
{
rc
=
pad_out
(
outfnc
,
outfncarg
,
' '
,
arg
->
width
-
n_extra
-
n
,
nbytes
);
if
(
rc
)
return
rc
;
}
return
0
;
}
/* "c" formatting. */
static
int
pr_char
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
argspec_t
arg
,
value_t
value
,
size_t
*
nbytes
)
{
int
rc
;
char
buf
[
1
];
if
(
arg
->
vt
!=
VALTYPE_INT
)
return
-1
;
buf
[
0
]
=
(
unsigned
int
)
value
.
a_int
;
rc
=
outfnc
(
outfncarg
,
buf
,
1
);
if
(
rc
)
return
rc
;
*
nbytes
+=
1
;
return
0
;
}
/* "s" formatting. */
static
int
pr_string
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
argspec_t
arg
,
value_t
value
,
size_t
*
nbytes
)
{
int
rc
;
size_t
n
;
const
char
*
string
,
*
s
;
if
(
arg
->
vt
!=
VALTYPE_STRING
)
return
-1
;
string
=
value
.
a_string
;
if
(
!
string
)
string
=
"(null)"
;
if
(
arg
->
precision
>=
0
)
{
for
(
n
=
0
,
s
=
string
;
*
s
&&
n
<
arg
->
precision
;
s
++
)
n
++
;
}
else
n
=
strlen
(
string
);
if
(
!
(
arg
->
flags
&
FLAG_LEFT_JUST
)
&&
arg
->
width
>=
0
&&
arg
->
width
>
n
)
{
rc
=
pad_out
(
outfnc
,
outfncarg
,
' '
,
arg
->
width
-
n
,
nbytes
);
if
(
rc
)
return
rc
;
}
rc
=
outfnc
(
outfncarg
,
string
,
n
);
if
(
rc
)
return
rc
;
*
nbytes
+=
n
;
if
((
arg
->
flags
&
FLAG_LEFT_JUST
)
&&
arg
->
width
>=
0
&&
arg
->
width
>
n
)
{
rc
=
pad_out
(
outfnc
,
outfncarg
,
' '
,
arg
->
width
-
n
,
nbytes
);
if
(
rc
)
return
rc
;
}
return
0
;
}
/* "p" formatting. */
static
int
pr_pointer
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
argspec_t
arg
,
value_t
value
,
size_t
*
nbytes
)
{
int
rc
;
#ifdef HAVE_LONG_LONG_INT
unsigned
long
long
aulong
;
#else
unsigned
long
aulong
;
#endif
char
numbuf
[
100
];
char
*
p
,
*
pend
;
if
(
arg
->
vt
!=
VALTYPE_POINTER
)
return
-1
;
/* We assume that a pointer can be converted to an unsigned long.
That is not correct for a 64 bit Windows, but then we assume that
long long is supported and usable for storing a pointer. */
#if defined(HAVE_LONG_LONG_INT) && (SIZEOF_UNSIGNED_LONG < SIZEOF_VOID_P)
aulong
=
(
unsigned
long
long
)
value
.
a_void_ptr
;
#else
aulong
=
(
unsigned
long
)
value
.
a_void_ptr
;
#endif
p
=
pend
=
numbuf
+
DIM
(
numbuf
);
do
{
*--
p
=
"0123456789abcdefx"
[(
aulong
%
16
)];
aulong
/=
16
;
}
while
(
aulong
);
while
((
pend
-
p
)
<
2
*
sizeof
(
aulong
))
*--
p
=
'0'
;
*--
p
=
'x'
;
*--
p
=
'0'
;
rc
=
outfnc
(
outfncarg
,
p
,
pend
-
p
);
if
(
rc
)
return
rc
;
*
nbytes
+=
pend
-
p
;
return
0
;
}
/* "n" pesudo format operation. */
static
int
pr_bytes_so_far
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
argspec_t
arg
,
value_t
value
,
size_t
*
nbytes
)
{
(
void
)
outfnc
;
(
void
)
outfncarg
;
switch
(
arg
->
vt
)
{
case
VALTYPE_SCHAR_PTR
:
*
value
.
a_schar_ptr
=
(
signed
char
)(
unsigned
int
)(
*
nbytes
);
break
;
case
VALTYPE_SHORT_PTR
:
*
value
.
a_short_ptr
=
(
short
)(
unsigned
int
)(
*
nbytes
);
break
;
case
VALTYPE_LONG_PTR
:
*
value
.
a_long_ptr
=
(
long
)(
*
nbytes
);
break
;
#ifdef HAVE_LONG_LONG_INT
case
VALTYPE_LONGLONG_PTR
:
*
value
.
a_longlong_ptr
=
(
long
long
)(
*
nbytes
);
break
;
#endif
#ifdef HAVE_INTMAX_T
case
VALTYPE_INTMAX_PTR
:
*
value
.
a_intmax_ptr
=
(
intmax_t
)(
*
nbytes
);
break
;
#endif
case
VALTYPE_SIZE_PTR
:
*
value
.
a_size_ptr
=
(
*
nbytes
);
break
;
#ifdef HAVE_PTRDIFF_T
case
VALTYPE_PTRDIFF_PTR
:
*
value
.
a_ptrdiff_ptr
=
(
ptrdiff_t
)(
*
nbytes
);
break
;
#endif
case
VALTYPE_INT_PTR
:
*
value
.
a_int_ptr
=
(
int
)(
*
nbytes
);
break
;
default
:
return
-1
;
/* An unsupported type has been used. */
}
return
0
;
}
/* Run the actual formatting. OUTFNC and OUTFNCARG are the output
functions. FORMAT is format string ARGSPECS is the parsed format
string, ARGSPECS_LEN the number of items in ARGSPECS. VALUETABLE
holds the values and may be directly addressed using the position
arguments given by ARGSPECS. MYERRNO is used for the "%m"
conversion. NBYTES well be updated to reflect the number of bytes
send to the output function. */
static
int
do_format
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
const
char
*
format
,
argspec_t
argspecs
,
size_t
argspecs_len
,
valueitem_t
valuetable
,
int
myerrno
,
size_t
*
nbytes
)
{
int
rc
=
0
;
const
char
*
s
;
argspec_t
arg
=
argspecs
;
int
argidx
=
0
;
/* Only used for assertion. */
size_t
n
;
value_t
value
;
s
=
format
;
while
(
*
s
)
{
if
(
*
s
!=
'%'
)
{
s
++
;
continue
;
}
if
(
s
!=
format
)
{
rc
=
outfnc
(
outfncarg
,
format
,
(
n
=
s
-
format
));
if
(
rc
)
return
rc
;
*
nbytes
+=
n
;
}
if
(
s
[
1
]
==
'%'
)
{
/* Note that this code ignores one trailing percent escape -
this is however okay as the args parser must have
detected this already. */
rc
=
outfnc
(
outfncarg
,
s
,
1
);
if
(
rc
)
return
rc
;
*
nbytes
+=
1
;
s
+=
2
;
format
=
s
;
continue
;
}
/* Save the next start. */
s
+=
arg
->
length
;
format
=
s
;
assert
(
argidx
<
argspecs_len
);
argidx
++
;
/* Apply indirect field width and precision values. */
if
(
arg
->
width
==
STAR_FIELD_VALUE
)
{
assert
(
valuetable
[
arg
->
width_pos
-1
].
vt
==
VALTYPE_INT
);
arg
->
width
=
valuetable
[
arg
->
width_pos
-1
].
value
.
a_int
;
if
(
arg
->
width
<
0
)
{
arg
->
width
=
-
arg
->
width
;
arg
->
flags
|=
FLAG_LEFT_JUST
;
}
}
if
(
arg
->
precision
==
STAR_FIELD_VALUE
)
{
assert
(
valuetable
[
arg
->
precision_pos
-1
].
vt
==
VALTYPE_INT
);
arg
->
precision
=
valuetable
[
arg
->
precision_pos
-1
].
value
.
a_int
;
if
(
arg
->
precision
<
0
)
arg
->
precision
=
NO_FIELD_VALUE
;
}
if
(
arg
->
arg_pos
==
-1
&&
arg
->
conspec
==
CONSPEC_STRERROR
)
value
.
a_string
=
strerror
(
myerrno
);
else
{
assert
(
arg
->
vt
==
valuetable
[
arg
->
arg_pos
-1
].
vt
);
value
=
valuetable
[
arg
->
arg_pos
-1
].
value
;
}
switch
(
arg
->
conspec
)
{
case
CONSPEC_UNKNOWN
:
assert
(
!
"bug"
);
break
;
case
CONSPEC_DECIMAL
:
case
CONSPEC_UNSIGNED
:
case
CONSPEC_OCTAL
:
case
CONSPEC_HEX
:
case
CONSPEC_HEX_UP
:
rc
=
pr_integer
(
outfnc
,
outfncarg
,
arg
,
value
,
nbytes
);
break
;
case
CONSPEC_FLOAT
:
case
CONSPEC_FLOAT_UP
:
case
CONSPEC_EXP
:
case
CONSPEC_EXP_UP
:
case
CONSPEC_F_OR_G
:
case
CONSPEC_F_OR_G_UP
:
case
CONSPEC_HEX_EXP
:
case
CONSPEC_HEX_EXP_UP
:
rc
=
pr_float
(
outfnc
,
outfncarg
,
arg
,
value
,
nbytes
);
break
;
case
CONSPEC_CHAR
:
rc
=
pr_char
(
outfnc
,
outfncarg
,
arg
,
value
,
nbytes
);
break
;
case
CONSPEC_STRING
:
case
CONSPEC_STRERROR
:
rc
=
pr_string
(
outfnc
,
outfncarg
,
arg
,
value
,
nbytes
);
break
;
case
CONSPEC_POINTER
:
rc
=
pr_pointer
(
outfnc
,
outfncarg
,
arg
,
value
,
nbytes
);
break
;
case
CONSPEC_BYTES_SO_FAR
:
rc
=
pr_bytes_so_far
(
outfnc
,
outfncarg
,
arg
,
value
,
nbytes
);
break
;
}
if
(
rc
)
return
rc
;
arg
++
;
}
/* Print out any trailing stuff. */
n
=
s
-
format
;
rc
=
n
?
outfnc
(
outfncarg
,
format
,
n
)
:
0
;
if
(
!
rc
)
*
nbytes
+=
n
;
return
rc
;
}
/* The versatile printf formatting routine. It expects a callback
function OUTFNC and an opaque argument OUTFNCARG used for actual
output of the formatted stuff. FORMAT is the format specification
and VAARGS a variable argumemt list matching the arguments of
FORMAT. */
int
estream_format
(
estream_printf_out_t
outfnc
,
void
*
outfncarg
,
const
char
*
format
,
va_list
vaargs
)
{
/* Buffer to hold the argspecs and a pointer to it.*/
struct
argspec_s
argspecs_buffer
[
DEFAULT_MAX_ARGSPECS
];
argspec_t
argspecs
=
argspecs_buffer
;
size_t
argspecs_len
;
/* Number of specifications in ARGSPECS. */
/* Buffer to hold the description for the values. */
struct
valueitem_s
valuetable_buffer
[
DEFAULT_MAX_VALUES
];
valueitem_t
valuetable
=
valuetable_buffer
;
int
rc
;
/* Return code. */
size_t
argidx
;
/* Used to index the argspecs array. */
size_t
validx
;
/* Used to index the valuetable. */
int
max_pos
;
/* Highest argument position. */
size_t
nbytes
=
0
;
/* Keep track of the number of bytes passed to
the output function. */
int
myerrno
=
errno
;
/* Save the errno for use with "%m". */
/* Parse the arguments to come up with descriptive list. We can't
do this on the fly because we need to support positional
arguments. */
rc
=
parse_format
(
format
,
&
argspecs
,
DIM
(
argspecs_buffer
),
&
argspecs_len
);
if
(
rc
)
goto
leave
;
/* Check that all ARG_POS fields are set. */
for
(
argidx
=
0
,
max_pos
=
0
;
argidx
<
argspecs_len
;
argidx
++
)
{
if
(
argspecs
[
argidx
].
arg_pos
!=
-1
&&
argspecs
[
argidx
].
arg_pos
>
max_pos
)
max_pos
=
argspecs
[
argidx
].
arg_pos
;
if
(
argspecs
[
argidx
].
width_pos
>
max_pos
)
max_pos
=
argspecs
[
argidx
].
width_pos
;
if
(
argspecs
[
argidx
].
precision_pos
>
max_pos
)
max_pos
=
argspecs
[
argidx
].
precision_pos
;
}
if
(
!
max_pos
)
{
/* Fill in all the positions. */
for
(
argidx
=
0
;
argidx
<
argspecs_len
;
argidx
++
)
{
if
(
argspecs
[
argidx
].
width
==
STAR_FIELD_VALUE
)
argspecs
[
argidx
].
width_pos
=
++
max_pos
;
if
(
argspecs
[
argidx
].
precision
==
STAR_FIELD_VALUE
)
argspecs
[
argidx
].
precision_pos
=
++
max_pos
;
if
(
argspecs
[
argidx
].
arg_pos
!=
-1
)
argspecs
[
argidx
].
arg_pos
=
++
max_pos
;
}
}
else
{
/* Check that they are all filled. More test are done later. */
for
(
argidx
=
0
;
argidx
<
argspecs_len
;
argidx
++
)
{
if
(
!
argspecs
[
argidx
].
arg_pos
||
(
argspecs
[
argidx
].
width
==
STAR_FIELD_VALUE
&&
!
argspecs
[
argidx
].
width_pos
)
||
(
argspecs
[
argidx
].
precision
==
STAR_FIELD_VALUE
&&
!
argspecs
[
argidx
].
precision_pos
))
goto
leave_einval
;
}
}
/* Check that there is no overflow in max_pos and that it has a
reasonable length. There may never be more elements than the
number of characters in FORMAT. */
if
(
max_pos
<
0
||
max_pos
>=
strlen
(
format
))
goto
leave_einval
;
#ifdef TEST
if
(
verbose
>
1
)
dump_argspecs
(
argspecs
,
argspecs_len
);
#endif
/* Allocate a table to hold the values. If it is small enough we
use a stack allocated buffer. */
if
(
max_pos
>
DIM
(
valuetable_buffer
))
{
valuetable
=
calloc
(
max_pos
,
sizeof
*
valuetable
);
if
(
!
valuetable
)
goto
leave_error
;
}
else
{
for
(
validx
=
0
;
validx
<
DIM
(
valuetable_buffer
);
validx
++
)
valuetable
[
validx
].
vt
=
VALTYPE_UNSUPPORTED
;
}
for
(
argidx
=
0
;
argidx
<
argspecs_len
;
argidx
++
)
{
if
(
argspecs
[
argidx
].
arg_pos
!=
-
1
)
{
validx
=
argspecs
[
argidx
].
arg_pos
-
1
;
if
(
valuetable
[
validx
].
vt
)
goto
leave_einval
;
/* Already defined. */
valuetable
[
validx
].
vt
=
argspecs
[
argidx
].
vt
;
}
if
(
argspecs
[
argidx
].
width
==
STAR_FIELD_VALUE
)
{
validx
=
argspecs
[
argidx
].
width_pos
-
1
;
if
(
valuetable
[
validx
].
vt
)
goto
leave_einval
;
/* Already defined. */
valuetable
[
validx
].
vt
=
VALTYPE_INT
;
}
if
(
argspecs
[
argidx
].
precision
==
STAR_FIELD_VALUE
)
{
validx
=
argspecs
[
argidx
].
precision_pos
-
1
;
if
(
valuetable
[
validx
].
vt
)
goto
leave_einval
;
/* Already defined. */
valuetable
[
validx
].
vt
=
VALTYPE_INT
;
}
}
/* Read all the arguments. This will error out for unsupported
types and for not given positional arguments. */
rc
=
read_values
(
valuetable
,
max_pos
,
vaargs
);
if
(
rc
)
goto
leave_einval
;
/* for (validx=0; validx < max_pos; validx++) */
/* fprintf (stderr, "%2d: vt=%d\n", validx, valuetable[validx].vt); */
/* Everything has been collected, go ahead with the formatting. */
rc
=
do_format
(
outfnc
,
outfncarg
,
format
,
argspecs
,
argspecs_len
,
valuetable
,
myerrno
,
&
nbytes
);
goto
leave
;
leave_einval
:
errno
=
EINVAL
;
leave_error
:
rc
=
-1
;
leave
:
if
(
valuetable
!=
valuetable_buffer
)
free
(
valuetable
);
if
(
argspecs
!=
argspecs_buffer
)
free
(
argspecs
);
return
rc
;
}
/* A simple output handler utilizing stdio. */
static
int
plain_stdio_out
(
void
*
outfncarg
,
const
char
*
buf
,
size_t
buflen
)
{
FILE
*
fp
=
(
FILE
*
)
outfncarg
;
if
(
fwrite
(
buf
,
buflen
,
1
,
fp
)
!=
1
)
return
-1
;
return
0
;
}
/* A replacement for printf. */
int
estream_printf
(
const
char
*
format
,
...)
{
int
rc
;
va_list
arg_ptr
;
va_start
(
arg_ptr
,
format
);
rc
=
estream_format
(
plain_stdio_out
,
stderr
,
format
,
arg_ptr
);
va_end
(
arg_ptr
);
return
rc
;
}
/* A replacement for fprintf. */
int
estream_fprintf
(
FILE
*
fp
,
const
char
*
format
,
...)
{
int
rc
;
va_list
arg_ptr
;
va_start
(
arg_ptr
,
format
);
rc
=
estream_format
(
plain_stdio_out
,
fp
,
format
,
arg_ptr
);
va_end
(
arg_ptr
);
return
rc
;
}
/* A replacement for vfprintf. */
int
estream_vfprintf
(
FILE
*
fp
,
const
char
*
format
,
va_list
arg_ptr
)
{
return
estream_format
(
plain_stdio_out
,
fp
,
format
,
arg_ptr
);
}
/* Communication object used between estream_snprintf and
fixed_buffer_out. */
struct
fixed_buffer_parm_s
{
size_t
size
;
/* Size of the buffer. */
size_t
count
;
/* Number of bytes requested for output. */
size_t
used
;
/* Used size of the buffer. */
char
*
buffer
;
/* Provided buffer. */
};
/* A simple malloced buffer output handler. */
static
int
fixed_buffer_out
(
void
*
outfncarg
,
const
char
*
buf
,
size_t
buflen
)
{
struct
fixed_buffer_parm_s
*
parm
=
outfncarg
;
parm
->
count
+=
buflen
;
if
(
!
parm
->
buffer
)
;
else
if
(
parm
->
used
+
buflen
<
parm
->
size
)
{
/* Handle the common case that everything fits into the buffer
separately. */
memcpy
(
parm
->
buffer
+
parm
->
used
,
buf
,
buflen
);
parm
->
used
+=
buflen
;
}
else
{
/* The slow version of above. */
for
(
;
buflen
&&
parm
->
used
<
parm
->
size
;
buflen
--
)
parm
->
buffer
[
parm
->
used
++
]
=
*
buf
++
;
}
return
0
;
}
/* A replacement for vsnprintf. */
int
estream_vsnprintf
(
char
*
buf
,
size_t
bufsize
,
const
char
*
format
,
va_list
arg_ptr
)
{
struct
fixed_buffer_parm_s
parm
;
int
rc
;
parm
.
size
=
bufsize
;
parm
.
count
=
0
;
parm
.
used
=
0
;
parm
.
buffer
=
bufsize
?
buf
:
NULL
;
rc
=
estream_format
(
fixed_buffer_out
,
&
parm
,
format
,
arg_ptr
);
if
(
!
rc
)
rc
=
fixed_buffer_out
(
&
parm
,
""
,
1
);
/* Print terminating Nul. */
if
(
rc
==
-1
)
return
-1
;
if
(
bufsize
&&
buf
&&
parm
.
size
&&
parm
.
count
>=
parm
.
size
)
buf
[
parm
.
size
-1
]
=
0
;
parm
.
count
--
;
/* Do not count the trailing nul. */
return
(
int
)
parm
.
count
;
/* Return number of bytes which would have
been written. */
}
/* A replacement for snprintf. */
int
estream_snprintf
(
char
*
buf
,
size_t
bufsize
,
const
char
*
format
,
...)
{
int
rc
;
va_list
arg_ptr
;
va_start
(
arg_ptr
,
format
);
rc
=
estream_vsnprintf
(
buf
,
bufsize
,
format
,
arg_ptr
);
va_end
(
arg_ptr
);
return
rc
;
}
/* Communication object used between estream_asprintf and
dynamic_buffer_out. */
struct
dynamic_buffer_parm_s
{
int
error_flag
;
/* Internal helper. */
size_t
alloced
;
/* Allocated size of the buffer. */
size_t
used
;
/* Used size of the buffer. */
char
*
buffer
;
/* Malloced buffer. */
};
/* A simple malloced buffer output handler. */
static
int
dynamic_buffer_out
(
void
*
outfncarg
,
const
char
*
buf
,
size_t
buflen
)
{
struct
dynamic_buffer_parm_s
*
parm
=
outfncarg
;
if
(
parm
->
error_flag
)
{
/* Just in case some formatting routine did not checked for an
error. */
errno
=
parm
->
error_flag
;
return
-1
;
}
if
(
parm
->
used
+
buflen
>=
parm
->
alloced
)
{
char
*
p
;
parm
->
alloced
+=
buflen
+
512
;
p
=
realloc
(
parm
->
buffer
,
parm
->
alloced
);
if
(
!
p
)
{
parm
->
error_flag
=
errno
?
errno
:
ENOMEM
;
/* Wipe out what we already accumulated. This is useful in
case sensitive data is formated. */
memset
(
parm
->
buffer
,
0
,
parm
->
used
);
return
-1
;
}
parm
->
buffer
=
p
;
}
memcpy
(
parm
->
buffer
+
parm
->
used
,
buf
,
buflen
);
parm
->
used
+=
buflen
;
return
0
;
}
/* A replacement for vasprintf. As with the BSD of vasprintf version -1
will be returned on error and NULL stored at BUFP. On success the
number of bytes printed will be returned. */
int
estream_vasprintf
(
char
**
bufp
,
const
char
*
format
,
va_list
arg_ptr
)
{
struct
dynamic_buffer_parm_s
parm
;
int
rc
;
parm
.
error_flag
=
0
;
parm
.
alloced
=
512
;
parm
.
used
=
0
;
parm
.
buffer
=
my_printf_malloc
(
parm
.
alloced
);
if
(
!
parm
.
buffer
)
{
*
bufp
=
NULL
;
return
-1
;
}
rc
=
estream_format
(
dynamic_buffer_out
,
&
parm
,
format
,
arg_ptr
);
if
(
!
rc
)
rc
=
dynamic_buffer_out
(
&
parm
,
""
,
1
);
/* Print terminating Nul. */
/* Fixme: Should we shrink the resulting buffer? */
if
(
rc
!=
-1
&&
parm
.
error_flag
)
{
rc
=
-1
;
errno
=
parm
.
error_flag
;
}
if
(
rc
==
-1
)
{
memset
(
parm
.
buffer
,
0
,
parm
.
used
);
my_printf_free
(
parm
.
buffer
);
*
bufp
=
NULL
;
return
-1
;
}
assert
(
parm
.
used
);
/* We have at least the terminating Nul. */
*
bufp
=
parm
.
buffer
;
return
parm
.
used
-
1
;
/* Do not include that Nul. */
}
/* A replacement for asprintf. As with the BSD of asprintf version -1
will be returned on error and NULL stored at BUFP. On success the
number of bytes printed will be returned. */
int
estream_asprintf
(
char
**
bufp
,
const
char
*
format
,
...)
{
int
rc
;
va_list
arg_ptr
;
va_start
(
arg_ptr
,
format
);
rc
=
estream_vasprintf
(
bufp
,
format
,
arg_ptr
);
va_end
(
arg_ptr
);
return
rc
;
}
#ifdef TEST
static
int
one_test
(
const
char
*
format
,
...)
{
#ifdef _WIN32
{
static
int
show
;
if
(
!
show
)
{
/* We do not have a system vasprintf. */
printf
(
"one-test: disabled under W32
\n
"
);
show
=
1
;
}
}
#else
int
rc1
,
rc2
;
va_list
arg_ptr
;
char
*
buf1
,
*
buf2
;
if
(
verbose
)
printf
(
"format: ->%s<-
\n
"
,
format
);
va_start
(
arg_ptr
,
format
);
rc1
=
vasprintf
(
&
buf1
,
format
,
arg_ptr
);
va_end
(
arg_ptr
);
if
(
rc1
==
-1
)
{
printf
(
" sys: errno=%d (%s)
\n
"
,
errno
,
strerror
(
errno
));
buf1
=
NULL
;
}
else
if
(
verbose
)
printf
(
" sys: ->%s<-
\n
"
,
buf1
);
va_start
(
arg_ptr
,
format
);
rc2
=
estream_vasprintf
(
&
buf2
,
format
,
arg_ptr
);
va_end
(
arg_ptr
);
if
(
rc2
==
-1
)
printf
(
" our: errno=%d (%s)
\n
"
,
errno
,
strerror
(
errno
));
else
if
(
verbose
)
printf
(
" our: ->%s<-
\n
"
,
buf2
);
if
(
rc1
!=
-1
&&
rc2
!=
-1
&&
strcmp
(
buf1
,
buf2
))
printf
(
"error: output does not match
\n
"
"format: ->%s<-
\n
sys: ->%s<-
\n
our: ->%s<-
\n
"
,
format
,
buf1
,
buf2
);
else
if
(
rc1
!=
rc2
)
printf
(
"error: return codes are different: sys_rc=%d our_rc=%d
\n
"
,
rc1
,
rc2
);
free
(
buf2
);
free
(
buf1
);
#endif
return
0
;
}
static
void
run_tests
(
void
)
{
/*one_test ("%d %% %'d", 17, 19681977);*/
one_test
(
"%d %% %d"
,
17
,
768114563
);
one_test
(
"%d %% %d"
,
17
,
-768114563
);
one_test
(
"%d"
,
17
);
one_test
(
"%4d"
,
17
);
one_test
(
"%40d"
,
17
);
one_test
(
"%-d"
,
17
);
one_test
(
"%-4d"
,
17
);
one_test
(
"%-140d"
,
17
);
one_test
(
"%d"
,
-17
);
one_test
(
"%4d"
,
-17
);
one_test
(
"%40d"
,
-17
);
one_test
(
"%-d"
,
-17
);
one_test
(
"%-4d"
,
-17
);
one_test
(
"%-40d"
,
-17
);
one_test
(
"%+4d"
,
17
);
one_test
(
"%+4d"
,
-17
);
one_test
(
"%-+4d"
,
17
);
one_test
(
"%-+4d"
,
-17
);
one_test
(
"% 4d"
,
17
);
one_test
(
"% 4d"
,
-17
);
one_test
(
"%- +4d"
,
17
);
one_test
(
"%- +4d"
,
-17
);
one_test
(
"%.4d"
,
17
);
one_test
(
"%.0d"
,
17
);
one_test
(
"%.0d"
,
0
);
one_test
(
"%.4d"
,
-17
);
one_test
(
"%.0d"
,
-17
);
one_test
(
"%6.4d"
,
17
);
one_test
(
"%6.4d"
,
-17
);
one_test
(
"%6.0d"
,
0
);
one_test
(
"%4.6d"
,
17
);
one_test
(
"%4.6d"
,
-17
);
one_test
(
"% 4.6d"
,
17
);
one_test
(
"% 6.0d"
,
0
);
one_test
(
"%.4d"
,
17
);
one_test
(
"%04d"
,
17
);
one_test
(
"%.4d"
,
-17
);
one_test
(
"%04d"
,
-17
);
one_test
(
"%0.d"
,
0
);
one_test
(
"%*d"
,
7
,
42
);
one_test
(
"%*d"
,
-7
,
42
);
one_test
(
"%.*d"
,
7
,
42
);
one_test
(
"%.*d"
,
-7
,
42
);
one_test
(
"%*.*d"
,
10
,
7
,
42
);
one_test
(
"%*.*d"
,
10
,
-7
,
42
);
one_test
(
"%*.*d"
,
-10
,
7
,
42
);
one_test
(
"%*.*d"
,
-10
,
-7
,
42
);
one_test
(
"%*x"
,
7
,
42
);
one_test
(
"%*x"
,
-7
,
42
);
one_test
(
"%.*x"
,
7
,
42
);
one_test
(
"%.*x"
,
-7
,
42
);
one_test
(
"%*.*x"
,
10
,
7
,
42
);
one_test
(
"%*.*x"
,
10
,
-7
,
42
);
one_test
(
"%*.*x"
,
-10
,
7
,
42
);
one_test
(
"%*.*x"
,
-10
,
-7
,
42
);
one_test
(
"%#*x"
,
7
,
42
);
one_test
(
"%#*x"
,
-7
,
42
);
one_test
(
"%#.*x"
,
7
,
42
);
one_test
(
"%#.*x"
,
-7
,
42
);
one_test
(
"%#*.*x"
,
10
,
7
,
42
);
one_test
(
"%#*.*x"
,
10
,
-7
,
42
);
one_test
(
"%#*.*x"
,
-10
,
7
,
42
);
one_test
(
"%#*.*x"
,
-10
,
-7
,
42
);
one_test
(
"%*X"
,
7
,
42
);
one_test
(
"%*X"
,
-7
,
42
);
one_test
(
"%.*X"
,
7
,
42
);
one_test
(
"%.*X"
,
-7
,
42
);
one_test
(
"%*.*X"
,
10
,
7
,
42
);
one_test
(
"%*.*X"
,
10
,
-7
,
42
);
one_test
(
"%*.*X"
,
-10
,
7
,
42
);
one_test
(
"%*.*X"
,
-10
,
-7
,
42
);
one_test
(
"%#*X"
,
7
,
42
);
one_test
(
"%#*X"
,
-7
,
42
);
one_test
(
"%#.*X"
,
7
,
42
);
one_test
(
"%#.*X"
,
-7
,
42
);
one_test
(
"%#*.*X"
,
10
,
7
,
42
);
one_test
(
"%#*.*X"
,
10
,
-7
,
42
);
one_test
(
"%#*.*X"
,
-10
,
7
,
42
);
one_test
(
"%#*.*X"
,
-10
,
-7
,
42
);
one_test
(
"%*o"
,
7
,
42
);
one_test
(
"%*o"
,
-7
,
42
);
one_test
(
"%.*o"
,
7
,
42
);
one_test
(
"%.*o"
,
-7
,
42
);
one_test
(
"%*.*o"
,
10
,
7
,
42
);
one_test
(
"%*.*o"
,
10
,
-7
,
42
);
one_test
(
"%*.*o"
,
-10
,
7
,
42
);
one_test
(
"%*.*o"
,
-10
,
-7
,
42
);
one_test
(
"%#*o"
,
7
,
42
);
one_test
(
"%#*o"
,
-7
,
42
);
one_test
(
"%#.*o"
,
7
,
42
);
one_test
(
"%#.*o"
,
-7
,
42
);
one_test
(
"%#*.*o"
,
10
,
7
,
42
);
one_test
(
"%#*.*o"
,
10
,
-7
,
42
);
one_test
(
"%#*.*o"
,
-10
,
7
,
42
);
one_test
(
"%#*.*o"
,
-10
,
-7
,
42
);
one_test
(
"%s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%.0s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%.10s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%.48s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%.49s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%.50s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%.51s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%48s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%49s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%50s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%51s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"%-51s"
,
"the quick brown fox jumps over the lazy dogs back"
);
one_test
(
"/%s="
,
"CN"
);
one_test
(
"%f"
,
3.1415926535
);
one_test
(
"%f"
,
-3.1415926535
);
one_test
(
"%.10f"
,
3.1415926535
);
one_test
(
"%.2f"
,
3.1415926535
);
one_test
(
"%.1f"
,
3.1415926535
);
one_test
(
"%.0f"
,
3.1415926535
);
one_test
(
"%.20f"
,
3.1415926535
);
one_test
(
"%10.10f"
,
3.1415926535
);
one_test
(
"%10.2f"
,
3.1415926535
);
one_test
(
"%10.1f"
,
3.1415926535
);
one_test
(
"%10.0f"
,
3.1415926535
);
one_test
(
"%30.20f"
,
3.1415926535
);
one_test
(
"%10.10f"
,
-3.1415926535
);
one_test
(
"%10.2f"
,
-3.1415926535
);
one_test
(
"%10.1f"
,
-3.1415926535
);
one_test
(
"%10.0f"
,
-3.1415926535
);
one_test
(
"%30.20f"
,
-3.1415926535
);
one_test
(
"%-10f"
,
3.1415926535
);
one_test
(
"%-10.10f"
,
3.1415926535
);
one_test
(
"%-10.2f"
,
3.1415926535
);
one_test
(
"%-10.1f"
,
3.1415926535
);
one_test
(
"%-10.0f"
,
3.1415926535
);
one_test
(
"%-30.20f"
,
3.1415926535
);
one_test
(
"%-10f"
,
-3.1415926535
);
one_test
(
"%-10.10f"
,
-3.1415926535
);
one_test
(
"%-10.2f"
,
-3.1415926535
);
one_test
(
"%-10.1f"
,
-3.1415926535
);
one_test
(
"%-10.0f"
,
-3.1415926535
);
one_test
(
"%-30.20f"
,
-3.1415926535
);
one_test
(
"%#.0f"
,
3.1415926535
);
one_test
(
"%#10.0f"
,
3.1415926535
);
one_test
(
"%#10.0f"
,
-3.1415926535
);
one_test
(
"%-#10.0f"
,
3.1415926535
);
one_test
(
"%-#10.0f"
,
-3.1415926535
);
one_test
(
"%e"
,
3.1415926535
);
one_test
(
"%g"
,
3.1415926535
);
one_test
(
"%a"
,
1
);
one_test
(
"%a"
,
-1
);
one_test
(
"%a"
,
3.1415926535
);
#ifdef HAVE_LONG_DOUBLE
one_test
(
"%La"
,
1
);
one_test
(
"%La"
,
-1
);
one_test
(
"%La"
,
3.1415926535
);
#endif
#ifdef __GLIBC__
/* "%m" is a glibc extension so this _test_ will only work on such a
system. */
errno
=
ENOENT
;
one_test
(
"%m"
);
errno
=
ENOENT
;
one_test
(
"%d=%m"
,
17
);
errno
=
ENOENT
;
one_test
(
"%2$d:%m:%1$d"
,
42
,
17
);
#endif
/*__GLIBC__*/
}
static
void
check_snprintf
(
void
)
{
char
buffer
[
20
];
int
rc
,
rc2
;
size_t
tmplen
,
blen
,
blen2
;
rc
=
estream_snprintf
(
buffer
,
0
,
"%*s"
,
18
,
""
);
if
(
rc
!=
18
)
printf
(
"rc=%d
\n
"
,
rc
);
rc
=
estream_snprintf
(
buffer
,
sizeof
buffer
,
"%*s"
,
18
,
""
);
if
(
rc
!=
18
)
printf
(
"rc=%d, strlen(buffer)=%d
\n
"
,
rc
,
(
int
)
strlen
(
buffer
));
rc
=
estream_snprintf
(
buffer
,
sizeof
buffer
,
"%*s"
,
19
,
""
);
if
(
rc
!=
19
)
printf
(
"rc=%d, strlen(buffer)=%d
\n
"
,
rc
,
(
int
)
strlen
(
buffer
));
rc
=
estream_snprintf
(
buffer
,
sizeof
buffer
,
"%*s"
,
20
,
""
);
if
(
rc
!=
20
)
printf
(
"rc=%d, strlen(buffer)=%d
\n
"
,
rc
,
(
int
)
strlen
(
buffer
));
rc
=
estream_snprintf
(
buffer
,
sizeof
buffer
,
"%*s"
,
21
,
""
);
if
(
rc
!=
21
)
printf
(
"rc=%d, strlen(buffer)=%d
\n
"
,
rc
,
(
int
)
strlen
(
buffer
));
for
(
tmplen
=
0
;
tmplen
<=
sizeof
buffer
;
tmplen
++
)
{
rc
=
estream_snprintf
(
buffer
,
tmplen
,
"%04d%02d%02dT%02d%02d%02d"
,
1998
,
9
,
7
,
16
,
56
,
05
);
blen
=
strlen
(
buffer
);
rc2
=
snprintf
(
buffer
,
tmplen
,
"%04d%02d%02dT%02d%02d%02d"
,
1998
,
9
,
7
,
16
,
56
,
05
);
blen2
=
strlen
(
buffer
);
if
(
rc
!=
rc2
||
blen
!=
blen2
)
printf
(
"snprintf test with len %u gives %d instead of %d (%d,%d)
\n
"
,
(
unsigned
int
)
tmplen
,
rc
,
rc2
,
blen
,
blen2
);
}
}
int
main
(
int
argc
,
char
**
argv
)
{
int
rc
;
if
(
argc
)
{
argc
--
;
argv
++
;
}
setlocale
(
LC_NUMERIC
,
""
);
while
(
argc
&&
!
strcmp
(
*
argv
,
"--verbose"
))
{
verbose
++
;
argc
--
;
argv
++
;
}
if
(
!
argc
)
{
run_tests
();
check_snprintf
()
;
}
else
{
rc
=
estream_vfprintf
(
stdout
,
argv
[
0
],
NULL
);
fflush
(
stdout
);
fprintf
(
stderr
,
"[estream_vfprintf returns: %d]
\n
"
,
rc
);
}
return
0
;
}
#endif
/*TEST*/
/*
Local Variables:
compile-command: "cc -Wall -O3 -g -I.. -DHAVE_CONFIG_H -DTEST -o estream-printf estream-printf.c"
End:
*/
File Metadata
Details
Attached
Mime Type
text/x-c
Expires
Fri, Aug 29, 7:56 AM (1 d, 14 h)
Storage Engine
local-disk
Storage Format
Raw Data
Storage Handle
fa/fb/ccf7de8bc3aafd3000e49107110b
Attached To
rS Scute
Event Timeline
Log In to Comment