Browse Source

edac: shell: Make more generic

The EDAC shell code previously contained a number of references to the
Intel IBECC driver. Make the shell code more generic to the defined EDAC
API, and make IBECC-specific code dependent on its config option.

Also make NMI control dependent on X86 as it is specific to that
platform.

Rather than looking for a node labeled "ibecc", the EDAC shell code
now simply requires that a chosen entry for "zephyr,edac" be present to
point to the desired device. The Intel Elkhart Lake DTS has been updated
to add this alias.

Signed-off-by: Robert Hancock <robert.hancock@calian.com>
pull/91704/merge
Robert Hancock 2 months ago committed by Anas Nashif
parent
commit
ba2f6f8945
  1. 206
      drivers/edac/shell.c
  2. 4
      dts/x86/intel/elkhart_lake.dtsi

206
drivers/edac/shell.c

@ -9,20 +9,19 @@ @@ -9,20 +9,19 @@
#include <zephyr/shell/shell.h>
#include <zephyr/drivers/edac.h>
#include "ibecc.h"
static const struct device *const edac_device = DEVICE_DT_GET_OR_NULL(DT_CHOSEN(zephyr_edac));
/**
* EDAC Error Injection interface
*
* edac inject addr [value] Physical memory address base
* edac inject mask [value] Physical memory address mask
* edac inject param1 [value] Show / Set EDAC injection parameter 1
* edac inject param2 [value] Show / Set EDAC injection parameter 2
* edac inject error_type Show / Set EDAC error type
* edac inject trigger Trigger injection
*
* edac inject test_default Set default injection parameters
*
* edac disable_nmi Experimental disable NMI
* edac enable_nmi Experimental enable NMI
* *
* edac disable_nmi Experimental disable NMI (X86 only)
* edac enable_nmi Experimental enable NMI (X86 only)
*
* EDAC Report interface
*
@ -35,7 +34,11 @@ @@ -35,7 +34,11 @@
* devmem [width [value]] Physical memory read / write
*/
static void decode_ecc_error(const struct shell *sh, uint64_t ecc_error)
#ifdef CONFIG_EDAC_IBECC
#include "ibecc.h"
static void decode_ibecc_error(const struct shell *sh, uint64_t ecc_error)
{
uint64_t erradd = ECC_ERROR_ERRADD(ecc_error);
unsigned long errsynd = ECC_ERROR_ERRSYND(ecc_error);
@ -56,6 +59,8 @@ static void decode_ecc_error(const struct shell *sh, uint64_t ecc_error) @@ -56,6 +59,8 @@ static void decode_ecc_error(const struct shell *sh, uint64_t ecc_error)
}
}
#endif /* CONFIG_EDAC_IBECC */
static int ecc_error_show(const struct shell *sh, const struct device *dev)
{
uint64_t error;
@ -69,9 +74,11 @@ static int ecc_error_show(const struct shell *sh, const struct device *dev) @@ -69,9 +74,11 @@ static int ecc_error_show(const struct shell *sh, const struct device *dev)
shell_fprintf(sh, SHELL_NORMAL, "ECC Error: 0x%llx\n", error);
#ifdef CONFIG_EDAC_IBECC
if (error != 0) {
decode_ecc_error(sh, error);
decode_ibecc_error(sh, error);
}
#endif /* CONFIG_EDAC_IBECC */
return 0;
}
@ -94,75 +101,57 @@ static int parity_error_show(const struct shell *sh, const struct device *dev) @@ -94,75 +101,57 @@ static int parity_error_show(const struct shell *sh, const struct device *dev)
static int cmd_edac_info(const struct shell *sh, size_t argc, char **argv)
{
const struct device *dev;
int err;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
shell_fprintf(sh, SHELL_NORMAL, "Show EDAC status\n");
err = ecc_error_show(sh, dev);
if (err != 0) {
return err;
}
(void)ecc_error_show(sh, edac_device);
err = parity_error_show(sh, dev);
if (err != 0) {
return err;
}
(void)parity_error_show(sh, edac_device);
shell_fprintf(sh, SHELL_NORMAL,
"Errors correctable: %d Errors uncorrectable %d\n",
edac_errors_cor_get(dev), edac_errors_uc_get(dev));
shell_fprintf(sh, SHELL_NORMAL, "Errors correctable: %d Errors uncorrectable: %d\n",
edac_errors_cor_get(edac_device), edac_errors_uc_get(edac_device));
return err;
return 0;
}
#if defined(CONFIG_EDAC_ERROR_INJECT)
static int cmd_inject_addr(const struct shell *sh, size_t argc, char **argv)
static int cmd_inject_param1(const struct shell *sh, size_t argc, char **argv)
{
const struct device *dev;
int err;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
if (argc > 2) {
/* Usage */
shell_fprintf(sh, SHELL_NORMAL,
"Usage: edac inject %s [addr]\n", argv[0]);
shell_fprintf(sh, SHELL_NORMAL, "Usage: edac inject %s [val]\n", argv[0]);
return -ENOTSUP;
}
if (argc == 1) {
uint64_t addr;
uint64_t value;
err = edac_inject_get_param1(dev, &addr);
err = edac_inject_get_param1(edac_device, &value);
if (err != 0) {
shell_error(sh, "Error getting address (err %d)",
err);
shell_error(sh, "Error getting param1 (err %d)", err);
return err;
}
shell_fprintf(sh, SHELL_NORMAL,
"Injection address base: 0x%llx\n", addr);
shell_fprintf(sh, SHELL_NORMAL, "Injection param1: 0x%llx\n", value);
} else {
unsigned long value = strtoul(argv[1], NULL, 16);
shell_fprintf(sh, SHELL_NORMAL,
"Set injection address base to: %s\n", argv[1]);
shell_fprintf(sh, SHELL_NORMAL, "Set injection param1 to: %s\n", argv[1]);
err = edac_inject_set_param1(dev, value);
err = edac_inject_set_param1(edac_device, value);
if (err != 0) {
shell_error(sh, "Error setting address (err %d)",
err);
shell_error(sh, "Error setting param1 (err %d)", err);
return err;
}
}
@ -170,44 +159,39 @@ static int cmd_inject_addr(const struct shell *sh, size_t argc, char **argv) @@ -170,44 +159,39 @@ static int cmd_inject_addr(const struct shell *sh, size_t argc, char **argv)
return err;
}
static int cmd_inject_mask(const struct shell *sh, size_t argc, char **argv)
static int cmd_inject_param2(const struct shell *sh, size_t argc, char **argv)
{
const struct device *dev;
int err;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
if (argc > 2) {
/* Usage */
shell_fprintf(sh, SHELL_NORMAL,
"Usage: edac inject %s [mask]\n", argv[0]);
shell_fprintf(sh, SHELL_NORMAL, "Usage: edac inject %s [val]\n", argv[0]);
return -ENOTSUP;
}
if (argc == 1) {
uint64_t mask;
uint64_t value;
err = edac_inject_get_param2(dev, &mask);
err = edac_inject_get_param2(edac_device, &value);
if (err != 0) {
shell_error(sh, "Error getting mask (err %d)", err);
shell_error(sh, "Error getting param2 (err %d)", err);
return err;
}
shell_fprintf(sh, SHELL_NORMAL,
"Injection address mask: 0x%llx\n", mask);
shell_fprintf(sh, SHELL_NORMAL, "Injection param2: 0x%llx\n", value);
} else {
uint64_t value = strtoul(argv[1], NULL, 16);
shell_fprintf(sh, SHELL_NORMAL,
"Set injection address mask to %llx\n", value);
shell_fprintf(sh, SHELL_NORMAL, "Set injection param2 to %llx\n", value);
err = edac_inject_set_param2(dev, value);
err = edac_inject_set_param2(edac_device, value);
if (err != 0) {
shell_error(sh, "Error setting mask (err %d)", err);
shell_error(sh, "Error setting param2 (err %d)", err);
return err;
}
}
@ -218,21 +202,20 @@ static int cmd_inject_mask(const struct shell *sh, size_t argc, char **argv) @@ -218,21 +202,20 @@ static int cmd_inject_mask(const struct shell *sh, size_t argc, char **argv)
static int cmd_inject_trigger(const struct shell *sh, size_t argc,
char **argv)
{
const struct device *dev;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
shell_fprintf(sh, SHELL_NORMAL, "Triggering injection\n");
edac_inject_error_trigger(dev);
edac_inject_error_trigger(edac_device);
return 0;
}
#ifdef CONFIG_X86
static int cmd_inject_disable_nmi(const struct shell *sh, size_t argc,
char **argv)
{
@ -249,6 +232,8 @@ static int cmd_inject_enable_nmi(const struct shell *sh, size_t argc, @@ -249,6 +232,8 @@ static int cmd_inject_enable_nmi(const struct shell *sh, size_t argc,
return 0;
}
#endif /* CONFIG_X86 */
static const char *get_error_type(uint32_t type)
{
switch (type) {
@ -264,17 +249,15 @@ static const char *get_error_type(uint32_t type) @@ -264,17 +249,15 @@ static const char *get_error_type(uint32_t type)
static int cmd_inject_error_type_show(const struct shell *sh, size_t argc,
char **argv)
{
const struct device *dev;
uint32_t error_type;
int err;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
err = edac_inject_get_error_type(dev, &error_type);
err = edac_inject_get_error_type(edac_device, &error_type);
if (err != 0) {
shell_error(sh, "Error getting error type (err %d)", err);
return err;
@ -288,18 +271,15 @@ static int cmd_inject_error_type_show(const struct shell *sh, size_t argc, @@ -288,18 +271,15 @@ static int cmd_inject_error_type_show(const struct shell *sh, size_t argc,
static int set_error_type(const struct shell *sh, uint32_t error_type)
{
const struct device *dev;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
shell_fprintf(sh, SHELL_NORMAL, "Set injection error type: %s\n",
get_error_type(error_type));
return edac_inject_set_error_type(dev, error_type);
return edac_inject_set_error_type(edac_device, error_type);
}
static int cmd_inject_error_type_cor(const struct shell *sh, size_t argc,
@ -314,24 +294,6 @@ static int cmd_inject_error_type_uc(const struct shell *sh, size_t argc, @@ -314,24 +294,6 @@ static int cmd_inject_error_type_uc(const struct shell *sh, size_t argc,
return set_error_type(sh, EDAC_ERROR_TYPE_DRAM_UC);
}
static int cmd_inject_test(const struct shell *sh, size_t argc, char **argv)
{
const struct device *dev;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
return -ENODEV;
}
edac_inject_set_param1(dev, 0x1000);
edac_inject_set_param2(dev, INJ_ADDR_BASE_MASK_MASK);
edac_inject_set_error_type(dev, EDAC_ERROR_TYPE_DRAM_COR);
edac_inject_error_trigger(dev);
return 0;
}
SHELL_STATIC_SUBCMD_SET_CREATE(sub_inject_error_type_cmds,
SHELL_CMD(correctable, NULL, "Set correctable error type",
cmd_inject_error_type_cor),
@ -341,18 +303,16 @@ SHELL_STATIC_SUBCMD_SET_CREATE(sub_inject_error_type_cmds, @@ -341,18 +303,16 @@ SHELL_STATIC_SUBCMD_SET_CREATE(sub_inject_error_type_cmds,
);
/* EDAC Error Injection shell commands */
SHELL_STATIC_SUBCMD_SET_CREATE(sub_inject_cmds,
SHELL_CMD(addr, NULL, "Get / Set physical address", cmd_inject_addr),
SHELL_CMD(mask, NULL, "Get / Set address mask", cmd_inject_mask),
SHELL_CMD_ARG(trigger, NULL, "Trigger injection", cmd_inject_trigger,
1, 0),
SHELL_CMD(error_type, &sub_inject_error_type_cmds,
"Get / Set injection error type",
SHELL_STATIC_SUBCMD_SET_CREATE(
sub_inject_cmds, SHELL_CMD(param1, NULL, "Get / Set injection param 1", cmd_inject_param1),
SHELL_CMD(param2, NULL, "Get / Set injection param 2", cmd_inject_param2),
SHELL_CMD_ARG(trigger, NULL, "Trigger injection", cmd_inject_trigger, 1, 0),
SHELL_CMD(error_type, &sub_inject_error_type_cmds, "Get / Set injection error type",
cmd_inject_error_type_show),
#ifdef CONFIG_X86
SHELL_CMD(disable_nmi, NULL, "Disable NMI", cmd_inject_disable_nmi),
SHELL_CMD(enable_nmi, NULL, "Enable NMI", cmd_inject_enable_nmi),
SHELL_CMD_ARG(test_default, NULL, "Test default injection parameters",
cmd_inject_test, 1, 0),
#endif /* CONFIG_X86 */
SHELL_SUBCMD_SET_END /* Array terminated */
);
#endif /* CONFIG_EDAC_ERROR_INJECT */
@ -360,30 +320,25 @@ SHELL_STATIC_SUBCMD_SET_CREATE(sub_inject_cmds, @@ -360,30 +320,25 @@ SHELL_STATIC_SUBCMD_SET_CREATE(sub_inject_cmds,
static int cmd_ecc_error_show(const struct shell *sh, size_t argc,
char **argv)
{
const struct device *dev;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
return ecc_error_show(sh, dev);
return ecc_error_show(sh, edac_device);
}
static int cmd_ecc_error_clear(const struct shell *sh, size_t argc,
char **argv)
{
const struct device *dev;
int err;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
err = edac_ecc_error_log_clear(dev);
err = edac_ecc_error_log_clear(edac_device);
if (err != 0) {
shell_error(sh, "Error clear ecc error log (err %d)",
err);
@ -404,30 +359,25 @@ SHELL_STATIC_SUBCMD_SET_CREATE(sub_ecc_error_cmds, @@ -404,30 +359,25 @@ SHELL_STATIC_SUBCMD_SET_CREATE(sub_ecc_error_cmds,
static int cmd_parity_error_show(const struct shell *sh, size_t argc,
char **argv)
{
const struct device *dev;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
return parity_error_show(sh, dev);
return parity_error_show(sh, edac_device);
}
static int cmd_parity_error_clear(const struct shell *sh, size_t argc,
char **argv)
{
const struct device *dev;
int err;
dev = DEVICE_DT_GET(DT_NODELABEL(ibecc));
if (!device_is_ready(dev)) {
shell_error(sh, "IBECC device not ready");
if (!device_is_ready(edac_device)) {
shell_error(sh, "EDAC device not ready");
return -ENODEV;
}
err = edac_parity_error_log_clear(dev);
err = edac_parity_error_log_clear(edac_device);
if (err != 0) {
shell_error(sh, "Error clear parity error log (err %d)",
err);

4
dts/x86/intel/elkhart_lake.dtsi

@ -23,6 +23,10 @@ @@ -23,6 +23,10 @@
};
chosen {
zephyr,edac = &ibecc;
};
dram0: memory@0 {
device_type = "memory";
reg = <0x0 DT_DRAM_SIZE>;

Loading…
Cancel
Save