[PATCH 04/12] ras: add estatus queuing and IRQ/NMI handling
Ahmed Tiba
ahmed.tiba at arm.com
Wed Dec 17 03:28:37 PST 2025
Hook the estatus core into the IRQ-work and NMI-safe queues that used to
live in GHES. The new code records CPER payloads into a per-source node,
ships them via irq_work so heavy processing happens out of the NMI path,
and adds the task_work hook used by memory-failure. Behaviour remains the
same as before, but the logic now lives alongside the rest of the estatus
infrastructure so both GHES and DeviceTree providers can reuse it.
Signed-off-by: Ahmed Tiba <ahmed.tiba at arm.com>
---
drivers/firmware/efi/estatus.c | 78 ++++++++++++++++++++++++++++++++++
include/linux/estatus.h | 3 +-
2 files changed, 79 insertions(+), 2 deletions(-)
diff --git a/drivers/firmware/efi/estatus.c b/drivers/firmware/efi/estatus.c
index 5a848d1b218e..8b62b23e2e93 100644
--- a/drivers/firmware/efi/estatus.c
+++ b/drivers/firmware/efi/estatus.c
@@ -973,3 +973,81 @@ void estatus_report_mem_error(int sev, struct cper_sec_mem_err *mem_err)
arch_apei_report_mem_error(sev, mem_err);
#endif
}
+
+int estatus_in_nmi_queue_one_entry(struct estatus_source *source, enum fixed_addresses fixmap_idx)
+{
+ estatus_generic_status *estatus, tmp_header;
+ struct estatus_node *estatus_node;
+ u32 len, node_len;
+ phys_addr_t buf_paddr;
+ int sev, rc;
+
+ if (!IS_ENABLED(CONFIG_ARCH_HAVE_NMI_SAFE_CMPXCHG))
+ return -EOPNOTSUPP;
+
+ rc = __estatus_peek_estatus(source, &tmp_header, &buf_paddr,
+ fixmap_idx);
+ if (rc) {
+ estatus_clear_estatus(source, &tmp_header, buf_paddr,
+ fixmap_idx);
+ return rc;
+ }
+
+ rc = __estatus_check_estatus(source, &tmp_header);
+ if (rc) {
+ estatus_clear_estatus(source, &tmp_header, buf_paddr,
+ fixmap_idx);
+ return rc;
+ }
+
+ len = estatus_len(&tmp_header);
+ node_len = ESTATUS_NODE_LEN(len);
+ estatus_node = (void *)gen_pool_alloc(estatus_pool, node_len);
+ if (!estatus_node)
+ return -ENOMEM;
+
+ estatus_node->source = source;
+ estatus = ESTATUS_FROM_NODE(estatus_node);
+
+ if (__estatus_read_estatus(source, estatus, buf_paddr, fixmap_idx,
+ len)) {
+ estatus_clear_estatus(source, estatus, buf_paddr, fixmap_idx);
+ rc = -ENOENT;
+ goto no_work;
+ }
+
+ sev = estatus_severity(estatus->error_severity);
+ if (sev >= ESTATUS_SEV_PANIC) {
+ estatus_print_queued_estatus();
+ __estatus_panic(source, estatus, buf_paddr, fixmap_idx);
+ }
+
+ estatus_clear_estatus(source, &tmp_header, buf_paddr, fixmap_idx);
+
+ /* This error has been reported before, don't process it again. */
+ if (estatus_cached(estatus))
+ goto no_work;
+
+ llist_add(&estatus_node->llnode, &estatus_llist);
+
+ return rc;
+
+no_work:
+ gen_pool_free(estatus_pool, (unsigned long)estatus_node,
+ node_len);
+
+ return rc;
+}
+EXPORT_SYMBOL_GPL(estatus_in_nmi_queue_one_entry);
+
+void estatus_register_report_chain(struct notifier_block *nb)
+{
+ atomic_notifier_chain_register(&estatus_report_chain, nb);
+}
+EXPORT_SYMBOL_GPL(estatus_register_report_chain);
+
+void estatus_unregister_report_chain(struct notifier_block *nb)
+{
+ atomic_notifier_chain_unregister(&estatus_report_chain, nb);
+}
+EXPORT_SYMBOL_GPL(estatus_unregister_report_chain);
diff --git a/include/linux/estatus.h b/include/linux/estatus.h
index 002a9533c85a..506a74ad60b9 100644
--- a/include/linux/estatus.h
+++ b/include/linux/estatus.h
@@ -122,8 +122,7 @@ enum {
};
int estatus_proc(struct estatus_source *ghes);
-int estatus_in_nmi_queue_one_entry(struct estatus_source *ghes,
- enum fixed_addresses fixmap_idx);
+int estatus_in_nmi_queue_one_entry(struct estatus_source *ghes, enum fixed_addresses fixmap_idx);
void estatus_proc_in_irq(struct irq_work *irq_work);
/**
--
2.43.0
More information about the linux-arm-kernel
mailing list