Linux kernel mirror (for testing) git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
kernel os linux

ACPI / PM: Fix corner case in acpi_bus_update_power()

The role of acpi_bus_update_power() is to update the given ACPI
device object's power.state field to reflect the current physical
state of the device (as inferred from the configuration of power
resources and _PSC, if available). For this purpose it calls
acpi_device_set_power() that should update the power resources'
reference counters and set power.state as appropriate. However,
that doesn't work if the "new" state is D1, D2 or D3hot and the
the current value of power.state means D3cold, because in that
case acpi_device_set_power() will refuse to transition the device
from D3cold to non-D0.

To address this problem, make acpi_bus_update_power() call
acpi_power_transition() directly to update the power resources'
reference counters and only use acpi_device_set_power() to put
the device into D0 if the current physical state of it cannot
be determined.

Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Cc: 3.9+ <stable@vger.kernel.org>

+18 -5
+18 -5
drivers/acpi/device_pm.c
··· 324 324 if (result) 325 325 return result; 326 326 327 - if (state == ACPI_STATE_UNKNOWN) 327 + if (state == ACPI_STATE_UNKNOWN) { 328 328 state = ACPI_STATE_D0; 329 - 330 - result = acpi_device_set_power(device, state); 331 - if (!result && state_p) 329 + result = acpi_device_set_power(device, state); 330 + if (result) 331 + return result; 332 + } else { 333 + if (device->power.flags.power_resources) { 334 + /* 335 + * We don't need to really switch the state, bu we need 336 + * to update the power resources' reference counters. 337 + */ 338 + result = acpi_power_transition(device, state); 339 + if (result) 340 + return result; 341 + } 342 + device->power.state = state; 343 + } 344 + if (state_p) 332 345 *state_p = state; 333 346 334 - return result; 347 + return 0; 335 348 } 336 349 EXPORT_SYMBOL_GPL(acpi_bus_update_power); 337 350