178acc472SPeter Tyser /* 278acc472SPeter Tyser * (C) Copyright 2000-2002 378acc472SPeter Tyser * Wolfgang Denk, DENX Software Engineering, wd@denx.de. 478acc472SPeter Tyser * 51a459660SWolfgang Denk * SPDX-License-Identifier: GPL-2.0+ 678acc472SPeter Tyser */ 778acc472SPeter Tyser 878acc472SPeter Tyser #include <common.h> 924b852a7SSimon Glass #include <console.h> 1033eac2dcSSimon Glass #include <div64.h> 11c6da9ae8SSimon Glass #include <inttypes.h> 1209c2e90cSAndreas Bießmann #include <version.h> 1378acc472SPeter Tyser #include <linux/ctype.h> 1478acc472SPeter Tyser #include <asm/io.h> 1578acc472SPeter Tyser 16*6c519f2dSSimon Glass char *display_options_get_banner_priv(bool newlines, const char *build_tag, 17*6c519f2dSSimon Glass char *buf, int size) 18*6c519f2dSSimon Glass { 19*6c519f2dSSimon Glass int len; 20*6c519f2dSSimon Glass 21*6c519f2dSSimon Glass len = snprintf(buf, size, "%s%s", newlines ? "\n\n" : "", 22*6c519f2dSSimon Glass version_string); 23*6c519f2dSSimon Glass if (build_tag && len < size) 24*6c519f2dSSimon Glass len += snprintf(buf + len, size - len, ", Build: %s", 25*6c519f2dSSimon Glass build_tag); 26*6c519f2dSSimon Glass if (len > size - 3) 27*6c519f2dSSimon Glass len = size - 3; 28*6c519f2dSSimon Glass strcpy(buf + len, "\n\n"); 29*6c519f2dSSimon Glass 30*6c519f2dSSimon Glass return buf; 31*6c519f2dSSimon Glass } 32*6c519f2dSSimon Glass 33*6c519f2dSSimon Glass #ifndef BUILD_TAG 34*6c519f2dSSimon Glass #define BUILD_TAG NULL 35*6c519f2dSSimon Glass #endif 36*6c519f2dSSimon Glass 37*6c519f2dSSimon Glass char *display_options_get_banner(bool newlines, char *buf, int size) 38*6c519f2dSSimon Glass { 39*6c519f2dSSimon Glass return display_options_get_banner_priv(newlines, BUILD_TAG, buf, size); 40*6c519f2dSSimon Glass } 41*6c519f2dSSimon Glass 4278acc472SPeter Tyser int display_options(void) 4378acc472SPeter Tyser { 44*6c519f2dSSimon Glass char buf[DISPLAY_OPTIONS_BANNER_LENGTH]; 45*6c519f2dSSimon Glass 46*6c519f2dSSimon Glass display_options_get_banner(true, buf, sizeof(buf)); 47*6c519f2dSSimon Glass printf("%s", buf); 48*6c519f2dSSimon Glass 4978acc472SPeter Tyser return 0; 5078acc472SPeter Tyser } 5178acc472SPeter Tyser 5233eac2dcSSimon Glass void print_freq(uint64_t freq, const char *s) 5333eac2dcSSimon Glass { 5480402f34SHeiko Schocher unsigned long m = 0; 5533eac2dcSSimon Glass uint32_t f; 5633eac2dcSSimon Glass static const char names[] = {'G', 'M', 'K'}; 5733eac2dcSSimon Glass unsigned long d = 1e9; 5833eac2dcSSimon Glass char c = 0; 5933eac2dcSSimon Glass unsigned int i; 6033eac2dcSSimon Glass 6133eac2dcSSimon Glass for (i = 0; i < ARRAY_SIZE(names); i++, d /= 1000) { 6233eac2dcSSimon Glass if (freq >= d) { 6333eac2dcSSimon Glass c = names[i]; 6433eac2dcSSimon Glass break; 6533eac2dcSSimon Glass } 6633eac2dcSSimon Glass } 6733eac2dcSSimon Glass 6833eac2dcSSimon Glass if (!c) { 6933eac2dcSSimon Glass printf("%" PRIu64 " Hz%s", freq, s); 7033eac2dcSSimon Glass return; 7133eac2dcSSimon Glass } 7233eac2dcSSimon Glass 7333eac2dcSSimon Glass f = do_div(freq, d); 7433eac2dcSSimon Glass 7533eac2dcSSimon Glass /* If there's a remainder, show the first few digits */ 7633eac2dcSSimon Glass if (f) { 7733eac2dcSSimon Glass m = f; 7833eac2dcSSimon Glass while (m > 1000) 7933eac2dcSSimon Glass m /= 10; 8033eac2dcSSimon Glass while (m && !(m % 10)) 8133eac2dcSSimon Glass m /= 10; 8233eac2dcSSimon Glass if (m >= 100) 8333eac2dcSSimon Glass m = (m / 10) + (m % 100 >= 50); 8433eac2dcSSimon Glass } 8533eac2dcSSimon Glass 86e9015b30SSuriyan Ramasami printf("%lu", (unsigned long) freq); 8733eac2dcSSimon Glass if (m) 8833eac2dcSSimon Glass printf(".%ld", m); 8933eac2dcSSimon Glass printf(" %cHz%s", c, s); 9033eac2dcSSimon Glass } 9133eac2dcSSimon Glass 92c6da9ae8SSimon Glass void print_size(uint64_t size, const char *s) 9378acc472SPeter Tyser { 9452dbac69STimur Tabi unsigned long m = 0, n; 95c6da9ae8SSimon Glass uint64_t f; 964b42c905STimur Tabi static const char names[] = {'E', 'P', 'T', 'G', 'M', 'K'}; 97f2d76ae4SNick Thompson unsigned long d = 10 * ARRAY_SIZE(names); 984b42c905STimur Tabi char c = 0; 994b42c905STimur Tabi unsigned int i; 10078acc472SPeter Tyser 101f2d76ae4SNick Thompson for (i = 0; i < ARRAY_SIZE(names); i++, d -= 10) { 102f2d76ae4SNick Thompson if (size >> d) { 1034b42c905STimur Tabi c = names[i]; 1044b42c905STimur Tabi break; 10578acc472SPeter Tyser } 10678acc472SPeter Tyser } 10778acc472SPeter Tyser 1084b42c905STimur Tabi if (!c) { 109c6da9ae8SSimon Glass printf("%" PRIu64 " Bytes%s", size, s); 1104b42c905STimur Tabi return; 1114b42c905STimur Tabi } 1124b42c905STimur Tabi 113f2d76ae4SNick Thompson n = size >> d; 114f2d76ae4SNick Thompson f = size & ((1ULL << d) - 1); 11578acc472SPeter Tyser 11678acc472SPeter Tyser /* If there's a remainder, deal with it */ 117f2d76ae4SNick Thompson if (f) { 118f2d76ae4SNick Thompson m = (10ULL * f + (1ULL << (d - 1))) >> d; 11978acc472SPeter Tyser 12078acc472SPeter Tyser if (m >= 10) { 12178acc472SPeter Tyser m -= 10; 12278acc472SPeter Tyser n += 1; 12378acc472SPeter Tyser } 12478acc472SPeter Tyser } 12578acc472SPeter Tyser 1264b42c905STimur Tabi printf ("%lu", n); 12778acc472SPeter Tyser if (m) { 12878acc472SPeter Tyser printf (".%ld", m); 12978acc472SPeter Tyser } 1304b42c905STimur Tabi printf (" %ciB%s", c, s); 13178acc472SPeter Tyser } 13278acc472SPeter Tyser 13378acc472SPeter Tyser #define MAX_LINE_LENGTH_BYTES (64) 13478acc472SPeter Tyser #define DEFAULT_LINE_LENGTH_BYTES (16) 135bda32ffcSSimon Glass int print_buffer(ulong addr, const void *data, uint width, uint count, 136bda32ffcSSimon Glass uint linelen) 13778acc472SPeter Tyser { 138150f7236SReinhard Meyer /* linebuf as a union causes proper alignment */ 139150f7236SReinhard Meyer union linebuf { 1404d1fd7f1SYork Sun #ifdef CONFIG_SYS_SUPPORT_64BIT_DATA 1414d1fd7f1SYork Sun uint64_t uq[MAX_LINE_LENGTH_BYTES/sizeof(uint64_t) + 1]; 1424d1fd7f1SYork Sun #endif 143150f7236SReinhard Meyer uint32_t ui[MAX_LINE_LENGTH_BYTES/sizeof(uint32_t) + 1]; 144150f7236SReinhard Meyer uint16_t us[MAX_LINE_LENGTH_BYTES/sizeof(uint16_t) + 1]; 145150f7236SReinhard Meyer uint8_t uc[MAX_LINE_LENGTH_BYTES/sizeof(uint8_t) + 1]; 146150f7236SReinhard Meyer } lb; 14778acc472SPeter Tyser int i; 1484d1fd7f1SYork Sun #ifdef CONFIG_SYS_SUPPORT_64BIT_DATA 14980402f34SHeiko Schocher uint64_t __maybe_unused x; 1504d1fd7f1SYork Sun #else 15180402f34SHeiko Schocher uint32_t __maybe_unused x; 1524d1fd7f1SYork Sun #endif 15378acc472SPeter Tyser 15478acc472SPeter Tyser if (linelen*width > MAX_LINE_LENGTH_BYTES) 15578acc472SPeter Tyser linelen = MAX_LINE_LENGTH_BYTES / width; 15678acc472SPeter Tyser if (linelen < 1) 15778acc472SPeter Tyser linelen = DEFAULT_LINE_LENGTH_BYTES / width; 15878acc472SPeter Tyser 15978acc472SPeter Tyser while (count) { 160efd7c114SAndreas Bießmann uint thislinelen = linelen; 16178acc472SPeter Tyser printf("%08lx:", addr); 16278acc472SPeter Tyser 16378acc472SPeter Tyser /* check for overflow condition */ 164efd7c114SAndreas Bießmann if (count < thislinelen) 165efd7c114SAndreas Bießmann thislinelen = count; 16678acc472SPeter Tyser 16778acc472SPeter Tyser /* Copy from memory into linebuf and print hex values */ 168efd7c114SAndreas Bießmann for (i = 0; i < thislinelen; i++) { 16964419e47SMike Frysinger if (width == 4) 170150f7236SReinhard Meyer x = lb.ui[i] = *(volatile uint32_t *)data; 1714d1fd7f1SYork Sun #ifdef CONFIG_SYS_SUPPORT_64BIT_DATA 1724d1fd7f1SYork Sun else if (width == 8) 1734d1fd7f1SYork Sun x = lb.uq[i] = *(volatile uint64_t *)data; 1744d1fd7f1SYork Sun #endif 17564419e47SMike Frysinger else if (width == 2) 176150f7236SReinhard Meyer x = lb.us[i] = *(volatile uint16_t *)data; 17764419e47SMike Frysinger else 178150f7236SReinhard Meyer x = lb.uc[i] = *(volatile uint8_t *)data; 1794d1fd7f1SYork Sun #ifdef 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