diff options
Diffstat (limited to 'drivers')
| -rw-r--r-- | drivers/pci/pci.c | 59 | ||||
| -rw-r--r-- | drivers/pci/setup-res.c | 2 |
2 files changed, 56 insertions, 5 deletions
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 1b34fc56067e..65ea7d25f691 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
| @@ -222,6 +222,37 @@ pci_find_parent_resource(const struct pci_dev *dev, struct resource *res) | |||
| 222 | } | 222 | } |
| 223 | 223 | ||
| 224 | /** | 224 | /** |
| 225 | * pci_restore_bars - restore a devices BAR values (e.g. after wake-up) | ||
| 226 | * @dev: PCI device to have its BARs restored | ||
| 227 | * | ||
| 228 | * Restore the BAR values for a given device, so as to make it | ||
| 229 | * accessible by its driver. | ||
| 230 | */ | ||
| 231 | void | ||
| 232 | pci_restore_bars(struct pci_dev *dev) | ||
| 233 | { | ||
| 234 | int i, numres; | ||
| 235 | |||
| 236 | switch (dev->hdr_type) { | ||
| 237 | case PCI_HEADER_TYPE_NORMAL: | ||
| 238 | numres = 6; | ||
| 239 | break; | ||
| 240 | case PCI_HEADER_TYPE_BRIDGE: | ||
| 241 | numres = 2; | ||
| 242 | break; | ||
| 243 | case PCI_HEADER_TYPE_CARDBUS: | ||
| 244 | numres = 1; | ||
| 245 | break; | ||
| 246 | default: | ||
| 247 | /* Should never get here, but just in case... */ | ||
| 248 | return; | ||
| 249 | } | ||
| 250 | |||
| 251 | for (i = 0; i < numres; i ++) | ||
| 252 | pci_update_resource(dev, &dev->resource[i], i); | ||
| 253 | } | ||
| 254 | |||
| 255 | /** | ||
| 225 | * pci_set_power_state - Set the power state of a PCI device | 256 | * pci_set_power_state - Set the power state of a PCI device |
| 226 | * @dev: PCI device to be suspended | 257 | * @dev: PCI device to be suspended |
| 227 | * @state: PCI power state (D0, D1, D2, D3hot, D3cold) we're entering | 258 | * @state: PCI power state (D0, D1, D2, D3hot, D3cold) we're entering |
| @@ -239,7 +270,7 @@ int (*platform_pci_set_power_state)(struct pci_dev *dev, pci_power_t t); | |||
| 239 | int | 270 | int |
| 240 | pci_set_power_state(struct pci_dev *dev, pci_power_t state) | 271 | pci_set_power_state(struct pci_dev *dev, pci_power_t state) |
| 241 | { | 272 | { |
| 242 | int pm; | 273 | int pm, need_restore = 0; |
| 243 | u16 pmcsr, pmc; | 274 | u16 pmcsr, pmc; |
| 244 | 275 | ||
| 245 | /* bound the state we're entering */ | 276 | /* bound the state we're entering */ |
| @@ -278,14 +309,17 @@ pci_set_power_state(struct pci_dev *dev, pci_power_t state) | |||
| 278 | return -EIO; | 309 | return -EIO; |
| 279 | } | 310 | } |
| 280 | 311 | ||
| 312 | pci_read_config_word(dev, pm + PCI_PM_CTRL, &pmcsr); | ||
| 313 | |||
| 281 | /* If we're in D3, force entire word to 0. | 314 | /* If we're in D3, force entire word to 0. |
| 282 | * This doesn't affect PME_Status, disables PME_En, and | 315 | * This doesn't affect PME_Status, disables PME_En, and |
| 283 | * sets PowerState to 0. | 316 | * sets PowerState to 0. |
| 284 | */ | 317 | */ |
| 285 | if (dev->current_state >= PCI_D3hot) | 318 | if (dev->current_state >= PCI_D3hot) { |
| 319 | if (!(pmcsr & PCI_PM_CTRL_NO_SOFT_RESET)) | ||
| 320 | need_restore = 1; | ||
| 286 | pmcsr = 0; | 321 | pmcsr = 0; |
| 287 | else { | 322 | } else { |
| 288 | pci_read_config_word(dev, pm + PCI_PM_CTRL, &pmcsr); | ||
| 289 | pmcsr &= ~PCI_PM_CTRL_STATE_MASK; | 323 | pmcsr &= ~PCI_PM_CTRL_STATE_MASK; |
| 290 | pmcsr |= state; | 324 | pmcsr |= state; |
| 291 | } | 325 | } |
| @@ -308,6 +342,22 @@ pci_set_power_state(struct pci_dev *dev, pci_power_t state) | |||
| 308 | platform_pci_set_power_state(dev, state); | 342 | platform_pci_set_power_state(dev, state); |
| 309 | 343 | ||
| 310 | dev->current_state = state; | 344 | dev->current_state = state; |
| 345 | |||
| 346 | /* According to section 5.4.1 of the "PCI BUS POWER MANAGEMENT | ||
| 347 | * INTERFACE SPECIFICATION, REV. 1.2", a device transitioning | ||
| 348 | * from D3hot to D0 _may_ perform an internal reset, thereby | ||
| 349 | * going to "D0 Uninitialized" rather than "D0 Initialized". | ||
| 350 | * For example, at least some versions of the 3c905B and the | ||
| 351 | * 3c556B exhibit this behaviour. | ||
| 352 | * | ||
| 353 | * At least some laptop BIOSen (e.g. the Thinkpad T21) leave | ||
| 354 | * devices in a D3hot state at boot. Consequently, we need to | ||
| 355 | * restore at least the BARs so that the device will be | ||
| 356 | * accessible to its driver. | ||
| 357 | */ | ||
| 358 | if (need_restore) | ||
| 359 | pci_restore_bars(dev); | ||
| 360 | |||
| 311 | return 0; | 361 | return 0; |
| 312 | } | 362 | } |
| 313 | 363 | ||
| @@ -805,6 +855,7 @@ struct pci_dev *isa_bridge; | |||
| 805 | EXPORT_SYMBOL(isa_bridge); | 855 | EXPORT_SYMBOL(isa_bridge); |
| 806 | #endif | 856 | #endif |
| 807 | 857 | ||
| 858 | EXPORT_SYMBOL_GPL(pci_restore_bars); | ||
| 808 | EXPORT_SYMBOL(pci_enable_device_bars); | 859 | EXPORT_SYMBOL(pci_enable_device_bars); |
| 809 | EXPORT_SYMBOL(pci_enable_device); | 860 | EXPORT_SYMBOL(pci_enable_device); |
| 810 | EXPORT_SYMBOL(pci_disable_device); | 861 | EXPORT_SYMBOL(pci_disable_device); |
diff --git a/drivers/pci/setup-res.c b/drivers/pci/setup-res.c index 1ca21d2ba11c..878fd0a65c02 100644 --- a/drivers/pci/setup-res.c +++ b/drivers/pci/setup-res.c | |||
| @@ -26,7 +26,7 @@ | |||
| 26 | #include "pci.h" | 26 | #include "pci.h" |
| 27 | 27 | ||
| 28 | 28 | ||
| 29 | static void | 29 | void |
| 30 | pci_update_resource(struct pci_dev *dev, struct resource *res, int resno) | 30 | pci_update_resource(struct pci_dev *dev, struct resource *res, int resno) |
| 31 | { | 31 | { |
| 32 | struct pci_bus_region region; | 32 | struct pci_bus_region region; |
