blob: a06abed49590ecbdbb8017f4df68e0c2208f3016 [file] [log] [blame]
Stefan Roese7d9cde12015-11-23 07:00:22 +01001/*
2 * Tiny printf version for SPL
3 *
4 * Copied from:
5 * http://www.sparetimelabs.com/printfrevisited/printfrevisited.php
6 *
7 * Copyright (C) 2004,2008 Kustaa Nyholm
8 *
9 * SPDX-License-Identifier: LGPL-2.1+
10 */
11
12#include <common.h>
13#include <stdarg.h>
14#include <serial.h>
15
16static char *bf;
Stefan Roese7d9cde12015-11-23 07:00:22 +010017static char zs;
18
19static void out(char c)
20{
21 *bf++ = c;
22}
23
24static void out_dgt(char dgt)
25{
Stefan Roesea5ecdd02015-11-16 15:26:34 +010026 out(dgt + (dgt < 10 ? '0' : 'a' - 10));
Stefan Roese7d9cde12015-11-23 07:00:22 +010027 zs = 1;
28}
29
Stefan Roesea5ecdd02015-11-16 15:26:34 +010030static void div_out(unsigned int *num, unsigned int div)
Stefan Roese7d9cde12015-11-23 07:00:22 +010031{
32 unsigned char dgt = 0;
33
Stefan Roesea5ecdd02015-11-16 15:26:34 +010034 while (*num >= div) {
35 *num -= div;
Stefan Roese7d9cde12015-11-23 07:00:22 +010036 dgt++;
37 }
38
39 if (zs || dgt > 0)
40 out_dgt(dgt);
41}
42
Sjoerd Simons962a43c2015-12-04 23:27:37 +010043int vprintf(const char *fmt, va_list va)
Stefan Roese7d9cde12015-11-23 07:00:22 +010044{
Stefan Roese7d9cde12015-11-23 07:00:22 +010045 char ch;
46 char *p;
Stefan Roesea5ecdd02015-11-16 15:26:34 +010047 unsigned int num;
48 char buf[12];
49 unsigned int div;
Stefan Roese7d9cde12015-11-23 07:00:22 +010050
Stefan Roese7d9cde12015-11-23 07:00:22 +010051 while ((ch = *(fmt++))) {
52 if (ch != '%') {
53 putc(ch);
54 } else {
55 char lz = 0;
56 char w = 0;
57
58 ch = *(fmt++);
59 if (ch == '0') {
60 ch = *(fmt++);
61 lz = 1;
62 }
63
64 if (ch >= '0' && ch <= '9') {
65 w = 0;
66 while (ch >= '0' && ch <= '9') {
Stefan Roesea5ecdd02015-11-16 15:26:34 +010067 w = (w * 10) + ch - '0';
Stefan Roese7d9cde12015-11-23 07:00:22 +010068 ch = *fmt++;
69 }
70 }
71 bf = buf;
72 p = bf;
73 zs = 0;
74
75 switch (ch) {
76 case 0:
77 goto abort;
78 case 'u':
79 case 'd':
80 num = va_arg(va, unsigned int);
81 if (ch == 'd' && (int)num < 0) {
82 num = -(int)num;
83 out('-');
84 }
Simon Glass74b13202016-01-05 09:30:57 -070085 if (!num) {
86 out_dgt(0);
87 } else {
88 for (div = 1000000000; div; div /= 10)
89 div_out(&num, div);
90 }
Stefan Roese7d9cde12015-11-23 07:00:22 +010091 break;
92 case 'x':
Stefan Roese7d9cde12015-11-23 07:00:22 +010093 num = va_arg(va, unsigned int);
Simon Glass74b13202016-01-05 09:30:57 -070094 if (!num) {
95 out_dgt(0);
96 } else {
97 for (div = 0x10000000; div; div /= 0x10)
98 div_out(&num, div);
99 }
Stefan Roese7d9cde12015-11-23 07:00:22 +0100100 break;
101 case 'c':
102 out((char)(va_arg(va, int)));
103 break;
104 case 's':
105 p = va_arg(va, char*);
106 break;
107 case '%':
108 out('%');
109 default:
110 break;
111 }
112
113 *bf = 0;
114 bf = p;
115 while (*bf++ && w > 0)
116 w--;
117 while (w-- > 0)
118 putc(lz ? '0' : ' ');
Simon Glass8e316812015-12-29 05:22:46 -0700119 if (p) {
120 while ((ch = *p++))
121 putc(ch);
122 }
Stefan Roese7d9cde12015-11-23 07:00:22 +0100123 }
124 }
125
126abort:
Stefan Roese7d9cde12015-11-23 07:00:22 +0100127 return 0;
128}
Sjoerd Simons962a43c2015-12-04 23:27:37 +0100129
130int printf(const char *fmt, ...)
131{
132 va_list va;
133 int ret;
134
135 va_start(va, fmt);
136 ret = vprintf(fmt, va);
137 va_end(va);
138
139 return ret;
140}