You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
241 lines
5.5 KiB
241 lines
5.5 KiB
/* |
|
* Copyright (c) 2019 Vestas Wind Systems A/S |
|
* Copyright (c) 2021 Lemonbeat GmbH |
|
* |
|
* SPDX-License-Identifier: Apache-2.0 |
|
*/ |
|
|
|
/** |
|
* @file |
|
* @brief EEPROM shell commands. |
|
*/ |
|
|
|
#include <zephyr/shell/shell.h> |
|
#include <zephyr/drivers/eeprom.h> |
|
#include <stdlib.h> |
|
|
|
struct args_index { |
|
uint8_t device; |
|
uint8_t offset; |
|
uint8_t length; |
|
uint8_t data; |
|
uint8_t pattern; |
|
}; |
|
|
|
static const struct args_index args_indx = { |
|
.device = 1, |
|
.offset = 2, |
|
.length = 3, |
|
.data = 3, |
|
.pattern = 4, |
|
}; |
|
|
|
static int cmd_read(const struct shell *sh, size_t argc, char **argv) |
|
{ |
|
const struct device *eeprom; |
|
size_t addr; |
|
size_t len; |
|
size_t pending; |
|
size_t upto; |
|
int err; |
|
|
|
addr = strtoul(argv[args_indx.offset], NULL, 0); |
|
len = strtoul(argv[args_indx.length], NULL, 0); |
|
|
|
eeprom = device_get_binding(argv[args_indx.device]); |
|
if (!eeprom) { |
|
shell_error(sh, "EEPROM device not found"); |
|
return -EINVAL; |
|
} |
|
|
|
shell_print(sh, "Reading %zu bytes from EEPROM, offset %zu...", len, |
|
addr); |
|
|
|
for (upto = 0; upto < len; upto += pending) { |
|
uint8_t data[SHELL_HEXDUMP_BYTES_IN_LINE]; |
|
|
|
pending = MIN(len - upto, SHELL_HEXDUMP_BYTES_IN_LINE); |
|
err = eeprom_read(eeprom, addr, data, pending); |
|
if (err) { |
|
shell_error(sh, "EEPROM read failed (err %d)", err); |
|
return err; |
|
} |
|
|
|
shell_hexdump_line(sh, addr, data, pending); |
|
addr += pending; |
|
} |
|
|
|
shell_print(sh, ""); |
|
return 0; |
|
} |
|
|
|
static int cmd_write(const struct shell *sh, size_t argc, char **argv) |
|
{ |
|
uint8_t wr_buf[CONFIG_EEPROM_SHELL_BUFFER_SIZE]; |
|
uint8_t rd_buf[CONFIG_EEPROM_SHELL_BUFFER_SIZE]; |
|
const struct device *eeprom; |
|
unsigned long byte; |
|
off_t offset; |
|
size_t len; |
|
int err; |
|
int i; |
|
|
|
offset = strtoul(argv[args_indx.offset], NULL, 0); |
|
len = argc - args_indx.data; |
|
|
|
if (len > sizeof(wr_buf)) { |
|
shell_error(sh, "Write buffer size (%zu bytes) exceeded", |
|
sizeof(wr_buf)); |
|
return -EINVAL; |
|
} |
|
|
|
for (i = 0; i < len; i++) { |
|
byte = strtoul(argv[args_indx.data + i], NULL, 0); |
|
if (byte > UINT8_MAX) { |
|
shell_error(sh, "Error parsing data byte %d", i); |
|
return -EINVAL; |
|
} |
|
wr_buf[i] = byte; |
|
} |
|
|
|
eeprom = device_get_binding(argv[args_indx.device]); |
|
if (!eeprom) { |
|
shell_error(sh, "EEPROM device not found"); |
|
return -EINVAL; |
|
} |
|
|
|
shell_print(sh, "Writing %zu bytes to EEPROM...", len); |
|
|
|
err = eeprom_write(eeprom, offset, wr_buf, len); |
|
if (err) { |
|
shell_error(sh, "EEPROM write failed (err %d)", err); |
|
return err; |
|
} |
|
|
|
shell_print(sh, "Verifying..."); |
|
|
|
err = eeprom_read(eeprom, offset, rd_buf, len); |
|
if (err) { |
|
shell_error(sh, "EEPROM read failed (err %d)", err); |
|
return err; |
|
} |
|
|
|
if (memcmp(wr_buf, rd_buf, len) != 0) { |
|
shell_error(sh, "Verify failed"); |
|
return -EIO; |
|
} |
|
|
|
shell_print(sh, "Verify OK"); |
|
|
|
return 0; |
|
} |
|
|
|
static int cmd_size(const struct shell *sh, size_t argc, char **argv) |
|
{ |
|
const struct device *eeprom; |
|
|
|
eeprom = device_get_binding(argv[args_indx.device]); |
|
if (!eeprom) { |
|
shell_error(sh, "EEPROM device not found"); |
|
return -EINVAL; |
|
} |
|
|
|
shell_print(sh, "%zu bytes", eeprom_get_size(eeprom)); |
|
return 0; |
|
} |
|
|
|
static int cmd_fill(const struct shell *sh, size_t argc, char **argv) |
|
{ |
|
uint8_t wr_buf[CONFIG_EEPROM_SHELL_BUFFER_SIZE]; |
|
uint8_t rd_buf[CONFIG_EEPROM_SHELL_BUFFER_SIZE]; |
|
const struct device *eeprom; |
|
unsigned long pattern; |
|
size_t addr; |
|
size_t initial_offset; |
|
size_t len; |
|
size_t pending; |
|
size_t upto; |
|
int err; |
|
|
|
initial_offset = strtoul(argv[args_indx.offset], NULL, 0); |
|
len = strtoul(argv[args_indx.length], NULL, 0); |
|
|
|
pattern = strtoul(argv[args_indx.pattern], NULL, 0); |
|
if (pattern > UINT8_MAX) { |
|
shell_error(sh, "Error parsing pattern byte"); |
|
return -EINVAL; |
|
} |
|
memset(wr_buf, pattern, MIN(len, CONFIG_EEPROM_SHELL_BUFFER_SIZE)); |
|
|
|
eeprom = device_get_binding(argv[args_indx.device]); |
|
if (!eeprom) { |
|
shell_error(sh, "EEPROM device not found"); |
|
return -EINVAL; |
|
} |
|
|
|
shell_print(sh, "Writing %zu bytes of 0x%02lx to EEPROM...", len, |
|
pattern); |
|
|
|
addr = initial_offset; |
|
|
|
for (upto = 0; upto < len; upto += pending) { |
|
pending = MIN(len - upto, CONFIG_EEPROM_SHELL_BUFFER_SIZE); |
|
err = eeprom_write(eeprom, addr, wr_buf, pending); |
|
if (err) { |
|
shell_error(sh, "EEPROM write failed (err %d)", err); |
|
return err; |
|
} |
|
addr += pending; |
|
} |
|
|
|
addr = initial_offset; |
|
|
|
shell_print(sh, "Verifying..."); |
|
|
|
for (upto = 0; upto < len; upto += pending) { |
|
pending = MIN(len - upto, CONFIG_EEPROM_SHELL_BUFFER_SIZE); |
|
err = eeprom_read(eeprom, addr, rd_buf, pending); |
|
if (err) { |
|
shell_error(sh, "EEPROM read failed (err %d)", err); |
|
return err; |
|
} |
|
|
|
if (memcmp(wr_buf, rd_buf, pending) != 0) { |
|
shell_error(sh, "Verify failed"); |
|
return -EIO; |
|
} |
|
|
|
addr += pending; |
|
} |
|
|
|
shell_print(sh, "Verify OK"); |
|
|
|
return 0; |
|
} |
|
|
|
/* Device name autocompletion support */ |
|
static void device_name_get(size_t idx, struct shell_static_entry *entry) |
|
{ |
|
const struct device *dev = shell_device_lookup(idx, NULL); |
|
|
|
entry->syntax = (dev != NULL) ? dev->name : NULL; |
|
entry->handler = NULL; |
|
entry->help = NULL; |
|
entry->subcmd = NULL; |
|
} |
|
|
|
SHELL_DYNAMIC_CMD_CREATE(dsub_device_name, device_name_get); |
|
|
|
SHELL_STATIC_SUBCMD_SET_CREATE(eeprom_cmds, |
|
SHELL_CMD_ARG(read, &dsub_device_name, |
|
"<device> <offset> <length>", cmd_read, 4, 0), |
|
SHELL_CMD_ARG(write, &dsub_device_name, |
|
"<device> <offset> [byte0] <byte1> .. <byteN>", cmd_write, |
|
4, CONFIG_EEPROM_SHELL_BUFFER_SIZE - 1), |
|
SHELL_CMD_ARG(size, &dsub_device_name, "<device>", cmd_size, 2, 0), |
|
SHELL_CMD_ARG(fill, &dsub_device_name, |
|
"<device> <offset> <length> <pattern>", cmd_fill, 5, 0), |
|
SHELL_SUBCMD_SET_END |
|
); |
|
|
|
SHELL_CMD_REGISTER(eeprom, &eeprom_cmds, "EEPROM shell commands", NULL);
|
|
|