Re: [PATCH 1/4] regulator: core: If consumers don't call regulator_set_load() assume max

From: David Collins
Date: Tue Aug 14 2018 - 21:32:20 EST


On 08/14/2018 04:56 PM, Doug Anderson wrote:
> On Tue, Aug 14, 2018 at 2:59 PM, David Collins <collinsd@xxxxxxxxxxxxxx> wrote:
>> On 08/14/2018 01:03 PM, Doug Anderson wrote:
>>> On Tue, Aug 14, 2018 at 11:30 AM, David Collins <collinsd@xxxxxxxxxxxxxx> wrote:>>> --- a/drivers/regulator/core.c
>>>>> +++ b/drivers/regulator/core.c
>>>>> @@ -732,6 +732,7 @@ static int drms_uA_update(struct regulator_dev *rdev)
>>>>> struct regulator *sibling;
>>>>> int current_uA = 0, output_uV, input_uV, err;
>>>>> unsigned int mode;
>>>>> + bool any_unset = false;
>>>>>
>>>>> lockdep_assert_held_once(&rdev->mutex);
>>>>>
>>>>> @@ -751,11 +752,17 @@ static int drms_uA_update(struct regulator_dev *rdev)
>>>>> return -EINVAL;
>>>>>
>>>>> /* calc total requested load */
>>>>> - list_for_each_entry(sibling, &rdev->consumer_list, list)
>>>>> + list_for_each_entry(sibling, &rdev->consumer_list, list) {
>>>>> current_uA += sibling->uA_load;
>>>>> + if (!sibling->uA_load_set)
>>>>> + any_unset = true;
>>>>> + }
>>>>>
>>>>> current_uA += rdev->constraints->system_load;
>>>>>
>>>>> + if (any_unset)
>>>>> + current_uA = INT_MAX;
>>>>> +
>>>>
>>>> This check will incorrectly result in a constant load request of INT_MAX
>>>> for all regulators that have at least one child regulator. This is the
>>>> case because such child regulators are present in rdev->consumer_list and
>>>> because regulator_set_load() requests are not propagated up to parent
>>>> regulators. Thus, the regulator structs for child regulators will always
>>>> have uA_load==0 and uA_load_set==false.
>>>
>>> Ah, interesting.
>>>
>>> ...but doesn't this same bug exist anyway, just in the opposite
>>> direction? Without my patch we'll try to request a 0 mA load in this
>>> case which seems just as wrong (or perhaps worse?). I guess on RPMh
>>> regulator you're "saved" because the RPMh hardware itself knows the
>>> parent/child relationship and knows to ignore this 0 mA load, but it's
>>> still a bug in the overall Linux framework...
>>
>> I'm not sure what existing bug you are referring to here. Where is a 0 mA
>> load being requested? Could you list the consumer function calls and the
>> behavior on the framework side that you're envisioning?
>
> Imagine you've got a tree like this:
>
> - regulatorParent (1.8 V)
> - regulatorChildA (1.7 V)
> - regulatorChildB (1.2 V)
> - regulatorChildC (1.5 V)
>
> ...and this set of calls:
>
> regulator_set_load(regulatorChildA, 1000);
> regulator_set_load(regulatorChildB, 2000);
> regulator_set_load(regulatorChildC, 3000);
>
> regulator_enable(regulatorChildA);
> regulator_enable(regulatorChildB);
> regulator_enable(regulatorChildC);
>
>
> With the existing code in Mark's tree then ChildA / ChildB / ChildC
> will presumably have a load high enough to be in high power mode.
> However, as you said, loads aren't propagated. ...so "Parent" will
> see no load requested (which translates to a load request of 0 mA).
>
> The "bug" is that when you did the
> "regulator_enable(regulatorChildA);" then that will propagate up and
> cause a "regulator_enable(regulatorParent)". In _regulator_enable()
> you can see drms_uA_update(). That will cause it to set the mode of
> regulatorParent based on a load of 0 mA. That is the bug. You may
> respond "but REGULATOR_CHANGE_DRMS isn't set for the parent! See
> below and for now imagine that REGULATOR_CHANGE_DRMS _is_ set for the
> parent.
>
> With my code in the same situation the parent will end up with a load
> of "MAX_INT" mA which I think is the bug you pointed out. ...while it
> would be good to fix this it does seem to be better to set the parent
> to a mode based on MAX_INT mA rather than 0 mA?

I agree that the existing behavior resulting from a lack of load current
propagation from child to parent regulators is wrong. However, I do not
agree that replacing it with something else that results in different
wrong behavior is the way forward.

Adding load current propagation would resolve your 0 mA concern but does
not necessary require making MAX_INT the default for all consumers.


>>> I have no idea how we ought to propagate regulator_set_load() to
>>> parents though. That seems like a tough thing to try to handle
>>> automagically.
>>
>> I attempted it seven years ago with two revisions of a patch series [1]
>> and [2]. The feature wasn't completed though. Perhaps something along
>> the same lines could be reintroduced now that child regulators are handled
>> the same way as ordinary consumers within the regulator framework.
>>
>> [1]: https://lkml.org/lkml/2011/3/28/246
>> [2]: https://lkml.org/lkml/2011/3/28/530
>
> Thanks for the links. My first instinct is just what Mark said there:
> you can't really take the child load and map it accurately to a parent
> load without loads of per-device complexity and even then you'd
> probably get nothing more than an approximation.
>
> IMO about the best we could hope to do would be to map "mode" from
> children to parent. AKA: perhaps you could assume that if a child is
> in a higher power mode that perhaps a parent should be too?

Propagating regulator framework modes (Standby, Idle, Normal, Fast) from
child to parent regulators is definitely not feasible in the general case.
The meaning of these modes varies between different types of regulators
as do load current thresholds between modes. I.e. just because a child
regulator changed from Idle to Normal mode does not imply that the parent
regulator needs to operate in Normal mode now.

As an example, consider the QTI PM8998 FTSMPS and NLDO regulators. The
FTSMPS regulators can supply up to 800 mA in PFM mode (Idle) and 4000 mA
in PWM mode (Fast). The child NLDO type regulators supplied by the
FTSMPSes can source up to 30 mA in LPM (Idle) and 300 - 1200 mA in HPM
(Normal) (depending upon subtype). Clearly, an NLDO switching from Idle
to Normal mode because 50 mA was requested does not mean that its parent
needs to switch to Fast mode.

Also note that in this case, the FTSMPS regulators have AUTO mode in
hardware what maps to Linux mode Normal. In practice, the FTSMPSes will
be configured to be in AUTO mode all the time unless there is a special
case that requires PWM mode (e.g. hardware that is particularly
susceptible to PFM voltage ripple). They will completely ignore load
current aggregation.

I agree that calculating the load current required from a parent based
upon the current output by a child regulator is challenging. However,
microamps are physically meaningful and consistent across different types
of regulators. Therefore, it is technically possible to propagate load
currents from child regulators up to parents. The same is not true of
regulator framework modes.

Take care,
David

--
The Qualcomm Innovation Center, Inc. is a member of the Code Aurora Forum,
a Linux Foundation Collaborative Project