diff options
author | Rafael J. Wysocki <rjw@sisk.pl> | 2010-02-17 17:36:58 -0500 |
---|---|---|
committer | Jesse Barnes <jbarnes@virtuousgeek.org> | 2010-02-22 19:20:24 -0500 |
commit | 58ff463396ad00828e922d50998787e97fd32512 (patch) | |
tree | 4964b07518a02736a23839d89bb7ac4a065dede8 /drivers/pci | |
parent | 6d3be84aab461815978d970aa45f5bc9e52dd772 (diff) |
PCI PM: Add function for checking PME status of devices
Add function pci_check_pme_status() that will check the PME status
bit of given device and clear it along with the PME enable bit. It
will be necessary for PCI run-time power management.
Based on a patch from Shaohua Li <shaohua.li@intel.com>
Signed-off-by: Rafael J. Wysocki <rjw@sisk.pl>
Signed-off-by: Jesse Barnes <jbarnes@virtuousgeek.org>
Diffstat (limited to 'drivers/pci')
-rw-r--r-- | drivers/pci/pci.c | 35 | ||||
-rw-r--r-- | drivers/pci/pci.h | 1 |
2 files changed, 36 insertions, 0 deletions
diff --git a/drivers/pci/pci.c b/drivers/pci/pci.c index 1f9be53c39ba..5723446544fd 100644 --- a/drivers/pci/pci.c +++ b/drivers/pci/pci.c | |||
@@ -1195,6 +1195,41 @@ int pci_set_pcie_reset_state(struct pci_dev *dev, enum pcie_reset_state state) | |||
1195 | } | 1195 | } |
1196 | 1196 | ||
1197 | /** | 1197 | /** |
1198 | * pci_check_pme_status - Check if given device has generated PME. | ||
1199 | * @dev: Device to check. | ||
1200 | * | ||
1201 | * Check the PME status of the device and if set, clear it and clear PME enable | ||
1202 | * (if set). Return 'true' if PME status and PME enable were both set or | ||
1203 | * 'false' otherwise. | ||
1204 | */ | ||
1205 | bool pci_check_pme_status(struct pci_dev *dev) | ||
1206 | { | ||
1207 | int pmcsr_pos; | ||
1208 | u16 pmcsr; | ||
1209 | bool ret = false; | ||
1210 | |||
1211 | if (!dev->pm_cap) | ||
1212 | return false; | ||
1213 | |||
1214 | pmcsr_pos = dev->pm_cap + PCI_PM_CTRL; | ||
1215 | pci_read_config_word(dev, pmcsr_pos, &pmcsr); | ||
1216 | if (!(pmcsr & PCI_PM_CTRL_PME_STATUS)) | ||
1217 | return false; | ||
1218 | |||
1219 | /* Clear PME status. */ | ||
1220 | pmcsr |= PCI_PM_CTRL_PME_STATUS; | ||
1221 | if (pmcsr & PCI_PM_CTRL_PME_ENABLE) { | ||
1222 | /* Disable PME to avoid interrupt flood. */ | ||
1223 | pmcsr &= ~PCI_PM_CTRL_PME_ENABLE; | ||
1224 | ret = true; | ||
1225 | } | ||
1226 | |||
1227 | pci_write_config_word(dev, pmcsr_pos, pmcsr); | ||
1228 | |||
1229 | return ret; | ||
1230 | } | ||
1231 | |||
1232 | /** | ||
1198 | * pci_pme_capable - check the capability of PCI device to generate PME# | 1233 | * pci_pme_capable - check the capability of PCI device to generate PME# |
1199 | * @dev: PCI device to handle. | 1234 | * @dev: PCI device to handle. |
1200 | * @state: PCI state from which device will issue PME#. | 1235 | * @state: PCI state from which device will issue PME#. |
diff --git a/drivers/pci/pci.h b/drivers/pci/pci.h index 5d169bc3ccd1..b95b0a077d31 100644 --- a/drivers/pci/pci.h +++ b/drivers/pci/pci.h | |||
@@ -49,6 +49,7 @@ struct pci_platform_pm_ops { | |||
49 | extern int pci_set_platform_pm(struct pci_platform_pm_ops *ops); | 49 | extern int pci_set_platform_pm(struct pci_platform_pm_ops *ops); |
50 | extern void pci_update_current_state(struct pci_dev *dev, pci_power_t state); | 50 | extern void pci_update_current_state(struct pci_dev *dev, pci_power_t state); |
51 | extern void pci_disable_enabled_device(struct pci_dev *dev); | 51 | extern void pci_disable_enabled_device(struct pci_dev *dev); |
52 | extern bool pci_check_pme_status(struct pci_dev *dev); | ||
52 | extern void pci_pm_init(struct pci_dev *dev); | 53 | extern void pci_pm_init(struct pci_dev *dev); |
53 | extern void platform_pci_wakeup_init(struct pci_dev *dev); | 54 | extern void platform_pci_wakeup_init(struct pci_dev *dev); |
54 | extern void pci_allocate_cap_save_buffers(struct pci_dev *dev); | 55 | extern void pci_allocate_cap_save_buffers(struct pci_dev *dev); |