xref: /openbmc/u-boot/lib/display_options.c (revision 772b9e8b)
183d290c5STom Rini // SPDX-License-Identifier: GPL-2.0+
278acc472SPeter Tyser /*
378acc472SPeter Tyser  * (C) Copyright 2000-2002
478acc472SPeter Tyser  * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
578acc472SPeter Tyser  */
678acc472SPeter Tyser 
778acc472SPeter Tyser #include <common.h>
824b852a7SSimon Glass #include <console.h>
933eac2dcSSimon Glass #include <div64.h>
10*772b9e8bSPali Rohár #include <version_string.h>
1178acc472SPeter Tyser #include <linux/ctype.h>
1278acc472SPeter Tyser #include <asm/io.h>
1378acc472SPeter Tyser 
display_options_get_banner_priv(bool newlines,const char * build_tag,char * buf,int size)146c519f2dSSimon Glass char *display_options_get_banner_priv(bool newlines, const char *build_tag,
156c519f2dSSimon Glass 				      char *buf, int size)
166c519f2dSSimon Glass {
176c519f2dSSimon Glass 	int len;
186c519f2dSSimon Glass 
196c519f2dSSimon Glass 	len = snprintf(buf, size, "%s%s", newlines ? "\n\n" : "",
206c519f2dSSimon Glass 		       version_string);
216c519f2dSSimon Glass 	if (build_tag && len < size)
226c519f2dSSimon Glass 		len += snprintf(buf + len, size - len, ", Build: %s",
236c519f2dSSimon Glass 				build_tag);
246c519f2dSSimon Glass 	if (len > size - 3)
256c519f2dSSimon Glass 		len = size - 3;
266c519f2dSSimon Glass 	strcpy(buf + len, "\n\n");
276c519f2dSSimon Glass 
286c519f2dSSimon Glass 	return buf;
296c519f2dSSimon Glass }
306c519f2dSSimon Glass 
316c519f2dSSimon Glass #ifndef BUILD_TAG
326c519f2dSSimon Glass #define BUILD_TAG NULL
336c519f2dSSimon Glass #endif
346c519f2dSSimon Glass 
display_options_get_banner(bool newlines,char * buf,int size)356c519f2dSSimon Glass char *display_options_get_banner(bool newlines, char *buf, int size)
366c519f2dSSimon Glass {
376c519f2dSSimon Glass 	return display_options_get_banner_priv(newlines, BUILD_TAG, buf, size);
386c519f2dSSimon Glass }
396c519f2dSSimon Glass 
display_options(void)4078acc472SPeter Tyser int display_options(void)
4178acc472SPeter Tyser {
426c519f2dSSimon Glass 	char buf[DISPLAY_OPTIONS_BANNER_LENGTH];
436c519f2dSSimon Glass 
446c519f2dSSimon Glass 	display_options_get_banner(true, buf, sizeof(buf));
456c519f2dSSimon Glass 	printf("%s", buf);
466c519f2dSSimon Glass 
4778acc472SPeter Tyser 	return 0;
4878acc472SPeter Tyser }
4978acc472SPeter Tyser 
print_freq(uint64_t freq,const char * s)5033eac2dcSSimon Glass void print_freq(uint64_t freq, const char *s)
5133eac2dcSSimon Glass {
5280402f34SHeiko Schocher 	unsigned long m = 0;
5333eac2dcSSimon Glass 	uint32_t f;
5433eac2dcSSimon Glass 	static const char names[] = {'G', 'M', 'K'};
5533eac2dcSSimon Glass 	unsigned long d = 1e9;
5633eac2dcSSimon Glass 	char c = 0;
5733eac2dcSSimon Glass 	unsigned int i;
5833eac2dcSSimon Glass 
5933eac2dcSSimon Glass 	for (i = 0; i < ARRAY_SIZE(names); i++, d /= 1000) {
6033eac2dcSSimon Glass 		if (freq >= d) {
6133eac2dcSSimon Glass 			c = names[i];
6233eac2dcSSimon Glass 			break;
6333eac2dcSSimon Glass 		}
6433eac2dcSSimon Glass 	}
6533eac2dcSSimon Glass 
6633eac2dcSSimon Glass 	if (!c) {
67dee37fc9SMasahiro Yamada 		printf("%llu Hz%s", freq, s);
6833eac2dcSSimon Glass 		return;
6933eac2dcSSimon Glass 	}
7033eac2dcSSimon Glass 
7133eac2dcSSimon Glass 	f = do_div(freq, d);
7233eac2dcSSimon Glass 
7333eac2dcSSimon Glass 	/* If there's a remainder, show the first few digits */
7433eac2dcSSimon Glass 	if (f) {
7533eac2dcSSimon Glass 		m = f;
7633eac2dcSSimon Glass 		while (m > 1000)
7733eac2dcSSimon Glass 			m /= 10;
7833eac2dcSSimon Glass 		while (m && !(m % 10))
7933eac2dcSSimon Glass 			m /= 10;
8033eac2dcSSimon Glass 		if (m >= 100)
8133eac2dcSSimon Glass 			m = (m / 10) + (m % 100 >= 50);
8233eac2dcSSimon Glass 	}
8333eac2dcSSimon Glass 
84e9015b30SSuriyan Ramasami 	printf("%lu", (unsigned long) freq);
8533eac2dcSSimon Glass 	if (m)
8633eac2dcSSimon Glass 		printf(".%ld", m);
8733eac2dcSSimon Glass 	printf(" %cHz%s", c, s);
8833eac2dcSSimon Glass }
8933eac2dcSSimon Glass 
print_size(uint64_t size,const char * s)90c6da9ae8SSimon Glass void print_size(uint64_t size, const char *s)
9178acc472SPeter Tyser {
9252dbac69STimur Tabi 	unsigned long m = 0, n;
93c6da9ae8SSimon Glass 	uint64_t f;
944b42c905STimur Tabi 	static const char names[] = {'E', 'P', 'T', 'G', 'M', 'K'};
95f2d76ae4SNick Thompson 	unsigned long d = 10 * ARRAY_SIZE(names);
964b42c905STimur Tabi 	char c = 0;
974b42c905STimur Tabi 	unsigned int i;
9878acc472SPeter Tyser 
99f2d76ae4SNick Thompson 	for (i = 0; i < ARRAY_SIZE(names); i++, d -= 10) {
100f2d76ae4SNick Thompson 		if (size >> d) {
1014b42c905STimur Tabi 			c = names[i];
1024b42c905STimur Tabi 			break;
10378acc472SPeter Tyser 		}
10478acc472SPeter Tyser 	}
10578acc472SPeter Tyser 
1064b42c905STimur Tabi 	if (!c) {
107dee37fc9SMasahiro Yamada 		printf("%llu Bytes%s", size, s);
1084b42c905STimur Tabi 		return;
1094b42c905STimur Tabi 	}
1104b42c905STimur Tabi 
111f2d76ae4SNick Thompson 	n = size >> d;
112f2d76ae4SNick Thompson 	f = size & ((1ULL << d) - 1);
11378acc472SPeter Tyser 
11478acc472SPeter Tyser 	/* If there's a remainder, deal with it */
115f2d76ae4SNick Thompson 	if (f) {
116f2d76ae4SNick Thompson 		m = (10ULL * f + (1ULL << (d - 1))) >> d;
11778acc472SPeter Tyser 
11878acc472SPeter Tyser 		if (m >= 10) {
11978acc472SPeter Tyser 			m -= 10;
12078acc472SPeter Tyser 			n += 1;
12178acc472SPeter Tyser 		}
12278acc472SPeter Tyser 	}
12378acc472SPeter Tyser 
1244b42c905STimur Tabi 	printf ("%lu", n);
12578acc472SPeter Tyser 	if (m) {
12678acc472SPeter Tyser 		printf (".%ld", m);
12778acc472SPeter Tyser 	}
1284b42c905STimur Tabi 	printf (" %ciB%s", c, s);
12978acc472SPeter Tyser }
13078acc472SPeter Tyser 
13178acc472SPeter Tyser #define MAX_LINE_LENGTH_BYTES (64)
13278acc472SPeter Tyser #define DEFAULT_LINE_LENGTH_BYTES (16)
print_buffer(ulong addr,const void * data,uint width,uint count,uint linelen)133bda32ffcSSimon Glass int print_buffer(ulong addr, const void *data, uint width, uint count,
134bda32ffcSSimon Glass 		 uint linelen)
13578acc472SPeter Tyser {
136150f7236SReinhard Meyer 	/* linebuf as a union causes proper alignment */
137150f7236SReinhard Meyer 	union linebuf {
1384d1fd7f1SYork Sun #ifdef CONFIG_SYS_SUPPORT_64BIT_DATA
1394d1fd7f1SYork Sun 		uint64_t uq[MAX_LINE_LENGTH_BYTES/sizeof(uint64_t) + 1];
1404d1fd7f1SYork Sun #endif
141150f7236SReinhard Meyer 		uint32_t ui[MAX_LINE_LENGTH_BYTES/sizeof(uint32_t) + 1];
142150f7236SReinhard Meyer 		uint16_t us[MAX_LINE_LENGTH_BYTES/sizeof(uint16_t) + 1];
143150f7236SReinhard Meyer 		uint8_t  uc[MAX_LINE_LENGTH_BYTES/sizeof(uint8_t) + 1];
144150f7236SReinhard Meyer 	} lb;
14578acc472SPeter Tyser 	int i;
1464d1fd7f1SYork Sun #ifdef CONFIG_SYS_SUPPORT_64BIT_DATA
14780402f34SHeiko Schocher 	uint64_t __maybe_unused x;
1484d1fd7f1SYork Sun #else
14980402f34SHeiko Schocher 	uint32_t __maybe_unused x;
1504d1fd7f1SYork Sun #endif
15178acc472SPeter Tyser 
15278acc472SPeter Tyser 	if (linelen*width > MAX_LINE_LENGTH_BYTES)
15378acc472SPeter Tyser 		linelen = MAX_LINE_LENGTH_BYTES / width;
15478acc472SPeter Tyser 	if (linelen < 1)
15578acc472SPeter Tyser 		linelen = DEFAULT_LINE_LENGTH_BYTES / width;
15678acc472SPeter Tyser 
15778acc472SPeter Tyser 	while (count) {
158efd7c114SAndreas Bießmann 		uint thislinelen = linelen;
15978acc472SPeter Tyser 		printf("%08lx:", addr);
16078acc472SPeter Tyser 
16178acc472SPeter Tyser 		/* check for overflow condition */
162efd7c114SAndreas Bießmann 		if (count < thislinelen)
163efd7c114SAndreas Bießmann 			thislinelen = count;
16478acc472SPeter Tyser 
16578acc472SPeter Tyser 		/* Copy from memory into linebuf and print hex values */
166efd7c114SAndreas Bießmann 		for (i = 0; i < thislinelen; i++) {
16764419e47SMike Frysinger 			if (width == 4)
168150f7236SReinhard Meyer 				x = lb.ui[i] = *(volatile uint32_t *)data;
1694d1fd7f1SYork Sun #ifdef CONFIG_SYS_SUPPORT_64BIT_DATA
1704d1fd7f1SYork Sun 			else if (width == 8)
1714d1fd7f1SYork Sun 				x = lb.uq[i] = *(volatile uint64_t *)data;
1724d1fd7f1SYork Sun #endif
17364419e47SMike Frysinger 			else if (width == 2)
174150f7236SReinhard Meyer 				x = lb.us[i] = *(volatile uint16_t *)data;
17564419e47SMike Frysinger 			else
176150f7236SReinhard Meyer 				x = lb.uc[i] = *(volatile uint8_t *)data;
177f60662deSSimon Glass #if defined(CONFIG_SPL_BUILD)
178f60662deSSimon Glass 			printf(" %x", (uint)x);
179f60662deSSimon Glass #elif defined(CONFIG_SYS_SUPPORT_64BIT_DATA)
18066da9bebSSimon Glass 			printf(" %0*llx", width * 2, (long long)x);
1814d1fd7f1SYork Sun #else
18264419e47SMike Frysinger 			printf(" %0*x", width * 2, x);
1834d1fd7f1SYork Sun #endif
18478acc472SPeter Tyser 			data += width;
18578acc472SPeter Tyser 		}
18678acc472SPeter Tyser 
187efd7c114SAndreas Bießmann 		while (thislinelen < linelen) {
188efd7c114SAndreas Bießmann 			/* fill line with whitespace for nice ASCII print */
189efd7c114SAndreas Bießmann 			for (i=0; i<width*2+1; i++)
190efd7c114SAndreas Bießmann 				puts(" ");
191efd7c114SAndreas Bießmann 			linelen--;
192efd7c114SAndreas Bießmann 		}
193efd7c114SAndreas Bießmann 
19478acc472SPeter Tyser 		/* Print data in ASCII characters */
195efd7c114SAndreas Bießmann 		for (i = 0; i < thislinelen * width; i++) {
196150f7236SReinhard Meyer 			if (!isprint(lb.uc[i]) || lb.uc[i] >= 0x80)
197150f7236SReinhard Meyer 				lb.uc[i] = '.';
198150f7236SReinhard Meyer 		}
199150f7236SReinhard Meyer 		lb.uc[i] = '\0';
200150f7236SReinhard Meyer 		printf("    %s\n", lb.uc);
20178acc472SPeter Tyser 
20278acc472SPeter Tyser 		/* update references */
203efd7c114SAndreas Bießmann 		addr += thislinelen * width;
204efd7c114SAndreas Bießmann 		count -= thislinelen;
20578acc472SPeter Tyser 
20678acc472SPeter Tyser 		if (ctrlc())
20778acc472SPeter Tyser 			return -1;
20878acc472SPeter Tyser 	}
20978acc472SPeter Tyser 
21078acc472SPeter Tyser 	return 0;
21178acc472SPeter Tyser }
212