xref: /openbmc/u-boot/lib/vsprintf.c (revision 3fdf7596dff87a79e2b41d07479c608d91d06cb3)
1 /*
2  *  linux/lib/vsprintf.c
3  *
4  *  Copyright (C) 1991, 1992  Linus Torvalds
5  */
6 
7 /* vsprintf.c -- Lars Wirzenius & Linus Torvalds. */
8 /*
9  * Wirzenius wrote this portably, Torvalds fucked it up :-)
10  *
11  * from hush: simple_itoa() was lifted from boa-0.93.15
12  */
13 
14 #include <stdarg.h>
15 #include <linux/types.h>
16 #include <linux/string.h>
17 #include <linux/ctype.h>
18 #include <errno.h>
19 
20 #include <common.h>
21 #if !defined(CONFIG_PANIC_HANG)
22 #include <command.h>
23 #endif
24 
25 #include <div64.h>
26 #define noinline __attribute__((noinline))
27 
28 /* some reluctance to put this into a new limits.h, so it is here */
29 #define INT_MAX		((int)(~0U>>1))
30 
31 static const char hex_asc[] = "0123456789abcdef";
32 #define hex_asc_lo(x)   hex_asc[((x) & 0x0f)]
33 #define hex_asc_hi(x)   hex_asc[((x) & 0xf0) >> 4]
34 
35 static inline char *pack_hex_byte(char *buf, u8 byte)
36 {
37 	*buf++ = hex_asc_hi(byte);
38 	*buf++ = hex_asc_lo(byte);
39 	return buf;
40 }
41 
42 unsigned long simple_strtoul(const char *cp, char **endp,
43 				unsigned int base)
44 {
45 	unsigned long result = 0;
46 	unsigned long value;
47 
48 	if (*cp == '0') {
49 		cp++;
50 		if ((*cp == 'x') && isxdigit(cp[1])) {
51 			base = 16;
52 			cp++;
53 		}
54 
55 		if (!base)
56 			base = 8;
57 	}
58 
59 	if (!base)
60 		base = 10;
61 
62 	while (isxdigit(*cp) && (value = isdigit(*cp) ? *cp-'0' : (islower(*cp)
63 	    ? toupper(*cp) : *cp)-'A'+10) < base) {
64 		result = result*base + value;
65 		cp++;
66 	}
67 
68 	if (endp)
69 		*endp = (char *)cp;
70 
71 	return result;
72 }
73 
74 int strict_strtoul(const char *cp, unsigned int base, unsigned long *res)
75 {
76 	char *tail;
77 	unsigned long val;
78 	size_t len;
79 
80 	*res = 0;
81 	len = strlen(cp);
82 	if (len == 0)
83 		return -EINVAL;
84 
85 	val = simple_strtoul(cp, &tail, base);
86 	if (tail == cp)
87 		return -EINVAL;
88 
89 	if ((*tail == '\0') ||
90 		((len == (size_t)(tail - cp) + 1) && (*tail == '\n'))) {
91 		*res = val;
92 		return 0;
93 	}
94 
95 	return -EINVAL;
96 }
97 
98 long simple_strtol(const char *cp, char **endp, unsigned int base)
99 {
100 	if (*cp == '-')
101 		return -simple_strtoul(cp + 1, endp, base);
102 
103 	return simple_strtoul(cp, endp, base);
104 }
105 
106 int ustrtoul(const char *cp, char **endp, unsigned int base)
107 {
108 	unsigned long result = simple_strtoul(cp, endp, base);
109 	switch (**endp) {
110 	case 'G':
111 		result *= 1024;
112 		/* fall through */
113 	case 'M':
114 		result *= 1024;
115 		/* fall through */
116 	case 'K':
117 	case 'k':
118 		result *= 1024;
119 		if ((*endp)[1] == 'i') {
120 			if ((*endp)[2] == 'B')
121 				(*endp) += 3;
122 			else
123 				(*endp) += 2;
124 		}
125 	}
126 	return result;
127 }
128 
129 unsigned long long simple_strtoull(const char *cp, char **endp,
130 					unsigned int base)
131 {
132 	unsigned long long result = 0, value;
133 
134 	if (*cp == '0') {
135 		cp++;
136 		if ((*cp == 'x') && isxdigit(cp[1])) {
137 			base = 16;
138 			cp++;
139 		}
140 
141 		if (!base)
142 			base = 8;
143 	}
144 
145 	if (!base)
146 		base = 10;
147 
148 	while (isxdigit(*cp) && (value = isdigit(*cp) ? *cp - '0'
149 		: (islower(*cp) ? toupper(*cp) : *cp) - 'A' + 10) < base) {
150 		result = result * base + value;
151 		cp++;
152 	}
153 
154 	if (endp)
155 		*endp = (char *) cp;
156 
157 	return result;
158 }
159 
160 /* we use this so that we can do without the ctype library */
161 #define is_digit(c)	((c) >= '0' && (c) <= '9')
162 
163 static int skip_atoi(const char **s)
164 {
165 	int i = 0;
166 
167 	while (is_digit(**s))
168 		i = i * 10 + *((*s)++) - '0';
169 
170 	return i;
171 }
172 
173 /* Decimal conversion is by far the most typical, and is used
174  * for /proc and /sys data. This directly impacts e.g. top performance
175  * with many processes running. We optimize it for speed
176  * using code from
177  * http://www.cs.uiowa.edu/~jones/bcd/decimal.html
178  * (with permission from the author, Douglas W. Jones). */
179 
180 /* Formats correctly any integer in [0,99999].
181  * Outputs from one to five digits depending on input.
182  * On i386 gcc 4.1.2 -O2: ~250 bytes of code. */
183 static char *put_dec_trunc(char *buf, unsigned q)
184 {
185 	unsigned d3, d2, d1, d0;
186 	d1 = (q>>4) & 0xf;
187 	d2 = (q>>8) & 0xf;
188 	d3 = (q>>12);
189 
190 	d0 = 6*(d3 + d2 + d1) + (q & 0xf);
191 	q = (d0 * 0xcd) >> 11;
192 	d0 = d0 - 10*q;
193 	*buf++ = d0 + '0'; /* least significant digit */
194 	d1 = q + 9*d3 + 5*d2 + d1;
195 	if (d1 != 0) {
196 		q = (d1 * 0xcd) >> 11;
197 		d1 = d1 - 10*q;
198 		*buf++ = d1 + '0'; /* next digit */
199 
200 		d2 = q + 2*d2;
201 		if ((d2 != 0) || (d3 != 0)) {
202 			q = (d2 * 0xd) >> 7;
203 			d2 = d2 - 10*q;
204 			*buf++ = d2 + '0'; /* next digit */
205 
206 			d3 = q + 4*d3;
207 			if (d3 != 0) {
208 				q = (d3 * 0xcd) >> 11;
209 				d3 = d3 - 10*q;
210 				*buf++ = d3 + '0';  /* next digit */
211 				if (q != 0)
212 					*buf++ = q + '0'; /* most sign. digit */
213 			}
214 		}
215 	}
216 	return buf;
217 }
218 /* Same with if's removed. Always emits five digits */
219 static char *put_dec_full(char *buf, unsigned q)
220 {
221 	/* BTW, if q is in [0,9999], 8-bit ints will be enough, */
222 	/* but anyway, gcc produces better code with full-sized ints */
223 	unsigned d3, d2, d1, d0;
224 	d1 = (q>>4) & 0xf;
225 	d2 = (q>>8) & 0xf;
226 	d3 = (q>>12);
227 
228 	/*
229 	 * Possible ways to approx. divide by 10
230 	 * gcc -O2 replaces multiply with shifts and adds
231 	 * (x * 0xcd) >> 11: 11001101 - shorter code than * 0x67 (on i386)
232 	 * (x * 0x67) >> 10:  1100111
233 	 * (x * 0x34) >> 9:    110100 - same
234 	 * (x * 0x1a) >> 8:     11010 - same
235 	 * (x * 0x0d) >> 7:      1101 - same, shortest code (on i386)
236 	 */
237 
238 	d0 = 6*(d3 + d2 + d1) + (q & 0xf);
239 	q = (d0 * 0xcd) >> 11;
240 	d0 = d0 - 10*q;
241 	*buf++ = d0 + '0';
242 	d1 = q + 9*d3 + 5*d2 + d1;
243 		q = (d1 * 0xcd) >> 11;
244 		d1 = d1 - 10*q;
245 		*buf++ = d1 + '0';
246 
247 		d2 = q + 2*d2;
248 			q = (d2 * 0xd) >> 7;
249 			d2 = d2 - 10*q;
250 			*buf++ = d2 + '0';
251 
252 			d3 = q + 4*d3;
253 				q = (d3 * 0xcd) >> 11; /* - shorter code */
254 				/* q = (d3 * 0x67) >> 10; - would also work */
255 				d3 = d3 - 10*q;
256 				*buf++ = d3 + '0';
257 					*buf++ = q + '0';
258 	return buf;
259 }
260 /* No inlining helps gcc to use registers better */
261 static noinline char *put_dec(char *buf, u64 num)
262 {
263 	while (1) {
264 		unsigned rem;
265 		if (num < 100000)
266 			return put_dec_trunc(buf, num);
267 		rem = do_div(num, 100000);
268 		buf = put_dec_full(buf, rem);
269 	}
270 }
271 
272 #define ZEROPAD	1		/* pad with zero */
273 #define SIGN	2		/* unsigned/signed long */
274 #define PLUS	4		/* show plus */
275 #define SPACE	8		/* space if plus */
276 #define LEFT	16		/* left justified */
277 #define SMALL	32		/* Must be 32 == 0x20 */
278 #define SPECIAL	64		/* 0x */
279 
280 #ifdef CONFIG_SYS_VSNPRINTF
281 /*
282  * Macro to add a new character to our output string, but only if it will
283  * fit. The macro moves to the next character position in the output string.
284  */
285 #define ADDCH(str, ch) do { \
286 	if ((str) < end) \
287 		*(str) = (ch); \
288 	++str; \
289 	} while (0)
290 #else
291 #define ADDCH(str, ch)	(*(str)++ = (ch))
292 #endif
293 
294 static char *number(char *buf, char *end, u64 num,
295 		int base, int size, int precision, int type)
296 {
297 	/* we are called with base 8, 10 or 16, only, thus don't need "G..."  */
298 	static const char digits[16] = "0123456789ABCDEF";
299 
300 	char tmp[66];
301 	char sign;
302 	char locase;
303 	int need_pfx = ((type & SPECIAL) && base != 10);
304 	int i;
305 
306 	/* locase = 0 or 0x20. ORing digits or letters with 'locase'
307 	 * produces same digits or (maybe lowercased) letters */
308 	locase = (type & SMALL);
309 	if (type & LEFT)
310 		type &= ~ZEROPAD;
311 	sign = 0;
312 	if (type & SIGN) {
313 		if ((s64) num < 0) {
314 			sign = '-';
315 			num = -(s64) num;
316 			size--;
317 		} else if (type & PLUS) {
318 			sign = '+';
319 			size--;
320 		} else if (type & SPACE) {
321 			sign = ' ';
322 			size--;
323 		}
324 	}
325 	if (need_pfx) {
326 		size--;
327 		if (base == 16)
328 			size--;
329 	}
330 
331 	/* generate full string in tmp[], in reverse order */
332 	i = 0;
333 	if (num == 0)
334 		tmp[i++] = '0';
335 	/* Generic code, for any base:
336 	else do {
337 		tmp[i++] = (digits[do_div(num,base)] | locase);
338 	} while (num != 0);
339 	*/
340 	else if (base != 10) { /* 8 or 16 */
341 		int mask = base - 1;
342 		int shift = 3;
343 
344 		if (base == 16)
345 			shift = 4;
346 
347 		do {
348 			tmp[i++] = (digits[((unsigned char)num) & mask]
349 					| locase);
350 			num >>= shift;
351 		} while (num);
352 	} else { /* base 10 */
353 		i = put_dec(tmp, num) - tmp;
354 	}
355 
356 	/* printing 100 using %2d gives "100", not "00" */
357 	if (i > precision)
358 		precision = i;
359 	/* leading space padding */
360 	size -= precision;
361 	if (!(type & (ZEROPAD + LEFT))) {
362 		while (--size >= 0)
363 			ADDCH(buf, ' ');
364 	}
365 	/* sign */
366 	if (sign)
367 		ADDCH(buf, sign);
368 	/* "0x" / "0" prefix */
369 	if (need_pfx) {
370 		ADDCH(buf, '0');
371 		if (base == 16)
372 			ADDCH(buf, 'X' | locase);
373 	}
374 	/* zero or space padding */
375 	if (!(type & LEFT)) {
376 		char c = (type & ZEROPAD) ? '0' : ' ';
377 
378 		while (--size >= 0)
379 			ADDCH(buf, c);
380 	}
381 	/* hmm even more zero padding? */
382 	while (i <= --precision)
383 		ADDCH(buf, '0');
384 	/* actual digits of result */
385 	while (--i >= 0)
386 		ADDCH(buf, tmp[i]);
387 	/* trailing space padding */
388 	while (--size >= 0)
389 		ADDCH(buf, ' ');
390 	return buf;
391 }
392 
393 static char *string(char *buf, char *end, char *s, int field_width,
394 		int precision, int flags)
395 {
396 	int len, i;
397 
398 	if (s == NULL)
399 		s = "<NULL>";
400 
401 	len = strnlen(s, precision);
402 
403 	if (!(flags & LEFT))
404 		while (len < field_width--)
405 			ADDCH(buf, ' ');
406 	for (i = 0; i < len; ++i)
407 		ADDCH(buf, *s++);
408 	while (len < field_width--)
409 		ADDCH(buf, ' ');
410 	return buf;
411 }
412 
413 #ifdef CONFIG_CMD_NET
414 static char *mac_address_string(char *buf, char *end, u8 *addr, int field_width,
415 				int precision, int flags)
416 {
417 	/* (6 * 2 hex digits), 5 colons and trailing zero */
418 	char mac_addr[6 * 3];
419 	char *p = mac_addr;
420 	int i;
421 
422 	for (i = 0; i < 6; i++) {
423 		p = pack_hex_byte(p, addr[i]);
424 		if (!(flags & SPECIAL) && i != 5)
425 			*p++ = ':';
426 	}
427 	*p = '\0';
428 
429 	return string(buf, end, mac_addr, field_width, precision,
430 		      flags & ~SPECIAL);
431 }
432 
433 static char *ip6_addr_string(char *buf, char *end, u8 *addr, int field_width,
434 			 int precision, int flags)
435 {
436 	/* (8 * 4 hex digits), 7 colons and trailing zero */
437 	char ip6_addr[8 * 5];
438 	char *p = ip6_addr;
439 	int i;
440 
441 	for (i = 0; i < 8; i++) {
442 		p = pack_hex_byte(p, addr[2 * i]);
443 		p = pack_hex_byte(p, addr[2 * i + 1]);
444 		if (!(flags & SPECIAL) && i != 7)
445 			*p++ = ':';
446 	}
447 	*p = '\0';
448 
449 	return string(buf, end, ip6_addr, field_width, precision,
450 		      flags & ~SPECIAL);
451 }
452 
453 static char *ip4_addr_string(char *buf, char *end, u8 *addr, int field_width,
454 			 int precision, int flags)
455 {
456 	/* (4 * 3 decimal digits), 3 dots and trailing zero */
457 	char ip4_addr[4 * 4];
458 	char temp[3];	/* hold each IP quad in reverse order */
459 	char *p = ip4_addr;
460 	int i, digits;
461 
462 	for (i = 0; i < 4; i++) {
463 		digits = put_dec_trunc(temp, addr[i]) - temp;
464 		/* reverse the digits in the quad */
465 		while (digits--)
466 			*p++ = temp[digits];
467 		if (i != 3)
468 			*p++ = '.';
469 	}
470 	*p = '\0';
471 
472 	return string(buf, end, ip4_addr, field_width, precision,
473 		      flags & ~SPECIAL);
474 }
475 #endif
476 
477 /*
478  * Show a '%p' thing.  A kernel extension is that the '%p' is followed
479  * by an extra set of alphanumeric characters that are extended format
480  * specifiers.
481  *
482  * Right now we handle:
483  *
484  * - 'M' For a 6-byte MAC address, it prints the address in the
485  *       usual colon-separated hex notation
486  * - 'I' [46] for IPv4/IPv6 addresses printed in the usual way (dot-separated
487  *       decimal for v4 and colon separated network-order 16 bit hex for v6)
488  * - 'i' [46] for 'raw' IPv4/IPv6 addresses, IPv6 omits the colons, IPv4 is
489  *       currently the same
490  *
491  * Note: The difference between 'S' and 'F' is that on ia64 and ppc64
492  * function pointers are really function descriptors, which contain a
493  * pointer to the real address.
494  */
495 static char *pointer(const char *fmt, char *buf, char *end, void *ptr,
496 		int field_width, int precision, int flags)
497 {
498 	/*
499 	 * Being a boot loader, we explicitly allow pointers to
500 	 * (physical) address null.
501 	 */
502 #if 0
503 	if (!ptr)
504 		return string(buf, end, "(null)", field_width, precision,
505 			      flags);
506 #endif
507 
508 #ifdef CONFIG_CMD_NET
509 	switch (*fmt) {
510 	case 'm':
511 		flags |= SPECIAL;
512 		/* Fallthrough */
513 	case 'M':
514 		return mac_address_string(buf, end, ptr, field_width,
515 					  precision, flags);
516 	case 'i':
517 		flags |= SPECIAL;
518 		/* Fallthrough */
519 	case 'I':
520 		if (fmt[1] == '6')
521 			return ip6_addr_string(buf, end, ptr, field_width,
522 					       precision, flags);
523 		if (fmt[1] == '4')
524 			return ip4_addr_string(buf, end, ptr, field_width,
525 					       precision, flags);
526 		flags &= ~SPECIAL;
527 		break;
528 	}
529 #endif
530 	flags |= SMALL;
531 	if (field_width == -1) {
532 		field_width = 2*sizeof(void *);
533 		flags |= ZEROPAD;
534 	}
535 	return number(buf, end, (unsigned long)ptr, 16, field_width,
536 		      precision, flags);
537 }
538 
539 static int vsnprintf_internal(char *buf, size_t size, const char *fmt,
540 			      va_list args)
541 {
542 	u64 num;
543 	int base;
544 	char *str;
545 
546 	int flags;		/* flags to number() */
547 
548 	int field_width;	/* width of output field */
549 	int precision;		/* min. # of digits for integers; max
550 				   number of chars for from string */
551 	int qualifier;		/* 'h', 'l', or 'L' for integer fields */
552 				/* 'z' support added 23/7/1999 S.H.    */
553 				/* 'z' changed to 'Z' --davidm 1/25/99 */
554 				/* 't' added for ptrdiff_t */
555 	char *end = buf + size;
556 
557 #ifdef CONFIG_SYS_VSNPRINTF
558 	/* Make sure end is always >= buf - do we want this in U-Boot? */
559 	if (end < buf) {
560 		end = ((void *)-1);
561 		size = end - buf;
562 	}
563 #endif
564 	str = buf;
565 
566 	for (; *fmt ; ++fmt) {
567 		if (*fmt != '%') {
568 			ADDCH(str, *fmt);
569 			continue;
570 		}
571 
572 		/* process flags */
573 		flags = 0;
574 repeat:
575 			++fmt;		/* this also skips first '%' */
576 			switch (*fmt) {
577 			case '-':
578 				flags |= LEFT;
579 				goto repeat;
580 			case '+':
581 				flags |= PLUS;
582 				goto repeat;
583 			case ' ':
584 				flags |= SPACE;
585 				goto repeat;
586 			case '#':
587 				flags |= SPECIAL;
588 				goto repeat;
589 			case '0':
590 				flags |= ZEROPAD;
591 				goto repeat;
592 			}
593 
594 		/* get field width */
595 		field_width = -1;
596 		if (is_digit(*fmt))
597 			field_width = skip_atoi(&fmt);
598 		else if (*fmt == '*') {
599 			++fmt;
600 			/* it's the next argument */
601 			field_width = va_arg(args, int);
602 			if (field_width < 0) {
603 				field_width = -field_width;
604 				flags |= LEFT;
605 			}
606 		}
607 
608 		/* get the precision */
609 		precision = -1;
610 		if (*fmt == '.') {
611 			++fmt;
612 			if (is_digit(*fmt))
613 				precision = skip_atoi(&fmt);
614 			else if (*fmt == '*') {
615 				++fmt;
616 				/* it's the next argument */
617 				precision = va_arg(args, int);
618 			}
619 			if (precision < 0)
620 				precision = 0;
621 		}
622 
623 		/* get the conversion qualifier */
624 		qualifier = -1;
625 		if (*fmt == 'h' || *fmt == 'l' || *fmt == 'L' ||
626 		    *fmt == 'Z' || *fmt == 'z' || *fmt == 't') {
627 			qualifier = *fmt;
628 			++fmt;
629 			if (qualifier == 'l' && *fmt == 'l') {
630 				qualifier = 'L';
631 				++fmt;
632 			}
633 		}
634 
635 		/* default base */
636 		base = 10;
637 
638 		switch (*fmt) {
639 		case 'c':
640 			if (!(flags & LEFT)) {
641 				while (--field_width > 0)
642 					ADDCH(str, ' ');
643 			}
644 			ADDCH(str, (unsigned char) va_arg(args, int));
645 			while (--field_width > 0)
646 				ADDCH(str, ' ');
647 			continue;
648 
649 		case 's':
650 			str = string(str, end, va_arg(args, char *),
651 				     field_width, precision, flags);
652 			continue;
653 
654 		case 'p':
655 			str = pointer(fmt + 1, str, end,
656 					va_arg(args, void *),
657 					field_width, precision, flags);
658 			/* Skip all alphanumeric pointer suffixes */
659 			while (isalnum(fmt[1]))
660 				fmt++;
661 			continue;
662 
663 		case 'n':
664 			if (qualifier == 'l') {
665 				long *ip = va_arg(args, long *);
666 				*ip = (str - buf);
667 			} else {
668 				int *ip = va_arg(args, int *);
669 				*ip = (str - buf);
670 			}
671 			continue;
672 
673 		case '%':
674 			ADDCH(str, '%');
675 			continue;
676 
677 		/* integer number formats - set up the flags and "break" */
678 		case 'o':
679 			base = 8;
680 			break;
681 
682 		case 'x':
683 			flags |= SMALL;
684 		case 'X':
685 			base = 16;
686 			break;
687 
688 		case 'd':
689 		case 'i':
690 			flags |= SIGN;
691 		case 'u':
692 			break;
693 
694 		default:
695 			ADDCH(str, '%');
696 			if (*fmt)
697 				ADDCH(str, *fmt);
698 			else
699 				--fmt;
700 			continue;
701 		}
702 		if (qualifier == 'L')  /* "quad" for 64 bit variables */
703 			num = va_arg(args, unsigned long long);
704 		else if (qualifier == 'l') {
705 			num = va_arg(args, unsigned long);
706 			if (flags & SIGN)
707 				num = (signed long) num;
708 		} else if (qualifier == 'Z' || qualifier == 'z') {
709 			num = va_arg(args, size_t);
710 		} else if (qualifier == 't') {
711 			num = va_arg(args, ptrdiff_t);
712 		} else if (qualifier == 'h') {
713 			num = (unsigned short) va_arg(args, int);
714 			if (flags & SIGN)
715 				num = (signed short) num;
716 		} else {
717 			num = va_arg(args, unsigned int);
718 			if (flags & SIGN)
719 				num = (signed int) num;
720 		}
721 		str = number(str, end, num, base, field_width, precision,
722 			     flags);
723 	}
724 
725 #ifdef CONFIG_SYS_VSNPRINTF
726 	if (size > 0) {
727 		ADDCH(str, '\0');
728 		if (str > end)
729 			end[-1] = '\0';
730 	}
731 #else
732 	*str = '\0';
733 #endif
734 	/* the trailing null byte doesn't count towards the total */
735 	return str - buf;
736 }
737 
738 #ifdef CONFIG_SYS_VSNPRINTF
739 int vsnprintf(char *buf, size_t size, const char *fmt,
740 			      va_list args)
741 {
742 	return vsnprintf_internal(buf, size, fmt, args);
743 }
744 
745 int vscnprintf(char *buf, size_t size, const char *fmt, va_list args)
746 {
747 	int i;
748 
749 	i = vsnprintf(buf, size, fmt, args);
750 
751 	if (likely(i < size))
752 		return i;
753 	if (size != 0)
754 		return size - 1;
755 	return 0;
756 }
757 
758 int snprintf(char *buf, size_t size, const char *fmt, ...)
759 {
760 	va_list args;
761 	int i;
762 
763 	va_start(args, fmt);
764 	i = vsnprintf(buf, size, fmt, args);
765 	va_end(args);
766 
767 	return i;
768 }
769 
770 int scnprintf(char *buf, size_t size, const char *fmt, ...)
771 {
772 	va_list args;
773 	int i;
774 
775 	va_start(args, fmt);
776 	i = vscnprintf(buf, size, fmt, args);
777 	va_end(args);
778 
779 	return i;
780 }
781 #endif /* CONFIG_SYS_VSNPRINT */
782 
783 /**
784  * Format a string and place it in a buffer (va_list version)
785  *
786  * @param buf	The buffer to place the result into
787  * @param fmt	The format string to use
788  * @param args	Arguments for the format string
789  *
790  * The function returns the number of characters written
791  * into @buf. Use vsnprintf() or vscnprintf() in order to avoid
792  * buffer overflows.
793  *
794  * If you're not already dealing with a va_list consider using sprintf().
795  */
796 int vsprintf(char *buf, const char *fmt, va_list args)
797 {
798 	return vsnprintf_internal(buf, INT_MAX, fmt, args);
799 }
800 
801 int sprintf(char *buf, const char *fmt, ...)
802 {
803 	va_list args;
804 	int i;
805 
806 	va_start(args, fmt);
807 	i = vsprintf(buf, fmt, args);
808 	va_end(args);
809 	return i;
810 }
811 
812 void panic(const char *fmt, ...)
813 {
814 	va_list args;
815 	va_start(args, fmt);
816 	vprintf(fmt, args);
817 	putc('\n');
818 	va_end(args);
819 #if defined(CONFIG_PANIC_HANG)
820 	hang();
821 #else
822 	udelay(100000);	/* allow messages to go out */
823 	do_reset(NULL, 0, 0, NULL);
824 #endif
825 	while (1)
826 		;
827 }
828 
829 void __assert_fail(const char *assertion, const char *file, unsigned line,
830 		   const char *function)
831 {
832 	/* This will not return */
833 	panic("%s:%u: %s: Assertion `%s' failed.", file, line, function,
834 	      assertion);
835 }
836 
837 char *simple_itoa(ulong i)
838 {
839 	/* 21 digits plus null terminator, good for 64-bit or smaller ints */
840 	static char local[22];
841 	char *p = &local[21];
842 
843 	*p-- = '\0';
844 	do {
845 		*p-- = '0' + i % 10;
846 		i /= 10;
847 	} while (i > 0);
848 	return p + 1;
849 }
850