mirror of
				https://github.com/smaeul/u-boot.git
				synced 2025-10-31 20:18:18 +00:00 
			
		
		
		
	These two functions are only used in lib/tiny-printf.c . Signed-off-by: Masahiro Yamada <yamada.masahiro@socionext.com> Tested-by: Andreas Färber <afaerber@suse.de> Reviewed-by: Stefan Roese <sr@denx.de>
		
			
				
	
	
		
			239 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			239 lines
		
	
	
		
			4.1 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*
 | |
|  * Tiny printf version for SPL
 | |
|  *
 | |
|  * Copied from:
 | |
|  * http://www.sparetimelabs.com/printfrevisited/printfrevisited.php
 | |
|  *
 | |
|  * Copyright (C) 2004,2008  Kustaa Nyholm
 | |
|  *
 | |
|  * SPDX-License-Identifier:	LGPL-2.1+
 | |
|  */
 | |
| 
 | |
| #include <common.h>
 | |
| #include <stdarg.h>
 | |
| #include <serial.h>
 | |
| 
 | |
| struct printf_info {
 | |
| 	char *bf;	/* Digit buffer */
 | |
| 	char zs;	/* non-zero if a digit has been written */
 | |
| 	char *outstr;	/* Next output position for sprintf() */
 | |
| 
 | |
| 	/* Output a character */
 | |
| 	void (*putc)(struct printf_info *info, char ch);
 | |
| };
 | |
| 
 | |
| static void putc_normal(struct printf_info *info, char ch)
 | |
| {
 | |
| 	putc(ch);
 | |
| }
 | |
| 
 | |
| static void out(struct printf_info *info, char c)
 | |
| {
 | |
| 	*info->bf++ = c;
 | |
| }
 | |
| 
 | |
| static void out_dgt(struct printf_info *info, char dgt)
 | |
| {
 | |
| 	out(info, dgt + (dgt < 10 ? '0' : 'a' - 10));
 | |
| 	info->zs = 1;
 | |
| }
 | |
| 
 | |
| static void div_out(struct printf_info *info, unsigned long *num,
 | |
| 		    unsigned long div)
 | |
| {
 | |
| 	unsigned char dgt = 0;
 | |
| 
 | |
| 	while (*num >= div) {
 | |
| 		*num -= div;
 | |
| 		dgt++;
 | |
| 	}
 | |
| 
 | |
| 	if (info->zs || dgt > 0)
 | |
| 		out_dgt(info, dgt);
 | |
| }
 | |
| 
 | |
| static int _vprintf(struct printf_info *info, const char *fmt, va_list va)
 | |
| {
 | |
| 	char ch;
 | |
| 	char *p;
 | |
| 	unsigned long num;
 | |
| 	char buf[12];
 | |
| 	unsigned long div;
 | |
| 
 | |
| 	while ((ch = *(fmt++))) {
 | |
| 		if (ch != '%') {
 | |
| 			info->putc(info, ch);
 | |
| 		} else {
 | |
| 			bool lz = false;
 | |
| 			int width = 0;
 | |
| 			bool islong = false;
 | |
| 
 | |
| 			ch = *(fmt++);
 | |
| 			if (ch == '-')
 | |
| 				ch = *(fmt++);
 | |
| 
 | |
| 			if (ch == '0') {
 | |
| 				ch = *(fmt++);
 | |
| 				lz = 1;
 | |
| 			}
 | |
| 
 | |
| 			if (ch >= '0' && ch <= '9') {
 | |
| 				width = 0;
 | |
| 				while (ch >= '0' && ch <= '9') {
 | |
| 					width = (width * 10) + ch - '0';
 | |
| 					ch = *fmt++;
 | |
| 				}
 | |
| 			}
 | |
| 			if (ch == 'l') {
 | |
| 				ch = *(fmt++);
 | |
| 				islong = true;
 | |
| 			}
 | |
| 
 | |
| 			info->bf = buf;
 | |
| 			p = info->bf;
 | |
| 			info->zs = 0;
 | |
| 
 | |
| 			switch (ch) {
 | |
| 			case '\0':
 | |
| 				goto abort;
 | |
| 			case 'u':
 | |
| 			case 'd':
 | |
| 				div = 1000000000;
 | |
| 				if (islong) {
 | |
| 					num = va_arg(va, unsigned long);
 | |
| 					if (sizeof(long) > 4)
 | |
| 						div *= div * 10;
 | |
| 				} else {
 | |
| 					num = va_arg(va, unsigned int);
 | |
| 				}
 | |
| 
 | |
| 				if (ch == 'd') {
 | |
| 					if (islong && (long)num < 0) {
 | |
| 						num = -(long)num;
 | |
| 						out(info, '-');
 | |
| 					} else if (!islong && (int)num < 0) {
 | |
| 						num = -(int)num;
 | |
| 						out(info, '-');
 | |
| 					}
 | |
| 				}
 | |
| 				if (!num) {
 | |
| 					out_dgt(info, 0);
 | |
| 				} else {
 | |
| 					for (; div; div /= 10)
 | |
| 						div_out(info, &num, div);
 | |
| 				}
 | |
| 				break;
 | |
| 			case 'x':
 | |
| 				if (islong) {
 | |
| 					num = va_arg(va, unsigned long);
 | |
| 					div = 1UL << (sizeof(long) * 8 - 4);
 | |
| 				} else {
 | |
| 					num = va_arg(va, unsigned int);
 | |
| 					div = 0x10000000;
 | |
| 				}
 | |
| 				if (!num) {
 | |
| 					out_dgt(info, 0);
 | |
| 				} else {
 | |
| 					for (; div; div /= 0x10)
 | |
| 						div_out(info, &num, div);
 | |
| 				}
 | |
| 				break;
 | |
| 			case 'c':
 | |
| 				out(info, (char)(va_arg(va, int)));
 | |
| 				break;
 | |
| 			case 's':
 | |
| 				p = va_arg(va, char*);
 | |
| 				break;
 | |
| 			case '%':
 | |
| 				out(info, '%');
 | |
| 			default:
 | |
| 				break;
 | |
| 			}
 | |
| 
 | |
| 			*info->bf = 0;
 | |
| 			info->bf = p;
 | |
| 			while (*info->bf++ && width > 0)
 | |
| 				width--;
 | |
| 			while (width-- > 0)
 | |
| 				info->putc(info, lz ? '0' : ' ');
 | |
| 			if (p) {
 | |
| 				while ((ch = *p++))
 | |
| 					info->putc(info, ch);
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| abort:
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| int vprintf(const char *fmt, va_list va)
 | |
| {
 | |
| 	struct printf_info info;
 | |
| 
 | |
| 	info.putc = putc_normal;
 | |
| 	return _vprintf(&info, fmt, va);
 | |
| }
 | |
| 
 | |
| int printf(const char *fmt, ...)
 | |
| {
 | |
| 	struct printf_info info;
 | |
| 
 | |
| 	va_list va;
 | |
| 	int ret;
 | |
| 
 | |
| 	info.putc = putc_normal;
 | |
| 	va_start(va, fmt);
 | |
| 	ret = _vprintf(&info, fmt, va);
 | |
| 	va_end(va);
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| static void putc_outstr(struct printf_info *info, char ch)
 | |
| {
 | |
| 	*info->outstr++ = ch;
 | |
| }
 | |
| 
 | |
| int sprintf(char *buf, const char *fmt, ...)
 | |
| {
 | |
| 	struct printf_info info;
 | |
| 	va_list va;
 | |
| 	int ret;
 | |
| 
 | |
| 	va_start(va, fmt);
 | |
| 	info.outstr = buf;
 | |
| 	info.putc = putc_outstr;
 | |
| 	ret = _vprintf(&info, fmt, va);
 | |
| 	va_end(va);
 | |
| 	*info.outstr = '\0';
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| /* Note that size is ignored */
 | |
| int snprintf(char *buf, size_t size, const char *fmt, ...)
 | |
| {
 | |
| 	struct printf_info info;
 | |
| 	va_list va;
 | |
| 	int ret;
 | |
| 
 | |
| 	va_start(va, fmt);
 | |
| 	info.outstr = buf;
 | |
| 	info.putc = putc_outstr;
 | |
| 	ret = _vprintf(&info, fmt, va);
 | |
| 	va_end(va);
 | |
| 	*info.outstr = '\0';
 | |
| 
 | |
| 	return ret;
 | |
| }
 | |
| 
 | |
| void __assert_fail(const char *assertion, const char *file, unsigned line,
 | |
| 		   const char *function)
 | |
| {
 | |
| 	/* This will not return */
 | |
| 	printf("%s:%u: %s: Assertion `%s' failed.", file, line, function,
 | |
| 	       assertion);
 | |
| 	hang();
 | |
| }
 |