Re: [RFC] PM: domains: Reverse the order of performance and enabling ops
From: Ulf Hansson
Date: Thu Jul 21 2022 - 12:48:53 EST
On Wed, 20 Jul 2022 at 13:03, Abel Vesa <abel.vesa@xxxxxxxxxx> wrote:
>
> Rather than enabling and then setting the performance state, which usually
> translates into two different levels (voltages) in order to get to the
> one required by the consumer, we could give a chance to the providers to
> cache the performance state needed by the consumer and then, when powering
> on the power domain, the provider could use the cached level instead.
I don't think it's really clear what you want to do here. Let's see
what the discussion below brings us to, but for the next version
please elaborate a bit more in the commit message.
Although, if I understand correctly (also from our offlist
discussions), you want to make it possible to move from two calls,
into one call into the FW from the genpd provider. So it's basically
an optimization, which to me, certainly sounds worth doing.
Furthermore, to get the complete picture, in the Qcom case, we set a
"default" low performance level from the genpd's ->power_on()
callback, which is needed to enable basic functionality for some
consumers.
The second call that I refer to is made when genpd calls the
->set_performance() callback (from genpd_runtime_suspend()), which is
done by genpd to potentially set a new value for an aggregated
performance state of the PM domain. In case when there actually is a
new performance state set in this path, we end up calling the FW twice
for the Qcom case, where this first one is unnecessary.
Did I get that right?
> Also the drop_performance and power_off have to be reversed so that
> when the last active consumer suspends, the level doesn't actually drop
> until the pd is disabled.
I don't quite get what this part helps with, is it really needed to
improve the behaviour?
>
> For the power domains that do not provide the set_performance, things
> remain unchanged, as does for the power domains that only provide the
> set_performance but do not provide the power_on/off.
Right, good points!
I get back to review the code soon, just wanted to make sure I have
the complete picture first.
Kind regards
Uffe
>
> Signed-off-by: Abel Vesa <abel.vesa@xxxxxxxxxx>
> ---
> drivers/base/power/domain.c | 30 +++++++++++++++---------------
> 1 file changed, 15 insertions(+), 15 deletions(-)
>
> diff --git a/drivers/base/power/domain.c b/drivers/base/power/domain.c
> index 5a2e0232862e..38647c304b73 100644
> --- a/drivers/base/power/domain.c
> +++ b/drivers/base/power/domain.c
> @@ -939,8 +939,8 @@ static int genpd_runtime_suspend(struct device *dev)
> return 0;
>
> genpd_lock(genpd);
> - gpd_data->rpm_pstate = genpd_drop_performance_state(dev);
> genpd_power_off(genpd, true, 0);
> + gpd_data->rpm_pstate = genpd_drop_performance_state(dev);
> genpd_unlock(genpd);
>
> return 0;
> @@ -978,9 +978,8 @@ static int genpd_runtime_resume(struct device *dev)
> goto out;
>
> genpd_lock(genpd);
> + genpd_restore_performance_state(dev, gpd_data->rpm_pstate);
> ret = genpd_power_on(genpd, 0);
> - if (!ret)
> - genpd_restore_performance_state(dev, gpd_data->rpm_pstate);
> genpd_unlock(genpd);
>
> if (ret)
> @@ -1018,8 +1017,8 @@ static int genpd_runtime_resume(struct device *dev)
> err_poweroff:
> if (!pm_runtime_is_irq_safe(dev) || genpd_is_irq_safe(genpd)) {
> genpd_lock(genpd);
> - gpd_data->rpm_pstate = genpd_drop_performance_state(dev);
> genpd_power_off(genpd, true, 0);
> + gpd_data->rpm_pstate = genpd_drop_performance_state(dev);
> genpd_unlock(genpd);
> }
>
> @@ -2747,17 +2746,6 @@ static int __genpd_dev_pm_attach(struct device *dev, struct device *base_dev,
> dev->pm_domain->detach = genpd_dev_pm_detach;
> dev->pm_domain->sync = genpd_dev_pm_sync;
>
> - if (power_on) {
> - genpd_lock(pd);
> - ret = genpd_power_on(pd, 0);
> - genpd_unlock(pd);
> - }
> -
> - if (ret) {
> - genpd_remove_device(pd, dev);
> - return -EPROBE_DEFER;
> - }
> -
> /* Set the default performance state */
> pstate = of_get_required_opp_performance_state(dev->of_node, index);
> if (pstate < 0 && pstate != -ENODEV && pstate != -EOPNOTSUPP) {
> @@ -2769,6 +2757,18 @@ static int __genpd_dev_pm_attach(struct device *dev, struct device *base_dev,
> goto err;
> dev_gpd_data(dev)->default_pstate = pstate;
> }
> +
> + if (power_on) {
> + genpd_lock(pd);
> + ret = genpd_power_on(pd, 0);
> + genpd_unlock(pd);
> + }
> +
> + if (ret) {
> + genpd_remove_device(pd, dev);
> + return -EPROBE_DEFER;
> + }
> +
> return 1;
>
> err:
> --
> 2.34.3
>