diff options
Diffstat (limited to 'drivers/pci')
-rw-r--r-- | drivers/pci/pci-acpi.c | 19 | ||||
-rw-r--r-- | drivers/pci/pci.c | 11 | ||||
-rw-r--r-- | drivers/pci/pci.h | 1 |
3 files changed, 29 insertions, 2 deletions
diff --git a/drivers/pci/pci-acpi.c b/drivers/pci/pci-acpi.c index 8eb599708de8..a0d43ea872df 100644 --- a/drivers/pci/pci-acpi.c +++ b/drivers/pci/pci-acpi.c | |||
@@ -253,6 +253,24 @@ static int acpi_pci_choose_state(struct pci_dev *pdev, pm_message_t state) | |||
253 | return -ENODEV; | 253 | return -ENODEV; |
254 | } | 254 | } |
255 | 255 | ||
256 | static int acpi_pci_set_power_state(struct pci_dev *dev, pci_power_t state) | ||
257 | { | ||
258 | acpi_handle handle = DEVICE_ACPI_HANDLE(&dev->dev); | ||
259 | static int state_conv[] = { | ||
260 | [0] = 0, | ||
261 | [1] = 1, | ||
262 | [2] = 2, | ||
263 | [3] = 3, | ||
264 | [4] = 3 | ||
265 | }; | ||
266 | int acpi_state = state_conv[(int __force) state]; | ||
267 | |||
268 | if (!handle) | ||
269 | return -ENODEV; | ||
270 | return acpi_bus_set_power(handle, acpi_state); | ||
271 | } | ||
272 | |||
273 | |||
256 | /* ACPI bus type */ | 274 | /* ACPI bus type */ |
257 | static int pci_acpi_find_device(struct device *dev, acpi_handle *handle) | 275 | static int pci_acpi_find_device(struct device *dev, acpi_handle *handle) |
258 | { | 276 | { |
@@ -300,6 +318,7 @@ static int __init pci_acpi_init(void) | |||
300 | if (ret) | 318 | if (ret) |
301 | return 0; | 319 | return 0; |
302 | platform_pci_choose_state = acpi_pci_choose_state; | 320 | platform_pci_choose_state = acpi_pci_choose_state; |
321 | platform_pci_set_power_state = acpi_pci_set_power_state; | ||
303 | return 0; | 322 | return 0; |
304 | } | 323 | } |
305 | arch_initcall(pci_acpi_init); | 324 | arch_initcall(pci_acpi_init); |
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 5af941807785..a1c66e8ea5f2 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
@@ -235,7 +235,7 @@ pci_find_parent_resource(const struct pci_dev *dev, struct resource *res) | |||
235 | * -EIO if device does not support PCI PM. | 235 | * -EIO if device does not support PCI PM. |
236 | * 0 if we can successfully change the power state. | 236 | * 0 if we can successfully change the power state. |
237 | */ | 237 | */ |
238 | 238 | int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t t) = NULL; | |
239 | int | 239 | int |
240 | pci_set_power_state(struct pci_dev *dev, pci_power_t state) | 240 | pci_set_power_state(struct pci_dev *dev, pci_power_t state) |
241 | { | 241 | { |
@@ -299,8 +299,15 @@ pci_set_power_state(struct pci_dev *dev, pci_power_t state) | |||
299 | msleep(10); | 299 | msleep(10); |
300 | else if (state == PCI_D2 || dev->current_state == PCI_D2) | 300 | else if (state == PCI_D2 || dev->current_state == PCI_D2) |
301 | udelay(200); | 301 | udelay(200); |
302 | dev->current_state = state; | ||
303 | 302 | ||
303 | /* | ||
304 | * Give firmware a chance to be called, such as ACPI _PRx, _PSx | ||
305 | * Firmware method after natice method ? | ||
306 | */ | ||
307 | if (platform_pci_set_power_state) | ||
308 | platform_pci_set_power_state(dev, state); | ||
309 | |||
310 | dev->current_state = state; | ||
304 | return 0; | 311 | return 0; |
305 | } | 312 | } |
306 | 313 | ||
diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h index 25c44922f7db..d94d7af4f7a0 100644 --- a/drivers/pci/pci.h +++ b/drivers/pci/pci.h | |||
@@ -13,6 +13,7 @@ extern int pci_bus_alloc_resource(struct pci_bus *bus, struct resource *res, | |||
13 | void *alignf_data); | 13 | void *alignf_data); |
14 | /* Firmware callbacks */ | 14 | /* Firmware callbacks */ |
15 | extern int (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state); | 15 | extern int (*platform_pci_choose_state)(struct pci_dev *dev, pm_message_t state); |
16 | extern int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t state); | ||
16 | 17 | ||
17 | /* PCI /proc functions */ | 18 | /* PCI /proc functions */ |
18 | #ifdef CONFIG_PROC_FS | 19 | #ifdef CONFIG_PROC_FS |