Commit d90fa2c6 authored by Rob Barnes's avatar Rob Barnes Committed by Tzung-Bi Shih

platform/chrome: cros_ec: Poll EC log on EC panic

Add handler for CrOS EC panic events. When a panic is reported,
immediately poll for EC log.

This should result in the log leading to the EC panic being
preserved.

ACPI_NOTIFY_CROS_EC_PANIC is defined in coreboot at
https://review.coreboot.org/plugins/gitiles/coreboot/+/refs/heads/master/src/ec/google/chromeec/acpi/ec.aslSigned-off-by: default avatarRob Barnes <robbarnes@google.com>
Reviewed-by: default avatarPrashant Malani <pmalani@chromium.org>
Signed-off-by: default avatarTzung-Bi Shih <tzungbi@kernel.org>
Link: https://lore.kernel.org/r/20230104011524.369764-2-robbarnes@google.com
parent aaab5af4
...@@ -49,6 +49,7 @@ struct cros_ec_debugfs { ...@@ -49,6 +49,7 @@ struct cros_ec_debugfs {
struct delayed_work log_poll_work; struct delayed_work log_poll_work;
/* EC panicinfo */ /* EC panicinfo */
struct debugfs_blob_wrapper panicinfo_blob; struct debugfs_blob_wrapper panicinfo_blob;
struct notifier_block notifier_panic;
}; };
/* /*
...@@ -437,6 +438,22 @@ static int cros_ec_create_panicinfo(struct cros_ec_debugfs *debug_info) ...@@ -437,6 +438,22 @@ static int cros_ec_create_panicinfo(struct cros_ec_debugfs *debug_info)
return ret; return ret;
} }
static int cros_ec_debugfs_panic_event(struct notifier_block *nb,
unsigned long queued_during_suspend, void *_notify)
{
struct cros_ec_debugfs *debug_info =
container_of(nb, struct cros_ec_debugfs, notifier_panic);
if (debug_info->log_buffer.buf) {
/* Force log poll work to run immediately */
mod_delayed_work(debug_info->log_poll_work.wq, &debug_info->log_poll_work, 0);
/* Block until log poll work finishes */
flush_delayed_work(&debug_info->log_poll_work);
}
return NOTIFY_DONE;
}
static int cros_ec_debugfs_probe(struct platform_device *pd) static int cros_ec_debugfs_probe(struct platform_device *pd)
{ {
struct cros_ec_dev *ec = dev_get_drvdata(pd->dev.parent); struct cros_ec_dev *ec = dev_get_drvdata(pd->dev.parent);
...@@ -473,6 +490,12 @@ static int cros_ec_debugfs_probe(struct platform_device *pd) ...@@ -473,6 +490,12 @@ static int cros_ec_debugfs_probe(struct platform_device *pd)
debugfs_create_u16("suspend_timeout_ms", 0664, debug_info->dir, debugfs_create_u16("suspend_timeout_ms", 0664, debug_info->dir,
&ec->ec_dev->suspend_timeout_ms); &ec->ec_dev->suspend_timeout_ms);
debug_info->notifier_panic.notifier_call = cros_ec_debugfs_panic_event;
ret = blocking_notifier_chain_register(&ec->ec_dev->panic_notifier,
&debug_info->notifier_panic);
if (ret)
goto remove_debugfs;
ec->debug_info = debug_info; ec->debug_info = debug_info;
dev_set_drvdata(&pd->dev, ec); dev_set_drvdata(&pd->dev, ec);
......
...@@ -320,6 +320,13 @@ static void cros_ec_lpc_acpi_notify(acpi_handle device, u32 value, void *data) ...@@ -320,6 +320,13 @@ static void cros_ec_lpc_acpi_notify(acpi_handle device, u32 value, void *data)
ec_dev->last_event_time = cros_ec_get_time_ns(); ec_dev->last_event_time = cros_ec_get_time_ns();
if (value == ACPI_NOTIFY_CROS_EC_PANIC) {
dev_emerg(ec_dev->dev, "CrOS EC Panic Reported. Shutdown is imminent!");
blocking_notifier_call_chain(&ec_dev->panic_notifier, 0, ec_dev);
/* Do not query for other events after a panic is reported */
return;
}
if (ec_dev->mkbp_event_supported) if (ec_dev->mkbp_event_supported)
do { do {
ret = cros_ec_get_next_event(ec_dev, NULL, ret = cros_ec_get_next_event(ec_dev, NULL,
......
...@@ -41,6 +41,13 @@ ...@@ -41,6 +41,13 @@
#define EC_MAX_REQUEST_OVERHEAD 1 #define EC_MAX_REQUEST_OVERHEAD 1
#define EC_MAX_RESPONSE_OVERHEAD 32 #define EC_MAX_RESPONSE_OVERHEAD 32
/*
* EC panic is not covered by the standard (0-F) ACPI notify values.
* Arbitrarily choosing B0 to notify ec panic, which is in the 84-BF
* device specific ACPI notify range.
*/
#define ACPI_NOTIFY_CROS_EC_PANIC 0xB0
/* /*
* Command interface between EC and AP, for LPC, I2C and SPI interfaces. * Command interface between EC and AP, for LPC, I2C and SPI interfaces.
*/ */
...@@ -176,6 +183,8 @@ struct cros_ec_device { ...@@ -176,6 +183,8 @@ struct cros_ec_device {
/* The platform devices used by the mfd driver */ /* The platform devices used by the mfd driver */
struct platform_device *ec; struct platform_device *ec;
struct platform_device *pd; struct platform_device *pd;
struct blocking_notifier_head panic_notifier;
}; };
/** /**
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment