178acc472SPeter Tyser /* 278acc472SPeter Tyser * (C) Copyright 2000-2002 378acc472SPeter Tyser * Wolfgang Denk, DENX Software Engineering, wd@denx.de. 478acc472SPeter Tyser * 578acc472SPeter Tyser * See file CREDITS for list of people who contributed to this 678acc472SPeter Tyser * project. 778acc472SPeter Tyser * 878acc472SPeter Tyser * This program is free software; you can redistribute it and/or 978acc472SPeter Tyser * modify it under the terms of the GNU General Public License as 1078acc472SPeter Tyser * published by the Free Software Foundation; either version 2 of 1178acc472SPeter Tyser * the License, or (at your option) any later version. 1278acc472SPeter Tyser * 1378acc472SPeter Tyser * This program is distributed in the hope that it will be useful, 1478acc472SPeter Tyser * but WITHOUT ANY WARRANTY; without even the implied warranty of 1578acc472SPeter Tyser * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the 1678acc472SPeter Tyser * GNU General Public License for more details. 1778acc472SPeter Tyser * 1878acc472SPeter Tyser * You should have received a copy of the GNU General Public License 1978acc472SPeter Tyser * along with this program; if not, write to the Free Software 2078acc472SPeter Tyser * Foundation, Inc., 59 Temple Place, Suite 330, Boston, 2178acc472SPeter Tyser * MA 02111-1307 USA 2278acc472SPeter Tyser */ 2378acc472SPeter Tyser 2478acc472SPeter Tyser #include <config.h> 2578acc472SPeter Tyser #include <common.h> 2678acc472SPeter Tyser #include <linux/ctype.h> 2778acc472SPeter Tyser #include <asm/io.h> 2878acc472SPeter Tyser 2978acc472SPeter Tyser int display_options (void) 3078acc472SPeter Tyser { 3178acc472SPeter Tyser extern char version_string[]; 3278acc472SPeter Tyser 3378acc472SPeter Tyser #if defined(BUILD_TAG) 3478acc472SPeter Tyser printf ("\n\n%s, Build: %s\n\n", version_string, BUILD_TAG); 3578acc472SPeter Tyser #else 3678acc472SPeter Tyser printf ("\n\n%s\n\n", version_string); 3778acc472SPeter Tyser #endif 3878acc472SPeter Tyser return 0; 3978acc472SPeter Tyser } 4078acc472SPeter Tyser 4178acc472SPeter Tyser /* 424b42c905STimur Tabi * print sizes as "xxx KiB", "xxx.y KiB", "xxx MiB", "xxx.y MiB", 434b42c905STimur Tabi * xxx GiB, xxx.y GiB, etc as needed; allow for optional trailing string 4478acc472SPeter Tyser * (like "\n") 4578acc472SPeter Tyser */ 464b42c905STimur Tabi void print_size(unsigned long long size, const char *s) 4778acc472SPeter Tyser { 4852dbac69STimur Tabi unsigned long m = 0, n; 49*f2d76ae4SNick Thompson unsigned long long f; 504b42c905STimur Tabi static const char names[] = {'E', 'P', 'T', 'G', 'M', 'K'}; 51*f2d76ae4SNick Thompson unsigned long d = 10 * ARRAY_SIZE(names); 524b42c905STimur Tabi char c = 0; 534b42c905STimur Tabi unsigned int i; 5478acc472SPeter Tyser 55*f2d76ae4SNick Thompson for (i = 0; i < ARRAY_SIZE(names); i++, d -= 10) { 56*f2d76ae4SNick Thompson if (size >> d) { 574b42c905STimur Tabi c = names[i]; 584b42c905STimur Tabi break; 5978acc472SPeter Tyser } 6078acc472SPeter Tyser } 6178acc472SPeter Tyser 624b42c905STimur Tabi if (!c) { 634b42c905STimur Tabi printf("%llu Bytes%s", size, s); 644b42c905STimur Tabi return; 654b42c905STimur Tabi } 664b42c905STimur Tabi 67*f2d76ae4SNick Thompson n = size >> d; 68*f2d76ae4SNick Thompson f = size & ((1ULL << d) - 1); 6978acc472SPeter Tyser 7078acc472SPeter Tyser /* If there's a remainder, deal with it */ 71*f2d76ae4SNick Thompson if (f) { 72*f2d76ae4SNick Thompson m = (10ULL * f + (1ULL << (d - 1))) >> d; 7378acc472SPeter Tyser 7478acc472SPeter Tyser if (m >= 10) { 7578acc472SPeter Tyser m -= 10; 7678acc472SPeter Tyser n += 1; 7778acc472SPeter Tyser } 7878acc472SPeter Tyser } 7978acc472SPeter Tyser 804b42c905STimur Tabi printf ("%lu", n); 8178acc472SPeter Tyser if (m) { 8278acc472SPeter Tyser printf (".%ld", m); 8378acc472SPeter Tyser } 844b42c905STimur Tabi printf (" %ciB%s", c, s); 8578acc472SPeter Tyser } 8678acc472SPeter Tyser 8778acc472SPeter Tyser /* 8878acc472SPeter Tyser * Print data buffer in hex and ascii form to the terminal. 8978acc472SPeter Tyser * 9078acc472SPeter Tyser * data reads are buffered so that each memory address is only read once. 9178acc472SPeter Tyser * Useful when displaying the contents of volatile registers. 9278acc472SPeter Tyser * 9378acc472SPeter Tyser * parameters: 9478acc472SPeter Tyser * addr: Starting address to display at start of line 9578acc472SPeter Tyser * data: pointer to data buffer 9678acc472SPeter Tyser * width: data value width. May be 1, 2, or 4. 9778acc472SPeter Tyser * count: number of values to display 9878acc472SPeter Tyser * linelen: Number of values to print per line; specify 0 for default length 9978acc472SPeter Tyser */ 10078acc472SPeter Tyser #define MAX_LINE_LENGTH_BYTES (64) 10178acc472SPeter Tyser #define DEFAULT_LINE_LENGTH_BYTES (16) 10278acc472SPeter Tyser int print_buffer (ulong addr, void* data, uint width, uint count, uint linelen) 10378acc472SPeter Tyser { 10478acc472SPeter Tyser uint8_t linebuf[MAX_LINE_LENGTH_BYTES]; 10578acc472SPeter Tyser uint32_t *uip = (void*)linebuf; 10678acc472SPeter Tyser uint16_t *usp = (void*)linebuf; 10778acc472SPeter Tyser uint8_t *ucp = (void*)linebuf; 10878acc472SPeter Tyser int i; 10978acc472SPeter Tyser 11078acc472SPeter Tyser if (linelen*width > MAX_LINE_LENGTH_BYTES) 11178acc472SPeter Tyser linelen = MAX_LINE_LENGTH_BYTES / width; 11278acc472SPeter Tyser if (linelen < 1) 11378acc472SPeter Tyser linelen = DEFAULT_LINE_LENGTH_BYTES / width; 11478acc472SPeter Tyser 11578acc472SPeter Tyser while (count) { 11678acc472SPeter Tyser printf("%08lx:", addr); 11778acc472SPeter Tyser 11878acc472SPeter Tyser /* check for overflow condition */ 11978acc472SPeter Tyser if (count < linelen) 12078acc472SPeter Tyser linelen = count; 12178acc472SPeter Tyser 12278acc472SPeter Tyser /* Copy from memory into linebuf and print hex values */ 12378acc472SPeter Tyser for (i = 0; i < linelen; i++) { 12478acc472SPeter Tyser if (width == 4) { 12578acc472SPeter Tyser uip[i] = *(volatile uint32_t *)data; 12678acc472SPeter Tyser printf(" %08x", uip[i]); 12778acc472SPeter Tyser } else if (width == 2) { 12878acc472SPeter Tyser usp[i] = *(volatile uint16_t *)data; 12978acc472SPeter Tyser printf(" %04x", usp[i]); 13078acc472SPeter Tyser } else { 13178acc472SPeter Tyser ucp[i] = *(volatile uint8_t *)data; 13278acc472SPeter Tyser printf(" %02x", ucp[i]); 13378acc472SPeter Tyser } 13478acc472SPeter Tyser data += width; 13578acc472SPeter Tyser } 13678acc472SPeter Tyser 13778acc472SPeter Tyser /* Print data in ASCII characters */ 13878acc472SPeter Tyser puts(" "); 13978acc472SPeter Tyser for (i = 0; i < linelen * width; i++) 14078acc472SPeter Tyser putc(isprint(ucp[i]) && (ucp[i] < 0x80) ? ucp[i] : '.'); 14178acc472SPeter Tyser putc ('\n'); 14278acc472SPeter Tyser 14378acc472SPeter Tyser /* update references */ 14478acc472SPeter Tyser addr += linelen * width; 14578acc472SPeter Tyser count -= linelen; 14678acc472SPeter Tyser 14778acc472SPeter Tyser if (ctrlc()) 14878acc472SPeter Tyser return -1; 14978acc472SPeter Tyser } 15078acc472SPeter Tyser 15178acc472SPeter Tyser return 0; 15278acc472SPeter Tyser } 153