ACPI / LPSS: power on when probe() and otherwise when remove()
When LPSS drivers are compiled as a module, which is usually the case, the second probe of that driver may fail because the driver is written in an assumption that device is powered on. That is not the case for all drivers. Moreover we would like not drain power in vain. Implement ->activate() and ->dismiss() callbacks in the ACPI LPSS custom power domain. -------- 8< -------- 8< -------- 8< -------- 8< -------- 8< -------- Case 1: The I2C probe() repeat. /sys/bus/platform/devices/808622C1:00 \_SB_.PCI0.I2C1 [D3hot] /sys/bus/platform/devices/808622C1:01 \_SB_.PCI0.I2C2 [D3hot] /sys/bus/platform/devices/808622C1:02 \_SB_.PCI0.I2C3 [D3hot] /sys/bus/platform/devices/808622C1:03 \_SB_.PCI0.I2C4 [D3hot] /sys/bus/platform/devices/808622C1:05 \_SB_.PCI0.I2C6 [D3hot] /sys/bus/platform/devices/808622C1:06 \_SB_.PCI0.I2C7 [D3hot] % modprobe i2c-designware-platform i2c_designware 808622C1:00: Unknown Synopsys component type: 0xffffffff i2c_designware 808622C1:01: Unknown Synopsys component type: 0xffffffff i2c_designware 808622C1:02: Unknown Synopsys component type: 0xffffffff i2c_designware 808622C1:03: Unknown Synopsys component type: 0xffffffff i2c_designware 808622C1:05: Unknown Synopsys component type: 0xffffffff i2c_designware 808622C1:06: Unknown Synopsys component type: 0xffffffff Case 2: The power drain in case of SDHCI. /sys/bus/platform/devices/80860F14:00 \_SB_.PCI0.SDHA [D3hot] /sys/bus/platform/devices/80860F14:01 \_SB_.PCI0.SDHC [D3hot] % modprobe -r sdhci-acpi mmc0: card 0001 removed /sys/bus/platform/devices/80860F14:00 \_SB_.PCI0.SDHA [D0] /sys/bus/platform/devices/80860F14:01 \_SB_.PCI0.SDHC [D0] -------- 8< -------- 8< -------- 8< -------- 8< -------- 8< -------- Patch fixes above problems. Signed-off-by: Andy Shevchenko <andriy.shevchenko@linux.intel.com> Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
This commit is contained in:
parent
02b9854043
commit
c3a49cf35e
|
@ -599,6 +599,34 @@ static void acpi_lpss_d3_to_d0_delay(struct lpss_private_data *pdata)
|
||||||
msleep(delay);
|
msleep(delay);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int acpi_lpss_activate(struct device *dev)
|
||||||
|
{
|
||||||
|
struct lpss_private_data *pdata = acpi_driver_data(ACPI_COMPANION(dev));
|
||||||
|
int ret;
|
||||||
|
|
||||||
|
ret = acpi_dev_runtime_resume(dev);
|
||||||
|
if (ret)
|
||||||
|
return ret;
|
||||||
|
|
||||||
|
acpi_lpss_d3_to_d0_delay(pdata);
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This is called only on ->probe() stage where a device is either in
|
||||||
|
* known state defined by BIOS or most likely powered off. Due to this
|
||||||
|
* we have to deassert reset line to be sure that ->probe() will
|
||||||
|
* recognize the device.
|
||||||
|
*/
|
||||||
|
if (pdata->dev_desc->flags & LPSS_SAVE_CTX)
|
||||||
|
lpss_deassert_reset(pdata);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void acpi_lpss_dismiss(struct device *dev)
|
||||||
|
{
|
||||||
|
acpi_dev_runtime_suspend(dev);
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef CONFIG_PM_SLEEP
|
#ifdef CONFIG_PM_SLEEP
|
||||||
static int acpi_lpss_suspend_late(struct device *dev)
|
static int acpi_lpss_suspend_late(struct device *dev)
|
||||||
{
|
{
|
||||||
|
@ -667,6 +695,10 @@ static int acpi_lpss_runtime_resume(struct device *dev)
|
||||||
#endif /* CONFIG_PM */
|
#endif /* CONFIG_PM */
|
||||||
|
|
||||||
static struct dev_pm_domain acpi_lpss_pm_domain = {
|
static struct dev_pm_domain acpi_lpss_pm_domain = {
|
||||||
|
#ifdef CONFIG_PM
|
||||||
|
.activate = acpi_lpss_activate,
|
||||||
|
.dismiss = acpi_lpss_dismiss,
|
||||||
|
#endif
|
||||||
.ops = {
|
.ops = {
|
||||||
#ifdef CONFIG_PM
|
#ifdef CONFIG_PM
|
||||||
#ifdef CONFIG_PM_SLEEP
|
#ifdef CONFIG_PM_SLEEP
|
||||||
|
|
Loading…
Reference in New Issue