Commit b913100d authored by David Shaohua Li's avatar David Shaohua Li Committed by Len Brown

[ACPI] pci_set_power_state() now calls

	platform_pci_set_power_state()
		and ACPI can answer

http://bugzilla.kernel.org/show_bug.cgi?id=4277Signed-off-by: default avatarDavid Shaohua Li <shaohua.li@intel.com>
Signed-off-by: default avatarLen Brown <len.brown@intel.com>
parent 0f64474b
...@@ -212,6 +212,12 @@ acpi_bus_set_power ( ...@@ -212,6 +212,12 @@ acpi_bus_set_power (
ACPI_DEBUG_PRINT((ACPI_DB_WARN, "Device is not power manageable\n")); ACPI_DEBUG_PRINT((ACPI_DB_WARN, "Device is not power manageable\n"));
return_VALUE(-ENODEV); return_VALUE(-ENODEV);
} }
/*
* Get device's current power state if it's unknown
* This means device power state isn't initialized or previous setting failed
*/
if (device->power.state == ACPI_STATE_UNKNOWN)
acpi_bus_get_power(device->handle, &device->power.state);
if (state == device->power.state) { if (state == device->power.state) {
ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Device is already at D%d\n", state)); ACPI_DEBUG_PRINT((ACPI_DB_INFO, "Device is already at D%d\n", state));
return_VALUE(0); return_VALUE(0);
...@@ -231,7 +237,7 @@ acpi_bus_set_power ( ...@@ -231,7 +237,7 @@ acpi_bus_set_power (
* On transitions to a high-powered state we first apply power (via * On transitions to a high-powered state we first apply power (via
* power resources) then evalute _PSx. Conversly for transitions to * power resources) then evalute _PSx. Conversly for transitions to
* a lower-powered state. * a lower-powered state.
*/ */
if (state < device->power.state) { if (state < device->power.state) {
if (device->power.flags.power_resources) { if (device->power.flags.power_resources) {
result = acpi_power_transition(device, state); result = acpi_power_transition(device, state);
......
...@@ -253,6 +253,24 @@ static int acpi_pci_choose_state(struct pci_dev *pdev, pm_message_t state) ...@@ -253,6 +253,24 @@ static int acpi_pci_choose_state(struct pci_dev *pdev, pm_message_t state)
return -ENODEV; return -ENODEV;
} }
static int acpi_pci_set_power_state(struct pci_dev *dev, pci_power_t state)
{
acpi_handle handle = DEVICE_ACPI_HANDLE(&dev->dev);
static int state_conv[] = {
[0] = 0,
[1] = 1,
[2] = 2,
[3] = 3,
[4] = 3
};
int acpi_state = state_conv[(int __force) state];
if (!handle)
return -ENODEV;
return acpi_bus_set_power(handle, acpi_state);
}
/* ACPI bus type */ /* ACPI bus type */
static int pci_acpi_find_device(struct device *dev, acpi_handle *handle) static int pci_acpi_find_device(struct device *dev, acpi_handle *handle)
{ {
...@@ -300,6 +318,7 @@ static int __init pci_acpi_init(void) ...@@ -300,6 +318,7 @@ static int __init pci_acpi_init(void)
if (ret) if (ret)
return 0; return 0;
platform_pci_choose_state = acpi_pci_choose_state; platform_pci_choose_state = acpi_pci_choose_state;
platform_pci_set_power_state = acpi_pci_set_power_state;
return 0; return 0;
} }
arch_initcall(pci_acpi_init); arch_initcall(pci_acpi_init);
...@@ -235,7 +235,7 @@ pci_find_parent_resource(const struct pci_dev *dev, struct resource *res) ...@@ -235,7 +235,7 @@ pci_find_parent_resource(const struct pci_dev *dev, struct resource *res)
* -EIO if device does not support PCI PM. * -EIO if device does not support PCI PM.
* 0 if we can successfully change the power state. * 0 if we can successfully change the power state.
*/ */
int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t t) = NULL;
int int
pci_set_power_state(struct pci_dev *dev, pci_power_t state) pci_set_power_state(struct pci_dev *dev, pci_power_t state)
{ {
...@@ -299,8 +299,15 @@ pci_set_power_state(struct pci_dev *dev, pci_power_t state) ...@@ -299,8 +299,15 @@ pci_set_power_state(struct pci_dev *dev, pci_power_t state)
msleep(10); msleep(10);
else if (state == PCI_D2 || dev->current_state == PCI_D2) else if (state == PCI_D2 || dev->current_state == PCI_D2)
udelay(200); udelay(200);
dev->current_state = state;
/*
* Give firmware a chance to be called, such as ACPI _PRx, _PSx
* Firmware method after natice method ?
*/
if (platform_pci_set_power_state)
platform_pci_set_power_state(dev, state);
dev->current_state = state;
return 0; return 0;
} }
......
...@@ -13,6 +13,7 @@ extern int pci_bus_alloc_resource(struct pci_bus *bus, struct resource *res, ...@@ -13,6 +13,7 @@ extern int pci_bus_alloc_resource(struct pci_bus *bus, struct resource *res,
void *alignf_data); void *alignf_data);
/* Firmware callbacks */ /* Firmware callbacks */
extern int (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state); extern int (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state);
extern int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t state);
/* PCI /proc functions */ /* PCI /proc functions */
#ifdef CONFIG_PROC_FS #ifdef CONFIG_PROC_FS
......
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