1*2e192b24SSimon Glass /* 2*2e192b24SSimon Glass * Command for accessing SPI flash. 3*2e192b24SSimon Glass * 4*2e192b24SSimon Glass * Copyright (C) 2008 Atmel Corporation 5*2e192b24SSimon Glass * 6*2e192b24SSimon Glass * SPDX-License-Identifier: GPL-2.0+ 7*2e192b24SSimon Glass */ 8*2e192b24SSimon Glass 9*2e192b24SSimon Glass #include <common.h> 10*2e192b24SSimon Glass #include <div64.h> 11*2e192b24SSimon Glass #include <dm.h> 12*2e192b24SSimon Glass #include <malloc.h> 13*2e192b24SSimon Glass #include <mapmem.h> 14*2e192b24SSimon Glass #include <spi.h> 15*2e192b24SSimon Glass #include <spi_flash.h> 16*2e192b24SSimon Glass #include <jffs2/jffs2.h> 17*2e192b24SSimon Glass #include <linux/mtd/mtd.h> 18*2e192b24SSimon Glass 19*2e192b24SSimon Glass #include <asm/io.h> 20*2e192b24SSimon Glass #include <dm/device-internal.h> 21*2e192b24SSimon Glass 22*2e192b24SSimon Glass static struct spi_flash *flash; 23*2e192b24SSimon Glass 24*2e192b24SSimon Glass /* 25*2e192b24SSimon Glass * This function computes the length argument for the erase command. 26*2e192b24SSimon Glass * The length on which the command is to operate can be given in two forms: 27*2e192b24SSimon Glass * 1. <cmd> offset len - operate on <'offset', 'len') 28*2e192b24SSimon Glass * 2. <cmd> offset +len - operate on <'offset', 'round_up(len)') 29*2e192b24SSimon Glass * If the second form is used and the length doesn't fall on the 30*2e192b24SSimon Glass * sector boundary, than it will be adjusted to the next sector boundary. 31*2e192b24SSimon Glass * If it isn't in the flash, the function will fail (return -1). 32*2e192b24SSimon Glass * Input: 33*2e192b24SSimon Glass * arg: length specification (i.e. both command arguments) 34*2e192b24SSimon Glass * Output: 35*2e192b24SSimon Glass * len: computed length for operation 36*2e192b24SSimon Glass * Return: 37*2e192b24SSimon Glass * 1: success 38*2e192b24SSimon Glass * -1: failure (bad format, bad address). 39*2e192b24SSimon Glass */ 40*2e192b24SSimon Glass static int sf_parse_len_arg(char *arg, ulong *len) 41*2e192b24SSimon Glass { 42*2e192b24SSimon Glass char *ep; 43*2e192b24SSimon Glass char round_up_len; /* indicates if the "+length" form used */ 44*2e192b24SSimon Glass ulong len_arg; 45*2e192b24SSimon Glass 46*2e192b24SSimon Glass round_up_len = 0; 47*2e192b24SSimon Glass if (*arg == '+') { 48*2e192b24SSimon Glass round_up_len = 1; 49*2e192b24SSimon Glass ++arg; 50*2e192b24SSimon Glass } 51*2e192b24SSimon Glass 52*2e192b24SSimon Glass len_arg = simple_strtoul(arg, &ep, 16); 53*2e192b24SSimon Glass if (ep == arg || *ep != '\0') 54*2e192b24SSimon Glass return -1; 55*2e192b24SSimon Glass 56*2e192b24SSimon Glass if (round_up_len && flash->sector_size > 0) 57*2e192b24SSimon Glass *len = ROUND(len_arg, flash->sector_size); 58*2e192b24SSimon Glass else 59*2e192b24SSimon Glass *len = len_arg; 60*2e192b24SSimon Glass 61*2e192b24SSimon Glass return 1; 62*2e192b24SSimon Glass } 63*2e192b24SSimon Glass 64*2e192b24SSimon Glass /** 65*2e192b24SSimon Glass * This function takes a byte length and a delta unit of time to compute the 66*2e192b24SSimon Glass * approximate bytes per second 67*2e192b24SSimon Glass * 68*2e192b24SSimon Glass * @param len amount of bytes currently processed 69*2e192b24SSimon Glass * @param start_ms start time of processing in ms 70*2e192b24SSimon Glass * @return bytes per second if OK, 0 on error 71*2e192b24SSimon Glass */ 72*2e192b24SSimon Glass static ulong bytes_per_second(unsigned int len, ulong start_ms) 73*2e192b24SSimon Glass { 74*2e192b24SSimon Glass /* less accurate but avoids overflow */ 75*2e192b24SSimon Glass if (len >= ((unsigned int) -1) / 1024) 76*2e192b24SSimon Glass return len / (max(get_timer(start_ms) / 1024, 1UL)); 77*2e192b24SSimon Glass else 78*2e192b24SSimon Glass return 1024 * len / max(get_timer(start_ms), 1UL); 79*2e192b24SSimon Glass } 80*2e192b24SSimon Glass 81*2e192b24SSimon Glass static int do_spi_flash_probe(int argc, char * const argv[]) 82*2e192b24SSimon Glass { 83*2e192b24SSimon Glass unsigned int bus = CONFIG_SF_DEFAULT_BUS; 84*2e192b24SSimon Glass unsigned int cs = CONFIG_SF_DEFAULT_CS; 85*2e192b24SSimon Glass unsigned int speed = CONFIG_SF_DEFAULT_SPEED; 86*2e192b24SSimon Glass unsigned int mode = CONFIG_SF_DEFAULT_MODE; 87*2e192b24SSimon Glass char *endp; 88*2e192b24SSimon Glass #ifdef CONFIG_DM_SPI_FLASH 89*2e192b24SSimon Glass struct udevice *new, *bus_dev; 90*2e192b24SSimon Glass int ret; 91*2e192b24SSimon Glass #else 92*2e192b24SSimon Glass struct spi_flash *new; 93*2e192b24SSimon Glass #endif 94*2e192b24SSimon Glass 95*2e192b24SSimon Glass if (argc >= 2) { 96*2e192b24SSimon Glass cs = simple_strtoul(argv[1], &endp, 0); 97*2e192b24SSimon Glass if (*argv[1] == 0 || (*endp != 0 && *endp != ':')) 98*2e192b24SSimon Glass return -1; 99*2e192b24SSimon Glass if (*endp == ':') { 100*2e192b24SSimon Glass if (endp[1] == 0) 101*2e192b24SSimon Glass return -1; 102*2e192b24SSimon Glass 103*2e192b24SSimon Glass bus = cs; 104*2e192b24SSimon Glass cs = simple_strtoul(endp + 1, &endp, 0); 105*2e192b24SSimon Glass if (*endp != 0) 106*2e192b24SSimon Glass return -1; 107*2e192b24SSimon Glass } 108*2e192b24SSimon Glass } 109*2e192b24SSimon Glass 110*2e192b24SSimon Glass if (argc >= 3) { 111*2e192b24SSimon Glass speed = simple_strtoul(argv[2], &endp, 0); 112*2e192b24SSimon Glass if (*argv[2] == 0 || *endp != 0) 113*2e192b24SSimon Glass return -1; 114*2e192b24SSimon Glass } 115*2e192b24SSimon Glass if (argc >= 4) { 116*2e192b24SSimon Glass mode = simple_strtoul(argv[3], &endp, 16); 117*2e192b24SSimon Glass if (*argv[3] == 0 || *endp != 0) 118*2e192b24SSimon Glass return -1; 119*2e192b24SSimon Glass } 120*2e192b24SSimon Glass 121*2e192b24SSimon Glass #ifdef CONFIG_DM_SPI_FLASH 122*2e192b24SSimon Glass /* Remove the old device, otherwise probe will just be a nop */ 123*2e192b24SSimon Glass ret = spi_find_bus_and_cs(bus, cs, &bus_dev, &new); 124*2e192b24SSimon Glass if (!ret) { 125*2e192b24SSimon Glass device_remove(new); 126*2e192b24SSimon Glass device_unbind(new); 127*2e192b24SSimon Glass } 128*2e192b24SSimon Glass flash = NULL; 129*2e192b24SSimon Glass ret = spi_flash_probe_bus_cs(bus, cs, speed, mode, &new); 130*2e192b24SSimon Glass if (ret) { 131*2e192b24SSimon Glass printf("Failed to initialize SPI flash at %u:%u (error %d)\n", 132*2e192b24SSimon Glass bus, cs, ret); 133*2e192b24SSimon Glass return 1; 134*2e192b24SSimon Glass } 135*2e192b24SSimon Glass 136*2e192b24SSimon Glass flash = dev_get_uclass_priv(new); 137*2e192b24SSimon Glass #else 138*2e192b24SSimon Glass if (flash) 139*2e192b24SSimon Glass spi_flash_free(flash); 140*2e192b24SSimon Glass 141*2e192b24SSimon Glass new = spi_flash_probe(bus, cs, speed, mode); 142*2e192b24SSimon Glass flash = new; 143*2e192b24SSimon Glass 144*2e192b24SSimon Glass if (!new) { 145*2e192b24SSimon Glass printf("Failed to initialize SPI flash at %u:%u\n", bus, cs); 146*2e192b24SSimon Glass return 1; 147*2e192b24SSimon Glass } 148*2e192b24SSimon Glass 149*2e192b24SSimon Glass flash = new; 150*2e192b24SSimon Glass #endif 151*2e192b24SSimon Glass 152*2e192b24SSimon Glass return 0; 153*2e192b24SSimon Glass } 154*2e192b24SSimon Glass 155*2e192b24SSimon Glass /** 156*2e192b24SSimon Glass * Write a block of data to SPI flash, first checking if it is different from 157*2e192b24SSimon Glass * what is already there. 158*2e192b24SSimon Glass * 159*2e192b24SSimon Glass * If the data being written is the same, then *skipped is incremented by len. 160*2e192b24SSimon Glass * 161*2e192b24SSimon Glass * @param flash flash context pointer 162*2e192b24SSimon Glass * @param offset flash offset to write 163*2e192b24SSimon Glass * @param len number of bytes to write 164*2e192b24SSimon Glass * @param buf buffer to write from 165*2e192b24SSimon Glass * @param cmp_buf read buffer to use to compare data 166*2e192b24SSimon Glass * @param skipped Count of skipped data (incremented by this function) 167*2e192b24SSimon Glass * @return NULL if OK, else a string containing the stage which failed 168*2e192b24SSimon Glass */ 169*2e192b24SSimon Glass static const char *spi_flash_update_block(struct spi_flash *flash, u32 offset, 170*2e192b24SSimon Glass size_t len, const char *buf, char *cmp_buf, size_t *skipped) 171*2e192b24SSimon Glass { 172*2e192b24SSimon Glass char *ptr = (char *)buf; 173*2e192b24SSimon Glass 174*2e192b24SSimon Glass debug("offset=%#x, sector_size=%#x, len=%#zx\n", 175*2e192b24SSimon Glass offset, flash->sector_size, len); 176*2e192b24SSimon Glass /* Read the entire sector so to allow for rewriting */ 177*2e192b24SSimon Glass if (spi_flash_read(flash, offset, flash->sector_size, cmp_buf)) 178*2e192b24SSimon Glass return "read"; 179*2e192b24SSimon Glass /* Compare only what is meaningful (len) */ 180*2e192b24SSimon Glass if (memcmp(cmp_buf, buf, len) == 0) { 181*2e192b24SSimon Glass debug("Skip region %x size %zx: no change\n", 182*2e192b24SSimon Glass offset, len); 183*2e192b24SSimon Glass *skipped += len; 184*2e192b24SSimon Glass return NULL; 185*2e192b24SSimon Glass } 186*2e192b24SSimon Glass /* Erase the entire sector */ 187*2e192b24SSimon Glass if (spi_flash_erase(flash, offset, flash->sector_size)) 188*2e192b24SSimon Glass return "erase"; 189*2e192b24SSimon Glass /* If it's a partial sector, copy the data into the temp-buffer */ 190*2e192b24SSimon Glass if (len != flash->sector_size) { 191*2e192b24SSimon Glass memcpy(cmp_buf, buf, len); 192*2e192b24SSimon Glass ptr = cmp_buf; 193*2e192b24SSimon Glass } 194*2e192b24SSimon Glass /* Write one complete sector */ 195*2e192b24SSimon Glass if (spi_flash_write(flash, offset, flash->sector_size, ptr)) 196*2e192b24SSimon Glass return "write"; 197*2e192b24SSimon Glass 198*2e192b24SSimon Glass return NULL; 199*2e192b24SSimon Glass } 200*2e192b24SSimon Glass 201*2e192b24SSimon Glass /** 202*2e192b24SSimon Glass * Update an area of SPI flash by erasing and writing any blocks which need 203*2e192b24SSimon Glass * to change. Existing blocks with the correct data are left unchanged. 204*2e192b24SSimon Glass * 205*2e192b24SSimon Glass * @param flash flash context pointer 206*2e192b24SSimon Glass * @param offset flash offset to write 207*2e192b24SSimon Glass * @param len number of bytes to write 208*2e192b24SSimon Glass * @param buf buffer to write from 209*2e192b24SSimon Glass * @return 0 if ok, 1 on error 210*2e192b24SSimon Glass */ 211*2e192b24SSimon Glass static int spi_flash_update(struct spi_flash *flash, u32 offset, 212*2e192b24SSimon Glass size_t len, const char *buf) 213*2e192b24SSimon Glass { 214*2e192b24SSimon Glass const char *err_oper = NULL; 215*2e192b24SSimon Glass char *cmp_buf; 216*2e192b24SSimon Glass const char *end = buf + len; 217*2e192b24SSimon Glass size_t todo; /* number of bytes to do in this pass */ 218*2e192b24SSimon Glass size_t skipped = 0; /* statistics */ 219*2e192b24SSimon Glass const ulong start_time = get_timer(0); 220*2e192b24SSimon Glass size_t scale = 1; 221*2e192b24SSimon Glass const char *start_buf = buf; 222*2e192b24SSimon Glass ulong delta; 223*2e192b24SSimon Glass 224*2e192b24SSimon Glass if (end - buf >= 200) 225*2e192b24SSimon Glass scale = (end - buf) / 100; 226*2e192b24SSimon Glass cmp_buf = memalign(ARCH_DMA_MINALIGN, flash->sector_size); 227*2e192b24SSimon Glass if (cmp_buf) { 228*2e192b24SSimon Glass ulong last_update = get_timer(0); 229*2e192b24SSimon Glass 230*2e192b24SSimon Glass for (; buf < end && !err_oper; buf += todo, offset += todo) { 231*2e192b24SSimon Glass todo = min_t(size_t, end - buf, flash->sector_size); 232*2e192b24SSimon Glass if (get_timer(last_update) > 100) { 233*2e192b24SSimon Glass printf(" \rUpdating, %zu%% %lu B/s", 234*2e192b24SSimon Glass 100 - (end - buf) / scale, 235*2e192b24SSimon Glass bytes_per_second(buf - start_buf, 236*2e192b24SSimon Glass start_time)); 237*2e192b24SSimon Glass last_update = get_timer(0); 238*2e192b24SSimon Glass } 239*2e192b24SSimon Glass err_oper = spi_flash_update_block(flash, offset, todo, 240*2e192b24SSimon Glass buf, cmp_buf, &skipped); 241*2e192b24SSimon Glass } 242*2e192b24SSimon Glass } else { 243*2e192b24SSimon Glass err_oper = "malloc"; 244*2e192b24SSimon Glass } 245*2e192b24SSimon Glass free(cmp_buf); 246*2e192b24SSimon Glass putc('\r'); 247*2e192b24SSimon Glass if (err_oper) { 248*2e192b24SSimon Glass printf("SPI flash failed in %s step\n", err_oper); 249*2e192b24SSimon Glass return 1; 250*2e192b24SSimon Glass } 251*2e192b24SSimon Glass 252*2e192b24SSimon Glass delta = get_timer(start_time); 253*2e192b24SSimon Glass printf("%zu bytes written, %zu bytes skipped", len - skipped, 254*2e192b24SSimon Glass skipped); 255*2e192b24SSimon Glass printf(" in %ld.%lds, speed %ld B/s\n", 256*2e192b24SSimon Glass delta / 1000, delta % 1000, bytes_per_second(len, start_time)); 257*2e192b24SSimon Glass 258*2e192b24SSimon Glass return 0; 259*2e192b24SSimon Glass } 260*2e192b24SSimon Glass 261*2e192b24SSimon Glass static int do_spi_flash_read_write(int argc, char * const argv[]) 262*2e192b24SSimon Glass { 263*2e192b24SSimon Glass unsigned long addr; 264*2e192b24SSimon Glass void *buf; 265*2e192b24SSimon Glass char *endp; 266*2e192b24SSimon Glass int ret = 1; 267*2e192b24SSimon Glass int dev = 0; 268*2e192b24SSimon Glass loff_t offset, len, maxsize; 269*2e192b24SSimon Glass 270*2e192b24SSimon Glass if (argc < 3) 271*2e192b24SSimon Glass return -1; 272*2e192b24SSimon Glass 273*2e192b24SSimon Glass addr = simple_strtoul(argv[1], &endp, 16); 274*2e192b24SSimon Glass if (*argv[1] == 0 || *endp != 0) 275*2e192b24SSimon Glass return -1; 276*2e192b24SSimon Glass 277*2e192b24SSimon Glass if (mtd_arg_off_size(argc - 2, &argv[2], &dev, &offset, &len, 278*2e192b24SSimon Glass &maxsize, MTD_DEV_TYPE_NOR, flash->size)) 279*2e192b24SSimon Glass return -1; 280*2e192b24SSimon Glass 281*2e192b24SSimon Glass /* Consistency checking */ 282*2e192b24SSimon Glass if (offset + len > flash->size) { 283*2e192b24SSimon Glass printf("ERROR: attempting %s past flash size (%#x)\n", 284*2e192b24SSimon Glass argv[0], flash->size); 285*2e192b24SSimon Glass return 1; 286*2e192b24SSimon Glass } 287*2e192b24SSimon Glass 288*2e192b24SSimon Glass buf = map_physmem(addr, len, MAP_WRBACK); 289*2e192b24SSimon Glass if (!buf) { 290*2e192b24SSimon Glass puts("Failed to map physical memory\n"); 291*2e192b24SSimon Glass return 1; 292*2e192b24SSimon Glass } 293*2e192b24SSimon Glass 294*2e192b24SSimon Glass if (strcmp(argv[0], "update") == 0) { 295*2e192b24SSimon Glass ret = spi_flash_update(flash, offset, len, buf); 296*2e192b24SSimon Glass } else if (strncmp(argv[0], "read", 4) == 0 || 297*2e192b24SSimon Glass strncmp(argv[0], "write", 5) == 0) { 298*2e192b24SSimon Glass int read; 299*2e192b24SSimon Glass 300*2e192b24SSimon Glass read = strncmp(argv[0], "read", 4) == 0; 301*2e192b24SSimon Glass if (read) 302*2e192b24SSimon Glass ret = spi_flash_read(flash, offset, len, buf); 303*2e192b24SSimon Glass else 304*2e192b24SSimon Glass ret = spi_flash_write(flash, offset, len, buf); 305*2e192b24SSimon Glass 306*2e192b24SSimon Glass printf("SF: %zu bytes @ %#x %s: ", (size_t)len, (u32)offset, 307*2e192b24SSimon Glass read ? "Read" : "Written"); 308*2e192b24SSimon Glass if (ret) 309*2e192b24SSimon Glass printf("ERROR %d\n", ret); 310*2e192b24SSimon Glass else 311*2e192b24SSimon Glass printf("OK\n"); 312*2e192b24SSimon Glass } 313*2e192b24SSimon Glass 314*2e192b24SSimon Glass unmap_physmem(buf, len); 315*2e192b24SSimon Glass 316*2e192b24SSimon Glass return ret == 0 ? 0 : 1; 317*2e192b24SSimon Glass } 318*2e192b24SSimon Glass 319*2e192b24SSimon Glass static int do_spi_flash_erase(int argc, char * const argv[]) 320*2e192b24SSimon Glass { 321*2e192b24SSimon Glass int ret; 322*2e192b24SSimon Glass int dev = 0; 323*2e192b24SSimon Glass loff_t offset, len, maxsize; 324*2e192b24SSimon Glass ulong size; 325*2e192b24SSimon Glass 326*2e192b24SSimon Glass if (argc < 3) 327*2e192b24SSimon Glass return -1; 328*2e192b24SSimon Glass 329*2e192b24SSimon Glass if (mtd_arg_off(argv[1], &dev, &offset, &len, &maxsize, 330*2e192b24SSimon Glass MTD_DEV_TYPE_NOR, flash->size)) 331*2e192b24SSimon Glass return -1; 332*2e192b24SSimon Glass 333*2e192b24SSimon Glass ret = sf_parse_len_arg(argv[2], &size); 334*2e192b24SSimon Glass if (ret != 1) 335*2e192b24SSimon Glass return -1; 336*2e192b24SSimon Glass 337*2e192b24SSimon Glass /* Consistency checking */ 338*2e192b24SSimon Glass if (offset + size > flash->size) { 339*2e192b24SSimon Glass printf("ERROR: attempting %s past flash size (%#x)\n", 340*2e192b24SSimon Glass argv[0], flash->size); 341*2e192b24SSimon Glass return 1; 342*2e192b24SSimon Glass } 343*2e192b24SSimon Glass 344*2e192b24SSimon Glass ret = spi_flash_erase(flash, offset, size); 345*2e192b24SSimon Glass printf("SF: %zu bytes @ %#x Erased: %s\n", (size_t)size, (u32)offset, 346*2e192b24SSimon Glass ret ? "ERROR" : "OK"); 347*2e192b24SSimon Glass 348*2e192b24SSimon Glass return ret == 0 ? 0 : 1; 349*2e192b24SSimon Glass } 350*2e192b24SSimon Glass 351*2e192b24SSimon Glass static int do_spi_protect(int argc, char * const argv[]) 352*2e192b24SSimon Glass { 353*2e192b24SSimon Glass int ret = 0; 354*2e192b24SSimon Glass loff_t start, len; 355*2e192b24SSimon Glass bool prot = false; 356*2e192b24SSimon Glass 357*2e192b24SSimon Glass if (argc != 4) 358*2e192b24SSimon Glass return -1; 359*2e192b24SSimon Glass 360*2e192b24SSimon Glass if (!str2off(argv[2], &start)) { 361*2e192b24SSimon Glass puts("start sector is not a valid number\n"); 362*2e192b24SSimon Glass return 1; 363*2e192b24SSimon Glass } 364*2e192b24SSimon Glass 365*2e192b24SSimon Glass if (!str2off(argv[3], &len)) { 366*2e192b24SSimon Glass puts("len is not a valid number\n"); 367*2e192b24SSimon Glass return 1; 368*2e192b24SSimon Glass } 369*2e192b24SSimon Glass 370*2e192b24SSimon Glass if (strcmp(argv[1], "lock") == 0) 371*2e192b24SSimon Glass prot = true; 372*2e192b24SSimon Glass else if (strcmp(argv[1], "unlock") == 0) 373*2e192b24SSimon Glass prot = false; 374*2e192b24SSimon Glass else 375*2e192b24SSimon Glass return -1; /* Unknown parameter */ 376*2e192b24SSimon Glass 377*2e192b24SSimon Glass ret = spi_flash_protect(flash, start, len, prot); 378*2e192b24SSimon Glass 379*2e192b24SSimon Glass return ret == 0 ? 0 : 1; 380*2e192b24SSimon Glass } 381*2e192b24SSimon Glass 382*2e192b24SSimon Glass #ifdef CONFIG_CMD_SF_TEST 383*2e192b24SSimon Glass enum { 384*2e192b24SSimon Glass STAGE_ERASE, 385*2e192b24SSimon Glass STAGE_CHECK, 386*2e192b24SSimon Glass STAGE_WRITE, 387*2e192b24SSimon Glass STAGE_READ, 388*2e192b24SSimon Glass 389*2e192b24SSimon Glass STAGE_COUNT, 390*2e192b24SSimon Glass }; 391*2e192b24SSimon Glass 392*2e192b24SSimon Glass static char *stage_name[STAGE_COUNT] = { 393*2e192b24SSimon Glass "erase", 394*2e192b24SSimon Glass "check", 395*2e192b24SSimon Glass "write", 396*2e192b24SSimon Glass "read", 397*2e192b24SSimon Glass }; 398*2e192b24SSimon Glass 399*2e192b24SSimon Glass struct test_info { 400*2e192b24SSimon Glass int stage; 401*2e192b24SSimon Glass int bytes; 402*2e192b24SSimon Glass unsigned base_ms; 403*2e192b24SSimon Glass unsigned time_ms[STAGE_COUNT]; 404*2e192b24SSimon Glass }; 405*2e192b24SSimon Glass 406*2e192b24SSimon Glass static void show_time(struct test_info *test, int stage) 407*2e192b24SSimon Glass { 408*2e192b24SSimon Glass uint64_t speed; /* KiB/s */ 409*2e192b24SSimon Glass int bps; /* Bits per second */ 410*2e192b24SSimon Glass 411*2e192b24SSimon Glass speed = (long long)test->bytes * 1000; 412*2e192b24SSimon Glass if (test->time_ms[stage]) 413*2e192b24SSimon Glass do_div(speed, test->time_ms[stage] * 1024); 414*2e192b24SSimon Glass bps = speed * 8; 415*2e192b24SSimon Glass 416*2e192b24SSimon Glass printf("%d %s: %d ticks, %d KiB/s %d.%03d Mbps\n", stage, 417*2e192b24SSimon Glass stage_name[stage], test->time_ms[stage], 418*2e192b24SSimon Glass (int)speed, bps / 1000, bps % 1000); 419*2e192b24SSimon Glass } 420*2e192b24SSimon Glass 421*2e192b24SSimon Glass static void spi_test_next_stage(struct test_info *test) 422*2e192b24SSimon Glass { 423*2e192b24SSimon Glass test->time_ms[test->stage] = get_timer(test->base_ms); 424*2e192b24SSimon Glass show_time(test, test->stage); 425*2e192b24SSimon Glass test->base_ms = get_timer(0); 426*2e192b24SSimon Glass test->stage++; 427*2e192b24SSimon Glass } 428*2e192b24SSimon Glass 429*2e192b24SSimon Glass /** 430*2e192b24SSimon Glass * Run a test on the SPI flash 431*2e192b24SSimon Glass * 432*2e192b24SSimon Glass * @param flash SPI flash to use 433*2e192b24SSimon Glass * @param buf Source buffer for data to write 434*2e192b24SSimon Glass * @param len Size of data to read/write 435*2e192b24SSimon Glass * @param offset Offset within flash to check 436*2e192b24SSimon Glass * @param vbuf Verification buffer 437*2e192b24SSimon Glass * @return 0 if ok, -1 on error 438*2e192b24SSimon Glass */ 439*2e192b24SSimon Glass static int spi_flash_test(struct spi_flash *flash, uint8_t *buf, ulong len, 440*2e192b24SSimon Glass ulong offset, uint8_t *vbuf) 441*2e192b24SSimon Glass { 442*2e192b24SSimon Glass struct test_info test; 443*2e192b24SSimon Glass int i; 444*2e192b24SSimon Glass 445*2e192b24SSimon Glass printf("SPI flash test:\n"); 446*2e192b24SSimon Glass memset(&test, '\0', sizeof(test)); 447*2e192b24SSimon Glass test.base_ms = get_timer(0); 448*2e192b24SSimon Glass test.bytes = len; 449*2e192b24SSimon Glass if (spi_flash_erase(flash, offset, len)) { 450*2e192b24SSimon Glass printf("Erase failed\n"); 451*2e192b24SSimon Glass return -1; 452*2e192b24SSimon Glass } 453*2e192b24SSimon Glass spi_test_next_stage(&test); 454*2e192b24SSimon Glass 455*2e192b24SSimon Glass if (spi_flash_read(flash, offset, len, vbuf)) { 456*2e192b24SSimon Glass printf("Check read failed\n"); 457*2e192b24SSimon Glass return -1; 458*2e192b24SSimon Glass } 459*2e192b24SSimon Glass for (i = 0; i < len; i++) { 460*2e192b24SSimon Glass if (vbuf[i] != 0xff) { 461*2e192b24SSimon Glass printf("Check failed at %d\n", i); 462*2e192b24SSimon Glass print_buffer(i, vbuf + i, 1, 463*2e192b24SSimon Glass min_t(uint, len - i, 0x40), 0); 464*2e192b24SSimon Glass return -1; 465*2e192b24SSimon Glass } 466*2e192b24SSimon Glass } 467*2e192b24SSimon Glass spi_test_next_stage(&test); 468*2e192b24SSimon Glass 469*2e192b24SSimon Glass if (spi_flash_write(flash, offset, len, buf)) { 470*2e192b24SSimon Glass printf("Write failed\n"); 471*2e192b24SSimon Glass return -1; 472*2e192b24SSimon Glass } 473*2e192b24SSimon Glass memset(vbuf, '\0', len); 474*2e192b24SSimon Glass spi_test_next_stage(&test); 475*2e192b24SSimon Glass 476*2e192b24SSimon Glass if (spi_flash_read(flash, offset, len, vbuf)) { 477*2e192b24SSimon Glass printf("Read failed\n"); 478*2e192b24SSimon Glass return -1; 479*2e192b24SSimon Glass } 480*2e192b24SSimon Glass spi_test_next_stage(&test); 481*2e192b24SSimon Glass 482*2e192b24SSimon Glass for (i = 0; i < len; i++) { 483*2e192b24SSimon Glass if (buf[i] != vbuf[i]) { 484*2e192b24SSimon Glass printf("Verify failed at %d, good data:\n", i); 485*2e192b24SSimon Glass print_buffer(i, buf + i, 1, 486*2e192b24SSimon Glass min_t(uint, len - i, 0x40), 0); 487*2e192b24SSimon Glass printf("Bad data:\n"); 488*2e192b24SSimon Glass print_buffer(i, vbuf + i, 1, 489*2e192b24SSimon Glass min_t(uint, len - i, 0x40), 0); 490*2e192b24SSimon Glass return -1; 491*2e192b24SSimon Glass } 492*2e192b24SSimon Glass } 493*2e192b24SSimon Glass printf("Test passed\n"); 494*2e192b24SSimon Glass for (i = 0; i < STAGE_COUNT; i++) 495*2e192b24SSimon Glass show_time(&test, i); 496*2e192b24SSimon Glass 497*2e192b24SSimon Glass return 0; 498*2e192b24SSimon Glass } 499*2e192b24SSimon Glass 500*2e192b24SSimon Glass static int do_spi_flash_test(int argc, char * const argv[]) 501*2e192b24SSimon Glass { 502*2e192b24SSimon Glass unsigned long offset; 503*2e192b24SSimon Glass unsigned long len; 504*2e192b24SSimon Glass uint8_t *buf, *from; 505*2e192b24SSimon Glass char *endp; 506*2e192b24SSimon Glass uint8_t *vbuf; 507*2e192b24SSimon Glass int ret; 508*2e192b24SSimon Glass 509*2e192b24SSimon Glass if (argc < 3) 510*2e192b24SSimon Glass return -1; 511*2e192b24SSimon Glass offset = simple_strtoul(argv[1], &endp, 16); 512*2e192b24SSimon Glass if (*argv[1] == 0 || *endp != 0) 513*2e192b24SSimon Glass return -1; 514*2e192b24SSimon Glass len = simple_strtoul(argv[2], &endp, 16); 515*2e192b24SSimon Glass if (*argv[2] == 0 || *endp != 0) 516*2e192b24SSimon Glass return -1; 517*2e192b24SSimon Glass 518*2e192b24SSimon Glass vbuf = memalign(ARCH_DMA_MINALIGN, len); 519*2e192b24SSimon Glass if (!vbuf) { 520*2e192b24SSimon Glass printf("Cannot allocate memory (%lu bytes)\n", len); 521*2e192b24SSimon Glass return 1; 522*2e192b24SSimon Glass } 523*2e192b24SSimon Glass buf = memalign(ARCH_DMA_MINALIGN, len); 524*2e192b24SSimon Glass if (!buf) { 525*2e192b24SSimon Glass free(vbuf); 526*2e192b24SSimon Glass printf("Cannot allocate memory (%lu bytes)\n", len); 527*2e192b24SSimon Glass return 1; 528*2e192b24SSimon Glass } 529*2e192b24SSimon Glass 530*2e192b24SSimon Glass from = map_sysmem(CONFIG_SYS_TEXT_BASE, 0); 531*2e192b24SSimon Glass memcpy(buf, from, len); 532*2e192b24SSimon Glass ret = spi_flash_test(flash, buf, len, offset, vbuf); 533*2e192b24SSimon Glass free(vbuf); 534*2e192b24SSimon Glass free(buf); 535*2e192b24SSimon Glass if (ret) { 536*2e192b24SSimon Glass printf("Test failed\n"); 537*2e192b24SSimon Glass return 1; 538*2e192b24SSimon Glass } 539*2e192b24SSimon Glass 540*2e192b24SSimon Glass return 0; 541*2e192b24SSimon Glass } 542*2e192b24SSimon Glass #endif /* CONFIG_CMD_SF_TEST */ 543*2e192b24SSimon Glass 544*2e192b24SSimon Glass static int do_spi_flash(cmd_tbl_t *cmdtp, int flag, int argc, 545*2e192b24SSimon Glass char * const argv[]) 546*2e192b24SSimon Glass { 547*2e192b24SSimon Glass const char *cmd; 548*2e192b24SSimon Glass int ret; 549*2e192b24SSimon Glass 550*2e192b24SSimon Glass /* need at least two arguments */ 551*2e192b24SSimon Glass if (argc < 2) 552*2e192b24SSimon Glass goto usage; 553*2e192b24SSimon Glass 554*2e192b24SSimon Glass cmd = argv[1]; 555*2e192b24SSimon Glass --argc; 556*2e192b24SSimon Glass ++argv; 557*2e192b24SSimon Glass 558*2e192b24SSimon Glass if (strcmp(cmd, "probe") == 0) { 559*2e192b24SSimon Glass ret = do_spi_flash_probe(argc, argv); 560*2e192b24SSimon Glass goto done; 561*2e192b24SSimon Glass } 562*2e192b24SSimon Glass 563*2e192b24SSimon Glass /* The remaining commands require a selected device */ 564*2e192b24SSimon Glass if (!flash) { 565*2e192b24SSimon Glass puts("No SPI flash selected. Please run `sf probe'\n"); 566*2e192b24SSimon Glass return 1; 567*2e192b24SSimon Glass } 568*2e192b24SSimon Glass 569*2e192b24SSimon Glass if (strcmp(cmd, "read") == 0 || strcmp(cmd, "write") == 0 || 570*2e192b24SSimon Glass strcmp(cmd, "update") == 0) 571*2e192b24SSimon Glass ret = do_spi_flash_read_write(argc, argv); 572*2e192b24SSimon Glass else if (strcmp(cmd, "erase") == 0) 573*2e192b24SSimon Glass ret = do_spi_flash_erase(argc, argv); 574*2e192b24SSimon Glass else if (strcmp(cmd, "protect") == 0) 575*2e192b24SSimon Glass ret = do_spi_protect(argc, argv); 576*2e192b24SSimon Glass #ifdef CONFIG_CMD_SF_TEST 577*2e192b24SSimon Glass else if (!strcmp(cmd, "test")) 578*2e192b24SSimon Glass ret = do_spi_flash_test(argc, argv); 579*2e192b24SSimon Glass #endif 580*2e192b24SSimon Glass else 581*2e192b24SSimon Glass ret = -1; 582*2e192b24SSimon Glass 583*2e192b24SSimon Glass done: 584*2e192b24SSimon Glass if (ret != -1) 585*2e192b24SSimon Glass return ret; 586*2e192b24SSimon Glass 587*2e192b24SSimon Glass usage: 588*2e192b24SSimon Glass return CMD_RET_USAGE; 589*2e192b24SSimon Glass } 590*2e192b24SSimon Glass 591*2e192b24SSimon Glass #ifdef CONFIG_CMD_SF_TEST 592*2e192b24SSimon Glass #define SF_TEST_HELP "\nsf test offset len " \ 593*2e192b24SSimon Glass "- run a very basic destructive test" 594*2e192b24SSimon Glass #else 595*2e192b24SSimon Glass #define SF_TEST_HELP 596*2e192b24SSimon Glass #endif 597*2e192b24SSimon Glass 598*2e192b24SSimon Glass U_BOOT_CMD( 599*2e192b24SSimon Glass sf, 5, 1, do_spi_flash, 600*2e192b24SSimon Glass "SPI flash sub-system", 601*2e192b24SSimon Glass "probe [[bus:]cs] [hz] [mode] - init flash device on given SPI bus\n" 602*2e192b24SSimon Glass " and chip select\n" 603*2e192b24SSimon Glass "sf read addr offset|partition len - read `len' bytes starting at\n" 604*2e192b24SSimon Glass " `offset' or from start of mtd\n" 605*2e192b24SSimon Glass " `partition'to memory at `addr'\n" 606*2e192b24SSimon Glass "sf write addr offset|partition len - write `len' bytes from memory\n" 607*2e192b24SSimon Glass " at `addr' to flash at `offset'\n" 608*2e192b24SSimon Glass " or to start of mtd `partition'\n" 609*2e192b24SSimon Glass "sf erase offset|partition [+]len - erase `len' bytes from `offset'\n" 610*2e192b24SSimon Glass " or from start of mtd `partition'\n" 611*2e192b24SSimon Glass " `+len' round up `len' to block size\n" 612*2e192b24SSimon Glass "sf update addr offset|partition len - erase and write `len' bytes from memory\n" 613*2e192b24SSimon Glass " at `addr' to flash at `offset'\n" 614*2e192b24SSimon Glass " or to start of mtd `partition'\n" 615*2e192b24SSimon Glass "sf protect lock/unlock sector len - protect/unprotect 'len' bytes starting\n" 616*2e192b24SSimon Glass " at address 'sector'\n" 617*2e192b24SSimon Glass SF_TEST_HELP 618*2e192b24SSimon Glass ); 619