[PATCH v1 3/4] usb: dwc3: exynos: Use devm_regulator_bulk_get_enable() helper function

Christophe JAILLET christophe.jaillet at wanadoo.fr
Mon Apr 8 13:26:16 PDT 2024


Le 08/04/2024 à 12:02, Anand Moon a écrit :
> Hi Christophe,
> 
> On Fri, 5 Apr 2024 at 21:42, Christophe JAILLET
> <christophe.jaillet at wanadoo.fr> wrote:
>>
>> Le 05/04/2024 à 08:10, Anand Moon a écrit :
>>>    Hi Christophe, Krzysztof,
>>>
>>> On Mon, 4 Mar 2024 at 17:16, Anand Moon <linux.amoon at gmail.com> wrote:
>>>>
>>>> Hi Christophe,
>>>>
>>>> On Sun, 3 Mar 2024 at 00:07, Christophe JAILLET
>>>> <christophe.jaillet at wanadoo.fr> wrote:
>>>>>
>>>>> Le 02/03/2024 à 17:48, Anand Moon a écrit :
>>>>>> Hi Christophe,
>>>>>>
>>>>>> On Sat, 2 Mar 2024 at 21:20, Christophe JAILLET
>>>>>> <christophe.jaillet at wanadoo.fr> wrote:
>>>>>>>
>>>>>>> Le 01/03/2024 à 20:38, Anand Moon a écrit :
>>>>>>>> Use devm_regulator_bulk_get_enable() instead of open coded
>>>>>>>> 'devm_regulator_get(), regulator_enable(), regulator_disable().
>>>>>>>>
>>>>>>>> Signed-off-by: Anand Moon <linux.amoon at gmail.com>
>>>>>>>> ---
>>>>>>>>      drivers/usb/dwc3/dwc3-exynos.c | 49 +++-------------------------------
>>>>>>>>      1 file changed, 4 insertions(+), 45 deletions(-)
>>>>>>>>
>>>>>>>> diff --git a/drivers/usb/dwc3/dwc3-exynos.c b/drivers/usb/dwc3/dwc3-exynos.c
>>>>>>>> index 5d365ca51771..7c77f3c69825 100644
>>>>>>>> --- a/drivers/usb/dwc3/dwc3-exynos.c
>>>>>>>> +++ b/drivers/usb/dwc3/dwc3-exynos.c
>>>>>>>> @@ -32,9 +32,6 @@ struct dwc3_exynos {
>>>>>>>>          struct clk              *clks[DWC3_EXYNOS_MAX_CLOCKS];
>>>>>>>>          int                     num_clks;
>>>>>>>>          int                     suspend_clk_idx;
>>>>>>>> -
>>>>>>>> -     struct regulator        *vdd33;
>>>>>>>> -     struct regulator        *vdd10;
>>>>>>>>      };
>>>>>>>>
>>>>>>>>      static int dwc3_exynos_probe(struct platform_device *pdev)
>>>>>>>> @@ -44,6 +41,7 @@ static int dwc3_exynos_probe(struct platform_device *pdev)
>>>>>>>>          struct device_node      *node = dev->of_node;
>>>>>>>>          const struct dwc3_exynos_driverdata *driver_data;
>>>>>>>>          int                     i, ret;
>>>>>>>> +     static const char * const regulators[] = { "vdd33", "vdd10" };
>>>>>>>>
>>>>>>>>          exynos = devm_kzalloc(dev, sizeof(*exynos), GFP_KERNEL);
>>>>>>>>          if (!exynos)
>>>>>>>> @@ -78,27 +76,9 @@ static int dwc3_exynos_probe(struct platform_device *pdev)
>>>>>>>>          if (exynos->suspend_clk_idx >= 0)
>>>>>>>>                  clk_prepare_enable(exynos->clks[exynos->suspend_clk_idx]);
>>>>>>>>
>>>>>>>> -     exynos->vdd33 = devm_regulator_get(dev, "vdd33");
>>>>>>>> -     if (IS_ERR(exynos->vdd33)) {
>>>>>>>> -             ret = PTR_ERR(exynos->vdd33);
>>>>>>>> -             goto vdd33_err;
>>>>>>>> -     }
>>>>>>>> -     ret = regulator_enable(exynos->vdd33);
>>>>>>>> -     if (ret) {
>>>>>>>> -             dev_err(dev, "Failed to enable VDD33 supply\n");
>>>>>>>> -             goto vdd33_err;
>>>>>>>> -     }
>>>>>>>> -
>>>>>>>> -     exynos->vdd10 = devm_regulator_get(dev, "vdd10");
>>>>>>>> -     if (IS_ERR(exynos->vdd10)) {
>>>>>>>> -             ret = PTR_ERR(exynos->vdd10);
>>>>>>>> -             goto vdd10_err;
>>>>>>>> -     }
>>>>>>>> -     ret = regulator_enable(exynos->vdd10);
>>>>>>>> -     if (ret) {
>>>>>>>> -             dev_err(dev, "Failed to enable VDD10 supply\n");
>>>>>>>> -             goto vdd10_err;
>>>>>>>> -     }
>>>>>>>> +     ret = devm_regulator_bulk_get_enable(dev, ARRAY_SIZE(regulators), regulators);
>>>>>>>> +     if (ret)
>>>>>>>> +             return dev_err_probe(dev, ret, "Failed to enable regulators\n");
>>>>>>>>
>>>>>>>>          if (node) {
>>>>>>>>                  ret = of_platform_populate(node, NULL, NULL, dev);
>>>>>>>> @@ -115,10 +95,6 @@ static int dwc3_exynos_probe(struct platform_device *pdev)
>>>>>>>>          return 0;
>>>>>>>>
>>>>>>>>      populate_err:
>>>>>>>> -     regulator_disable(exynos->vdd10);
>>>>>>>> -vdd10_err:
>>>>>>>> -     regulator_disable(exynos->vdd33);
>>>>>>>> -vdd33_err:
>>>>>>>>          for (i = exynos->num_clks - 1; i >= 0; i--)
>>>>>>>>                  clk_disable_unprepare(exynos->clks[i]);
>>>>>>>>
>>>>>>>> @@ -140,9 +116,6 @@ static void dwc3_exynos_remove(struct platform_device *pdev)
>>>>>>>>
>>>>>>>>          if (exynos->suspend_clk_idx >= 0)
>>>>>>>>                  clk_disable_unprepare(exynos->clks[exynos->suspend_clk_idx]);
>>>>>>>> -
>>>>>>>> -     regulator_disable(exynos->vdd33);
>>>>>>>> -     regulator_disable(exynos->vdd10);
>>>>>>>>      }
>>>>>>>>
>>>>>>>>      static const struct dwc3_exynos_driverdata exynos5250_drvdata = {
>>>>>>>> @@ -196,9 +169,6 @@ static int dwc3_exynos_suspend(struct device *dev)
>>>>>>>>          for (i = exynos->num_clks - 1; i >= 0; i--)
>>>>>>>>                  clk_disable_unprepare(exynos->clks[i]);
>>>>>>>>
>>>>>>>> -     regulator_disable(exynos->vdd33);
>>>>>>>> -     regulator_disable(exynos->vdd10);
>>>>>>>
>>>>>>> Hi,
>>>>>>>
>>>>>>> Same here, I don't think that removing regulator_[en|dis]able from the
>>>>>>> suspend and resume function is correct.
>>>>>>>
>>>>>>> The goal is to stop some hardware when the system is suspended, in order
>>>>>>> to save some power.
>>>>>> Ok,
>>>>>>>
>>>>>>> Why did you removed it?
>>>>>>
>>>>>> As per the description of the function  devm_regulator_bulk_get_enable
>>>>>>
>>>>>> * This helper function allows drivers to get several regulator
>>>>>>     * consumers in one operation with management, the regulators will
>>>>>>     * automatically be freed when the device is unbound.  If any of the
>>>>>>     * regulators cannot be acquired then any regulators that were
>>>>>>     * allocated will be freed before returning to the caller.
>>>>>
>>>>> The code in suspend/resume is not about freeing some resources. It is
>>>>> about enabling/disabling some hardware to save some power.
>>>>>
>>>>> Think to the probe/remove functions as the software in the kernel that
>>>>> knows how to handle some hardawre, and the suspend/resume as the on/off
>>>>> button to power-on and off the electrical chips.
>>>>>
>>>>> When the system is suspended, the software is still around. But some
>>>>> hardware can be set in a low consumption mode to save some power.
>>>>>
>>>>> IMHO, part of the code you removed changed this behaviour and increase
>>>>> the power consumption when the system is suspended.
>>>>>
>>>>
>>>> You are correct, I have changed the regulator API from
>>>> devm_regulator_get_enable to devm_regulator_bulk_get_enable
>>>> which changes this behavior.
>>>> I will fix it in the next version.
>>>>
>>>>> CJ
>>>
>>> I could not find any example in the kernel to support
>>> devm_regulator_bulk_disable
>>> but here is my modified file.
>>>
>>> If you have any suggestions for this plz let me know.
>>
>> I don't think that your approach is correct, and I don't think that the
>> proposed patch does what you expect it to do.
>>
>> Calling a devm_ function in suspend/resume functions looks really
>> strange to me and is likely broken.
>>
>> Especially here, devm_regulator_bulk_get_enable() in the resume function
>> allocates some memory that is not freed in
>> devm_regulator_bulk_disable(), because the API is not designed to work
>> like that. So this could generate a kind of memory leak.
>>
>>
>> *I think that the code is good enough as-is*, but if you really want to
>> change something, maybe:
>>      - devm_regulator_get()+regulator_enable() in the probe could be
>> changed to devm_regulator_get_enable()
>>      - the resume/suspend function should be left as-is with
>> regulator_disable()/regulator_ensable()
>>      - remove regulator_disable() from the error handling path of the
>> probe and from the remove function.
>>
>> I *think* it would work.
>>
> No devm_regulator_get_enable use the same logic as
> devm_regulator_bulk_get_enable
> to enable the regulator.

Yes, the logic is the same, but you get a pointer to the "struct 
regulator" which can be used to disable/enable in the suspend/resume 
functions.

With the bulk version, you can not do that.

See my first reply on your 3/4 patch.

> 
> [0] https://elixir.bootlin.com/linux/latest/source/drivers/regulator/devres.c#L126
> 
> So as of now I am dropping the changes on the regulator in this patch series.

I do agree that it is certainly the way to go here.

CJ

> 
>> CJ
>>
> Thanks for your inputs.
> 
> Thanks
> 
> -Anand
> 
> 




More information about the linux-arm-kernel mailing list