mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-08-05 16:54:27 +00:00

When the HED driver is built-in, it initializes after evged because they both are at the same initcall level, so the initialization ordering depends on the Makefile order. However, this prevents RAS records coming in between the evged driver initialization and the HED driver initialization from being handled. If the number of such RAS records is above the APEI HEST error source number, the HEST resources may be exhausted, and that may affect subsequent RAS error reporting. To fix this issue, change the initcall level of HED to subsys_initcall and prevent the driver from being built as a module by changing ACPI_HED in Kconfig from "tristate" to "bool". Signed-off-by: Xiaofei Tan <tanxiaofei@huawei.com> Link: https://patch.msgid.link/20250212063408.927666-1-tanxiaofei@huawei.com [ rjw: Changelog edits ] Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
92 lines
2.2 KiB
C
92 lines
2.2 KiB
C
// SPDX-License-Identifier: GPL-2.0-only
|
|
/*
|
|
* ACPI Hardware Error Device (PNP0C33) Driver
|
|
*
|
|
* Copyright (C) 2010, Intel Corp.
|
|
* Author: Huang Ying <ying.huang@intel.com>
|
|
*
|
|
* ACPI Hardware Error Device is used to report some hardware errors
|
|
* notified via SCI, mainly the corrected errors.
|
|
*/
|
|
|
|
#include <linux/kernel.h>
|
|
#include <linux/module.h>
|
|
#include <linux/init.h>
|
|
#include <linux/acpi.h>
|
|
#include <acpi/hed.h>
|
|
|
|
static const struct acpi_device_id acpi_hed_ids[] = {
|
|
{"PNP0C33", 0},
|
|
{"", 0},
|
|
};
|
|
MODULE_DEVICE_TABLE(acpi, acpi_hed_ids);
|
|
|
|
static acpi_handle hed_handle;
|
|
|
|
static BLOCKING_NOTIFIER_HEAD(acpi_hed_notify_list);
|
|
|
|
int register_acpi_hed_notifier(struct notifier_block *nb)
|
|
{
|
|
return blocking_notifier_chain_register(&acpi_hed_notify_list, nb);
|
|
}
|
|
EXPORT_SYMBOL_GPL(register_acpi_hed_notifier);
|
|
|
|
void unregister_acpi_hed_notifier(struct notifier_block *nb)
|
|
{
|
|
blocking_notifier_chain_unregister(&acpi_hed_notify_list, nb);
|
|
}
|
|
EXPORT_SYMBOL_GPL(unregister_acpi_hed_notifier);
|
|
|
|
/*
|
|
* SCI to report hardware error is forwarded to the listeners of HED,
|
|
* it is used by HEST Generic Hardware Error Source with notify type
|
|
* SCI.
|
|
*/
|
|
static void acpi_hed_notify(acpi_handle handle, u32 event, void *data)
|
|
{
|
|
blocking_notifier_call_chain(&acpi_hed_notify_list, 0, NULL);
|
|
}
|
|
|
|
static int acpi_hed_add(struct acpi_device *device)
|
|
{
|
|
int err;
|
|
|
|
/* Only one hardware error device */
|
|
if (hed_handle)
|
|
return -EINVAL;
|
|
hed_handle = device->handle;
|
|
|
|
err = acpi_dev_install_notify_handler(device, ACPI_DEVICE_NOTIFY,
|
|
acpi_hed_notify, device);
|
|
if (err)
|
|
hed_handle = NULL;
|
|
|
|
return err;
|
|
}
|
|
|
|
static void acpi_hed_remove(struct acpi_device *device)
|
|
{
|
|
acpi_dev_remove_notify_handler(device, ACPI_DEVICE_NOTIFY,
|
|
acpi_hed_notify);
|
|
hed_handle = NULL;
|
|
}
|
|
|
|
static struct acpi_driver acpi_hed_driver = {
|
|
.name = "hardware_error_device",
|
|
.class = "hardware_error",
|
|
.ids = acpi_hed_ids,
|
|
.ops = {
|
|
.add = acpi_hed_add,
|
|
.remove = acpi_hed_remove,
|
|
},
|
|
};
|
|
|
|
static int __init acpi_hed_driver_init(void)
|
|
{
|
|
return acpi_bus_register_driver(&acpi_hed_driver);
|
|
}
|
|
subsys_initcall(acpi_hed_driver_init);
|
|
|
|
MODULE_AUTHOR("Huang Ying");
|
|
MODULE_DESCRIPTION("ACPI Hardware Error Device Driver");
|
|
MODULE_LICENSE("GPL");
|