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 /* 42*4b42c905STimur Tabi * print sizes as "xxx KiB", "xxx.y KiB", "xxx MiB", "xxx.y MiB", 43*4b42c905STimur Tabi * xxx GiB, xxx.y GiB, etc as needed; allow for optional trailing string 4478acc472SPeter Tyser * (like "\n") 4578acc472SPeter Tyser */ 46*4b42c905STimur Tabi void print_size(unsigned long long size, const char *s) 4778acc472SPeter Tyser { 4852dbac69STimur Tabi unsigned long m = 0, n; 49*4b42c905STimur Tabi static const char names[] = {'E', 'P', 'T', 'G', 'M', 'K'}; 50*4b42c905STimur Tabi unsigned long long d = 1ULL << (10 * ARRAY_SIZE(names)); 51*4b42c905STimur Tabi char c = 0; 52*4b42c905STimur Tabi unsigned int i; 5378acc472SPeter Tyser 54*4b42c905STimur Tabi for (i = 0; i < ARRAY_SIZE(names); i++, d >>= 10) { 55*4b42c905STimur Tabi if (size >= d) { 56*4b42c905STimur Tabi c = names[i]; 57*4b42c905STimur Tabi break; 5878acc472SPeter Tyser } 5978acc472SPeter Tyser } 6078acc472SPeter Tyser 61*4b42c905STimur Tabi if (!c) { 62*4b42c905STimur Tabi printf("%llu Bytes%s", size, s); 63*4b42c905STimur Tabi return; 64*4b42c905STimur Tabi } 65*4b42c905STimur Tabi 6678acc472SPeter Tyser n = size / d; 6778acc472SPeter Tyser 6878acc472SPeter Tyser /* If there's a remainder, deal with it */ 6978acc472SPeter Tyser if(size % d) { 7078acc472SPeter Tyser m = (10 * (size - (n * d)) + (d / 2) ) / d; 7178acc472SPeter Tyser 7278acc472SPeter Tyser if (m >= 10) { 7378acc472SPeter Tyser m -= 10; 7478acc472SPeter Tyser n += 1; 7578acc472SPeter Tyser } 7678acc472SPeter Tyser } 7778acc472SPeter Tyser 78*4b42c905STimur Tabi printf ("%lu", n); 7978acc472SPeter Tyser if (m) { 8078acc472SPeter Tyser printf (".%ld", m); 8178acc472SPeter Tyser } 82*4b42c905STimur Tabi printf (" %ciB%s", c, s); 8378acc472SPeter Tyser } 8478acc472SPeter Tyser 8578acc472SPeter Tyser /* 8678acc472SPeter Tyser * Print data buffer in hex and ascii form to the terminal. 8778acc472SPeter Tyser * 8878acc472SPeter Tyser * data reads are buffered so that each memory address is only read once. 8978acc472SPeter Tyser * Useful when displaying the contents of volatile registers. 9078acc472SPeter Tyser * 9178acc472SPeter Tyser * parameters: 9278acc472SPeter Tyser * addr: Starting address to display at start of line 9378acc472SPeter Tyser * data: pointer to data buffer 9478acc472SPeter Tyser * width: data value width. May be 1, 2, or 4. 9578acc472SPeter Tyser * count: number of values to display 9678acc472SPeter Tyser * linelen: Number of values to print per line; specify 0 for default length 9778acc472SPeter Tyser */ 9878acc472SPeter Tyser #define MAX_LINE_LENGTH_BYTES (64) 9978acc472SPeter Tyser #define DEFAULT_LINE_LENGTH_BYTES (16) 10078acc472SPeter Tyser int print_buffer (ulong addr, void* data, uint width, uint count, uint linelen) 10178acc472SPeter Tyser { 10278acc472SPeter Tyser uint8_t linebuf[MAX_LINE_LENGTH_BYTES]; 10378acc472SPeter Tyser uint32_t *uip = (void*)linebuf; 10478acc472SPeter Tyser uint16_t *usp = (void*)linebuf; 10578acc472SPeter Tyser uint8_t *ucp = (void*)linebuf; 10678acc472SPeter Tyser int i; 10778acc472SPeter Tyser 10878acc472SPeter Tyser if (linelen*width > MAX_LINE_LENGTH_BYTES) 10978acc472SPeter Tyser linelen = MAX_LINE_LENGTH_BYTES / width; 11078acc472SPeter Tyser if (linelen < 1) 11178acc472SPeter Tyser linelen = DEFAULT_LINE_LENGTH_BYTES / width; 11278acc472SPeter Tyser 11378acc472SPeter Tyser while (count) { 11478acc472SPeter Tyser printf("%08lx:", addr); 11578acc472SPeter Tyser 11678acc472SPeter Tyser /* check for overflow condition */ 11778acc472SPeter Tyser if (count < linelen) 11878acc472SPeter Tyser linelen = count; 11978acc472SPeter Tyser 12078acc472SPeter Tyser /* Copy from memory into linebuf and print hex values */ 12178acc472SPeter Tyser for (i = 0; i < linelen; i++) { 12278acc472SPeter Tyser if (width == 4) { 12378acc472SPeter Tyser uip[i] = *(volatile uint32_t *)data; 12478acc472SPeter Tyser printf(" %08x", uip[i]); 12578acc472SPeter Tyser } else if (width == 2) { 12678acc472SPeter Tyser usp[i] = *(volatile uint16_t *)data; 12778acc472SPeter Tyser printf(" %04x", usp[i]); 12878acc472SPeter Tyser } else { 12978acc472SPeter Tyser ucp[i] = *(volatile uint8_t *)data; 13078acc472SPeter Tyser printf(" %02x", ucp[i]); 13178acc472SPeter Tyser } 13278acc472SPeter Tyser data += width; 13378acc472SPeter Tyser } 13478acc472SPeter Tyser 13578acc472SPeter Tyser /* Print data in ASCII characters */ 13678acc472SPeter Tyser puts(" "); 13778acc472SPeter Tyser for (i = 0; i < linelen * width; i++) 13878acc472SPeter Tyser putc(isprint(ucp[i]) && (ucp[i] < 0x80) ? ucp[i] : '.'); 13978acc472SPeter Tyser putc ('\n'); 14078acc472SPeter Tyser 14178acc472SPeter Tyser /* update references */ 14278acc472SPeter Tyser addr += linelen * width; 14378acc472SPeter Tyser count -= linelen; 14478acc472SPeter Tyser 14578acc472SPeter Tyser if (ctrlc()) 14678acc472SPeter Tyser return -1; 14778acc472SPeter Tyser } 14878acc472SPeter Tyser 14978acc472SPeter Tyser return 0; 15078acc472SPeter Tyser } 151