[PATCH v7 01/16] ACPI/AEST: Register arm64_ras platform devices from AEST v2

Ruidong Tian tianruidong at linux.alibaba.com
Tue Jun 2 00:15:24 PDT 2026


Parse the ARM Error Source Table (AEST) v2 [1] and present each
error-source node to the RAS subsystem as a generic platform device.

Rather than letting the RAS driver consume AEST-specific structures
directly, all per-node metadata (interface type, register bases,
group format, record bitmaps, GSIVs, vendor data) is conveyed via
fwnode software-node properties. This keeps every AEST encoding
detail in the ACPI front-end and lets the same back-end driver bind
unchanged when a Device Tree front-end is added later.

If the interface flags indicate an associated ACPI namespace device
(AEST_XFACE_FLAG_ERROR_DEVICE), the companion ACPI device is looked up
and attached so that downstream drivers can reach it.

[1]: https://developer.arm.com/documentation/den0085/0200/

Signed-off-by: Ruidong Tian <tianruidong at linux.alibaba.com>
---
 MAINTAINERS                 |   8 ++
 drivers/acpi/arm64/Kconfig  |  10 ++
 drivers/acpi/arm64/Makefile |   1 +
 drivers/acpi/arm64/aest.c   | 256 ++++++++++++++++++++++++++++++++++++
 include/linux/acpi_aest.h   |  19 +++
 5 files changed, 294 insertions(+)
 create mode 100644 drivers/acpi/arm64/aest.c
 create mode 100644 include/linux/acpi_aest.h

diff --git a/MAINTAINERS b/MAINTAINERS
index c3fe46d7c4bc..16c80a7ea72c 100644
--- a/MAINTAINERS
+++ b/MAINTAINERS
@@ -344,6 +344,14 @@ S:	Maintained
 F:	drivers/acpi/arm64
 F:	include/linux/acpi_iort.h
 
+ACPI AEST
+M:	Ruidong Tian <tianruidong at linux.alibaba.com>
+L:	linux-acpi at vger.kernel.org
+L:	linux-arm-kernel at lists.infradead.org
+S:	Supported
+F:	drivers/acpi/arm64/aest.c
+F:	include/linux/acpi_aest.h
+
 ACPI FOR RISC-V (ACPI/riscv)
 M:	Sunil V L <sunilvl at ventanamicro.com>
 L:	linux-acpi at vger.kernel.org
diff --git a/drivers/acpi/arm64/Kconfig b/drivers/acpi/arm64/Kconfig
index f2fd79f22e7d..49b487bba928 100644
--- a/drivers/acpi/arm64/Kconfig
+++ b/drivers/acpi/arm64/Kconfig
@@ -24,3 +24,13 @@ config ACPI_APMT
 
 config ACPI_MPAM
 	bool
+
+config ACPI_AEST
+	bool "ARM Error Source Table Support"
+	depends on ARM64_RAS_EXTN
+	help
+	  The Arm Error Source Table (AEST) provides details on ACPI
+	  extensions that enable kernel-first handling of errors in a
+	  system that supports the Armv8 RAS extensions.
+
+	  If set, the kernel will report and log hardware errors.
diff --git a/drivers/acpi/arm64/Makefile b/drivers/acpi/arm64/Makefile
index 9390b57cb564..bad77fdbf8dd 100644
--- a/drivers/acpi/arm64/Makefile
+++ b/drivers/acpi/arm64/Makefile
@@ -7,5 +7,6 @@ obj-$(CONFIG_ACPI_IORT) 	+= iort.o
 obj-$(CONFIG_ACPI_MPAM) 	+= mpam.o
 obj-$(CONFIG_ACPI_PROCESSOR_IDLE) += cpuidle.o
 obj-$(CONFIG_ARM_AMBA)		+= amba.o
+obj-$(CONFIG_ACPI_AEST) 	+= aest.o
 obj-y				+= dma.o init.o
 obj-y				+= thermal_cpufreq.o
diff --git a/drivers/acpi/arm64/aest.c b/drivers/acpi/arm64/aest.c
new file mode 100644
index 000000000000..8cf24467d0c2
--- /dev/null
+++ b/drivers/acpi/arm64/aest.c
@@ -0,0 +1,256 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * ARM Error Source Table Support
+ *
+ * Copyright (c) 2025, Alibaba Group.
+ */
+
+#include <linux/cleanup.h>
+#include <linux/platform_device.h>
+#include <linux/property.h>
+#include <linux/acpi_aest.h>
+
+#include "init.h"
+
+#undef pr_fmt
+#define pr_fmt(fmt) "ACPI AEST: " fmt
+
+/*
+ * Fill the per-AEST-entry inner properties (node-type / interface-type /
+ * group-format / record bitmaps / register bases ...).
+ */
+static int __init
+aest_init_node_props(struct acpi_aest_hdr *hdr, struct property_entry *props,
+		     int *p, struct platform_device *pdev)
+{
+	struct acpi_aest_node_interface_header *interface;
+	struct acpi_aest_node_interface_common *common = NULL;
+	u64 *record_implemented = NULL;
+	u64 *status_reporting = NULL;
+	u64 *addressing_mode = NULL;
+	int group_len = 0, i;
+	size_t len;
+
+	interface = ACPI_ADD_PTR(struct acpi_aest_node_interface_header,
+				 hdr, hdr->node_interface_offset);
+	switch (interface->group_format) {
+	case ACPI_AEST_NODE_GROUP_FORMAT_4K: {
+		struct acpi_aest_node_interface_4k *itf =
+			(struct acpi_aest_node_interface_4k *)(interface + 1);
+
+		record_implemented = &itf->error_record_implemented;
+		status_reporting   = &itf->error_status_reporting;
+		addressing_mode    = &itf->addressing_mode;
+		group_len = 1;
+		common = &itf->common;
+		break;
+	}
+	case ACPI_AEST_NODE_GROUP_FORMAT_16K: {
+		struct acpi_aest_node_interface_16k *itf =
+			(struct acpi_aest_node_interface_16k *)(interface + 1);
+
+		record_implemented = itf->error_record_implemented;
+		status_reporting   = itf->error_status_reporting;
+		addressing_mode    = itf->addressing_mode;
+		group_len = 4;
+		common = &itf->common;
+		break;
+	}
+	case ACPI_AEST_NODE_GROUP_FORMAT_64K: {
+		struct acpi_aest_node_interface_64k *itf =
+			(struct acpi_aest_node_interface_64k *)(interface + 1);
+
+		record_implemented = itf->error_record_implemented;
+		status_reporting   = itf->error_status_reporting;
+		addressing_mode    = itf->addressing_mode;
+		group_len = 14;
+		common = &itf->common;
+		break;
+	}
+	default:
+		pr_err("invalid group format: %d\n", interface->group_format);
+		return -EINVAL;
+	}
+
+	if (interface->flags & AEST_XFACE_FLAG_ERROR_DEVICE) {
+		struct acpi_device *companion;
+		char uid[16];
+		int n;
+
+		n = snprintf(uid, sizeof(uid), "%u",
+			     common->error_node_device);
+		if (n > 0 && n < sizeof(uid)) {
+			companion = acpi_dev_get_first_match_dev("ARMHE000",
+								 uid, -1);
+			if (companion) {
+				ACPI_COMPANION_SET(&pdev->dev, companion);
+				acpi_dev_put(companion);
+			} else {
+				pr_debug("MSC.%u: missing namespace entry\n",
+					 common->error_node_device);
+			}
+		}
+	}
+
+	props[(*p)++] = PROPERTY_ENTRY_U8("arm,node-type", hdr->type);
+	props[(*p)++] = PROPERTY_ENTRY_U8("arm,group-format",
+					  interface->group_format);
+	props[(*p)++] = PROPERTY_ENTRY_U32("arm,error-records-count",
+					   interface->error_record_count);
+	props[(*p)++] = PROPERTY_ENTRY_U32("arm,error-records-index",
+					   interface->error_record_index);
+	props[(*p)++] = PROPERTY_ENTRY_U32("arm,interface-flags",
+					   interface->flags);
+	props[(*p)++] = PROPERTY_ENTRY_U64_ARRAY_LEN("arm,record-implemented",
+						     record_implemented,
+						     group_len);
+	props[(*p)++] = PROPERTY_ENTRY_U64_ARRAY_LEN("arm,status-reporting",
+						     status_reporting,
+						     group_len);
+	props[(*p)++] = PROPERTY_ENTRY_U64("arm,error-group-base",
+					   common->error_group_register_base);
+
+	len = hdr->node_interface_offset - hdr->node_specific_offset;
+	props[(*p)++] =
+		PROPERTY_ENTRY_U8_ARRAY_LEN("arm,node-specific-data",
+					    ACPI_ADD_PTR(u8, hdr, hdr->node_specific_offset), len);
+
+	return 0;
+}
+
+static int __init
+aest_create_node_fwnode(struct acpi_aest_hdr *hdr, struct platform_device *pdev)
+{
+	struct property_entry props[10] = { };
+	int p = 0;
+	int ret;
+
+	ret = aest_init_node_props(hdr, props, &p, pdev);
+	if (ret)
+		return ret;
+
+	return device_create_managed_software_node(&pdev->dev, props, NULL);
+}
+
+static int aest_node_mem_size(u8 group_format)
+{
+	switch (group_format) {
+	case ACPI_AEST_NODE_GROUP_FORMAT_4K:
+		return SZ_4K;
+	case ACPI_AEST_NODE_GROUP_FORMAT_16K:
+		return SZ_16K;
+	case ACPI_AEST_NODE_GROUP_FORMAT_64K:
+		return SZ_64K;
+	default:
+		return SZ_4K;
+	}
+}
+
+DEFINE_FREE(res, struct resource *, if (_T) kfree(_T))
+
+static struct platform_device *__init
+acpi_aest_alloc_pdev(struct acpi_aest_hdr *aest_hdr)
+{
+	struct platform_device *pdev __free(platform_device_put) =
+		platform_device_alloc("arm64_ras", PLATFORM_DEVID_AUTO);
+	struct resource *res __free(res) = NULL;
+	struct acpi_aest_node_interface_header *interface;
+	int ret, j = 0;
+
+	if (!pdev)
+		return ERR_PTR(-ENOMEM);
+
+	res = kcalloc(1, sizeof(*res), GFP_KERNEL);
+	if (!res)
+		return ERR_PTR(-ENOMEM);
+
+	interface = ACPI_ADD_PTR(struct acpi_aest_node_interface_header,
+				 aest_hdr, aest_hdr->node_interface_offset);
+	if (interface->type != ACPI_AEST_NODE_SYSTEM_REGISTER) {
+		res[j].name = AEST_NODE_NAME;
+		res[j].start = interface->address;
+		res[j].end = res[j].start + aest_node_mem_size(interface->group_format) - 1;
+		res[j].flags = IORESOURCE_MEM;
+		j++;
+	}
+
+	ret = platform_device_add_resources(pdev, res, j);
+	if (ret)
+		return ERR_PTR(ret);
+
+	return_ptr(pdev);
+}
+
+static int __init acpi_aest_init_node(struct acpi_aest_hdr *aest_hdr)
+{
+	struct platform_device *pdev __free(platform_device_put) = NULL;
+	int ret;
+
+	pdev = acpi_aest_alloc_pdev(aest_hdr);
+	if (IS_ERR(pdev))
+		return PTR_ERR(pdev);
+
+	ret = aest_create_node_fwnode(aest_hdr, pdev);
+	if (ret)
+		return ret;
+
+	ret = platform_device_add(pdev);
+	if (ret)
+		return ret;
+	pr_debug("Platform device added for AEST node: %s.%d\n",
+		 pdev->name, pdev->id);
+	retain_and_null_ptr(pdev);
+
+	return 0;
+}
+
+static int __init acpi_aest_init_nodes(struct acpi_table_header *aest_table)
+{
+	struct acpi_aest_hdr *aest_node, *aest_end;
+	struct acpi_table_aest *aest;
+	int rc;
+
+	aest = (struct acpi_table_aest *)aest_table;
+	aest_node = ACPI_ADD_PTR(struct acpi_aest_hdr, aest,
+				 sizeof(struct acpi_table_header));
+	aest_end = ACPI_ADD_PTR(struct acpi_aest_hdr, aest, aest_table->length);
+
+	while (aest_node < aest_end) {
+		if (((u64)aest_node + aest_node->length) > (u64)aest_end) {
+			pr_warn(FW_WARN
+				"AEST node pointer overflow, bad table.\n");
+			return -EINVAL;
+		}
+
+		rc = acpi_aest_init_node(aest_node);
+		if (rc)
+			return rc;
+
+		aest_node = ACPI_ADD_PTR(struct acpi_aest_hdr, aest_node,
+					 aest_node->length);
+	}
+
+	return 0;
+}
+
+static int __init acpi_aest_init(void)
+{
+	int ret;
+
+	if (acpi_disabled)
+		return 0;
+
+	struct acpi_table_header *aest_table __free(acpi_put_table) =
+		acpi_get_table_pointer(ACPI_SIG_AEST, 0);
+	if (IS_ERR(aest_table))
+		return 0;
+
+	ret = acpi_aest_init_nodes(aest_table);
+	if (ret) {
+		pr_err("Failed init aest node %d\n", ret);
+		return ret;
+	}
+
+	return 0;
+}
+subsys_initcall_sync(acpi_aest_init);
diff --git a/include/linux/acpi_aest.h b/include/linux/acpi_aest.h
new file mode 100644
index 000000000000..e485a6236891
--- /dev/null
+++ b/include/linux/acpi_aest.h
@@ -0,0 +1,19 @@
+/* SPDX-License-Identifier: GPL-2.0 */
+#ifndef __ACPI_AEST_H__
+#define __ACPI_AEST_H__
+
+#include <linux/acpi.h>
+
+/* AEST resource name */
+#define AEST_NODE_NAME "AEST:NODE"
+
+/* AEST interface */
+#define AEST_XFACE_FLAG_SHARED		BIT(0)
+#define AEST_XFACE_FLAG_CLEAR_MISC	BIT(1)
+#define AEST_XFACE_FLAG_ERROR_DEVICE	BIT(2)
+#define AEST_XFACE_FLAG_AFFINITY	BIT(3)
+#define AEST_XFACE_FLAG_ERROR_GROUP	BIT(4)
+#define AEST_XFACE_FLAG_FAULT_INJECT	BIT(5)
+#define AEST_XFACE_FLAG_INT_CONFIG	BIT(6)
+
+#endif /* __ACPI_AEST_H__ */
-- 
2.51.2.612.gdc70283dfc




More information about the linux-arm-kernel mailing list