From: Eduardo Valentin <eduval@xxxxxxxxxx> This commit will register a reboot notifier block callback for giving opportunity tearing down thermal zones, e.g. remove their workqueues, and for giving the governors the opportunity to leave the hardware in a known safe state. Cc: "Rafael J. Wysocki" <rafael@xxxxxxxxxx> (supporter:THERMAL) Cc: Daniel Lezcano <daniel.lezcano@xxxxxxxxxx> (supporter:THERMAL) Cc: Amit Kucheria <amitk@xxxxxxxxxx> (reviewer:THERMAL) Cc: Zhang Rui <rui.zhang@xxxxxxxxx> (reviewer:THERMAL) Cc: Jonathan Corbet <corbet@xxxxxxx> (maintainer:DOCUMENTATION) Cc: linux-pm@xxxxxxxxxxxxxxx (open list:THERMAL) Cc: linux-doc@xxxxxxxxxxxxxxx (open list:DOCUMENTATION) Cc: linux-kernel@xxxxxxxxxxxxxxx (open list) Signed-off-by: Eduardo Valentin <eduval@xxxxxxxxxx> --- drivers/thermal/thermal_core.c | 39 ++++++++++++++++++++++++++++++++++ 1 file changed, 39 insertions(+) diff --git a/drivers/thermal/thermal_core.c b/drivers/thermal/thermal_core.c index 359e7b2ff0e3..66a255fb650b 100644 --- a/drivers/thermal/thermal_core.c +++ b/drivers/thermal/thermal_core.c @@ -319,6 +319,12 @@ static void handle_error_temperature(struct thermal_zone_device *tz, int error) tz->governor->check_error(tz, error); } +static void handle_reboot_prepare(struct thermal_zone_device *tz) +{ + if (tz->governor && tz->governor->reboot_prepare) + tz->governor->reboot_prepare(tz); +} + void thermal_zone_device_critical(struct thermal_zone_device *tz) { /* @@ -1508,6 +1514,30 @@ struct thermal_zone_device *thermal_zone_get_zone_by_name(const char *name) } EXPORT_SYMBOL_GPL(thermal_zone_get_zone_by_name); +/* Best effort attempt to leave thermal zones at a safe state across reboots */ +static void thermal_reboot_prepare(void) +{ + struct thermal_zone_device *tz; + + list_for_each_entry(tz, &thermal_tz_list, node) { + cancel_delayed_work(&tz->poll_queue); + handle_reboot_prepare(tz); + } +} + +static int thermal_reboot_notify(struct notifier_block *nb, + unsigned long mode, void *_unused) +{ + switch (mode) { + case SYS_RESTART: + thermal_reboot_prepare(); + break; + default: + break; + } + return 0; +} + static int thermal_pm_notify(struct notifier_block *nb, unsigned long mode, void *_unused) { @@ -1539,6 +1569,10 @@ static struct notifier_block thermal_pm_nb = { .notifier_call = thermal_pm_notify, }; +static struct notifier_block thermal_reboot_nb = { + .notifier_call = thermal_reboot_notify, +}; + static int __init thermal_init(void) { int result; @@ -1572,6 +1606,11 @@ static int __init thermal_init(void) pr_warn("Thermal: Can not register suspend notifier, return %d\n", result); + result = register_reboot_notifier(&thermal_reboot_nb); + if (result) + pr_warn("Thermal: Can not register reboot notifier, return %d\n", + result); + return 0; unregister_governors: -- 2.34.1