]>
Commit | Line | Data |
---|---|---|
83d290c5 | 1 | // SPDX-License-Identifier: LGPL-2.1+ |
7d9cde10 SR |
2 | /* |
3 | * Tiny printf version for SPL | |
4 | * | |
5 | * Copied from: | |
6 | * http://www.sparetimelabs.com/printfrevisited/printfrevisited.php | |
7 | * | |
8 | * Copyright (C) 2004,2008 Kustaa Nyholm | |
7d9cde10 SR |
9 | */ |
10 | ||
11 | #include <common.h> | |
12 | #include <stdarg.h> | |
13 | #include <serial.h> | |
cdce1f76 | 14 | #include <linux/ctype.h> |
7d9cde10 | 15 | |
45313e83 SG |
16 | struct printf_info { |
17 | char *bf; /* Digit buffer */ | |
18 | char zs; /* non-zero if a digit has been written */ | |
19 | char *outstr; /* Next output position for sprintf() */ | |
7d9cde10 | 20 | |
45313e83 SG |
21 | /* Output a character */ |
22 | void (*putc)(struct printf_info *info, char ch); | |
23 | }; | |
5c411d88 | 24 | |
45313e83 | 25 | static void out(struct printf_info *info, char c) |
7d9cde10 | 26 | { |
45313e83 | 27 | *info->bf++ = c; |
7d9cde10 SR |
28 | } |
29 | ||
45313e83 SG |
30 | static void out_dgt(struct printf_info *info, char dgt) |
31 | { | |
32 | out(info, dgt + (dgt < 10 ? '0' : 'a' - 10)); | |
33 | info->zs = 1; | |
34 | } | |
35 | ||
a28e1d98 AP |
36 | static void div_out(struct printf_info *info, unsigned long *num, |
37 | unsigned long div) | |
7d9cde10 SR |
38 | { |
39 | unsigned char dgt = 0; | |
40 | ||
a5ecdd08 SR |
41 | while (*num >= div) { |
42 | *num -= div; | |
7d9cde10 SR |
43 | dgt++; |
44 | } | |
45 | ||
45313e83 SG |
46 | if (info->zs || dgt > 0) |
47 | out_dgt(info, dgt); | |
7d9cde10 SR |
48 | } |
49 | ||
cdce1f76 V |
50 | #ifdef CONFIG_SPL_NET_SUPPORT |
51 | static void string(struct printf_info *info, char *s) | |
52 | { | |
53 | char ch; | |
54 | ||
55 | while ((ch = *s++)) | |
56 | out(info, ch); | |
57 | } | |
58 | ||
59 | static const char hex_asc[] = "0123456789abcdef"; | |
60 | #define hex_asc_lo(x) hex_asc[((x) & 0x0f)] | |
61 | #define hex_asc_hi(x) hex_asc[((x) & 0xf0) >> 4] | |
62 | ||
63 | static inline char *pack_hex_byte(char *buf, u8 byte) | |
64 | { | |
65 | *buf++ = hex_asc_hi(byte); | |
66 | *buf++ = hex_asc_lo(byte); | |
67 | return buf; | |
68 | } | |
69 | ||
70 | static void mac_address_string(struct printf_info *info, u8 *addr, | |
71 | bool separator) | |
72 | { | |
73 | /* (6 * 2 hex digits), 5 colons and trailing zero */ | |
74 | char mac_addr[6 * 3]; | |
75 | char *p = mac_addr; | |
76 | int i; | |
77 | ||
78 | for (i = 0; i < 6; i++) { | |
79 | p = pack_hex_byte(p, addr[i]); | |
80 | if (separator && i != 5) | |
81 | *p++ = ':'; | |
82 | } | |
83 | *p = '\0'; | |
84 | ||
85 | string(info, mac_addr); | |
86 | } | |
87 | ||
88 | static char *put_dec_trunc(char *buf, unsigned int q) | |
89 | { | |
90 | unsigned int d3, d2, d1, d0; | |
91 | d1 = (q >> 4) & 0xf; | |
92 | d2 = (q >> 8) & 0xf; | |
93 | d3 = (q >> 12); | |
94 | ||
95 | d0 = 6 * (d3 + d2 + d1) + (q & 0xf); | |
96 | q = (d0 * 0xcd) >> 11; | |
97 | d0 = d0 - 10 * q; | |
98 | *buf++ = d0 + '0'; /* least significant digit */ | |
99 | d1 = q + 9 * d3 + 5 * d2 + d1; | |
100 | if (d1 != 0) { | |
101 | q = (d1 * 0xcd) >> 11; | |
102 | d1 = d1 - 10 * q; | |
103 | *buf++ = d1 + '0'; /* next digit */ | |
104 | ||
105 | d2 = q + 2 * d2; | |
106 | if ((d2 != 0) || (d3 != 0)) { | |
107 | q = (d2 * 0xd) >> 7; | |
108 | d2 = d2 - 10 * q; | |
109 | *buf++ = d2 + '0'; /* next digit */ | |
110 | ||
111 | d3 = q + 4 * d3; | |
112 | if (d3 != 0) { | |
113 | q = (d3 * 0xcd) >> 11; | |
114 | d3 = d3 - 10 * q; | |
115 | *buf++ = d3 + '0'; /* next digit */ | |
116 | if (q != 0) | |
117 | *buf++ = q + '0'; /* most sign. digit */ | |
118 | } | |
119 | } | |
120 | } | |
121 | return buf; | |
122 | } | |
123 | ||
124 | static void ip4_addr_string(struct printf_info *info, u8 *addr) | |
125 | { | |
126 | /* (4 * 3 decimal digits), 3 dots and trailing zero */ | |
127 | char ip4_addr[4 * 4]; | |
128 | char temp[3]; /* hold each IP quad in reverse order */ | |
129 | char *p = ip4_addr; | |
130 | int i, digits; | |
131 | ||
132 | for (i = 0; i < 4; i++) { | |
133 | digits = put_dec_trunc(temp, addr[i]) - temp; | |
134 | /* reverse the digits in the quad */ | |
135 | while (digits--) | |
136 | *p++ = temp[digits]; | |
137 | if (i != 3) | |
138 | *p++ = '.'; | |
139 | } | |
140 | *p = '\0'; | |
141 | ||
142 | string(info, ip4_addr); | |
143 | } | |
144 | #endif | |
145 | ||
146 | /* | |
147 | * Show a '%p' thing. A kernel extension is that the '%p' is followed | |
148 | * by an extra set of characters that are extended format | |
149 | * specifiers. | |
150 | * | |
151 | * Right now we handle: | |
152 | * | |
153 | * - 'M' For a 6-byte MAC address, it prints the address in the | |
154 | * usual colon-separated hex notation. | |
155 | * - 'm' Same as above except there is no colon-separator. | |
156 | * - 'I4'for IPv4 addresses printed in the usual way (dot-separated | |
157 | * decimal). | |
158 | */ | |
159 | ||
831c1611 SG |
160 | static void __maybe_unused pointer(struct printf_info *info, const char *fmt, |
161 | void *ptr) | |
cdce1f76 V |
162 | { |
163 | #ifdef DEBUG | |
164 | unsigned long num = (uintptr_t)ptr; | |
165 | unsigned long div; | |
166 | #endif | |
167 | ||
168 | switch (*fmt) { | |
169 | #ifdef DEBUG | |
170 | case 'a': | |
171 | ||
172 | switch (fmt[1]) { | |
173 | case 'p': | |
174 | default: | |
175 | num = *(phys_addr_t *)ptr; | |
176 | break; | |
177 | } | |
178 | break; | |
179 | #endif | |
180 | #ifdef CONFIG_SPL_NET_SUPPORT | |
181 | case 'm': | |
182 | return mac_address_string(info, ptr, false); | |
183 | case 'M': | |
184 | return mac_address_string(info, ptr, true); | |
185 | case 'I': | |
186 | if (fmt[1] == '4') | |
187 | return ip4_addr_string(info, ptr); | |
188 | #endif | |
189 | default: | |
190 | break; | |
191 | } | |
192 | #ifdef DEBUG | |
193 | div = 1UL << (sizeof(long) * 8 - 4); | |
194 | for (; div; div /= 0x10) | |
195 | div_out(info, &num, div); | |
196 | #endif | |
197 | } | |
198 | ||
433cbfb3 | 199 | static int _vprintf(struct printf_info *info, const char *fmt, va_list va) |
7d9cde10 | 200 | { |
7d9cde10 SR |
201 | char ch; |
202 | char *p; | |
a28e1d98 | 203 | unsigned long num; |
a5ecdd08 | 204 | char buf[12]; |
a28e1d98 | 205 | unsigned long div; |
7d9cde10 | 206 | |
7d9cde10 SR |
207 | while ((ch = *(fmt++))) { |
208 | if (ch != '%') { | |
45313e83 | 209 | info->putc(info, ch); |
7d9cde10 | 210 | } else { |
1fb67608 SG |
211 | bool lz = false; |
212 | int width = 0; | |
a28e1d98 | 213 | bool islong = false; |
7d9cde10 SR |
214 | |
215 | ch = *(fmt++); | |
1c853629 AP |
216 | if (ch == '-') |
217 | ch = *(fmt++); | |
218 | ||
7d9cde10 SR |
219 | if (ch == '0') { |
220 | ch = *(fmt++); | |
221 | lz = 1; | |
222 | } | |
223 | ||
224 | if (ch >= '0' && ch <= '9') { | |
1fb67608 | 225 | width = 0; |
7d9cde10 | 226 | while (ch >= '0' && ch <= '9') { |
1fb67608 | 227 | width = (width * 10) + ch - '0'; |
7d9cde10 SR |
228 | ch = *fmt++; |
229 | } | |
230 | } | |
a28e1d98 AP |
231 | if (ch == 'l') { |
232 | ch = *(fmt++); | |
233 | islong = true; | |
234 | } | |
235 | ||
45313e83 SG |
236 | info->bf = buf; |
237 | p = info->bf; | |
238 | info->zs = 0; | |
7d9cde10 SR |
239 | |
240 | switch (ch) { | |
1fb67608 | 241 | case '\0': |
7d9cde10 SR |
242 | goto abort; |
243 | case 'u': | |
244 | case 'd': | |
e7882f65 | 245 | case 'i': |
a28e1d98 AP |
246 | div = 1000000000; |
247 | if (islong) { | |
248 | num = va_arg(va, unsigned long); | |
249 | if (sizeof(long) > 4) | |
250 | div *= div * 10; | |
251 | } else { | |
252 | num = va_arg(va, unsigned int); | |
253 | } | |
254 | ||
e7882f65 | 255 | if (ch != 'u') { |
a28e1d98 AP |
256 | if (islong && (long)num < 0) { |
257 | num = -(long)num; | |
258 | out(info, '-'); | |
259 | } else if (!islong && (int)num < 0) { | |
260 | num = -(int)num; | |
261 | out(info, '-'); | |
262 | } | |
7d9cde10 | 263 | } |
74b1320a | 264 | if (!num) { |
45313e83 | 265 | out_dgt(info, 0); |
74b1320a | 266 | } else { |
a28e1d98 | 267 | for (; div; div /= 10) |
45313e83 | 268 | div_out(info, &num, div); |
74b1320a | 269 | } |
7d9cde10 | 270 | break; |
831c1611 SG |
271 | case 'p': |
272 | #ifdef DEBUG | |
273 | pointer(info, fmt, va_arg(va, void *)); | |
274 | /* | |
275 | * Skip this because it pulls in _ctype which is | |
276 | * 256 bytes, and we don't generally implement | |
277 | * pointer anyway | |
278 | */ | |
279 | while (isalnum(fmt[0])) | |
280 | fmt++; | |
281 | break; | |
282 | #else | |
283 | islong = true; | |
284 | /* no break */ | |
285 | #endif | |
7d9cde10 | 286 | case 'x': |
a28e1d98 AP |
287 | if (islong) { |
288 | num = va_arg(va, unsigned long); | |
289 | div = 1UL << (sizeof(long) * 8 - 4); | |
290 | } else { | |
291 | num = va_arg(va, unsigned int); | |
292 | div = 0x10000000; | |
293 | } | |
74b1320a | 294 | if (!num) { |
45313e83 | 295 | out_dgt(info, 0); |
74b1320a | 296 | } else { |
a28e1d98 | 297 | for (; div; div /= 0x10) |
45313e83 | 298 | div_out(info, &num, div); |
74b1320a | 299 | } |
7d9cde10 SR |
300 | break; |
301 | case 'c': | |
45313e83 | 302 | out(info, (char)(va_arg(va, int))); |
7d9cde10 SR |
303 | break; |
304 | case 's': | |
305 | p = va_arg(va, char*); | |
306 | break; | |
307 | case '%': | |
45313e83 | 308 | out(info, '%'); |
7d9cde10 SR |
309 | default: |
310 | break; | |
311 | } | |
312 | ||
45313e83 SG |
313 | *info->bf = 0; |
314 | info->bf = p; | |
315 | while (*info->bf++ && width > 0) | |
1fb67608 SG |
316 | width--; |
317 | while (width-- > 0) | |
45313e83 | 318 | info->putc(info, lz ? '0' : ' '); |
8e31681c SG |
319 | if (p) { |
320 | while ((ch = *p++)) | |
45313e83 | 321 | info->putc(info, ch); |
8e31681c | 322 | } |
7d9cde10 SR |
323 | } |
324 | } | |
325 | ||
326 | abort: | |
7d9cde10 SR |
327 | return 0; |
328 | } | |
962a43cc | 329 | |
4f1eed75 AK |
330 | #if CONFIG_IS_ENABLED(PRINTF) |
331 | static void putc_normal(struct printf_info *info, char ch) | |
332 | { | |
333 | putc(ch); | |
334 | } | |
335 | ||
da70b4d1 HG |
336 | int vprintf(const char *fmt, va_list va) |
337 | { | |
45313e83 SG |
338 | struct printf_info info; |
339 | ||
340 | info.putc = putc_normal; | |
341 | return _vprintf(&info, fmt, va); | |
da70b4d1 HG |
342 | } |
343 | ||
962a43cc SS |
344 | int printf(const char *fmt, ...) |
345 | { | |
45313e83 SG |
346 | struct printf_info info; |
347 | ||
962a43cc SS |
348 | va_list va; |
349 | int ret; | |
350 | ||
45313e83 | 351 | info.putc = putc_normal; |
962a43cc | 352 | va_start(va, fmt); |
45313e83 | 353 | ret = _vprintf(&info, fmt, va); |
5c411d88 SG |
354 | va_end(va); |
355 | ||
356 | return ret; | |
357 | } | |
4f1eed75 | 358 | #endif |
5c411d88 | 359 | |
45313e83 | 360 | static void putc_outstr(struct printf_info *info, char ch) |
5c411d88 | 361 | { |
45313e83 | 362 | *info->outstr++ = ch; |
5c411d88 SG |
363 | } |
364 | ||
abeb272d | 365 | int sprintf(char *buf, const char *fmt, ...) |
5c411d88 | 366 | { |
45313e83 | 367 | struct printf_info info; |
5c411d88 SG |
368 | va_list va; |
369 | int ret; | |
370 | ||
371 | va_start(va, fmt); | |
45313e83 SG |
372 | info.outstr = buf; |
373 | info.putc = putc_outstr; | |
374 | ret = _vprintf(&info, fmt, va); | |
962a43cc | 375 | va_end(va); |
45313e83 | 376 | *info.outstr = '\0'; |
962a43cc SS |
377 | |
378 | return ret; | |
379 | } | |
abeb272d | 380 | |
9b3fbb2b SS |
381 | #if CONFIG_IS_ENABLED(LOG) |
382 | /* Note that size is ignored */ | |
383 | int vsnprintf(char *buf, size_t size, const char *fmt, va_list va) | |
384 | { | |
385 | struct printf_info info; | |
386 | int ret; | |
387 | ||
388 | info.outstr = buf; | |
389 | info.putc = putc_outstr; | |
390 | ret = _vprintf(&info, fmt, va); | |
391 | *info.outstr = '\0'; | |
392 | ||
393 | return ret; | |
394 | } | |
395 | #endif | |
396 | ||
abeb272d MV |
397 | /* Note that size is ignored */ |
398 | int snprintf(char *buf, size_t size, const char *fmt, ...) | |
399 | { | |
45313e83 | 400 | struct printf_info info; |
abeb272d MV |
401 | va_list va; |
402 | int ret; | |
403 | ||
404 | va_start(va, fmt); | |
45313e83 SG |
405 | info.outstr = buf; |
406 | info.putc = putc_outstr; | |
407 | ret = _vprintf(&info, fmt, va); | |
abeb272d | 408 | va_end(va); |
45313e83 | 409 | *info.outstr = '\0'; |
abeb272d MV |
410 | |
411 | return ret; | |
412 | } | |
dee74e6c SG |
413 | |
414 | void print_grouped_ull(unsigned long long int_val, int digits) | |
415 | { | |
416 | /* Don't try to print the upper 32-bits */ | |
417 | printf("%ld ", (ulong)int_val); | |
418 | } |