PM / Domains: Fix genpd to deal with drivers returning 1 from ->prepare()
authorUlf Hansson <ulf.hansson@linaro.org>
Wed, 8 Nov 2017 09:11:02 +0000 (10:11 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Sun, 10 Dec 2017 12:40:38 +0000 (13:40 +0100)
[ Upstream commit 5241ab40f6e742f8a1631f8826faf6dc6412b3b5 ]

During system-wide PM, genpd relies on its PM callbacks to be invoked for
all its attached devices, as to deal with powering off/on the PM domain. In
other words, genpd is not compatible with the direct_complete path, if
executed by the PM core for any of its attached devices.

However, when genpd's ->prepare() callback invokes pm_generic_prepare(), it
does not take into account that it may return 1. Instead it treats that as
an error internally and expects the PM core to abort the prepare phase and
roll back. This leads to genpd not properly powering on/off the PM domain,
because its internal counters gets wrongly balanced.

To fix the behaviour, allow drivers to return 1 from their ->prepare()
callbacks, but let's return 0 from genpd's ->prepare() callback in such
case, as that prevents the PM core from running the direct_complete path
for the device.

Signed-off-by: Ulf Hansson <ulf.hansson@linaro.org>
Signed-off-by: Rafael J. Wysocki <rafael.j.wysocki@intel.com>
Signed-off-by: Sasha Levin <alexander.levin@verizon.com>
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
drivers/base/power/domain.c

index e8ca5e2cf1e51e5ad17b08274a26e3251fdb0f93..70f8904f46a31d4df75b99c5abbfe9cb83904aa2 100644 (file)
@@ -921,7 +921,7 @@ static int pm_genpd_prepare(struct device *dev)
        genpd_unlock(genpd);
 
        ret = pm_generic_prepare(dev);
-       if (ret) {
+       if (ret < 0) {
                genpd_lock(genpd);
 
                genpd->prepared_count--;
@@ -929,7 +929,8 @@ static int pm_genpd_prepare(struct device *dev)
                genpd_unlock(genpd);
        }
 
-       return ret;
+       /* Never return 1, as genpd don't cope with the direct_complete path. */
+       return ret >= 0 ? 0 : ret;
 }
 
 /**