/* * Copyright (c) 2020 ML!PA Consulting GmbH * * This file is subject to the terms and conditions of the GNU Lesser * General Public License v2.1. See the file LICENSE in the top level * directory for more details. */ /** * @ingroup tests * @{ * * @file * @brief Application for testing MTD implementations * * @author Benjamin Valentin <benjamin.valentin@ml-pa.com> * * @} */ #include <assert.h> #include <inttypes.h> #include <stdbool.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include "od.h" #include "mtd.h" #include "shell.h" #include "board.h" #include "macros/units.h" #ifndef MTD_NUMOF #ifdef MTD_0 #define MTD_NUMOF 1 #else #define MTD_NUMOF 0 #endif #endif static mtd_dev_t *_get_mtd_dev(unsigned idx) { switch (idx) { #ifdef MTD_0 case 0: return MTD_0; #endif #ifdef MTD_1 case 1: return MTD_1; #endif #ifdef MTD_2 case 2: return MTD_2; #endif #ifdef MTD_3 case 3: return MTD_3; #endif } return NULL; } static mtd_dev_t *_get_dev(int argc, char **argv) { if (argc < 2) { printf("%s: please specify the MTD device\n", argv[0]); return NULL; } unsigned idx = atoi(argv[1]); if (idx > MTD_NUMOF) { printf("%s: invalid device: %s\n", argv[0], argv[1]); return NULL; } return _get_mtd_dev(idx); } static uint64_t _get_size(mtd_dev_t *dev) { return (uint64_t)dev->sector_count * dev->pages_per_sector * dev->page_size; } static int cmd_read(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); uint32_t addr, len; if (argc < 4 || dev == NULL) { printf("usage: %s <dev> <addr> <len>\n", argv[0]); return -1; } addr = atoi(argv[2]); len = atoi(argv[3]); void *buffer = malloc(len); if (buffer == NULL) { puts("out of memory"); return -1; } /* don't print random data if read fails */ memset(buffer, 0x3F, len); int res = mtd_read(dev, buffer, addr, len); od_hex_dump_ext(buffer, len, 0, addr); free(buffer); if (res) { printf("error: %i\n", res); } return res; } static int cmd_read_page(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); uint32_t page, offset, len; if (argc < 5 || dev == NULL) { printf("usage: %s <dev> <page> <offset> <len>\n", argv[0]); return -1; } page = atoi(argv[2]); offset = atoi(argv[3]); len = atoi(argv[4]); void *buffer = malloc(len); if (buffer == NULL) { puts("out of memory"); return -1; } int res = mtd_read_page(dev, buffer, page, offset, len); od_hex_dump_ext(buffer, len, 0, page * dev->page_size + offset); free(buffer); if (res) { printf("error: %i\n", res); } return res; } static int cmd_write(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); uint32_t addr, len; if (argc < 4 || dev == NULL) { printf("usage: %s <dev> <addr> <data>\n", argv[0]); return -1; } addr = atoi(argv[2]); len = strlen(argv[3]); int res = mtd_write(dev, argv[3], addr, len); if (res) { printf("error: %i\n", res); } return res; } static int cmd_write_page_raw(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); uint32_t page, offset, len; if (argc < 5 || dev == NULL) { printf("usage: %s <dev> <page> <offset> <data>\n", argv[0]); return -1; } page = atoi(argv[2]); offset = atoi(argv[3]); len = strlen(argv[4]); int res = mtd_write_page_raw(dev, argv[4], page, offset, len); if (res) { printf("error: %i\n", res); } return res; } static int cmd_write_page(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); uint32_t page, offset, len; if (argc < 5 || dev == NULL) { printf("usage: %s <dev> <page> <offset> <data>\n", argv[0]); return -1; } page = atoi(argv[2]); offset = atoi(argv[3]); len = strlen(argv[4]); int res = mtd_write_page(dev, argv[4], page, offset, len); if (res) { printf("error: %i\n", res); } return res; } static int cmd_erase(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); uint32_t addr; uint32_t len; if (argc < 4 || dev == NULL) { printf("usage: %s <dev> <addr> <len>\n", argv[0]); return -1; } addr = atoi(argv[2]); len = atoi(argv[3]); int res = mtd_erase(dev, addr, len); if (res) { printf("error: %i\n", res); } return res; } static int cmd_erase_sector(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); uint32_t sector, count = 1; if (argc < 3 || dev == NULL) { printf("usage: %s <dev> <sector> [count]\n", argv[0]); return -1; } sector = atoi(argv[2]); if (argc > 3) { count = atoi(argv[3]); } int res = mtd_erase_sector(dev, sector, count); if (res) { printf("error: %i\n", res); } return res; } static void _print_size(uint64_t size) { unsigned long len; const char *unit; if (size == 0) { len = 0; unit = "byte"; } else if ((size & (GiB(1) - 1)) == 0) { len = size / GiB(1); unit = "GiB"; } else if ((size & (MiB(1) - 1)) == 0) { len = size / MiB(1); unit = "MiB"; } else if ((size & (KiB(1) - 1)) == 0) { len = size / KiB(1); unit = "kiB"; } else { len = size; unit = "byte"; } printf("%lu %s", len, unit); } static void _print_info(mtd_dev_t *dev) { printf("sectors: %"PRIu32"\n", dev->sector_count); printf("pages per sector: %"PRIu32"\n", dev->pages_per_sector); printf("page size: %"PRIu32"\n", dev->page_size); printf("total: "); _print_size(_get_size(dev)); puts(""); } static int cmd_info(int argc, char **argv) { if (argc < 2) { printf("mtd devices: %d\n", MTD_NUMOF); for (int i = 0; i < MTD_NUMOF; ++i) { printf(" -=[ MTD_%d ]=-\n", i); _print_info(_get_mtd_dev(i)); } return 0; } mtd_dev_t *dev = _get_dev(argc, argv); if (dev == NULL) { return -1; } _print_info(dev); return 0; } static inline int _print_power_usage(const char *progname) { printf("usage: %s <dev> <on|off>\n", progname); return -1; } static int cmd_power(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); enum mtd_power_state state; if (argc < 3 || dev == NULL) { return _print_power_usage(argv[0]); } if (strcmp(argv[2], "off") == 0) { state = MTD_POWER_DOWN; } else if (strcmp(argv[2], "on") == 0) { state = MTD_POWER_UP; } else { return _print_power_usage(argv[0]); } mtd_power(dev, state); return 0; } static bool mem_is_all_set(const uint8_t *buf, uint8_t c, size_t n) { for (const uint8_t *end = buf + n; buf != end; ++buf) { if (*buf != c) { return false; } } return true; } static int cmd_test(int argc, char **argv) { mtd_dev_t *dev = _get_dev(argc, argv); uint32_t sector; if (argc < 2 || dev == NULL) { printf("usage: %s <dev>\n", argv[0]); return -1; } if (dev->sector_count < 2) { return -1; } if (argc > 2) { sector = atoi(argv[2]); } else { sector = dev->sector_count - 2; } uint8_t *buffer = malloc(dev->page_size); if (buffer == NULL) { puts("out of memory"); return -1; } uint32_t page_0 = dev->pages_per_sector * sector; uint32_t page_1 = dev->pages_per_sector * (sector + 1); uint32_t page_size = dev->page_size; puts("[START]"); /* write dummy data to sectors */ memset(buffer, 0x23, dev->page_size); assert(mtd_write_page_raw(dev, buffer, page_0, 0, page_size) == 0); assert(mtd_write_page_raw(dev, buffer, page_1, 0, page_size) == 0); /* erase two sectors and check if they have been erased */ assert(mtd_erase_sector(dev, sector, 2) == 0); assert(mtd_read_page(dev, buffer, page_0, 0, page_size) == 0); assert(mem_is_all_set(buffer, 0xFF, page_size) || mem_is_all_set(buffer, 0x00, page_size)); assert(mtd_read_page(dev, buffer, page_1, 0, page_size) == 0); assert(mem_is_all_set(buffer, 0xFF, page_size) || mem_is_all_set(buffer, 0x00, page_size)); /* write test data & read it back */ const char test_str[] = "0123456789"; uint32_t offset = 5; assert(mtd_write_page_raw(dev, test_str, page_0, offset, sizeof(test_str)) == 0); assert(mtd_read_page(dev, buffer, page_0, offset, sizeof(test_str)) == 0); assert(memcmp(test_str, buffer, sizeof(test_str)) == 0); /* write across page boundary */ offset = page_size - sizeof(test_str) / 2; assert(mtd_write_page_raw(dev, test_str, page_0, offset, sizeof(test_str)) == 0); assert(mtd_read_page(dev, buffer, page_0, offset, sizeof(test_str)) == 0); assert(memcmp(test_str, buffer, sizeof(test_str)) == 0); /* write across sector boundary */ offset = page_size - sizeof(test_str) / 2 + (dev->pages_per_sector - 1) * page_size; assert(mtd_write_page_raw(dev, test_str, page_0, offset, sizeof(test_str)) == 0); assert(mtd_read_page(dev, buffer, page_0, offset, sizeof(test_str)) == 0); assert(memcmp(test_str, buffer, sizeof(test_str)) == 0); /* overwrite first test string, rely on MTD for read-modify-write */ const char test_str_2[] = "Hello World!"; offset = 5; assert(mtd_write_page(dev, test_str_2, page_0, offset, sizeof(test_str_2)) == 0); assert(mtd_read_page(dev, buffer, page_0, offset, sizeof(test_str_2)) == 0); assert(memcmp(test_str_2, buffer, sizeof(test_str_2)) == 0); puts("[SUCCESS]"); free(buffer); return 0; } static const shell_command_t shell_commands[] = { { "info", "Print properties of the MTD device", cmd_info }, { "power", "Turn the MTD device on/off", cmd_power }, { "read", "Read a region of memory on the MTD device", cmd_read }, { "read_page", "Read a region of memory on the MTD device (pagewise addressing)", cmd_read_page }, { "write", "Write a region of memory on the MTD device", cmd_write }, { "write_page_raw", "Write a region of memory on the MTD device (pagewise addressing)", cmd_write_page_raw }, { "write_page", "Write a region of memory on the MTD device (pagewise addressing, read-modify-write)", cmd_write_page }, { "erase", "Erase a region of memory on the MTD device", cmd_erase }, { "erase_sector", "Erase a sector of memory on the MTD device", cmd_erase_sector }, { "test", "Erase & write test data to the last two sectors", cmd_test }, { NULL, NULL, NULL } }; int main(void) { puts("Manual MTD test"); if (MTD_NUMOF == 0) { puts("no MTD device present on the board."); } for (int i = 0; i < MTD_NUMOF; ++i) { printf("init MTD_%d… ", i); mtd_dev_t *dev = _get_mtd_dev(i); int res = mtd_init(dev); if (res) { printf("error: %d\n", res); continue; } printf("OK ("); _print_size(_get_size(dev)); puts(")"); mtd_power(dev, MTD_POWER_UP); } /* run the shell */ char line_buf[SHELL_DEFAULT_BUFSIZE]; shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE); return 0; }