1 /* Decomposed printf argument list.
2 Copyright (C) 1999, 2002-2003, 2005-2007, 2009-2024 Free Software
5 This file is free software: you can redistribute it and/or modify
6 it under the terms of the GNU Lesser General Public License as
7 published by the Free Software Foundation; either version 2.1 of the
8 License, or (at your option) any later version.
10 This file is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU Lesser General Public License for more details.
15 You should have received a copy of the GNU Lesser General Public License
16 along with this program. If not, see <https://www.gnu.org/licenses/>. */
18 /* This file can be parametrized with the following macros:
19 ENABLE_UNISTDIO Set to 1 to enable the unistdio extensions.
20 PRINTF_FETCHARGS Name of the function to be defined.
21 STATIC Set to 'static' to declare the function static. */
23 #ifndef PRINTF_FETCHARGS
28 #ifndef PRINTF_FETCHARGS
29 # include "printf-args.h"
39 PRINTF_FETCHARGS (va_list args
, arguments
*a
)
44 for (i
= 0, ap
= &a
->arg
[0]; i
< a
->count
; i
++, ap
++)
48 ap
->a
.a_schar
= va_arg (args
, /*signed char*/ int);
51 ap
->a
.a_uchar
= va_arg (args
, /*unsigned char*/ int);
54 ap
->a
.a_short
= va_arg (args
, /*short*/ int);
57 ap
->a
.a_ushort
= va_arg (args
, /*unsigned short*/ int);
60 ap
->a
.a_int
= va_arg (args
, int);
63 ap
->a
.a_uint
= va_arg (args
, unsigned int);
66 ap
->a
.a_longint
= va_arg (args
, long int);
69 ap
->a
.a_ulongint
= va_arg (args
, unsigned long int);
71 case TYPE_LONGLONGINT
:
72 ap
->a
.a_longlongint
= va_arg (args
, long long int);
74 case TYPE_ULONGLONGINT
:
75 ap
->a
.a_ulonglongint
= va_arg (args
, unsigned long long int);
78 #if INT8_WIDTH < INT_WIDTH
79 ap
->a
.a_int8_t
= va_arg (args
, /* int8_t */ int);
81 ap
->a
.a_int8_t
= va_arg (args
, int8_t);
85 #if UINT8_WIDTH < INT_WIDTH
86 ap
->a
.a_uint8_t
= va_arg (args
, /* uint8_t */ int);
88 ap
->a
.a_uint8_t
= va_arg (args
, uint8_t);
92 #if INT16_WIDTH < INT_WIDTH
93 ap
->a
.a_int16_t
= va_arg (args
, /* int16_t */ int);
95 ap
->a
.a_int16_t
= va_arg (args
, int16_t);
99 #if UINT16_WIDTH < INT_WIDTH
100 ap
->a
.a_uint16_t
= va_arg (args
, /* uint16_t */ int);
102 ap
->a
.a_uint16_t
= va_arg (args
, uint16_t);
106 #if INT32_WIDTH < INT_WIDTH
107 ap
->a
.a_int32_t
= va_arg (args
, /* int32_t */ int);
109 ap
->a
.a_int32_t
= va_arg (args
, int32_t);
113 #if UINT32_WIDTH < INT_WIDTH
114 ap
->a
.a_uint32_t
= va_arg (args
, /* uint32_t */ int);
116 ap
->a
.a_uint32_t
= va_arg (args
, uint32_t);
120 ap
->a
.a_int64_t
= va_arg (args
, int64_t);
123 ap
->a
.a_uint64_t
= va_arg (args
, uint64_t);
125 case TYPE_INT_FAST8_T
:
126 #if INT_FAST8_WIDTH < INT_WIDTH
127 ap
->a
.a_int_fast8_t
= va_arg (args
, /* int_fast8_t */ int);
129 ap
->a
.a_int_fast8_t
= va_arg (args
, int_fast8_t);
132 case TYPE_UINT_FAST8_T
:
133 #if UINT_FAST8_WIDTH < INT_WIDTH
134 ap
->a
.a_uint_fast8_t
= va_arg (args
, /* uint_fast8_t */ int);
136 ap
->a
.a_uint_fast8_t
= va_arg (args
, uint_fast8_t);
139 case TYPE_INT_FAST16_T
:
140 #if INT_FAST16_WIDTH < INT_WIDTH
141 ap
->a
.a_int_fast16_t
= va_arg (args
, /* int_fast16_t */ int);
143 ap
->a
.a_int_fast16_t
= va_arg (args
, int_fast16_t);
146 case TYPE_UINT_FAST16_T
:
147 #if UINT_FAST16_WIDTH < INT_WIDTH
148 ap
->a
.a_uint_fast16_t
= va_arg (args
, /* uint_fast16_t */ int);
150 ap
->a
.a_uint_fast16_t
= va_arg (args
, uint_fast16_t);
153 case TYPE_INT_FAST32_T
:
154 #if INT_FAST32_WIDTH < INT_WIDTH
155 ap
->a
.a_int_fast32_t
= va_arg (args
, /* int_fast32_t */ int);
157 ap
->a
.a_int_fast32_t
= va_arg (args
, int_fast32_t);
160 case TYPE_UINT_FAST32_T
:
161 #if UINT_FAST32_WIDTH < INT_WIDTH
162 ap
->a
.a_uint_fast32_t
= va_arg (args
, /* uint_fast32_t */ int);
164 ap
->a
.a_uint_fast32_t
= va_arg (args
, uint_fast32_t);
167 case TYPE_INT_FAST64_T
:
168 ap
->a
.a_int_fast64_t
= va_arg (args
, int_fast64_t);
170 case TYPE_UINT_FAST64_T
:
171 ap
->a
.a_uint_fast64_t
= va_arg (args
, uint_fast64_t);
174 ap
->a
.a_double
= va_arg (args
, double);
176 case TYPE_LONGDOUBLE
:
177 ap
->a
.a_longdouble
= va_arg (args
, long double);
180 ap
->a
.a_char
= va_arg (args
, int);
184 /* Although ISO C 99 7.24.1.(2) says that wint_t is "unchanged by
185 default argument promotions", this is not the case in mingw32,
186 where wint_t is 'unsigned short'. */
188 (sizeof (wint_t) < sizeof (int)
189 ? (wint_t) va_arg (args
, int)
190 : va_arg (args
, wint_t));
194 ap
->a
.a_string
= va_arg (args
, const char *);
195 /* A null pointer is an invalid argument for "%s", but in practice
196 it occurs quite frequently in printf statements that produce
197 debug output. Use a fallback in this case. */
198 if (ap
->a
.a_string
== NULL
)
199 ap
->a
.a_string
= "(NULL)";
201 case TYPE_WIDE_STRING
:
202 ap
->a
.a_wide_string
= va_arg (args
, const wchar_t *);
203 /* A null pointer is an invalid argument for "%ls", but in practice
204 it occurs quite frequently in printf statements that produce
205 debug output. Use a fallback in this case. */
206 if (ap
->a
.a_wide_string
== NULL
)
208 static const wchar_t wide_null_string
[] =
211 (wchar_t)'N', (wchar_t)'U', (wchar_t)'L', (wchar_t)'L',
215 ap
->a
.a_wide_string
= wide_null_string
;
219 ap
->a
.a_pointer
= va_arg (args
, void *);
221 case TYPE_COUNT_SCHAR_POINTER
:
222 ap
->a
.a_count_schar_pointer
= va_arg (args
, signed char *);
224 case TYPE_COUNT_SHORT_POINTER
:
225 ap
->a
.a_count_short_pointer
= va_arg (args
, short *);
227 case TYPE_COUNT_INT_POINTER
:
228 ap
->a
.a_count_int_pointer
= va_arg (args
, int *);
230 case TYPE_COUNT_LONGINT_POINTER
:
231 ap
->a
.a_count_longint_pointer
= va_arg (args
, long int *);
233 case TYPE_COUNT_LONGLONGINT_POINTER
:
234 ap
->a
.a_count_longlongint_pointer
= va_arg (args
, long long int *);
236 case TYPE_COUNT_INT8_T_POINTER
:
237 ap
->a
.a_count_int8_t_pointer
= va_arg (args
, int8_t *);
239 case TYPE_COUNT_INT16_T_POINTER
:
240 ap
->a
.a_count_int16_t_pointer
= va_arg (args
, int16_t *);
242 case TYPE_COUNT_INT32_T_POINTER
:
243 ap
->a
.a_count_int32_t_pointer
= va_arg (args
, int32_t *);
245 case TYPE_COUNT_INT64_T_POINTER
:
246 ap
->a
.a_count_int64_t_pointer
= va_arg (args
, int64_t *);
248 case TYPE_COUNT_INT_FAST8_T_POINTER
:
249 ap
->a
.a_count_int_fast8_t_pointer
= va_arg (args
, int_fast8_t *);
251 case TYPE_COUNT_INT_FAST16_T_POINTER
:
252 ap
->a
.a_count_int_fast16_t_pointer
= va_arg (args
, int_fast16_t *);
254 case TYPE_COUNT_INT_FAST32_T_POINTER
:
255 ap
->a
.a_count_int_fast32_t_pointer
= va_arg (args
, int_fast32_t *);
257 case TYPE_COUNT_INT_FAST64_T_POINTER
:
258 ap
->a
.a_count_int_fast64_t_pointer
= va_arg (args
, int_fast64_t *);
261 /* The unistdio extensions. */
263 ap
->a
.a_u8_string
= va_arg (args
, const uint8_t *);
264 /* A null pointer is an invalid argument for "%U", but in practice
265 it occurs quite frequently in printf statements that produce
266 debug output. Use a fallback in this case. */
267 if (ap
->a
.a_u8_string
== NULL
)
269 static const uint8_t u8_null_string
[] =
270 { '(', 'N', 'U', 'L', 'L', ')', 0 };
271 ap
->a
.a_u8_string
= u8_null_string
;
274 case TYPE_U16_STRING
:
275 ap
->a
.a_u16_string
= va_arg (args
, const uint16_t *);
276 /* A null pointer is an invalid argument for "%lU", but in practice
277 it occurs quite frequently in printf statements that produce
278 debug output. Use a fallback in this case. */
279 if (ap
->a
.a_u16_string
== NULL
)
281 static const uint16_t u16_null_string
[] =
282 { '(', 'N', 'U', 'L', 'L', ')', 0 };
283 ap
->a
.a_u16_string
= u16_null_string
;
286 case TYPE_U32_STRING
:
287 ap
->a
.a_u32_string
= va_arg (args
, const uint32_t *);
288 /* A null pointer is an invalid argument for "%llU", but in practice
289 it occurs quite frequently in printf statements that produce
290 debug output. Use a fallback in this case. */
291 if (ap
->a
.a_u32_string
== NULL
)
293 static const uint32_t u32_null_string
[] =
294 { '(', 'N', 'U', 'L', 'L', ')', 0 };
295 ap
->a
.a_u32_string
= u32_null_string
;