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)";
202 case TYPE_WIDE_STRING
:
203 ap
->a
.a_wide_string
= va_arg (args
, const wchar_t *);
204 /* A null pointer is an invalid argument for "%ls", but in practice
205 it occurs quite frequently in printf statements that produce
206 debug output. Use a fallback in this case. */
207 if (ap
->a
.a_wide_string
== NULL
)
209 static const wchar_t wide_null_string
[] =
212 (wchar_t)'N', (wchar_t)'U', (wchar_t)'L', (wchar_t)'L',
216 ap
->a
.a_wide_string
= wide_null_string
;
221 ap
->a
.a_pointer
= va_arg (args
, void *);
223 case TYPE_COUNT_SCHAR_POINTER
:
224 ap
->a
.a_count_schar_pointer
= va_arg (args
, signed char *);
226 case TYPE_COUNT_SHORT_POINTER
:
227 ap
->a
.a_count_short_pointer
= va_arg (args
, short *);
229 case TYPE_COUNT_INT_POINTER
:
230 ap
->a
.a_count_int_pointer
= va_arg (args
, int *);
232 case TYPE_COUNT_LONGINT_POINTER
:
233 ap
->a
.a_count_longint_pointer
= va_arg (args
, long int *);
235 case TYPE_COUNT_LONGLONGINT_POINTER
:
236 ap
->a
.a_count_longlongint_pointer
= va_arg (args
, long long int *);
238 case TYPE_COUNT_INT8_T_POINTER
:
239 ap
->a
.a_count_int8_t_pointer
= va_arg (args
, int8_t *);
241 case TYPE_COUNT_INT16_T_POINTER
:
242 ap
->a
.a_count_int16_t_pointer
= va_arg (args
, int16_t *);
244 case TYPE_COUNT_INT32_T_POINTER
:
245 ap
->a
.a_count_int32_t_pointer
= va_arg (args
, int32_t *);
247 case TYPE_COUNT_INT64_T_POINTER
:
248 ap
->a
.a_count_int64_t_pointer
= va_arg (args
, int64_t *);
250 case TYPE_COUNT_INT_FAST8_T_POINTER
:
251 ap
->a
.a_count_int_fast8_t_pointer
= va_arg (args
, int_fast8_t *);
253 case TYPE_COUNT_INT_FAST16_T_POINTER
:
254 ap
->a
.a_count_int_fast16_t_pointer
= va_arg (args
, int_fast16_t *);
256 case TYPE_COUNT_INT_FAST32_T_POINTER
:
257 ap
->a
.a_count_int_fast32_t_pointer
= va_arg (args
, int_fast32_t *);
259 case TYPE_COUNT_INT_FAST64_T_POINTER
:
260 ap
->a
.a_count_int_fast64_t_pointer
= va_arg (args
, int_fast64_t *);
263 /* The unistdio extensions. */
265 ap
->a
.a_u8_string
= va_arg (args
, const uint8_t *);
266 /* A null pointer is an invalid argument for "%U", but in practice
267 it occurs quite frequently in printf statements that produce
268 debug output. Use a fallback in this case. */
269 if (ap
->a
.a_u8_string
== NULL
)
271 static const uint8_t u8_null_string
[] =
272 { '(', 'N', 'U', 'L', 'L', ')', 0 };
273 ap
->a
.a_u8_string
= u8_null_string
;
276 case TYPE_U16_STRING
:
277 ap
->a
.a_u16_string
= va_arg (args
, const uint16_t *);
278 /* A null pointer is an invalid argument for "%lU", but in practice
279 it occurs quite frequently in printf statements that produce
280 debug output. Use a fallback in this case. */
281 if (ap
->a
.a_u16_string
== NULL
)
283 static const uint16_t u16_null_string
[] =
284 { '(', 'N', 'U', 'L', 'L', ')', 0 };
285 ap
->a
.a_u16_string
= u16_null_string
;
288 case TYPE_U32_STRING
:
289 ap
->a
.a_u32_string
= va_arg (args
, const uint32_t *);
290 /* A null pointer is an invalid argument for "%llU", but in practice
291 it occurs quite frequently in printf statements that produce
292 debug output. Use a fallback in this case. */
293 if (ap
->a
.a_u32_string
== NULL
)
295 static const uint32_t u32_null_string
[] =
296 { '(', 'N', 'U', 'L', 'L', ')', 0 };
297 ap
->a
.a_u32_string
= u32_null_string
;