Commit 9b7ff25d authored by Rafael J. Wysocki's avatar Rafael J. Wysocki

ACPI: power: Refine turning off unused power resources

Commit 7e4fdeaf ("ACPI: power: Turn off unused power resources
unconditionally") dropped the power resource state check from
acpi_turn_off_unused_power_resources(), because according to the
ACPI specification (e.g. ACPI 6.4, Section 7.2.2) the OS "may run
the _OFF method repeatedly, even if the resource is already off".

However, it turns out that some systems do not follow the
specification in this particular respect and that commit introduced
boot issues on them, so refine acpi_turn_off_unused_power_resources()
to only turn off power resources without any users after device
enumeration and restore its previous behavior in the system-wide
resume path.

Fixes: 7e4fdeaf ("ACPI: power: Turn off unused power resources unconditionally")
Link: https://uefi.org/specs/ACPI/6.4/07_Power_and_Performance_Mgmt/declaring-a-power-resource-object.html#off
BugLink: https://bugzilla.kernel.org/show_bug.cgi?id=213019Reported-by: default avatarZhang Rui <rui.zhang@intel.com>
Tested-by: default avatarZhang Rui <rui.zhang@intel.com>
Reported-by: default avatarDave Olsthoorn <dave@bewaar.me>
Tested-by: default avatarDave Olsthoorn <dave@bewaar.me>
Reported-by: default avatarShujun Wang <wsj20369@163.com>
Tested-by: default avatarShujun Wang <wsj20369@163.com>
Signed-off-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
parent c4681547
...@@ -134,7 +134,7 @@ int acpi_power_init(void); ...@@ -134,7 +134,7 @@ int acpi_power_init(void);
void acpi_power_resources_list_free(struct list_head *list); void acpi_power_resources_list_free(struct list_head *list);
int acpi_extract_power_resources(union acpi_object *package, unsigned int start, int acpi_extract_power_resources(union acpi_object *package, unsigned int start,
struct list_head *list); struct list_head *list);
int acpi_add_power_resource(acpi_handle handle); struct acpi_device *acpi_add_power_resource(acpi_handle handle);
void acpi_power_add_remove_device(struct acpi_device *adev, bool add); void acpi_power_add_remove_device(struct acpi_device *adev, bool add);
int acpi_power_wakeup_list_init(struct list_head *list, int *system_level); int acpi_power_wakeup_list_init(struct list_head *list, int *system_level);
int acpi_device_sleep_wake(struct acpi_device *dev, int acpi_device_sleep_wake(struct acpi_device *dev,
...@@ -142,7 +142,7 @@ int acpi_device_sleep_wake(struct acpi_device *dev, ...@@ -142,7 +142,7 @@ int acpi_device_sleep_wake(struct acpi_device *dev,
int acpi_power_get_inferred_state(struct acpi_device *device, int *state); int acpi_power_get_inferred_state(struct acpi_device *device, int *state);
int acpi_power_on_resources(struct acpi_device *device, int state); int acpi_power_on_resources(struct acpi_device *device, int state);
int acpi_power_transition(struct acpi_device *device, int state); int acpi_power_transition(struct acpi_device *device, int state);
void acpi_turn_off_unused_power_resources(void); void acpi_turn_off_unused_power_resources(bool init);
/* -------------------------------------------------------------------------- /* --------------------------------------------------------------------------
Device Power Management Device Power Management
......
...@@ -52,6 +52,7 @@ struct acpi_power_resource { ...@@ -52,6 +52,7 @@ struct acpi_power_resource {
u32 system_level; u32 system_level;
u32 order; u32 order;
unsigned int ref_count; unsigned int ref_count;
unsigned int users;
bool wakeup_enabled; bool wakeup_enabled;
struct mutex resource_lock; struct mutex resource_lock;
struct list_head dependents; struct list_head dependents;
...@@ -147,6 +148,7 @@ int acpi_extract_power_resources(union acpi_object *package, unsigned int start, ...@@ -147,6 +148,7 @@ int acpi_extract_power_resources(union acpi_object *package, unsigned int start,
for (i = start; i < package->package.count; i++) { for (i = start; i < package->package.count; i++) {
union acpi_object *element = &package->package.elements[i]; union acpi_object *element = &package->package.elements[i];
struct acpi_device *rdev;
acpi_handle rhandle; acpi_handle rhandle;
if (element->type != ACPI_TYPE_LOCAL_REFERENCE) { if (element->type != ACPI_TYPE_LOCAL_REFERENCE) {
...@@ -163,13 +165,16 @@ int acpi_extract_power_resources(union acpi_object *package, unsigned int start, ...@@ -163,13 +165,16 @@ int acpi_extract_power_resources(union acpi_object *package, unsigned int start,
if (acpi_power_resource_is_dup(package, start, i)) if (acpi_power_resource_is_dup(package, start, i))
continue; continue;
err = acpi_add_power_resource(rhandle); rdev = acpi_add_power_resource(rhandle);
if (err) if (!rdev) {
err = -ENODEV;
break; break;
}
err = acpi_power_resources_list_add(rhandle, list); err = acpi_power_resources_list_add(rhandle, list);
if (err) if (err)
break; break;
to_power_resource(rdev)->users++;
} }
if (err) if (err)
acpi_power_resources_list_free(list); acpi_power_resources_list_free(list);
...@@ -907,7 +912,7 @@ static void acpi_power_add_resource_to_list(struct acpi_power_resource *resource ...@@ -907,7 +912,7 @@ static void acpi_power_add_resource_to_list(struct acpi_power_resource *resource
mutex_unlock(&power_resource_list_lock); mutex_unlock(&power_resource_list_lock);
} }
int acpi_add_power_resource(acpi_handle handle) struct acpi_device *acpi_add_power_resource(acpi_handle handle)
{ {
struct acpi_power_resource *resource; struct acpi_power_resource *resource;
struct acpi_device *device = NULL; struct acpi_device *device = NULL;
...@@ -918,11 +923,11 @@ int acpi_add_power_resource(acpi_handle handle) ...@@ -918,11 +923,11 @@ int acpi_add_power_resource(acpi_handle handle)
acpi_bus_get_device(handle, &device); acpi_bus_get_device(handle, &device);
if (device) if (device)
return 0; return device;
resource = kzalloc(sizeof(*resource), GFP_KERNEL); resource = kzalloc(sizeof(*resource), GFP_KERNEL);
if (!resource) if (!resource)
return -ENOMEM; return NULL;
device = &resource->device; device = &resource->device;
acpi_init_device_object(device, handle, ACPI_BUS_TYPE_POWER); acpi_init_device_object(device, handle, ACPI_BUS_TYPE_POWER);
...@@ -959,11 +964,11 @@ int acpi_add_power_resource(acpi_handle handle) ...@@ -959,11 +964,11 @@ int acpi_add_power_resource(acpi_handle handle)
acpi_power_add_resource_to_list(resource); acpi_power_add_resource_to_list(resource);
acpi_device_add_finalize(device); acpi_device_add_finalize(device);
return 0; return device;
err: err:
acpi_release_power_resource(&device->dev); acpi_release_power_resource(&device->dev);
return result; return NULL;
} }
#ifdef CONFIG_ACPI_SLEEP #ifdef CONFIG_ACPI_SLEEP
...@@ -997,7 +1002,38 @@ void acpi_resume_power_resources(void) ...@@ -997,7 +1002,38 @@ void acpi_resume_power_resources(void)
} }
#endif #endif
void acpi_turn_off_unused_power_resources(void) static void acpi_power_turn_off_if_unused(struct acpi_power_resource *resource,
bool init)
{
if (resource->ref_count > 0)
return;
if (init) {
if (resource->users > 0)
return;
} else {
int result, state;
result = acpi_power_get_state(resource->device.handle, &state);
if (result || state == ACPI_POWER_RESOURCE_STATE_OFF)
return;
}
dev_info(&resource->device.dev, "Turning OFF\n");
__acpi_power_off(resource);
}
/**
* acpi_turn_off_unused_power_resources - Turn off power resources not in use.
* @init: Control switch.
*
* If @ainit is set, unconditionally turn off all of the ACPI power resources
* without any users.
*
* Otherwise, turn off all ACPI power resources without active references (that
* is, the ones that should be "off" at the moment) that are "on".
*/
void acpi_turn_off_unused_power_resources(bool init)
{ {
struct acpi_power_resource *resource; struct acpi_power_resource *resource;
...@@ -1006,10 +1042,7 @@ void acpi_turn_off_unused_power_resources(void) ...@@ -1006,10 +1042,7 @@ void acpi_turn_off_unused_power_resources(void)
list_for_each_entry_reverse(resource, &acpi_power_resource_list, list_node) { list_for_each_entry_reverse(resource, &acpi_power_resource_list, list_node) {
mutex_lock(&resource->resource_lock); mutex_lock(&resource->resource_lock);
if (!resource->ref_count) { acpi_power_turn_off_if_unused(resource, init);
dev_info(&resource->device.dev, "Turning OFF\n");
__acpi_power_off(resource);
}
mutex_unlock(&resource->resource_lock); mutex_unlock(&resource->resource_lock);
} }
......
...@@ -2360,7 +2360,7 @@ int __init acpi_scan_init(void) ...@@ -2360,7 +2360,7 @@ int __init acpi_scan_init(void)
} }
} }
acpi_turn_off_unused_power_resources(); acpi_turn_off_unused_power_resources(true);
acpi_scan_initialized = true; acpi_scan_initialized = true;
......
...@@ -504,7 +504,7 @@ static void acpi_pm_start(u32 acpi_state) ...@@ -504,7 +504,7 @@ static void acpi_pm_start(u32 acpi_state)
*/ */
static void acpi_pm_end(void) static void acpi_pm_end(void)
{ {
acpi_turn_off_unused_power_resources(); acpi_turn_off_unused_power_resources(false);
acpi_scan_lock_release(); acpi_scan_lock_release();
/* /*
* This is necessary in case acpi_pm_finish() is not called during a * This is necessary in case acpi_pm_finish() is not called during a
......
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