Commit 94c7993f authored by Mika Westerberg's avatar Mika Westerberg Committed by Bjorn Helgaas

PCI/portdrv: Add runtime PM hooks for port service drivers

When PCIe port is runtime suspended/resumed some extra steps might be
needed to be executed from the port service driver side. For instance we
may need to disable PCIe hotplug interrupt to prevent it from triggering
immediately when PCIe link to the downstream component goes down.

To make the above possible add optional ->runtime_suspend() and
->runtime_resume() callbacks to struct pcie_port_service_driver and call
them for each port service in runtime suspend/resume callbacks of portdrv.
Signed-off-by: default avatarMika Westerberg <mika.westerberg@linux.intel.com>
[bhelgaas: adjust "slot->state" for 5790a9c7 ("PCI: pciehp: Unify
controller and slot structs")]
Signed-off-by: default avatarBjorn Helgaas <bhelgaas@google.com>
Reviewed-by: default avatarRafael J. Wysocki <rafael.j.wysocki@intel.com>
parent 52be9464
...@@ -76,6 +76,8 @@ struct pcie_port_service_driver { ...@@ -76,6 +76,8 @@ struct pcie_port_service_driver {
int (*suspend) (struct pcie_device *dev); int (*suspend) (struct pcie_device *dev);
int (*resume_noirq) (struct pcie_device *dev); int (*resume_noirq) (struct pcie_device *dev);
int (*resume) (struct pcie_device *dev); int (*resume) (struct pcie_device *dev);
int (*runtime_suspend) (struct pcie_device *dev);
int (*runtime_resume) (struct pcie_device *dev);
/* Device driver may resume normal operations */ /* Device driver may resume normal operations */
void (*error_resume)(struct pci_dev *dev); void (*error_resume)(struct pci_dev *dev);
...@@ -109,6 +111,8 @@ int pcie_port_device_register(struct pci_dev *dev); ...@@ -109,6 +111,8 @@ int pcie_port_device_register(struct pci_dev *dev);
int pcie_port_device_suspend(struct device *dev); int pcie_port_device_suspend(struct device *dev);
int pcie_port_device_resume_noirq(struct device *dev); int pcie_port_device_resume_noirq(struct device *dev);
int pcie_port_device_resume(struct device *dev); int pcie_port_device_resume(struct device *dev);
int pcie_port_device_runtime_suspend(struct device *dev);
int pcie_port_device_runtime_resume(struct device *dev);
#endif #endif
void pcie_port_device_remove(struct pci_dev *dev); void pcie_port_device_remove(struct pci_dev *dev);
int __must_check pcie_port_bus_register(void); int __must_check pcie_port_bus_register(void);
......
...@@ -395,6 +395,26 @@ int pcie_port_device_resume(struct device *dev) ...@@ -395,6 +395,26 @@ int pcie_port_device_resume(struct device *dev)
size_t off = offsetof(struct pcie_port_service_driver, resume); size_t off = offsetof(struct pcie_port_service_driver, resume);
return device_for_each_child(dev, &off, pm_iter); return device_for_each_child(dev, &off, pm_iter);
} }
/**
* pcie_port_device_runtime_suspend - runtime suspend port services
* @dev: PCI Express port to handle
*/
int pcie_port_device_runtime_suspend(struct device *dev)
{
size_t off = offsetof(struct pcie_port_service_driver, runtime_suspend);
return device_for_each_child(dev, &off, pm_iter);
}
/**
* pcie_port_device_runtime_resume - runtime resume port services
* @dev: PCI Express port to handle
*/
int pcie_port_device_runtime_resume(struct device *dev)
{
size_t off = offsetof(struct pcie_port_service_driver, runtime_resume);
return device_for_each_child(dev, &off, pm_iter);
}
#endif /* PM */ #endif /* PM */
static int remove_iter(struct device *dev, void *data) static int remove_iter(struct device *dev, void *data)
......
...@@ -45,12 +45,10 @@ __setup("pcie_ports=", pcie_port_setup); ...@@ -45,12 +45,10 @@ __setup("pcie_ports=", pcie_port_setup);
#ifdef CONFIG_PM #ifdef CONFIG_PM
static int pcie_port_runtime_suspend(struct device *dev) static int pcie_port_runtime_suspend(struct device *dev)
{ {
return to_pci_dev(dev)->bridge_d3 ? 0 : -EBUSY; if (!to_pci_dev(dev)->bridge_d3)
} return -EBUSY;
static int pcie_port_runtime_resume(struct device *dev) return pcie_port_device_runtime_suspend(dev);
{
return 0;
} }
static int pcie_port_runtime_idle(struct device *dev) static int pcie_port_runtime_idle(struct device *dev)
...@@ -73,7 +71,7 @@ static const struct dev_pm_ops pcie_portdrv_pm_ops = { ...@@ -73,7 +71,7 @@ static const struct dev_pm_ops pcie_portdrv_pm_ops = {
.restore_noirq = pcie_port_device_resume_noirq, .restore_noirq = pcie_port_device_resume_noirq,
.restore = pcie_port_device_resume, .restore = pcie_port_device_resume,
.runtime_suspend = pcie_port_runtime_suspend, .runtime_suspend = pcie_port_runtime_suspend,
.runtime_resume = pcie_port_runtime_resume, .runtime_resume = pcie_port_device_runtime_resume,
.runtime_idle = pcie_port_runtime_idle, .runtime_idle = pcie_port_runtime_idle,
}; };
......
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