diff mbox series

[v2] clk: starfive: jh7110-sys: Fix lower rate of CPUfreq by setting PLL0 rate to 1.5GHz

Message ID 20230821152915.208366-1-xingyu.wu@starfivetech.com (mailing list archive)
State Changes Requested, archived
Headers show
Series [v2] clk: starfive: jh7110-sys: Fix lower rate of CPUfreq by setting PLL0 rate to 1.5GHz | expand

Commit Message

Xingyu Wu Aug. 21, 2023, 3:29 p.m. UTC
CPUfreq supports 4 cpu frequency loads on 375/500/750/1500MHz.
But now PLL0 rate is 1GHz and the cpu frequency loads become
333/500/500/1000MHz in fact.

So PLL0 rate should be set to 1.5GHz. Change the parent of cpu_root clock
and the divider of cpu_core before the setting.

Reviewed-by: Hal Feng <hal.feng@starfivetech.com>
Fixes: e2c510d6d630 ("riscv: dts: starfive: Add cpu scaling for JH7110 SoC")
Signed-off-by: Xingyu Wu <xingyu.wu@starfivetech.com>
---

Hi Stephen and Emil,

This patch fixes the issue about lower rate of CPUfreq[1]
and sets PLL0 rate to 1.5GHz. In order not to affect the cpu 
operation, the cpu_root's parent clock should be changed first.
And the divider of the cpu_core clock should be set to 2 so they
won't crash when setting 1.5GHz without voltage regulation.

[1]: https://github.com/starfive-tech/VisionFive2/issues/55

This patch is based on linux-next(20230818) which has merge PLL driver
on the StarFive JH7110 SoC.

Thanks,
Xingyu Wu

---
 .../clk/starfive/clk-starfive-jh7110-sys.c    | 47 ++++++++++++++++++-
 1 file changed, 46 insertions(+), 1 deletion(-)

Comments

Emil Renner Berthing Aug. 21, 2023, 3:38 p.m. UTC | #1
On Mon, 21 Aug 2023 at 17:29, Xingyu Wu <xingyu.wu@starfivetech.com> wrote:
>
> CPUfreq supports 4 cpu frequency loads on 375/500/750/1500MHz.
> But now PLL0 rate is 1GHz and the cpu frequency loads become
> 333/500/500/1000MHz in fact.
>
> So PLL0 rate should be set to 1.5GHz. Change the parent of cpu_root clock
> and the divider of cpu_core before the setting.
>
> Reviewed-by: Hal Feng <hal.feng@starfivetech.com>
> Fixes: e2c510d6d630 ("riscv: dts: starfive: Add cpu scaling for JH7110 SoC")
> Signed-off-by: Xingyu Wu <xingyu.wu@starfivetech.com>
> ---
>
> Hi Stephen and Emil,
>
> This patch fixes the issue about lower rate of CPUfreq[1]
> and sets PLL0 rate to 1.5GHz. In order not to affect the cpu
> operation, the cpu_root's parent clock should be changed first.
> And the divider of the cpu_core clock should be set to 2 so they
> won't crash when setting 1.5GHz without voltage regulation.

Hi Xingyu,

It sounds like this is something the driver should handle
automatically whenever clk_set_rate() is called on the PLL0 clock.
Then we should be able to use regular assigned-clocks /
assigned-clock-rates stanzas in the device tree instead of having this
1.5GHz rate hard-coded in the driver.

/Emil

> [1]: https://github.com/starfive-tech/VisionFive2/issues/55
>
> This patch is based on linux-next(20230818) which has merge PLL driver
> on the StarFive JH7110 SoC.
>
> Thanks,
> Xingyu Wu
>
> ---
>  .../clk/starfive/clk-starfive-jh7110-sys.c    | 47 ++++++++++++++++++-
>  1 file changed, 46 insertions(+), 1 deletion(-)
>
> diff --git a/drivers/clk/starfive/clk-starfive-jh7110-sys.c b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> index 3884eff9fe93..b6b9e967dfc7 100644
> --- a/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> +++ b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> @@ -501,7 +501,52 @@ static int __init jh7110_syscrg_probe(struct platform_device *pdev)
>         if (ret)
>                 return ret;
>
> -       return jh7110_reset_controller_register(priv, "rst-sys", 0);
> +       ret = jh7110_reset_controller_register(priv, "rst-sys", 0);
> +       if (ret)
> +               return ret;
> +
> +       /*
> +        * Set PLL0 rate to 1.5GHz
> +        * In order to not affect the cpu when the PLL0 rate is changing,
> +        * we need to switch the parent of cpu_root clock to osc clock first,
> +        * and then switch back after setting the PLL0 rate.
> +        */
> +       pllclk = clk_get(priv->dev, "pll0_out");
> +       if (!IS_ERR(pllclk)) {
> +               struct clk *osc = clk_get(&pdev->dev, "osc");
> +               struct clk *cpu_root = priv->reg[JH7110_SYSCLK_CPU_ROOT].hw.clk;
> +               struct clk *cpu_core = priv->reg[JH7110_SYSCLK_CPU_CORE].hw.clk;
> +
> +               if (IS_ERR(osc)) {
> +                       clk_put(pllclk);
> +                       return PTR_ERR(osc);
> +               }
> +
> +               /*
> +                * CPU need voltage regulation by CPUfreq if set 1.5GHz.
> +                * So in this driver, cpu_core need to be set the divider to be 2 first
> +                * and will be 750M after setting parent.
> +                */
> +               ret = clk_set_rate(cpu_core, clk_get_rate(cpu_core) / 2);
> +               if (ret)
> +                       goto failed_set;
> +
> +               ret = clk_set_parent(cpu_root, osc);
> +               if (ret)
> +                       goto failed_set;
> +
> +               ret = clk_set_rate(pllclk, 1500000000);
> +               if (ret)
> +                       goto failed_set;
> +
> +               ret = clk_set_parent(cpu_root, pllclk);
> +
> +failed_set:
> +               clk_put(pllclk);
> +               clk_put(osc);
> +       }
> +
> +       return ret;
>  }
>
>  static const struct of_device_id jh7110_syscrg_match[] = {
> --
> 2.25.1
>
Xingyu Wu Aug. 22, 2023, 3:27 a.m. UTC | #2
On 2023/8/21 23:38, Emil Renner Berthing wrote:
> On Mon, 21 Aug 2023 at 17:29, Xingyu Wu <xingyu.wu@starfivetech.com> wrote:
>>
>> CPUfreq supports 4 cpu frequency loads on 375/500/750/1500MHz.
>> But now PLL0 rate is 1GHz and the cpu frequency loads become
>> 333/500/500/1000MHz in fact.
>>
>> So PLL0 rate should be set to 1.5GHz. Change the parent of cpu_root clock
>> and the divider of cpu_core before the setting.
>>
>> Reviewed-by: Hal Feng <hal.feng@starfivetech.com>
>> Fixes: e2c510d6d630 ("riscv: dts: starfive: Add cpu scaling for JH7110 SoC")
>> Signed-off-by: Xingyu Wu <xingyu.wu@starfivetech.com>
>> ---
>>
>> Hi Stephen and Emil,
>>
>> This patch fixes the issue about lower rate of CPUfreq[1]
>> and sets PLL0 rate to 1.5GHz. In order not to affect the cpu
>> operation, the cpu_root's parent clock should be changed first.
>> And the divider of the cpu_core clock should be set to 2 so they
>> won't crash when setting 1.5GHz without voltage regulation.
> 
> Hi Xingyu,
> 
> It sounds like this is something the driver should handle
> automatically whenever clk_set_rate() is called on the PLL0 clock.
> Then we should be able to use regular assigned-clocks /
> assigned-clock-rates stanzas in the device tree instead of having this
> 1.5GHz rate hard-coded in the driver.
> 
> /Emil

Hi Emil,

The frequency of PLL0 is set according to this process to avoid crash:
1. The divider of the cpu_core clock should be set to 2 if PLL0 is set to 1.5GHz.
2. The parent of cpu_root is changed from pll0 to osc.
3. The PLL0 is set to 1.5GHz.
4. The parent of cpu_root is changed from osc to pll0 back.
I don't think assigned-clock-rates/assigned-clock-parents can do such a complicated job.

Best regards,
Xingyu Wu

> 
>> [1]: https://github.com/starfive-tech/VisionFive2/issues/55
>>
>> This patch is based on linux-next(20230818) which has merge PLL driver
>> on the StarFive JH7110 SoC.
>>
>> Thanks,
>> Xingyu Wu
>>
>> ---
>>  .../clk/starfive/clk-starfive-jh7110-sys.c    | 47 ++++++++++++++++++-
>>  1 file changed, 46 insertions(+), 1 deletion(-)
>>
>> diff --git a/drivers/clk/starfive/clk-starfive-jh7110-sys.c b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
>> index 3884eff9fe93..b6b9e967dfc7 100644
>> --- a/drivers/clk/starfive/clk-starfive-jh7110-sys.c
>> +++ b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
>> @@ -501,7 +501,52 @@ static int __init jh7110_syscrg_probe(struct platform_device *pdev)
>>         if (ret)
>>                 return ret;
>>
>> -       return jh7110_reset_controller_register(priv, "rst-sys", 0);
>> +       ret = jh7110_reset_controller_register(priv, "rst-sys", 0);
>> +       if (ret)
>> +               return ret;
>> +
>> +       /*
>> +        * Set PLL0 rate to 1.5GHz
>> +        * In order to not affect the cpu when the PLL0 rate is changing,
>> +        * we need to switch the parent of cpu_root clock to osc clock first,
>> +        * and then switch back after setting the PLL0 rate.
>> +        */
>> +       pllclk = clk_get(priv->dev, "pll0_out");
>> +       if (!IS_ERR(pllclk)) {
>> +               struct clk *osc = clk_get(&pdev->dev, "osc");
>> +               struct clk *cpu_root = priv->reg[JH7110_SYSCLK_CPU_ROOT].hw.clk;
>> +               struct clk *cpu_core = priv->reg[JH7110_SYSCLK_CPU_CORE].hw.clk;
>> +
>> +               if (IS_ERR(osc)) {
>> +                       clk_put(pllclk);
>> +                       return PTR_ERR(osc);
>> +               }
>> +
>> +               /*
>> +                * CPU need voltage regulation by CPUfreq if set 1.5GHz.
>> +                * So in this driver, cpu_core need to be set the divider to be 2 first
>> +                * and will be 750M after setting parent.
>> +                */
>> +               ret = clk_set_rate(cpu_core, clk_get_rate(cpu_core) / 2);
>> +               if (ret)
>> +                       goto failed_set;
>> +
>> +               ret = clk_set_parent(cpu_root, osc);
>> +               if (ret)
>> +                       goto failed_set;
>> +
>> +               ret = clk_set_rate(pllclk, 1500000000);
>> +               if (ret)
>> +                       goto failed_set;
>> +
>> +               ret = clk_set_parent(cpu_root, pllclk);
>> +
>> +failed_set:
>> +               clk_put(pllclk);
>> +               clk_put(osc);
>> +       }
>> +
>> +       return ret;
>>  }
>>
>>  static const struct of_device_id jh7110_syscrg_match[] = {
>> --
>> 2.25.1
>>
Emil Renner Berthing Aug. 22, 2023, 8:56 a.m. UTC | #3
On Tue, 22 Aug 2023 at 05:33, Xingyu Wu <xingyu.wu@starfivetech.com> wrote:
> On 2023/8/21 23:38, Emil Renner Berthing wrote:
> > On Mon, 21 Aug 2023 at 17:29, Xingyu Wu <xingyu.wu@starfivetech.com> wrote:
> >>
> >> CPUfreq supports 4 cpu frequency loads on 375/500/750/1500MHz.
> >> But now PLL0 rate is 1GHz and the cpu frequency loads become
> >> 333/500/500/1000MHz in fact.
> >>
> >> So PLL0 rate should be set to 1.5GHz. Change the parent of cpu_root clock
> >> and the divider of cpu_core before the setting.
> >>
> >> Reviewed-by: Hal Feng <hal.feng@starfivetech.com>
> >> Fixes: e2c510d6d630 ("riscv: dts: starfive: Add cpu scaling for JH7110 SoC")
> >> Signed-off-by: Xingyu Wu <xingyu.wu@starfivetech.com>
> >> ---
> >>
> >> Hi Stephen and Emil,
> >>
> >> This patch fixes the issue about lower rate of CPUfreq[1]
> >> and sets PLL0 rate to 1.5GHz. In order not to affect the cpu
> >> operation, the cpu_root's parent clock should be changed first.
> >> And the divider of the cpu_core clock should be set to 2 so they
> >> won't crash when setting 1.5GHz without voltage regulation.
> >
> > Hi Xingyu,
> >
> > It sounds like this is something the driver should handle
> > automatically whenever clk_set_rate() is called on the PLL0 clock.
> > Then we should be able to use regular assigned-clocks /
> > assigned-clock-rates stanzas in the device tree instead of having this
> > 1.5GHz rate hard-coded in the driver.
> >
> > /Emil
>
> Hi Emil,
>
> The frequency of PLL0 is set according to this process to avoid crash:
> 1. The divider of the cpu_core clock should be set to 2 if PLL0 is set to 1.5GHz.
> 2. The parent of cpu_root is changed from pll0 to osc.
> 3. The PLL0 is set to 1.5GHz.
> 4. The parent of cpu_root is changed from osc to pll0 back.
> I don't think assigned-clock-rates/assigned-clock-parents can do such a complicated job.

Right, but what I'm saying is that if calling clk_set_rate() on the
PLL0 clock causes a crash, that sounds like a bug in the driver. If
you fix that bug, and make clk_set_rate() on the PLL0 clock do the
procedure above when changing the rate, then the PLL0 clock can work
just like any other clock and assigned-clock-rates would work.

/Emil

>
> Best regards,
> Xingyu Wu
>
> >
> >> [1]: https://github.com/starfive-tech/VisionFive2/issues/55
> >>
> >> This patch is based on linux-next(20230818) which has merge PLL driver
> >> on the StarFive JH7110 SoC.
> >>
> >> Thanks,
> >> Xingyu Wu
> >>
> >> ---
> >>  .../clk/starfive/clk-starfive-jh7110-sys.c    | 47 ++++++++++++++++++-
> >>  1 file changed, 46 insertions(+), 1 deletion(-)
> >>
> >> diff --git a/drivers/clk/starfive/clk-starfive-jh7110-sys.c b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> >> index 3884eff9fe93..b6b9e967dfc7 100644
> >> --- a/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> >> +++ b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> >> @@ -501,7 +501,52 @@ static int __init jh7110_syscrg_probe(struct platform_device *pdev)
> >>         if (ret)
> >>                 return ret;
> >>
> >> -       return jh7110_reset_controller_register(priv, "rst-sys", 0);
> >> +       ret = jh7110_reset_controller_register(priv, "rst-sys", 0);
> >> +       if (ret)
> >> +               return ret;
> >> +
> >> +       /*
> >> +        * Set PLL0 rate to 1.5GHz
> >> +        * In order to not affect the cpu when the PLL0 rate is changing,
> >> +        * we need to switch the parent of cpu_root clock to osc clock first,
> >> +        * and then switch back after setting the PLL0 rate.
> >> +        */
> >> +       pllclk = clk_get(priv->dev, "pll0_out");
> >> +       if (!IS_ERR(pllclk)) {
> >> +               struct clk *osc = clk_get(&pdev->dev, "osc");
> >> +               struct clk *cpu_root = priv->reg[JH7110_SYSCLK_CPU_ROOT].hw.clk;
> >> +               struct clk *cpu_core = priv->reg[JH7110_SYSCLK_CPU_CORE].hw.clk;
> >> +
> >> +               if (IS_ERR(osc)) {
> >> +                       clk_put(pllclk);
> >> +                       return PTR_ERR(osc);
> >> +               }
> >> +
> >> +               /*
> >> +                * CPU need voltage regulation by CPUfreq if set 1.5GHz.
> >> +                * So in this driver, cpu_core need to be set the divider to be 2 first
> >> +                * and will be 750M after setting parent.
> >> +                */
> >> +               ret = clk_set_rate(cpu_core, clk_get_rate(cpu_core) / 2);
> >> +               if (ret)
> >> +                       goto failed_set;
> >> +
> >> +               ret = clk_set_parent(cpu_root, osc);
> >> +               if (ret)
> >> +                       goto failed_set;
> >> +
> >> +               ret = clk_set_rate(pllclk, 1500000000);
> >> +               if (ret)
> >> +                       goto failed_set;
> >> +
> >> +               ret = clk_set_parent(cpu_root, pllclk);
> >> +
> >> +failed_set:
> >> +               clk_put(pllclk);
> >> +               clk_put(osc);
> >> +       }
> >> +
> >> +       return ret;
> >>  }
> >>
> >>  static const struct of_device_id jh7110_syscrg_match[] = {
> >> --
> >> 2.25.1
> >>
>
Xingyu Wu Oct. 8, 2023, 9:16 a.m. UTC | #4
On 2023/8/22 16:56, Emil Renner Berthing wrote:
> On Tue, 22 Aug 2023 at 05:33, Xingyu Wu <xingyu.wu@starfivetech.com> wrote:
>> On 2023/8/21 23:38, Emil Renner Berthing wrote:
>> > On Mon, 21 Aug 2023 at 17:29, Xingyu Wu <xingyu.wu@starfivetech.com> wrote:
>> >>
>> >> CPUfreq supports 4 cpu frequency loads on 375/500/750/1500MHz.
>> >> But now PLL0 rate is 1GHz and the cpu frequency loads become
>> >> 333/500/500/1000MHz in fact.
>> >>
>> >> So PLL0 rate should be set to 1.5GHz. Change the parent of cpu_root clock
>> >> and the divider of cpu_core before the setting.
>> >>
>> >> Reviewed-by: Hal Feng <hal.feng@starfivetech.com>
>> >> Fixes: e2c510d6d630 ("riscv: dts: starfive: Add cpu scaling for JH7110 SoC")
>> >> Signed-off-by: Xingyu Wu <xingyu.wu@starfivetech.com>
>> >> ---
>> >>
>> >> Hi Stephen and Emil,
>> >>
>> >> This patch fixes the issue about lower rate of CPUfreq[1]
>> >> and sets PLL0 rate to 1.5GHz. In order not to affect the cpu
>> >> operation, the cpu_root's parent clock should be changed first.
>> >> And the divider of the cpu_core clock should be set to 2 so they
>> >> won't crash when setting 1.5GHz without voltage regulation.
>> >
>> > Hi Xingyu,
>> >
>> > It sounds like this is something the driver should handle
>> > automatically whenever clk_set_rate() is called on the PLL0 clock.
>> > Then we should be able to use regular assigned-clocks /
>> > assigned-clock-rates stanzas in the device tree instead of having this
>> > 1.5GHz rate hard-coded in the driver.
>> >
>> > /Emil
>>
>> Hi Emil,
>>
>> The frequency of PLL0 is set according to this process to avoid crash:
>> 1. The divider of the cpu_core clock should be set to 2 if PLL0 is set to 1.5GHz.
>> 2. The parent of cpu_root is changed from pll0 to osc.
>> 3. The PLL0 is set to 1.5GHz.
>> 4. The parent of cpu_root is changed from osc to pll0 back.
>> I don't think assigned-clock-rates/assigned-clock-parents can do such a complicated job.
> 
> Right, but what I'm saying is that if calling clk_set_rate() on the
> PLL0 clock causes a crash, that sounds like a bug in the driver. If
> you fix that bug, and make clk_set_rate() on the PLL0 clock do the
> procedure above when changing the rate, then the PLL0 clock can work
> just like any other clock and assigned-clock-rates would work.
> 
> /Emil

Yeah, if fix this bug, I should add there steps when setting rate in the PLL clock driver. But how to get and use the clocks of cpu_root and cpu_core in the PLL driver? It seem to be only two ways, clk_get() or ioremap(). If use clk_get(), the pll node should add syscrg clocks. It shows that the PLL driver depends on the SYSCRG driver, which seems confusing because normally the syscrg depends on the pll. Do you have a better suggestion?

Thanks.
Xingyu Wu

> 
>> >
>> >> [1]: https://github.com/starfive-tech/VisionFive2/issues/55
>> >>
>> >> This patch is based on linux-next(20230818) which has merge PLL driver
>> >> on the StarFive JH7110 SoC.
>> >>
>> >> Thanks,
>> >> Xingyu Wu
>> >>
>> >> ---
>> >>  .../clk/starfive/clk-starfive-jh7110-sys.c    | 47 ++++++++++++++++++-
>> >>  1 file changed, 46 insertions(+), 1 deletion(-)
>> >>
>> >> diff --git a/drivers/clk/starfive/clk-starfive-jh7110-sys.c b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
>> >> index 3884eff9fe93..b6b9e967dfc7 100644
>> >> --- a/drivers/clk/starfive/clk-starfive-jh7110-sys.c
>> >> +++ b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
>> >> @@ -501,7 +501,52 @@ static int __init jh7110_syscrg_probe(struct platform_device *pdev)
>> >>         if (ret)
>> >>                 return ret;
>> >>
>> >> -       return jh7110_reset_controller_register(priv, "rst-sys", 0);
>> >> +       ret = jh7110_reset_controller_register(priv, "rst-sys", 0);
>> >> +       if (ret)
>> >> +               return ret;
>> >> +
>> >> +       /*
>> >> +        * Set PLL0 rate to 1.5GHz
>> >> +        * In order to not affect the cpu when the PLL0 rate is changing,
>> >> +        * we need to switch the parent of cpu_root clock to osc clock first,
>> >> +        * and then switch back after setting the PLL0 rate.
>> >> +        */
>> >> +       pllclk = clk_get(priv->dev, "pll0_out");
>> >> +       if (!IS_ERR(pllclk)) {
>> >> +               struct clk *osc = clk_get(&pdev->dev, "osc");
>> >> +               struct clk *cpu_root = priv->reg[JH7110_SYSCLK_CPU_ROOT].hw.clk;
>> >> +               struct clk *cpu_core = priv->reg[JH7110_SYSCLK_CPU_CORE].hw.clk;
>> >> +
>> >> +               if (IS_ERR(osc)) {
>> >> +                       clk_put(pllclk);
>> >> +                       return PTR_ERR(osc);
>> >> +               }
>> >> +
>> >> +               /*
>> >> +                * CPU need voltage regulation by CPUfreq if set 1.5GHz.
>> >> +                * So in this driver, cpu_core need to be set the divider to be 2 first
>> >> +                * and will be 750M after setting parent.
>> >> +                */
>> >> +               ret = clk_set_rate(cpu_core, clk_get_rate(cpu_core) / 2);
>> >> +               if (ret)
>> >> +                       goto failed_set;
>> >> +
>> >> +               ret = clk_set_parent(cpu_root, osc);
>> >> +               if (ret)
>> >> +                       goto failed_set;
>> >> +
>> >> +               ret = clk_set_rate(pllclk, 1500000000);
>> >> +               if (ret)
>> >> +                       goto failed_set;
>> >> +
>> >> +               ret = clk_set_parent(cpu_root, pllclk);
>> >> +
>> >> +failed_set:
>> >> +               clk_put(pllclk);
>> >> +               clk_put(osc);
>> >> +       }
>> >> +
>> >> +       return ret;
>> >>  }
>> >>
>> >>  static const struct of_device_id jh7110_syscrg_match[] = {
>> >> --
>> >> 2.25.1
>> >>
>>
Xingyu Wu March 15, 2024, 3:50 a.m. UTC | #5
> On 2023/8/22 16:56, Emil Renner Berthing wrote:
> > On Tue, 22 Aug 2023 at 05:33, Xingyu Wu <xingyu.wu@starfivetech.com> wrote:
> >> On 2023/8/21 23:38, Emil Renner Berthing wrote:
> >> > On Mon, 21 Aug 2023 at 17:29, Xingyu Wu <xingyu.wu@starfivetech.com>
> wrote:
> >> >>
> >> >> CPUfreq supports 4 cpu frequency loads on 375/500/750/1500MHz.
> >> >> But now PLL0 rate is 1GHz and the cpu frequency loads become
> >> >> 333/500/500/1000MHz in fact.
> >> >>
> >> >> So PLL0 rate should be set to 1.5GHz. Change the parent of
> >> >> cpu_root clock and the divider of cpu_core before the setting.
> >> >>
> >> >> Reviewed-by: Hal Feng <hal.feng@starfivetech.com>
> >> >> Fixes: e2c510d6d630 ("riscv: dts: starfive: Add cpu scaling for
> >> >> JH7110 SoC")
> >> >> Signed-off-by: Xingyu Wu <xingyu.wu@starfivetech.com>
> >> >> ---
> >> >>
> >> >> Hi Stephen and Emil,
> >> >>
> >> >> This patch fixes the issue about lower rate of CPUfreq[1] and sets
> >> >> PLL0 rate to 1.5GHz. In order not to affect the cpu operation, the
> >> >> cpu_root's parent clock should be changed first.
> >> >> And the divider of the cpu_core clock should be set to 2 so they
> >> >> won't crash when setting 1.5GHz without voltage regulation.
> >> >
> >> > Hi Xingyu,
> >> >
> >> > It sounds like this is something the driver should handle
> >> > automatically whenever clk_set_rate() is called on the PLL0 clock.
> >> > Then we should be able to use regular assigned-clocks /
> >> > assigned-clock-rates stanzas in the device tree instead of having
> >> > this 1.5GHz rate hard-coded in the driver.
> >> >
> >> > /Emil
> >>
> >> Hi Emil,
> >>
> >> The frequency of PLL0 is set according to this process to avoid crash:
> >> 1. The divider of the cpu_core clock should be set to 2 if PLL0 is set to 1.5GHz.
> >> 2. The parent of cpu_root is changed from pll0 to osc.
> >> 3. The PLL0 is set to 1.5GHz.
> >> 4. The parent of cpu_root is changed from osc to pll0 back.
> >> I don't think assigned-clock-rates/assigned-clock-parents can do such a
> complicated job.
> >
> > Right, but what I'm saying is that if calling clk_set_rate() on the
> > PLL0 clock causes a crash, that sounds like a bug in the driver. If
> > you fix that bug, and make clk_set_rate() on the PLL0 clock do the
> > procedure above when changing the rate, then the PLL0 clock can work
> > just like any other clock and assigned-clock-rates would work.
> >
> > /Emil
> 
> Yeah, if fix this bug, I should add there steps when setting rate in the PLL clock driver.
> But how to get and use the clocks of cpu_root and cpu_core in the PLL driver?
> It seem to be only two ways, clk_get() or ioremap(). If use clk_get(), the pll node
> should add syscrg clocks.
> It looks confusing because normally the SYSCRG driver depends on the PLL driver.
> 

Hi Emil,

I'm sorry for forgetting this patch. I tested that I added this step while setting the pll0 rate. I used clk_get() to get the clocks and added the syscrg clock property in the pll node, but the system was not work and crashed. Can I only use ioremap()?

Best regards,
Xingyu Wu

> >>
> >> >
> >> >> [1]: https://github.com/starfive-tech/VisionFive2/issues/55
> >> >>
> >> >> This patch is based on linux-next(20230818) which has merge PLL
> >> >> driver on the StarFive JH7110 SoC.
> >> >>
> >> >> Thanks,
> >> >> Xingyu Wu
> >> >>
> >> >> ---
> >> >>  .../clk/starfive/clk-starfive-jh7110-sys.c    | 47 ++++++++++++++++++-
> >> >>  1 file changed, 46 insertions(+), 1 deletion(-)
> >> >>
> >> >> diff --git a/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> >> >> b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> >> >> index 3884eff9fe93..b6b9e967dfc7 100644
> >> >> --- a/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> >> >> +++ b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
> >> >> @@ -501,7 +501,52 @@ static int __init jh7110_syscrg_probe(struct
> platform_device *pdev)
> >> >>         if (ret)
> >> >>                 return ret;
> >> >>
> >> >> -       return jh7110_reset_controller_register(priv, "rst-sys", 0);
> >> >> +       ret = jh7110_reset_controller_register(priv, "rst-sys", 0);
> >> >> +       if (ret)
> >> >> +               return ret;
> >> >> +
> >> >> +       /*
> >> >> +        * Set PLL0 rate to 1.5GHz
> >> >> +        * In order to not affect the cpu when the PLL0 rate is changing,
> >> >> +        * we need to switch the parent of cpu_root clock to osc clock first,
> >> >> +        * and then switch back after setting the PLL0 rate.
> >> >> +        */
> >> >> +       pllclk = clk_get(priv->dev, "pll0_out");
> >> >> +       if (!IS_ERR(pllclk)) {
> >> >> +               struct clk *osc = clk_get(&pdev->dev, "osc");
> >> >> +               struct clk *cpu_root =
> priv->reg[JH7110_SYSCLK_CPU_ROOT].hw.clk;
> >> >> +               struct clk *cpu_core =
> >> >> + priv->reg[JH7110_SYSCLK_CPU_CORE].hw.clk;
> >> >> +
> >> >> +               if (IS_ERR(osc)) {
> >> >> +                       clk_put(pllclk);
> >> >> +                       return PTR_ERR(osc);
> >> >> +               }
> >> >> +
> >> >> +               /*
> >> >> +                * CPU need voltage regulation by CPUfreq if set 1.5GHz.
> >> >> +                * So in this driver, cpu_core need to be set the divider to
> be 2 first
> >> >> +                * and will be 750M after setting parent.
> >> >> +                */
> >> >> +               ret = clk_set_rate(cpu_core, clk_get_rate(cpu_core) / 2);
> >> >> +               if (ret)
> >> >> +                       goto failed_set;
> >> >> +
> >> >> +               ret = clk_set_parent(cpu_root, osc);
> >> >> +               if (ret)
> >> >> +                       goto failed_set;
> >> >> +
> >> >> +               ret = clk_set_rate(pllclk, 1500000000);
> >> >> +               if (ret)
> >> >> +                       goto failed_set;
> >> >> +
> >> >> +               ret = clk_set_parent(cpu_root, pllclk);
> >> >> +
> >> >> +failed_set:
> >> >> +               clk_put(pllclk);
> >> >> +               clk_put(osc);
> >> >> +       }
> >> >> +
> >> >> +       return ret;
> >> >>  }
> >> >>
> >> >>  static const struct of_device_id jh7110_syscrg_match[] = {
> >> >> --
> >> >> 2.25.1
> >> >>
> >>
diff mbox series

Patch

diff --git a/drivers/clk/starfive/clk-starfive-jh7110-sys.c b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
index 3884eff9fe93..b6b9e967dfc7 100644
--- a/drivers/clk/starfive/clk-starfive-jh7110-sys.c
+++ b/drivers/clk/starfive/clk-starfive-jh7110-sys.c
@@ -501,7 +501,52 @@  static int __init jh7110_syscrg_probe(struct platform_device *pdev)
 	if (ret)
 		return ret;
 
-	return jh7110_reset_controller_register(priv, "rst-sys", 0);
+	ret = jh7110_reset_controller_register(priv, "rst-sys", 0);
+	if (ret)
+		return ret;
+
+	/*
+	 * Set PLL0 rate to 1.5GHz
+	 * In order to not affect the cpu when the PLL0 rate is changing,
+	 * we need to switch the parent of cpu_root clock to osc clock first,
+	 * and then switch back after setting the PLL0 rate.
+	 */
+	pllclk = clk_get(priv->dev, "pll0_out");
+	if (!IS_ERR(pllclk)) {
+		struct clk *osc = clk_get(&pdev->dev, "osc");
+		struct clk *cpu_root = priv->reg[JH7110_SYSCLK_CPU_ROOT].hw.clk;
+		struct clk *cpu_core = priv->reg[JH7110_SYSCLK_CPU_CORE].hw.clk;
+
+		if (IS_ERR(osc)) {
+			clk_put(pllclk);
+			return PTR_ERR(osc);
+		}
+
+		/*
+		 * CPU need voltage regulation by CPUfreq if set 1.5GHz.
+		 * So in this driver, cpu_core need to be set the divider to be 2 first
+		 * and will be 750M after setting parent.
+		 */
+		ret = clk_set_rate(cpu_core, clk_get_rate(cpu_core) / 2);
+		if (ret)
+			goto failed_set;
+
+		ret = clk_set_parent(cpu_root, osc);
+		if (ret)
+			goto failed_set;
+
+		ret = clk_set_rate(pllclk, 1500000000);
+		if (ret)
+			goto failed_set;
+
+		ret = clk_set_parent(cpu_root, pllclk);
+
+failed_set:
+		clk_put(pllclk);
+		clk_put(osc);
+	}
+
+	return ret;
 }
 
 static const struct of_device_id jh7110_syscrg_match[] = {