PCI PM: Restore standard config registers of all devices early
authorRafael J. Wysocki <rjw@sisk.pl>
Fri, 16 Jan 2009 20:54:43 +0000 (21:54 +0100)
committerJesse Barnes <jbarnes@virtuousgeek.org>
Fri, 16 Jan 2009 20:57:58 +0000 (12:57 -0800)
commitaa8c6c93747f7b55fa11e1624fec8ca33763a805
treee40bf643ec9916dd2738ef9aaafdfa49ad8b4781
parent0db29af1e767464d71b89410d61a1e5b668d0370
PCI PM: Restore standard config registers of all devices early

There is a problem in our handling of suspend-resume of PCI devices that
many of them have their standard config registers restored with
interrupts enabled and they are put into the full power state with
interrupts enabled as well.  This may lead to the following scenario:
  * an interrupt vector is shared between two or more devices
  * one device is resumed earlier and generates an interrupt
  * the interrupt handler of another device tries to handle it and
    attempts to access the device the config space of which hasn't been
    restored yet and/or which still is in a low power state
  * the system crashes as a result

To prevent this from happening we should restore the standard
configuration registers of all devices with interrupts disabled and we
should put them into the D0 power state right after that.
Unfortunately, this cannot be done using the existing
pci_set_power_state(), because it can sleep.  Also, to do it we have to
make sure that the config spaces of all devices were actually saved
during suspend.

Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Acked-by: Linus Torvalds <torvalds@linux-foundation.org>
Signed-off-by: Jesse Barnes <jbarnes@virtuousgeek.org>
drivers/pci/pci-driver.c
drivers/pci/pci.c
drivers/pci/pci.h
include/linux/pci.h