Move crosec commands from drivers/misc/cros_ec.c to cmd/cros_ec.c Acked-by: Simon Glass <sjg@chromium.org> Signed-off-by: Moritz Fischer <moritz.fischer@ettus.com> Cc: Simon Glass <sjg@chromium.org> Cc: Heiko Schocher <hs@denx.de> Cc: Bin Meng <bmeng.cn@gmail.com> Cc: Miao Yan <yanmiaobest@gmail.com> Cc: Masahiro Yamada <yamada.masahiro@socionext.com> Cc: Stefan Roese <sr@denx.de> Cc: Przemyslaw Marczak <p.marczak@samsung.com> Cc: Maxime Ripard <maxime.ripard@free-electrons.com> Cc: Nishanth Menon <nm@ti.com> Cc: u-boot@lists.denx.demaster
parent
2f159402d9
commit
bfeba0173a
@ -0,0 +1,365 @@ |
||||
/*
|
||||
* Chromium OS cros_ec driver |
||||
* |
||||
* Copyright (c) 2016 The Chromium OS Authors. |
||||
* Copyright (c) 2016 National Instruments Corp |
||||
* |
||||
* SPDX-License-Identifier: GPL-2.0+ |
||||
*/ |
||||
|
||||
#include <common.h> |
||||
#include <command.h> |
||||
#include <cros_ec.h> |
||||
#include <dm.h> |
||||
#include <dm/device-internal.h> |
||||
#include <dm/uclass-internal.h> |
||||
|
||||
/* Note: depends on enum ec_current_image */ |
||||
static const char * const ec_current_image_name[] = {"unknown", "RO", "RW"}; |
||||
|
||||
DECLARE_GLOBAL_DATA_PTR; |
||||
|
||||
/**
|
||||
* Perform a flash read or write command |
||||
* |
||||
* @param dev CROS-EC device to read/write |
||||
* @param is_write 1 do to a write, 0 to do a read |
||||
* @param argc Number of arguments |
||||
* @param argv Arguments (2 is region, 3 is address) |
||||
* @return 0 for ok, 1 for a usage error or -ve for ec command error |
||||
* (negative EC_RES_...) |
||||
*/ |
||||
static int do_read_write(struct cros_ec_dev *dev, int is_write, int argc, |
||||
char * const argv[]) |
||||
{ |
||||
uint32_t offset, size = -1U, region_size; |
||||
unsigned long addr; |
||||
char *endp; |
||||
int region; |
||||
int ret; |
||||
|
||||
region = cros_ec_decode_region(argc - 2, argv + 2); |
||||
if (region == -1) |
||||
return 1; |
||||
if (argc < 4) |
||||
return 1; |
||||
addr = simple_strtoul(argv[3], &endp, 16); |
||||
if (*argv[3] == 0 || *endp != 0) |
||||
return 1; |
||||
if (argc > 4) { |
||||
size = simple_strtoul(argv[4], &endp, 16); |
||||
if (*argv[4] == 0 || *endp != 0) |
||||
return 1; |
||||
} |
||||
|
||||
ret = cros_ec_flash_offset(dev, region, &offset, ®ion_size); |
||||
if (ret) { |
||||
debug("%s: Could not read region info\n", __func__); |
||||
return ret; |
||||
} |
||||
if (size == -1U) |
||||
size = region_size; |
||||
|
||||
ret = is_write ? |
||||
cros_ec_flash_write(dev, (uint8_t *)addr, offset, size) : |
||||
cros_ec_flash_read(dev, (uint8_t *)addr, offset, size); |
||||
if (ret) { |
||||
debug("%s: Could not %s region\n", __func__, |
||||
is_write ? "write" : "read"); |
||||
return ret; |
||||
} |
||||
|
||||
return 0; |
||||
} |
||||
|
||||
static int do_cros_ec(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[]) |
||||
{ |
||||
struct cros_ec_dev *dev; |
||||
struct udevice *udev; |
||||
const char *cmd; |
||||
int ret = 0; |
||||
|
||||
if (argc < 2) |
||||
return CMD_RET_USAGE; |
||||
|
||||
cmd = argv[1]; |
||||
if (0 == strcmp("init", cmd)) { |
||||
/* Remove any existing device */ |
||||
ret = uclass_find_device(UCLASS_CROS_EC, 0, &udev); |
||||
if (!ret) |
||||
device_remove(udev); |
||||
ret = uclass_get_device(UCLASS_CROS_EC, 0, &udev); |
||||
if (ret) { |
||||
printf("Could not init cros_ec device (err %d)\n", ret); |
||||
return 1; |
||||
} |
||||
return 0; |
||||
} |
||||
|
||||
ret = uclass_get_device(UCLASS_CROS_EC, 0, &udev); |
||||
if (ret) { |
||||
printf("Cannot get cros-ec device (err=%d)\n", ret); |
||||
return 1; |
||||
} |
||||
dev = dev_get_uclass_priv(udev); |
||||
if (0 == strcmp("id", cmd)) { |
||||
char id[MSG_BYTES]; |
||||
|
||||
if (cros_ec_read_id(dev, id, sizeof(id))) { |
||||
debug("%s: Could not read KBC ID\n", __func__); |
||||
return 1; |
||||
} |
||||
printf("%s\n", id); |
||||
} else if (0 == strcmp("info", cmd)) { |
||||
struct ec_response_mkbp_info info; |
||||
|
||||
if (cros_ec_info(dev, &info)) { |
||||
debug("%s: Could not read KBC info\n", __func__); |
||||
return 1; |
||||
} |
||||
printf("rows = %u\n", info.rows); |
||||
printf("cols = %u\n", info.cols); |
||||
printf("switches = %#x\n", info.switches); |
||||
} else if (0 == strcmp("curimage", cmd)) { |
||||
enum ec_current_image image; |
||||
|
||||
if (cros_ec_read_current_image(dev, &image)) { |
||||
debug("%s: Could not read KBC image\n", __func__); |
||||
return 1; |
||||
} |
||||
printf("%d\n", image); |
||||
} else if (0 == strcmp("hash", cmd)) { |
||||
struct ec_response_vboot_hash hash; |
||||
int i; |
||||
|
||||
if (cros_ec_read_hash(dev, &hash)) { |
||||
debug("%s: Could not read KBC hash\n", __func__); |
||||
return 1; |
||||
} |
||||
|
||||
if (hash.hash_type == EC_VBOOT_HASH_TYPE_SHA256) |
||||
printf("type: SHA-256\n"); |
||||
else |
||||
printf("type: %d\n", hash.hash_type); |
||||
|
||||
printf("offset: 0x%08x\n", hash.offset); |
||||
printf("size: 0x%08x\n", hash.size); |
||||
|
||||
printf("digest: "); |
||||
for (i = 0; i < hash.digest_size; i++) |
||||
printf("%02x", hash.hash_digest[i]); |
||||
printf("\n"); |
||||
} else if (0 == strcmp("reboot", cmd)) { |
||||
int region; |
||||
enum ec_reboot_cmd cmd; |
||||
|
||||
if (argc >= 3 && !strcmp(argv[2], "cold")) { |
||||
cmd = EC_REBOOT_COLD; |
||||
} else { |
||||
region = cros_ec_decode_region(argc - 2, argv + 2); |
||||
if (region == EC_FLASH_REGION_RO) |
||||
cmd = EC_REBOOT_JUMP_RO; |
||||
else if (region == EC_FLASH_REGION_RW) |
||||
cmd = EC_REBOOT_JUMP_RW; |
||||
else |
||||
return CMD_RET_USAGE; |
||||
} |
||||
|
||||
if (cros_ec_reboot(dev, cmd, 0)) { |
||||
debug("%s: Could not reboot KBC\n", __func__); |
||||
return 1; |
||||
} |
||||
} else if (0 == strcmp("events", cmd)) { |
||||
uint32_t events; |
||||
|
||||
if (cros_ec_get_host_events(dev, &events)) { |
||||
debug("%s: Could not read host events\n", __func__); |
||||
return 1; |
||||
} |
||||
printf("0x%08x\n", events); |
||||
} else if (0 == strcmp("clrevents", cmd)) { |
||||
uint32_t events = 0x7fffffff; |
||||
|
||||
if (argc >= 3) |
||||
events = simple_strtol(argv[2], NULL, 0); |
||||
|
||||
if (cros_ec_clear_host_events(dev, events)) { |
||||
debug("%s: Could not clear host events\n", __func__); |
||||
return 1; |
||||
} |
||||
} else if (0 == strcmp("read", cmd)) { |
||||
ret = do_read_write(dev, 0, argc, argv); |
||||
if (ret > 0) |
||||
return CMD_RET_USAGE; |
||||
} else if (0 == strcmp("write", cmd)) { |
||||
ret = do_read_write(dev, 1, argc, argv); |
||||
if (ret > 0) |
||||
return CMD_RET_USAGE; |
||||
} else if (0 == strcmp("erase", cmd)) { |
||||
int region = cros_ec_decode_region(argc - 2, argv + 2); |
||||
uint32_t offset, size; |
||||
|
||||
if (region == -1) |
||||
return CMD_RET_USAGE; |
||||
if (cros_ec_flash_offset(dev, region, &offset, &size)) { |
||||
debug("%s: Could not read region info\n", __func__); |
||||
ret = -1; |
||||
} else { |
||||
ret = cros_ec_flash_erase(dev, offset, size); |
||||
if (ret) { |
||||
debug("%s: Could not erase region\n", |
||||
__func__); |
||||
} |
||||
} |
||||
} else if (0 == strcmp("regioninfo", cmd)) { |
||||
int region = cros_ec_decode_region(argc - 2, argv + 2); |
||||
uint32_t offset, size; |
||||
|
||||
if (region == -1) |
||||
return CMD_RET_USAGE; |
||||
ret = cros_ec_flash_offset(dev, region, &offset, &size); |
||||
if (ret) { |
||||
debug("%s: Could not read region info\n", __func__); |
||||
} else { |
||||
printf("Region: %s\n", region == EC_FLASH_REGION_RO ? |
||||
"RO" : "RW"); |
||||
printf("Offset: %x\n", offset); |
||||
printf("Size: %x\n", size); |
||||
} |
||||
} else if (0 == strcmp("flashinfo", cmd)) { |
||||
struct ec_response_flash_info p; |
||||
|
||||
ret = cros_ec_read_flashinfo(dev, &p); |
||||
if (!ret) { |
||||
printf("Flash size: %u\n", p.flash_size); |
||||
printf("Write block size: %u\n", p.write_block_size); |
||||
printf("Erase block size: %u\n", p.erase_block_size); |
||||
} |
||||
} else if (0 == strcmp("vbnvcontext", cmd)) { |
||||
uint8_t block[EC_VBNV_BLOCK_SIZE]; |
||||
char buf[3]; |
||||
int i, len; |
||||
unsigned long result; |
||||
|
||||
if (argc <= 2) { |
||||
ret = cros_ec_read_vbnvcontext(dev, block); |
||||
if (!ret) { |
||||
printf("vbnv_block: "); |
||||
for (i = 0; i < EC_VBNV_BLOCK_SIZE; i++) |
||||
printf("%02x", block[i]); |
||||
putc('\n'); |
||||
} |
||||
} else { |
||||
/*
|
||||
* TODO(clchiou): Move this to a utility function as |
||||
* cmd_spi might want to call it. |
||||
*/ |
||||
memset(block, 0, EC_VBNV_BLOCK_SIZE); |
||||
len = strlen(argv[2]); |
||||
buf[2] = '\0'; |
||||
for (i = 0; i < EC_VBNV_BLOCK_SIZE; i++) { |
||||
if (i * 2 >= len) |
||||
break; |
||||
buf[0] = argv[2][i * 2]; |
||||
if (i * 2 + 1 >= len) |
||||
buf[1] = '0'; |
||||
else |
||||
buf[1] = argv[2][i * 2 + 1]; |
||||
strict_strtoul(buf, 16, &result); |
||||
block[i] = result; |
||||
} |
||||
ret = cros_ec_write_vbnvcontext(dev, block); |
||||
} |
||||
if (ret) { |
||||
debug("%s: Could not %s VbNvContext\n", __func__, |
||||
argc <= 2 ? "read" : "write"); |
||||
} |
||||
} else if (0 == strcmp("test", cmd)) { |
||||
int result = cros_ec_test(dev); |
||||
|
||||
if (result) |
||||
printf("Test failed with error %d\n", result); |
||||
else |
||||
puts("Test passed\n"); |
||||
} else if (0 == strcmp("version", cmd)) { |
||||
struct ec_response_get_version *p; |
||||
char *build_string; |
||||
|
||||
ret = cros_ec_read_version(dev, &p); |
||||
if (!ret) { |
||||
/* Print versions */ |
||||
printf("RO version: %1.*s\n", |
||||
(int)sizeof(p->version_string_ro), |
||||
p->version_string_ro); |
||||
printf("RW version: %1.*s\n", |
||||
(int)sizeof(p->version_string_rw), |
||||
p->version_string_rw); |
||||
printf("Firmware copy: %s\n", |
||||
(p->current_image < |
||||
ARRAY_SIZE(ec_current_image_name) ? |
||||
ec_current_image_name[p->current_image] : |
||||
"?")); |
||||
ret = cros_ec_read_build_info(dev, &build_string); |
||||
if (!ret) |
||||
printf("Build info: %s\n", build_string); |
||||
} |
||||
} else if (0 == strcmp("ldo", cmd)) { |
||||
uint8_t index, state; |
||||
char *endp; |
||||
|
||||
if (argc < 3) |
||||
return CMD_RET_USAGE; |
||||
index = simple_strtoul(argv[2], &endp, 10); |
||||
if (*argv[2] == 0 || *endp != 0) |
||||
return CMD_RET_USAGE; |
||||
if (argc > 3) { |
||||
state = simple_strtoul(argv[3], &endp, 10); |
||||
if (*argv[3] == 0 || *endp != 0) |
||||
return CMD_RET_USAGE; |
||||
ret = cros_ec_set_ldo(udev, index, state); |
||||
} else { |
||||
ret = cros_ec_get_ldo(udev, index, &state); |
||||
if (!ret) { |
||||
printf("LDO%d: %s\n", index, |
||||
state == EC_LDO_STATE_ON ? |
||||
"on" : "off"); |
||||
} |
||||
} |
||||
|
||||
if (ret) { |
||||
debug("%s: Could not access LDO%d\n", __func__, index); |
||||
return ret; |
||||
} |
||||
} else { |
||||
return CMD_RET_USAGE; |
||||
} |
||||
|
||||
if (ret < 0) { |
||||
printf("Error: CROS-EC command failed (error %d)\n", ret); |
||||
ret = 1; |
||||
} |
||||
|
||||
return ret; |
||||
} |
||||
|
||||
U_BOOT_CMD( |
||||
crosec, 6, 1, do_cros_ec, |
||||
"CROS-EC utility command", |
||||
"init Re-init CROS-EC (done on startup automatically)\n" |
||||
"crosec id Read CROS-EC ID\n" |
||||
"crosec info Read CROS-EC info\n" |
||||
"crosec curimage Read CROS-EC current image\n" |
||||
"crosec hash Read CROS-EC hash\n" |
||||
"crosec reboot [rw | ro | cold] Reboot CROS-EC\n" |
||||
"crosec events Read CROS-EC host events\n" |
||||
"crosec clrevents [mask] Clear CROS-EC host events\n" |
||||
"crosec regioninfo <ro|rw> Read image info\n" |
||||
"crosec flashinfo Read flash info\n" |
||||
"crosec erase <ro|rw> Erase EC image\n" |
||||
"crosec read <ro|rw> <addr> [<size>] Read EC image\n" |
||||
"crosec write <ro|rw> <addr> [<size>] Write EC image\n" |
||||
"crosec vbnvcontext [hexstring] Read [write] VbNvContext from EC\n" |
||||
"crosec ldo <idx> [<state>] Switch/Read LDO state\n" |
||||
"crosec test run tests on cros_ec\n" |
||||
"crosec version Read CROS-EC version" |
||||
); |
Loading…
Reference in new issue