/* * Copyright (C) 2018 Inria * * 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 Manual test application for the EEPROM peripheral drivers * * @author Alexandre Abadie * * @} */ #include #include #include #include "architecture.h" #include "shell.h" #include "test_utils/expect.h" #include "periph/eeprom.h" #ifndef BUFFER_SIZE #define BUFFER_SIZE (42U) #endif static char buffer[BUFFER_SIZE + 1]; static int cmd_info(int argc, char **argv) { (void)argc; (void)argv; #ifdef EEPROM_START_ADDR printf("EEPROM start addr:\t0x%08x\n", (int)EEPROM_START_ADDR); #endif printf("EEPROM size:\t\t%i\n", (int)EEPROM_SIZE); return 0; } static int cmd_read(int argc, char **argv) { if (argc < 3) { printf("usage: %s \n", argv[0]); return 1; } uint32_t pos = atoi(argv[1]); uint8_t count = atoi(argv[2]); if (!count) { puts("Count should be greater than 0"); return 1; } if (count > BUFFER_SIZE) { puts("Count exceeds buffer size"); return 1; } if (pos + count > EEPROM_SIZE) { puts("Failed: cannot read out of eeprom bounds"); return 1; } size_t ret = eeprom_read(pos, (uint8_t *)buffer, count); buffer[count] = '\0'; printf("Data read from EEPROM (%" PRIuSIZE " bytes): %s\n", ret, buffer); return 0; } static int cmd_read_byte(int argc, char **argv) { if (argc < 2) { printf("usage: %s \n", argv[0]); return 1; } uint32_t pos = atoi(argv[1]); if (pos >= EEPROM_SIZE) { puts("Failed: cannot read out of eeprom bounds"); return 1; } uint8_t byte = eeprom_read_byte(pos); printf("Byte read from EEPROM: 0x%02X (%c)\n", byte, byte); return 0; } static int cmd_write(int argc, char **argv) { if (argc < 3) { printf("usage: %s \n", argv[0]); return 1; } uint32_t pos = atoi(argv[1]); if (pos + strlen(argv[2]) > EEPROM_SIZE) { puts("Failed: cannot write out of eeprom bounds"); return 1; } size_t ret = eeprom_write(pos, (uint8_t *)argv[2], strlen(argv[2])); printf("%" PRIuSIZE " bytes written to EEPROM\n", ret); return 0; } static int cmd_write_byte(int argc, char **argv) { if (argc < 3) { printf("usage: %s \n", argv[0]); return 1; } uint32_t pos = atoi(argv[1]); if (pos >= EEPROM_SIZE) { puts("Failed: cannot write out of eeprom bounds"); return 1; } eeprom_write_byte(pos, *(uint8_t *)argv[2]); printf("Byte written to EEPROM\n"); return 0; } static int cmd_set(int argc, char **argv) { if (argc < 4) { printf("usage: %s \n", argv[0]); return 1; } uint32_t pos = atoi(argv[1]); uint32_t count = atoi(argv[3]); if (strlen(argv[2]) != 1) { puts("Failed: char must be a single digit"); return 1; } uint8_t c = (uint8_t)argv[2][0]; if (pos + count > EEPROM_SIZE) { puts("Failed: cannot clear out of eeprom bounds"); return 1; } size_t ret = eeprom_set(pos, c, count); printf("%" PRIuSIZE " bytes set to %c in EEPROM\n", ret, c); return 0; } static int cmd_clear(int argc, char **argv) { if (argc < 3) { printf("usage: %s \n", argv[0]); return 1; } uint32_t pos = atoi(argv[1]); uint32_t count = atoi(argv[2]); if (pos + count > EEPROM_SIZE) { puts("Failed: cannot clear out of eeprom bounds"); return 1; } size_t ret = eeprom_clear(pos, count); printf("%" PRIuSIZE " bytes cleared in EEPROM\n", ret); return 0; } static int cmd_erase(int argc, char **argv) { if (argc != 1) { printf("usage: %s\n", argv[0]); return 1; } size_t ret = eeprom_erase(); if (ret == EEPROM_SIZE) { puts("EEPROM erased with success"); } else { puts("EEPROM erase failed"); return 1; } return 0; } static int cmd_test(int argc, char **argv) { (void)argv; if (argc != 1) { puts("FAILED"); return 1; } const char *expected = "test"; /* test read/write function */ /* read/write from beginning of EEPROM */ size_t ret = eeprom_write(0, (uint8_t *)expected, 4); expect(ret == 4); char result[4]; ret = eeprom_read(0, (uint8_t *)result, 4); expect(memcmp(result, expected, 4) == 0); expect(ret == 4); /* read/write at end of EEPROM */ ret = eeprom_write(EEPROM_SIZE - 4, (uint8_t *)expected, 4); expect(ret == 4); memset(result, 0, 4); ret = eeprom_read(EEPROM_SIZE - 4, (uint8_t *)result, 4); expect(memcmp(result, expected, 4) == 0); expect(ret == 4); /* read/write single byte */ eeprom_write_byte(0, 'A'); expect(eeprom_read_byte(0) == 'A'); eeprom_write_byte(EEPROM_SIZE - 1, 'A'); expect(eeprom_read_byte(EEPROM_SIZE - 1) == 'A'); eeprom_write_byte(EEPROM_SIZE / 2, 'A'); expect(eeprom_read_byte(EEPROM_SIZE / 2) == 'A'); /* clear some bytes */ const uint8_t cleared[4] = { EEPROM_CLEAR_BYTE, EEPROM_CLEAR_BYTE, EEPROM_CLEAR_BYTE, EEPROM_CLEAR_BYTE, }; eeprom_clear(0, 4); memset(result, 0, 4); ret = eeprom_read(0, (uint8_t *)result, 4); expect(memcmp(result, cleared, 4) == 0); expect(ret == 4); eeprom_clear(EEPROM_SIZE - 4, 4); ret = eeprom_read(EEPROM_SIZE - 4, (uint8_t *)result, 4); expect(memcmp(result, cleared, 4) == 0); expect(ret == 4); /* set some bytes */ eeprom_set(0, 'A', 4); ret = eeprom_read(0, (uint8_t *)result, 4); expect(memcmp(result, "AAAA", 4) == 0); expect(ret == 4); memset(result, 0, 4); eeprom_set(EEPROM_SIZE - 4, 'A', 4); ret = eeprom_read(EEPROM_SIZE - 4, (uint8_t *)result, 4); expect(memcmp(result, "AAAA", 4) == 0); expect(ret == 4); puts("SUCCESS"); return 0; } static const shell_command_t shell_commands[] = { { "info", "Print information about eeprom", cmd_info }, { "read", "Read bytes from eeprom", cmd_read }, { "write", "Write bytes to eeprom", cmd_write }, { "read_byte", "Read a single byte from eeprom", cmd_read_byte }, { "write_byte", "Write a single byte to eeprom", cmd_write_byte }, { "set", "Set bytes to eeprom", cmd_set}, { "clear", "Clear bytes to eeprom", cmd_clear}, { "erase", "Erase whole eeprom", cmd_erase}, { "test", "Test the EEPROM implementation", cmd_test }, { NULL, NULL, NULL } }; int main(void) { puts("EEPROM read write test\n"); puts("Please refer to the README.md for more details\n"); cmd_info(0, NULL); /* run the shell */ char line_buf[SHELL_DEFAULT_BUFSIZE]; shell_run(shell_commands, line_buf, SHELL_DEFAULT_BUFSIZE); return 0; }