* [PATCH] PM / Domains: Call driver's noirq callbacks @ 2017-06-20 12:05 Mikko Perttunen 2017-06-20 12:47 ` Ulf Hansson 0 siblings, 1 reply; 10+ messages in thread From: Mikko Perttunen @ 2017-06-20 12:05 UTC (permalink / raw) To: rjw, khilman, ulf.hansson; +Cc: linux-pm, linux-kernel, Mikko Perttunen Currently genpd installs its own suspend_noirq and resume_noirq callbacks, but never calls down to the driver's corresponding callbacks. Add these calls. Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> --- drivers/base/power/domain.c | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c index d3f1d96f75e9..c3b6e6018c02 100644 --- a/drivers/base/power/domain.c +++ b/drivers/base/power/domain.c @@ -925,6 +925,10 @@ static int pm_genpd_suspend_noirq(struct device *dev) return ret; } + ret = pm_generic_suspend_noirq(dev); + if (ret) + return ret; + genpd_lock(genpd); genpd->suspended_count++; genpd_sync_power_off(genpd, true, 0); @@ -958,6 +962,10 @@ static int pm_genpd_resume_noirq(struct device *dev) genpd->suspended_count--; genpd_unlock(genpd); + ret = pm_generic_resume_noirq(dev); + if (ret) + return ret; + if (genpd->dev_ops.stop && genpd->dev_ops.start) ret = pm_runtime_force_resume(dev); -- 2.1.4 ^ permalink raw reply related [flat|nested] 10+ messages in thread
* Re: [PATCH] PM / Domains: Call driver's noirq callbacks 2017-06-20 12:05 [PATCH] PM / Domains: Call driver's noirq callbacks Mikko Perttunen @ 2017-06-20 12:47 ` Ulf Hansson 2017-06-20 13:04 ` Mikko Perttunen 2017-06-20 13:38 ` [PATCH v2] " Mikko Perttunen 0 siblings, 2 replies; 10+ messages in thread From: Ulf Hansson @ 2017-06-20 12:47 UTC (permalink / raw) To: Mikko Perttunen Cc: Rafael J. Wysocki, Kevin Hilman, linux-pm@vger.kernel.org, linux-kernel@vger.kernel.org On 20 June 2017 at 14:05, Mikko Perttunen <mperttunen@nvidia.com> wrote: > Currently genpd installs its own suspend_noirq and resume_noirq > callbacks, but never calls down to the driver's corresponding > callbacks. Add these calls. > > Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> > --- > > drivers/base/power/domain.c | 8 ++++++++ > 1 file changed, 8 insertions(+) > > diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c > index d3f1d96f75e9..c3b6e6018c02 100644 > --- a/drivers/base/power/domain.c > +++ b/drivers/base/power/domain.c > @@ -925,6 +925,10 @@ static int pm_genpd_suspend_noirq(struct device *dev) > return ret; > } > > + ret = pm_generic_suspend_noirq(dev); > + if (ret) > + return ret; Two things: 1) I would move this a couple of lines further above, as to avoid pm_runtime_force_suspend() from being called before, as it may suspend the device before the noirq callbacks gets to run. 2) pm_genpd_suspend_noirq() is also assigned to the ->poweroff_noirq() callback. Certainly we don't want run the *suspend* variants of the callback, but rather the *poweroff* variants in that case. > + > genpd_lock(genpd); > genpd->suspended_count++; > genpd_sync_power_off(genpd, true, 0); > @@ -958,6 +962,10 @@ static int pm_genpd_resume_noirq(struct device *dev) > genpd->suspended_count--; > genpd_unlock(genpd); > > + ret = pm_generic_resume_noirq(dev); > + if (ret) > + return ret; > + > if (genpd->dev_ops.stop && genpd->dev_ops.start) > ret = pm_runtime_force_resume(dev); > > -- > 2.1.4 > Kind regards Uffe ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH] PM / Domains: Call driver's noirq callbacks 2017-06-20 12:47 ` Ulf Hansson @ 2017-06-20 13:04 ` Mikko Perttunen 2017-06-20 13:38 ` [PATCH v2] " Mikko Perttunen 1 sibling, 0 replies; 10+ messages in thread From: Mikko Perttunen @ 2017-06-20 13:04 UTC (permalink / raw) To: Ulf Hansson Cc: Rafael J. Wysocki, Kevin Hilman, linux-pm@vger.kernel.org, linux-kernel@vger.kernel.org On 20.06.2017 15:47, Ulf Hansson wrote: > On 20 June 2017 at 14:05, Mikko Perttunen <mperttunen@nvidia.com> wrote: >> Currently genpd installs its own suspend_noirq and resume_noirq >> callbacks, but never calls down to the driver's corresponding >> callbacks. Add these calls. >> >> Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> >> --- >> >> drivers/base/power/domain.c | 8 ++++++++ >> 1 file changed, 8 insertions(+) >> >> diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c >> index d3f1d96f75e9..c3b6e6018c02 100644 >> --- a/drivers/base/power/domain.c >> +++ b/drivers/base/power/domain.c >> @@ -925,6 +925,10 @@ static int pm_genpd_suspend_noirq(struct device *dev) >> return ret; >> } >> >> + ret = pm_generic_suspend_noirq(dev); >> + if (ret) >> + return ret; > > Two things: > 1) I would move this a couple of lines further above, as to avoid > pm_runtime_force_suspend() from being called before, as it may suspend > the device before the noirq callbacks gets to run. Sure. > 2) pm_genpd_suspend_noirq() is also assigned to the ->poweroff_noirq() > callback. Certainly we don't want run the *suspend* variants of the > callback, but rather the *poweroff* variants in that case. Ah, didn't notice this. Will fix. Thanks for the comments, I'll post a v2. Mikko > >> + >> genpd_lock(genpd); >> genpd->suspended_count++; >> genpd_sync_power_off(genpd, true, 0); >> @@ -958,6 +962,10 @@ static int pm_genpd_resume_noirq(struct device *dev) >> genpd->suspended_count--; >> genpd_unlock(genpd); >> >> + ret = pm_generic_resume_noirq(dev); >> + if (ret) >> + return ret; >> + >> if (genpd->dev_ops.stop && genpd->dev_ops.start) >> ret = pm_runtime_force_resume(dev); >> >> -- >> 2.1.4 >> > > Kind regards > Uffe > ^ permalink raw reply [flat|nested] 10+ messages in thread
* [PATCH v2] PM / Domains: Call driver's noirq callbacks 2017-06-20 12:47 ` Ulf Hansson 2017-06-20 13:04 ` Mikko Perttunen @ 2017-06-20 13:38 ` Mikko Perttunen 2017-06-20 14:18 ` Ulf Hansson 2017-06-21 14:52 ` [PATCH v2] " kbuild test robot 1 sibling, 2 replies; 10+ messages in thread From: Mikko Perttunen @ 2017-06-20 13:38 UTC (permalink / raw) To: rjw, khilman, ulf.hansson; +Cc: linux-pm, linux-kernel, Mikko Perttunen Currently genpd installs its own suspend_noirq, resume_noirq, and poweroff_noirq callbacks, but never calls down to the driver's corresponding callbacks. Add these calls. Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> --- v2: - Moved pm_generic_suspend_noirq to before pm_runtime_force_suspend, and correspondingly pm_generic_resume_noirq after pm_runtime_force_resume - Added new pm_genpd_poweroff_noirq callback that is identical to pm_genpd_suspend_noirq but calls the appropriate driver callback drivers/base/power/domain.c | 50 ++++++++++++++++++++++++++++++++++++++++++++- 1 file changed, 49 insertions(+), 1 deletion(-) diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c index d3f1d96f75e9..b070ee58186d 100644 --- a/drivers/base/power/domain.c +++ b/drivers/base/power/domain.c @@ -919,6 +919,10 @@ static int pm_genpd_suspend_noirq(struct device *dev) if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) return 0; + ret = pm_generic_suspend_noirq(dev); + if (ret) + return ret; + if (genpd->dev_ops.stop && genpd->dev_ops.start) { ret = pm_runtime_force_suspend(dev); if (ret) @@ -961,6 +965,10 @@ static int pm_genpd_resume_noirq(struct device *dev) if (genpd->dev_ops.stop && genpd->dev_ops.start) ret = pm_runtime_force_resume(dev); + ret = pm_generic_resume_noirq(dev); + if (ret) + return ret; + return ret; } @@ -1015,6 +1023,46 @@ static int pm_genpd_thaw_noirq(struct device *dev) } /** + * pm_genpd_poweroff_noirq - Completion of hibernation of device in an + * I/O PM domain. + * @dev: Device to poweroff. + * + * Stop the device and remove power from the domain if all devices in it have + * been stopped. + */ +static int pm_genpd_poweroff_noirq(struct device *dev) +{ + struct generic_pm_domain *genpd; + int ret; + + dev_dbg(dev, "%s()\n", __func__); + + genpd = dev_to_genpd(dev); + if (IS_ERR(genpd)) + return -EINVAL; + + if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) + return 0; + + ret = pm_generic_poweroff_noirq(dev); + if (ret) + return ret; + + if (genpd->dev_ops.stop && genpd->dev_ops.start) { + ret = pm_runtime_force_suspend(dev); + if (ret) + return ret; + } + + genpd_lock(genpd); + genpd->suspended_count++; + genpd_sync_power_off(genpd, true, 0); + genpd_unlock(genpd); + + return 0; +} + +/** * pm_genpd_restore_noirq - Start of restore of device in an I/O PM domain. * @dev: Device to resume. * @@ -1493,7 +1541,7 @@ int pm_genpd_init(struct generic_pm_domain *genpd, genpd->domain.ops.resume_noirq = pm_genpd_resume_noirq; genpd->domain.ops.freeze_noirq = pm_genpd_freeze_noirq; genpd->domain.ops.thaw_noirq = pm_genpd_thaw_noirq; - genpd->domain.ops.poweroff_noirq = pm_genpd_suspend_noirq; + genpd->domain.ops.poweroff_noirq = pm_genpd_poweroff_noirq; genpd->domain.ops.restore_noirq = pm_genpd_restore_noirq; genpd->domain.ops.complete = pm_genpd_complete; -- 2.1.4 ^ permalink raw reply related [flat|nested] 10+ messages in thread
* Re: [PATCH v2] PM / Domains: Call driver's noirq callbacks 2017-06-20 13:38 ` [PATCH v2] " Mikko Perttunen @ 2017-06-20 14:18 ` Ulf Hansson 2017-06-21 7:06 ` Mikko Perttunen 2017-06-22 7:18 ` [PATCH v3] " Mikko Perttunen 2017-06-21 14:52 ` [PATCH v2] " kbuild test robot 1 sibling, 2 replies; 10+ messages in thread From: Ulf Hansson @ 2017-06-20 14:18 UTC (permalink / raw) To: Mikko Perttunen Cc: Rafael J. Wysocki, Kevin Hilman, linux-pm@vger.kernel.org, linux-kernel@vger.kernel.org On 20 June 2017 at 15:38, Mikko Perttunen <mperttunen@nvidia.com> wrote: > Currently genpd installs its own suspend_noirq, resume_noirq, > and poweroff_noirq callbacks, but never calls down to the driver's > corresponding callbacks. Add these calls. > > Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> > --- > v2: > - Moved pm_generic_suspend_noirq to before pm_runtime_force_suspend, > and correspondingly pm_generic_resume_noirq after > pm_runtime_force_resume > - Added new pm_genpd_poweroff_noirq callback that is identical to > pm_genpd_suspend_noirq but calls the appropriate driver callback > > drivers/base/power/domain.c | 50 ++++++++++++++++++++++++++++++++++++++++++++- > 1 file changed, 49 insertions(+), 1 deletion(-) > > diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c > index d3f1d96f75e9..b070ee58186d 100644 > --- a/drivers/base/power/domain.c > +++ b/drivers/base/power/domain.c > @@ -919,6 +919,10 @@ static int pm_genpd_suspend_noirq(struct device *dev) > if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) > return 0; > > + ret = pm_generic_suspend_noirq(dev); > + if (ret) > + return ret; > + > if (genpd->dev_ops.stop && genpd->dev_ops.start) { > ret = pm_runtime_force_suspend(dev); > if (ret) > @@ -961,6 +965,10 @@ static int pm_genpd_resume_noirq(struct device *dev) > if (genpd->dev_ops.stop && genpd->dev_ops.start) > ret = pm_runtime_force_resume(dev); > > + ret = pm_generic_resume_noirq(dev); > + if (ret) > + return ret; > + > return ret; > } > > @@ -1015,6 +1023,46 @@ static int pm_genpd_thaw_noirq(struct device *dev) > } > > /** > + * pm_genpd_poweroff_noirq - Completion of hibernation of device in an > + * I/O PM domain. > + * @dev: Device to poweroff. > + * > + * Stop the device and remove power from the domain if all devices in it have > + * been stopped. > + */ > +static int pm_genpd_poweroff_noirq(struct device *dev) > +{ > + struct generic_pm_domain *genpd; > + int ret; > + > + dev_dbg(dev, "%s()\n", __func__); > + > + genpd = dev_to_genpd(dev); > + if (IS_ERR(genpd)) > + return -EINVAL; > + > + if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) > + return 0; > + > + ret = pm_generic_poweroff_noirq(dev); The only difference between pm_genpd_suspend_noirq() and pm_genpd_poweroff_noirq() is the above line. Can we re-factor the code so we avoid open code here, please. > + if (ret) > + return ret; > + > + if (genpd->dev_ops.stop && genpd->dev_ops.start) { > + ret = pm_runtime_force_suspend(dev); > + if (ret) > + return ret; > + } > + > + genpd_lock(genpd); > + genpd->suspended_count++; > + genpd_sync_power_off(genpd, true, 0); > + genpd_unlock(genpd); > + > + return 0; > +} > + > +/** > * pm_genpd_restore_noirq - Start of restore of device in an I/O PM domain. > * @dev: Device to resume. > * > @@ -1493,7 +1541,7 @@ int pm_genpd_init(struct generic_pm_domain *genpd, > genpd->domain.ops.resume_noirq = pm_genpd_resume_noirq; > genpd->domain.ops.freeze_noirq = pm_genpd_freeze_noirq; > genpd->domain.ops.thaw_noirq = pm_genpd_thaw_noirq; > - genpd->domain.ops.poweroff_noirq = pm_genpd_suspend_noirq; > + genpd->domain.ops.poweroff_noirq = pm_genpd_poweroff_noirq; > genpd->domain.ops.restore_noirq = pm_genpd_restore_noirq; The pm_genpd_restore_noirq() doesn't invokes the lower level ->restore_noirq() callbacks. If you are going to change that for the *poweroff* callback, certainly we should change that also for the *restore* callbacks as well. Don't you think? Moreover, what about the freeze and thaw callbacks, should these also walk the lower level callbacks? > genpd->domain.ops.complete = pm_genpd_complete; > > -- > 2.1.4 > Kind regards Uffe ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH v2] PM / Domains: Call driver's noirq callbacks 2017-06-20 14:18 ` Ulf Hansson @ 2017-06-21 7:06 ` Mikko Perttunen 2017-06-22 7:18 ` [PATCH v3] " Mikko Perttunen 1 sibling, 0 replies; 10+ messages in thread From: Mikko Perttunen @ 2017-06-21 7:06 UTC (permalink / raw) To: Ulf Hansson Cc: Rafael J. Wysocki, Kevin Hilman, linux-pm@vger.kernel.org, linux-kernel@vger.kernel.org On 20.06.2017 17:18, Ulf Hansson wrote: > On 20 June 2017 at 15:38, Mikko Perttunen <mperttunen@nvidia.com> wrote: >> Currently genpd installs its own suspend_noirq, resume_noirq, >> and poweroff_noirq callbacks, but never calls down to the driver's >> corresponding callbacks. Add these calls. >> >> Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> >> --- >> v2: >> - Moved pm_generic_suspend_noirq to before pm_runtime_force_suspend, >> and correspondingly pm_generic_resume_noirq after >> pm_runtime_force_resume >> - Added new pm_genpd_poweroff_noirq callback that is identical to >> pm_genpd_suspend_noirq but calls the appropriate driver callback >> >> drivers/base/power/domain.c | 50 ++++++++++++++++++++++++++++++++++++++++++++- >> 1 file changed, 49 insertions(+), 1 deletion(-) >> >> diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c >> index d3f1d96f75e9..b070ee58186d 100644 >> --- a/drivers/base/power/domain.c >> +++ b/drivers/base/power/domain.c >> @@ -919,6 +919,10 @@ static int pm_genpd_suspend_noirq(struct device *dev) >> if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) >> return 0; >> >> + ret = pm_generic_suspend_noirq(dev); >> + if (ret) >> + return ret; >> + >> if (genpd->dev_ops.stop && genpd->dev_ops.start) { >> ret = pm_runtime_force_suspend(dev); >> if (ret) >> @@ -961,6 +965,10 @@ static int pm_genpd_resume_noirq(struct device *dev) >> if (genpd->dev_ops.stop && genpd->dev_ops.start) >> ret = pm_runtime_force_resume(dev); >> >> + ret = pm_generic_resume_noirq(dev); >> + if (ret) >> + return ret; >> + >> return ret; >> } >> >> @@ -1015,6 +1023,46 @@ static int pm_genpd_thaw_noirq(struct device *dev) >> } >> >> /** >> + * pm_genpd_poweroff_noirq - Completion of hibernation of device in an >> + * I/O PM domain. >> + * @dev: Device to poweroff. >> + * >> + * Stop the device and remove power from the domain if all devices in it have >> + * been stopped. >> + */ >> +static int pm_genpd_poweroff_noirq(struct device *dev) >> +{ >> + struct generic_pm_domain *genpd; >> + int ret; >> + >> + dev_dbg(dev, "%s()\n", __func__); >> + >> + genpd = dev_to_genpd(dev); >> + if (IS_ERR(genpd)) >> + return -EINVAL; >> + >> + if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) >> + return 0; >> + >> + ret = pm_generic_poweroff_noirq(dev); > > The only difference between pm_genpd_suspend_noirq() and > pm_genpd_poweroff_noirq() is the above line. Can we re-factor the code > so we avoid open code here, please. I wasn't sure if the functions' complexity warranted adding a helper function, but sure, I'll refactor this with a helper function. > >> + if (ret) >> + return ret; >> + >> + if (genpd->dev_ops.stop && genpd->dev_ops.start) { >> + ret = pm_runtime_force_suspend(dev); >> + if (ret) >> + return ret; >> + } >> + >> + genpd_lock(genpd); >> + genpd->suspended_count++; >> + genpd_sync_power_off(genpd, true, 0); >> + genpd_unlock(genpd); >> + >> + return 0; >> +} >> + >> +/** >> * pm_genpd_restore_noirq - Start of restore of device in an I/O PM domain. >> * @dev: Device to resume. >> * >> @@ -1493,7 +1541,7 @@ int pm_genpd_init(struct generic_pm_domain *genpd, >> genpd->domain.ops.resume_noirq = pm_genpd_resume_noirq; >> genpd->domain.ops.freeze_noirq = pm_genpd_freeze_noirq; >> genpd->domain.ops.thaw_noirq = pm_genpd_thaw_noirq; >> - genpd->domain.ops.poweroff_noirq = pm_genpd_suspend_noirq; >> + genpd->domain.ops.poweroff_noirq = pm_genpd_poweroff_noirq; >> genpd->domain.ops.restore_noirq = pm_genpd_restore_noirq; > > The pm_genpd_restore_noirq() doesn't invokes the lower level > ->restore_noirq() callbacks. If you are going to change that for the > *poweroff* callback, certainly we should change that also for the > *restore* callbacks as well. Don't you think? > > Moreover, what about the freeze and thaw callbacks, should these also > walk the lower level callbacks? Yes, I'll add the calls to the rest of the ops as well. Thanks, Mikko. > >> genpd->domain.ops.complete = pm_genpd_complete; >> >> -- >> 2.1.4 >> > > Kind regards > Uffe > ^ permalink raw reply [flat|nested] 10+ messages in thread
* [PATCH v3] PM / Domains: Call driver's noirq callbacks 2017-06-20 14:18 ` Ulf Hansson 2017-06-21 7:06 ` Mikko Perttunen @ 2017-06-22 7:18 ` Mikko Perttunen 2017-06-27 22:46 ` Rafael J. Wysocki 2017-06-28 9:00 ` Ulf Hansson 1 sibling, 2 replies; 10+ messages in thread From: Mikko Perttunen @ 2017-06-22 7:18 UTC (permalink / raw) To: rjw, khilman, ulf.hansson; +Cc: linux-pm, linux-kernel, cyndis, Mikko Perttunen Currently genpd installs its own noirq callbacks, but never calls down to the driver's corresponding callbacks. Add these calls. Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> --- v3: - Factored out common code in pm_genpd_{suspend,poweroff}_noirq - Added pm_generic_* calls to rest of callbacks drivers/base/power/domain.c | 68 +++++++++++++++++++++++++++++++++++++++------ 1 file changed, 59 insertions(+), 9 deletions(-) diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c index d3f1d96f75e9..248c5f79f588 100644 --- a/drivers/base/power/domain.c +++ b/drivers/base/power/domain.c @@ -899,19 +899,19 @@ static int pm_genpd_prepare(struct device *dev) } /** - * pm_genpd_suspend_noirq - Completion of suspend of device in an I/O PM domain. + * genpd_finish_suspend - Completion of suspend or hibernation of device in an + * I/O pm domain. * @dev: Device to suspend. + * @poweroff: Specifies if this is a poweroff_noirq or suspend_noirq callback. * * Stop the device and remove power from the domain if all devices in it have * been stopped. */ -static int pm_genpd_suspend_noirq(struct device *dev) +static int genpd_finish_suspend(struct device *dev, bool poweroff) { struct generic_pm_domain *genpd; int ret; - dev_dbg(dev, "%s()\n", __func__); - genpd = dev_to_genpd(dev); if (IS_ERR(genpd)) return -EINVAL; @@ -919,6 +919,13 @@ static int pm_genpd_suspend_noirq(struct device *dev) if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) return 0; + if (poweroff) + ret = pm_generic_poweroff_noirq(dev); + else + ret = pm_generic_suspend_noirq(dev); + if (ret) + return ret; + if (genpd->dev_ops.stop && genpd->dev_ops.start) { ret = pm_runtime_force_suspend(dev); if (ret) @@ -934,6 +941,20 @@ static int pm_genpd_suspend_noirq(struct device *dev) } /** + * pm_genpd_suspend_noirq - Completion of suspend of device in an I/O PM domain. + * @dev: Device to suspend. + * + * Stop the device and remove power from the domain if all devices in it have + * been stopped. + */ +static int pm_genpd_suspend_noirq(struct device *dev) +{ + dev_dbg(dev, "%s()\n", __func__); + + return genpd_finish_suspend(dev, false); +} + +/** * pm_genpd_resume_noirq - Start of resume of device in an I/O PM domain. * @dev: Device to resume. * @@ -961,6 +982,10 @@ static int pm_genpd_resume_noirq(struct device *dev) if (genpd->dev_ops.stop && genpd->dev_ops.start) ret = pm_runtime_force_resume(dev); + ret = pm_generic_resume_noirq(dev); + if (ret) + return ret; + return ret; } @@ -984,6 +1009,10 @@ static int pm_genpd_freeze_noirq(struct device *dev) if (IS_ERR(genpd)) return -EINVAL; + ret = pm_generic_freeze_noirq(dev); + if (ret) + return ret; + if (genpd->dev_ops.stop && genpd->dev_ops.start) ret = pm_runtime_force_suspend(dev); @@ -1008,10 +1037,28 @@ static int pm_genpd_thaw_noirq(struct device *dev) if (IS_ERR(genpd)) return -EINVAL; - if (genpd->dev_ops.stop && genpd->dev_ops.start) + if (genpd->dev_ops.stop && genpd->dev_ops.start) { ret = pm_runtime_force_resume(dev); + if (ret) + return ret; + } - return ret; + return pm_generic_thaw_noirq(dev); +} + +/** + * pm_genpd_poweroff_noirq - Completion of hibernation of device in an + * I/O PM domain. + * @dev: Device to poweroff. + * + * Stop the device and remove power from the domain if all devices in it have + * been stopped. + */ +static int pm_genpd_poweroff_noirq(struct device *dev) +{ + dev_dbg(dev, "%s()\n", __func__); + + return genpd_finish_suspend(dev, true); } /** @@ -1048,10 +1095,13 @@ static int pm_genpd_restore_noirq(struct device *dev) genpd_sync_power_on(genpd, true, 0); genpd_unlock(genpd); - if (genpd->dev_ops.stop && genpd->dev_ops.start) + if (genpd->dev_ops.stop && genpd->dev_ops.start) { ret = pm_runtime_force_resume(dev); + if (ret) + return ret; + } - return ret; + return pm_generic_restore_noirq(dev); } /** @@ -1493,7 +1543,7 @@ int pm_genpd_init(struct generic_pm_domain *genpd, genpd->domain.ops.resume_noirq = pm_genpd_resume_noirq; genpd->domain.ops.freeze_noirq = pm_genpd_freeze_noirq; genpd->domain.ops.thaw_noirq = pm_genpd_thaw_noirq; - genpd->domain.ops.poweroff_noirq = pm_genpd_suspend_noirq; + genpd->domain.ops.poweroff_noirq = pm_genpd_poweroff_noirq; genpd->domain.ops.restore_noirq = pm_genpd_restore_noirq; genpd->domain.ops.complete = pm_genpd_complete; -- 2.1.4 ^ permalink raw reply related [flat|nested] 10+ messages in thread
* Re: [PATCH v3] PM / Domains: Call driver's noirq callbacks 2017-06-22 7:18 ` [PATCH v3] " Mikko Perttunen @ 2017-06-27 22:46 ` Rafael J. Wysocki 2017-06-28 9:00 ` Ulf Hansson 1 sibling, 0 replies; 10+ messages in thread From: Rafael J. Wysocki @ 2017-06-27 22:46 UTC (permalink / raw) To: Mikko Perttunen, ulf.hansson; +Cc: khilman, linux-pm, linux-kernel, cyndis On Thursday, June 22, 2017 10:18:33 AM Mikko Perttunen wrote: > Currently genpd installs its own noirq callbacks, but never calls down > to the driver's corresponding callbacks. Add these calls. > > Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> Ulf? > --- > v3: > - Factored out common code in pm_genpd_{suspend,poweroff}_noirq > - Added pm_generic_* calls to rest of callbacks > > drivers/base/power/domain.c | 68 +++++++++++++++++++++++++++++++++++++++------ > 1 file changed, 59 insertions(+), 9 deletions(-) > > diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c > index d3f1d96f75e9..248c5f79f588 100644 > --- a/drivers/base/power/domain.c > +++ b/drivers/base/power/domain.c > @@ -899,19 +899,19 @@ static int pm_genpd_prepare(struct device *dev) > } > > /** > - * pm_genpd_suspend_noirq - Completion of suspend of device in an I/O PM domain. > + * genpd_finish_suspend - Completion of suspend or hibernation of device in an > + * I/O pm domain. > * @dev: Device to suspend. > + * @poweroff: Specifies if this is a poweroff_noirq or suspend_noirq callback. > * > * Stop the device and remove power from the domain if all devices in it have > * been stopped. > */ > -static int pm_genpd_suspend_noirq(struct device *dev) > +static int genpd_finish_suspend(struct device *dev, bool poweroff) > { > struct generic_pm_domain *genpd; > int ret; > > - dev_dbg(dev, "%s()\n", __func__); > - > genpd = dev_to_genpd(dev); > if (IS_ERR(genpd)) > return -EINVAL; > @@ -919,6 +919,13 @@ static int pm_genpd_suspend_noirq(struct device *dev) > if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) > return 0; > > + if (poweroff) > + ret = pm_generic_poweroff_noirq(dev); > + else > + ret = pm_generic_suspend_noirq(dev); > + if (ret) > + return ret; > + > if (genpd->dev_ops.stop && genpd->dev_ops.start) { > ret = pm_runtime_force_suspend(dev); > if (ret) > @@ -934,6 +941,20 @@ static int pm_genpd_suspend_noirq(struct device *dev) > } > > /** > + * pm_genpd_suspend_noirq - Completion of suspend of device in an I/O PM domain. > + * @dev: Device to suspend. > + * > + * Stop the device and remove power from the domain if all devices in it have > + * been stopped. > + */ > +static int pm_genpd_suspend_noirq(struct device *dev) > +{ > + dev_dbg(dev, "%s()\n", __func__); > + > + return genpd_finish_suspend(dev, false); > +} > + > +/** > * pm_genpd_resume_noirq - Start of resume of device in an I/O PM domain. > * @dev: Device to resume. > * > @@ -961,6 +982,10 @@ static int pm_genpd_resume_noirq(struct device *dev) > if (genpd->dev_ops.stop && genpd->dev_ops.start) > ret = pm_runtime_force_resume(dev); > > + ret = pm_generic_resume_noirq(dev); > + if (ret) > + return ret; > + > return ret; > } > > @@ -984,6 +1009,10 @@ static int pm_genpd_freeze_noirq(struct device *dev) > if (IS_ERR(genpd)) > return -EINVAL; > > + ret = pm_generic_freeze_noirq(dev); > + if (ret) > + return ret; > + > if (genpd->dev_ops.stop && genpd->dev_ops.start) > ret = pm_runtime_force_suspend(dev); > > @@ -1008,10 +1037,28 @@ static int pm_genpd_thaw_noirq(struct device *dev) > if (IS_ERR(genpd)) > return -EINVAL; > > - if (genpd->dev_ops.stop && genpd->dev_ops.start) > + if (genpd->dev_ops.stop && genpd->dev_ops.start) { > ret = pm_runtime_force_resume(dev); > + if (ret) > + return ret; > + } > > - return ret; > + return pm_generic_thaw_noirq(dev); > +} > + > +/** > + * pm_genpd_poweroff_noirq - Completion of hibernation of device in an > + * I/O PM domain. > + * @dev: Device to poweroff. > + * > + * Stop the device and remove power from the domain if all devices in it have > + * been stopped. > + */ > +static int pm_genpd_poweroff_noirq(struct device *dev) > +{ > + dev_dbg(dev, "%s()\n", __func__); > + > + return genpd_finish_suspend(dev, true); > } > > /** > @@ -1048,10 +1095,13 @@ static int pm_genpd_restore_noirq(struct device *dev) > genpd_sync_power_on(genpd, true, 0); > genpd_unlock(genpd); > > - if (genpd->dev_ops.stop && genpd->dev_ops.start) > + if (genpd->dev_ops.stop && genpd->dev_ops.start) { > ret = pm_runtime_force_resume(dev); > + if (ret) > + return ret; > + } > > - return ret; > + return pm_generic_restore_noirq(dev); > } > > /** > @@ -1493,7 +1543,7 @@ int pm_genpd_init(struct generic_pm_domain *genpd, > genpd->domain.ops.resume_noirq = pm_genpd_resume_noirq; > genpd->domain.ops.freeze_noirq = pm_genpd_freeze_noirq; > genpd->domain.ops.thaw_noirq = pm_genpd_thaw_noirq; > - genpd->domain.ops.poweroff_noirq = pm_genpd_suspend_noirq; > + genpd->domain.ops.poweroff_noirq = pm_genpd_poweroff_noirq; > genpd->domain.ops.restore_noirq = pm_genpd_restore_noirq; > genpd->domain.ops.complete = pm_genpd_complete; > > ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH v3] PM / Domains: Call driver's noirq callbacks 2017-06-22 7:18 ` [PATCH v3] " Mikko Perttunen 2017-06-27 22:46 ` Rafael J. Wysocki @ 2017-06-28 9:00 ` Ulf Hansson 1 sibling, 0 replies; 10+ messages in thread From: Ulf Hansson @ 2017-06-28 9:00 UTC (permalink / raw) To: Mikko Perttunen Cc: Rafael J. Wysocki, Kevin Hilman, linux-pm@vger.kernel.org, linux-kernel@vger.kernel.org, cyndis On 22 June 2017 at 09:18, Mikko Perttunen <mperttunen@nvidia.com> wrote: > Currently genpd installs its own noirq callbacks, but never calls down > to the driver's corresponding callbacks. Add these calls. > > Signed-off-by: Mikko Perttunen <mperttunen@nvidia.com> Acked-by: Ulf Hansson <ulf.hansson@linaro.org> Kind regards Uffe > --- > v3: > - Factored out common code in pm_genpd_{suspend,poweroff}_noirq > - Added pm_generic_* calls to rest of callbacks > > drivers/base/power/domain.c | 68 +++++++++++++++++++++++++++++++++++++++------ > 1 file changed, 59 insertions(+), 9 deletions(-) > > diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c > index d3f1d96f75e9..248c5f79f588 100644 > --- a/drivers/base/power/domain.c > +++ b/drivers/base/power/domain.c > @@ -899,19 +899,19 @@ static int pm_genpd_prepare(struct device *dev) > } > > /** > - * pm_genpd_suspend_noirq - Completion of suspend of device in an I/O PM domain. > + * genpd_finish_suspend - Completion of suspend or hibernation of device in an > + * I/O pm domain. > * @dev: Device to suspend. > + * @poweroff: Specifies if this is a poweroff_noirq or suspend_noirq callback. > * > * Stop the device and remove power from the domain if all devices in it have > * been stopped. > */ > -static int pm_genpd_suspend_noirq(struct device *dev) > +static int genpd_finish_suspend(struct device *dev, bool poweroff) > { > struct generic_pm_domain *genpd; > int ret; > > - dev_dbg(dev, "%s()\n", __func__); > - > genpd = dev_to_genpd(dev); > if (IS_ERR(genpd)) > return -EINVAL; > @@ -919,6 +919,13 @@ static int pm_genpd_suspend_noirq(struct device *dev) > if (dev->power.wakeup_path && genpd_dev_active_wakeup(genpd, dev)) > return 0; > > + if (poweroff) > + ret = pm_generic_poweroff_noirq(dev); > + else > + ret = pm_generic_suspend_noirq(dev); > + if (ret) > + return ret; > + > if (genpd->dev_ops.stop && genpd->dev_ops.start) { > ret = pm_runtime_force_suspend(dev); > if (ret) > @@ -934,6 +941,20 @@ static int pm_genpd_suspend_noirq(struct device *dev) > } > > /** > + * pm_genpd_suspend_noirq - Completion of suspend of device in an I/O PM domain. > + * @dev: Device to suspend. > + * > + * Stop the device and remove power from the domain if all devices in it have > + * been stopped. > + */ > +static int pm_genpd_suspend_noirq(struct device *dev) > +{ > + dev_dbg(dev, "%s()\n", __func__); > + > + return genpd_finish_suspend(dev, false); > +} > + > +/** > * pm_genpd_resume_noirq - Start of resume of device in an I/O PM domain. > * @dev: Device to resume. > * > @@ -961,6 +982,10 @@ static int pm_genpd_resume_noirq(struct device *dev) > if (genpd->dev_ops.stop && genpd->dev_ops.start) > ret = pm_runtime_force_resume(dev); > > + ret = pm_generic_resume_noirq(dev); > + if (ret) > + return ret; > + > return ret; > } > > @@ -984,6 +1009,10 @@ static int pm_genpd_freeze_noirq(struct device *dev) > if (IS_ERR(genpd)) > return -EINVAL; > > + ret = pm_generic_freeze_noirq(dev); > + if (ret) > + return ret; > + > if (genpd->dev_ops.stop && genpd->dev_ops.start) > ret = pm_runtime_force_suspend(dev); > > @@ -1008,10 +1037,28 @@ static int pm_genpd_thaw_noirq(struct device *dev) > if (IS_ERR(genpd)) > return -EINVAL; > > - if (genpd->dev_ops.stop && genpd->dev_ops.start) > + if (genpd->dev_ops.stop && genpd->dev_ops.start) { > ret = pm_runtime_force_resume(dev); > + if (ret) > + return ret; > + } > > - return ret; > + return pm_generic_thaw_noirq(dev); > +} > + > +/** > + * pm_genpd_poweroff_noirq - Completion of hibernation of device in an > + * I/O PM domain. > + * @dev: Device to poweroff. > + * > + * Stop the device and remove power from the domain if all devices in it have > + * been stopped. > + */ > +static int pm_genpd_poweroff_noirq(struct device *dev) > +{ > + dev_dbg(dev, "%s()\n", __func__); > + > + return genpd_finish_suspend(dev, true); > } > > /** > @@ -1048,10 +1095,13 @@ static int pm_genpd_restore_noirq(struct device *dev) > genpd_sync_power_on(genpd, true, 0); > genpd_unlock(genpd); > > - if (genpd->dev_ops.stop && genpd->dev_ops.start) > + if (genpd->dev_ops.stop && genpd->dev_ops.start) { > ret = pm_runtime_force_resume(dev); > + if (ret) > + return ret; > + } > > - return ret; > + return pm_generic_restore_noirq(dev); > } > > /** > @@ -1493,7 +1543,7 @@ int pm_genpd_init(struct generic_pm_domain *genpd, > genpd->domain.ops.resume_noirq = pm_genpd_resume_noirq; > genpd->domain.ops.freeze_noirq = pm_genpd_freeze_noirq; > genpd->domain.ops.thaw_noirq = pm_genpd_thaw_noirq; > - genpd->domain.ops.poweroff_noirq = pm_genpd_suspend_noirq; > + genpd->domain.ops.poweroff_noirq = pm_genpd_poweroff_noirq; > genpd->domain.ops.restore_noirq = pm_genpd_restore_noirq; > genpd->domain.ops.complete = pm_genpd_complete; > > -- > 2.1.4 > ^ permalink raw reply [flat|nested] 10+ messages in thread
* Re: [PATCH v2] PM / Domains: Call driver's noirq callbacks 2017-06-20 13:38 ` [PATCH v2] " Mikko Perttunen 2017-06-20 14:18 ` Ulf Hansson @ 2017-06-21 14:52 ` kbuild test robot 1 sibling, 0 replies; 10+ messages in thread From: kbuild test robot @ 2017-06-21 14:52 UTC (permalink / raw) Cc: kbuild-all, rjw, khilman, ulf.hansson, linux-pm, linux-kernel, Mikko Perttunen [-- Attachment #1: Type: text/plain, Size: 2022 bytes --] Hi Mikko, [auto build test ERROR on pm/linux-next] [also build test ERROR on v4.12-rc6 next-20170621] [if your patch is applied to the wrong git tree, please drop us a note to help improve the system] url: https://github.com/0day-ci/linux/commits/Mikko-Perttunen/PM-Domains-Call-driver-s-noirq-callbacks/20170621-222245 base: https://git.kernel.org/pub/scm/linux/kernel/git/rafael/linux-pm.git linux-next config: sh-allmodconfig (attached as .config) compiler: sh4-linux-gnu-gcc (Debian 6.3.0-18) 6.3.0 20170516 reproduce: wget https://raw.githubusercontent.com/01org/lkp-tests/master/sbin/make.cross -O ~/bin/make.cross chmod +x ~/bin/make.cross # save the attached .config to linux build tree make.cross ARCH=sh All errors (new ones prefixed by >>): drivers/base/power/domain.c: In function 'pm_genpd_init': >> drivers/base/power/domain.c:1544:37: error: 'pm_genpd_poweroff_noirq' undeclared (first use in this function) genpd->domain.ops.poweroff_noirq = pm_genpd_poweroff_noirq; ^~~~~~~~~~~~~~~~~~~~~~~ drivers/base/power/domain.c:1544:37: note: each undeclared identifier is reported only once for each function it appears in vim +/pm_genpd_poweroff_noirq +1544 drivers/base/power/domain.c 1538 genpd->domain.ops.runtime_resume = genpd_runtime_resume; 1539 genpd->domain.ops.prepare = pm_genpd_prepare; 1540 genpd->domain.ops.suspend_noirq = pm_genpd_suspend_noirq; 1541 genpd->domain.ops.resume_noirq = pm_genpd_resume_noirq; 1542 genpd->domain.ops.freeze_noirq = pm_genpd_freeze_noirq; 1543 genpd->domain.ops.thaw_noirq = pm_genpd_thaw_noirq; > 1544 genpd->domain.ops.poweroff_noirq = pm_genpd_poweroff_noirq; 1545 genpd->domain.ops.restore_noirq = pm_genpd_restore_noirq; 1546 genpd->domain.ops.complete = pm_genpd_complete; 1547 --- 0-DAY kernel test infrastructure Open Source Technology Center https://lists.01.org/pipermail/kbuild-all Intel Corporation [-- Attachment #2: .config.gz --] [-- Type: application/gzip, Size: 45304 bytes --] ^ permalink raw reply [flat|nested] 10+ messages in thread
end of thread, other threads:[~2017-06-28 9:00 UTC | newest] Thread overview: 10+ messages (download: mbox.gz follow: Atom feed -- links below jump to the message on this page -- 2017-06-20 12:05 [PATCH] PM / Domains: Call driver's noirq callbacks Mikko Perttunen 2017-06-20 12:47 ` Ulf Hansson 2017-06-20 13:04 ` Mikko Perttunen 2017-06-20 13:38 ` [PATCH v2] " Mikko Perttunen 2017-06-20 14:18 ` Ulf Hansson 2017-06-21 7:06 ` Mikko Perttunen 2017-06-22 7:18 ` [PATCH v3] " Mikko Perttunen 2017-06-27 22:46 ` Rafael J. Wysocki 2017-06-28 9:00 ` Ulf Hansson 2017-06-21 14:52 ` [PATCH v2] " kbuild test robot
This is a public inbox, see mirroring instructions for how to clone and mirror all data and code used for this inbox; as well as URLs for NNTP newsgroup(s).