/* * (C) Copyright 2000 * Wolfgang Denk, DENX Software Engineering, wd@denx.de. * * See file CREDITS for list of people who contributed to this * project. * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License as * published by the Free Software Foundation; either version 2 of * the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, * MA 02111-1307 USA */ /* * Memory Functions * * Copied from FADS ROM, Dan Malek (dmalek@jlc.net) */ #include #include #include #include #include #include #include #include #include #ifdef CMD_MEM_DEBUG #define PRINTF(fmt,args...) printf (fmt ,##args) #else #define PRINTF(fmt,args...) #endif #define RW_BUF_SIZE (ulong)4096 static char *rw_buf; /* Memory Display * * Syntax: * md{.b, .w, .l} {addr} {len} */ #define DISP_LINE_LEN 16 int memory_display(char *addr, ulong offs, ulong nbytes, int size) { ulong linebytes, i; u_char *cp; /* Print the lines. * * We buffer all read data, so we can make sure data is read only * once, and all accesses are with the specified bus width. */ do { char linebuf[DISP_LINE_LEN]; uint *uip = (uint *)linebuf; ushort *usp = (ushort *)linebuf; u_char *ucp = (u_char *)linebuf; printf("%08lx:", offs); linebytes = (nbytes>DISP_LINE_LEN)?DISP_LINE_LEN:nbytes; for (i=0; i 0x7e)) putchar('.'); else printf("%c", *cp); cp++; } putchar('\n'); nbytes -= linebytes; if (ctrlc()) { return -EINTR; } } while (nbytes > 0); return 0; } static int mem_parse_options(int argc, char *argv[], char *optstr, int *mode, char **filename) { int opt; getopt_reset(); while((opt = getopt(argc, argv, optstr)) > 0) { switch(opt) { case 'b': *mode = O_RWSIZE_1; break; case 'w': *mode = O_RWSIZE_2; break; case 'l': *mode = O_RWSIZE_4; break; case 'f': *filename = optarg; break; default: return -1; } } return 0; } static int do_mem_md ( cmd_tbl_t *cmdtp, int argc, char *argv[]) { ulong start = 0, size = 0x100; int r, now; int ret = 0; int fd; char *filename = "/dev/mem"; int mode = O_RWSIZE_4; errno = 0; if (mem_parse_options(argc, argv, "bwlf:", &mode, &filename) < 0) return 1; if (optind < argc) { parse_area_spec(argv[optind], &start, &size); if (size == ~0) size = 0x100; } fd = open(filename, mode | O_RDONLY); if (fd < 0) { perror("open"); return 1; } if (lseek(fd, start, SEEK_SET)) { perror("lseek"); goto out; } do { now = min(size, RW_BUF_SIZE); r = read(fd, rw_buf, now); if (r < 0) { perror("read"); goto out; } if (!r) goto out; if ((ret = memory_display(rw_buf, start, r, mode >> O_RWSIZE_SHIFT))) goto out; start += r; size -= r; } while (size); out: close(fd); return errno; } static __maybe_unused char cmd_md_help[] = "Usage md [OPTIONS] \n" "display (hexdump) a memory region.\n" "options:\n" " -f display file (default /dev/mem)\n" " -b output in bytes\n" " -w output in halfwords (16bit)\n" " -l output in words (32bit)\n" "\n" "Memory regions:\n" "Memory regions can be specified in two different forms: start+size\n" "or start-end, If is ommitted it defaults to 0. If end is ommited it\n" "defaults to the end of the device, except for interactive commands like md\n" "and mw for which it defaults to 0x100.\n" "Sizes can be specified as decimal, or if prefixed with 0x as hexadecimal.\n" "an optional suffix of k, M or G is for kibibytes, Megabytes or Gigabytes,\n" "respectively\n"; U_BOOT_CMD_START(md) .maxargs = CONFIG_MAXARGS, .cmd = do_mem_md, .usage = "memory display", U_BOOT_CMD_HELP(cmd_md_help) U_BOOT_CMD_END static int do_mem_mw ( cmd_tbl_t *cmdtp, int argc, char *argv[]) { int ret = 0; int fd; char *filename = "/dev/mem"; int mode = O_RWSIZE_4; ulong adr; errno = 0; if (mem_parse_options(argc, argv, "bwlf:", &mode, &filename) < 0) return 1; if (optind + 1 >= argc) { printf ("Usage:\n%s\n", cmdtp->usage); return 1; } fd = open(filename, mode | O_WRONLY); if (fd < 0) { perror("open"); return 1; } adr = strtoul_suffix(argv[optind++], NULL, 0); if (lseek(fd, adr, SEEK_SET)) { perror("lseek"); goto out; } while (optind < argc) { u8 val8; u16 val16; u32 val32; switch (mode) { case O_RWSIZE_1: val8 = simple_strtoul(argv[optind], NULL, 0); ret = write(fd, &val8, 1); break; case O_RWSIZE_2: val16 = simple_strtoul(argv[optind], NULL, 0); ret = write(fd, &val16, 2); break; case O_RWSIZE_4: val32 = simple_strtoul(argv[optind], NULL, 0); ret = write(fd, &val32, 4); break; } if (ret < 0) { perror("write"); break; } optind++; } out: close(fd); return errno; } static __maybe_unused char cmd_mw_help[] = "Usage: mw [OPTIONS] \n" "Write value(s) to the specifies region.\n" "see 'help md' for supported options.\n"; U_BOOT_CMD_START(mw) .maxargs = CONFIG_MAXARGS, .cmd = do_mem_mw, .usage = "memory write (fill)", U_BOOT_CMD_HELP(cmd_mw_help) U_BOOT_CMD_END static int do_mem_cmp (cmd_tbl_t *cmdtp, int argc, char *argv[]) { ulong addr1, addr2, count, ngood; int size; int mode = O_RWSIZE_4; if (mem_parse_options(argc, argv, "bwl", &mode, NULL) < 0) return 1; if (optind + 3 != argc) { printf ("Usage:\n%s\n", cmdtp->usage); return 1; } addr1 = simple_strtoul(argv[optind++], NULL, 0); addr2 = simple_strtoul(argv[optind++], NULL, 0); size = (mode & O_RWSIZE_MASK) >> O_RWSIZE_SHIFT; count = simple_strtoul(argv[optind], NULL, 0) / size; while (count-- > 0) { if (size == 4) { ulong word1 = *(ulong *)addr1; ulong word2 = *(ulong *)addr2; if (word1 != word2) { printf("word at 0x%08lx (0x%08lx) " "!= word at 0x%08lx (0x%08lx)\n", addr1, word1, addr2, word2); return 1; } } else if (size == 2) { ushort hword1 = *(ushort *)addr1; ushort hword2 = *(ushort *)addr2; if (hword1 != hword2) { printf("halfword at 0x%08lx (0x%04x) " "!= halfword at 0x%08lx (0x%04x)\n", addr1, hword1, addr2, hword2); return 1; } } else { u_char byte1 = *(u_char *)addr1; u_char byte2 = *(u_char *)addr2; if (byte1 != byte2) { printf("byte at 0x%08lx (0x%02x) " "!= byte at 0x%08lx (0x%02x)\n", addr1, byte1, addr2, byte2); return 1; } } ngood++; addr1 += size; addr2 += size; } printf("OK\n"); return 0; } static __maybe_unused char cmd_memcmp_help[] = "Usage: memcmp [OPTIONS] \n" "\n" "options:\n" " -b, -w, -l use byte, halfword, or word accesses\n" "\n" "Compare memory regions specified with addr1 and addr2\n" "of size bytes.\n"; U_BOOT_CMD_START(memcmp) .maxargs = CONFIG_MAXARGS, .cmd = do_mem_cmp, .usage = "memory compare", U_BOOT_CMD_HELP(cmd_memcmp_help) U_BOOT_CMD_END static int do_mem_cp (cmd_tbl_t *cmdtp, int argc, char *argv[]) { ulong count; ulong dest, src; int mode = O_RWSIZE_4; int size; if (mem_parse_options(argc, argv, "bwl", &mode, NULL) < 0) return 1; if (optind + 3 != argc) { printf ("Usage:\n%s\n", cmdtp->usage); return 1; } src = simple_strtoul(argv[optind++], NULL, 0); dest = simple_strtoul(argv[optind++], NULL, 0); size = (mode & O_RWSIZE_MASK) >> O_RWSIZE_SHIFT; count = simple_strtoul(argv[optind], NULL, 0) / size; while (count-- > 0) { if (size == 4) *((ulong *)dest) = *((ulong *)src); else if (size == 2) *((ushort *)dest) = *((ushort *)src); else *((u_char *)dest) = *((u_char *)src); src += size; dest += size; } return 0; } static __maybe_unused char cmd_memcpy_help[] = "Usage: memcpy [OPTIONS] \n" "\n" "options:\n" " -b, -w, -l use byte, halfword, or word accesses\n" "\n" "Copy memory at of bytes to \n"; U_BOOT_CMD_START(memcpy) .maxargs = CONFIG_MAXARGS, .cmd = do_mem_cp, .usage = "memory copy", U_BOOT_CMD_HELP(cmd_memcpy_help) U_BOOT_CMD_END static struct device_d mem_dev = { .name = "mem", .id = "mem", .map_base = 0, .size = ~0, /* FIXME: should be 0x100000000, ahem... */ }; static struct driver_d mem_drv = { .name = "mem", .probe = dummy_probe, .read = mem_read, .write = mem_write, }; static struct driver_d ram_drv = { .name = "ram", .probe = dummy_probe, .read = mem_read, .write = mem_write, .type = DEVICE_TYPE_DRAM, }; static struct driver_d rom_drv = { .name = "rom", .probe = dummy_probe, .read = mem_read, }; static int mem_init(void) { rw_buf = malloc(RW_BUF_SIZE); if(!rw_buf) { printf("%s: Out of memory\n", __FUNCTION__); return -1; } register_device(&mem_dev); register_driver(&mem_drv); register_driver(&ram_drv); register_driver(&rom_drv); return 0; } device_initcall(mem_init);