Re: [PATCH] ARC: [plat-hsdk]: Temporary fix to set CPU frequency to 1GHz

From: Eugeniy Paltsev
Date: Fri Sep 29 2017 - 14:54:43 EST


On Fri, 2017-09-29 at 11:34 -0700, Vineet Gupta wrote:
> On 09/28/2017 07:33 AM, Eugeniy Paltsev wrote:
> > Add temporary fix to HSDK platform code to setup CPU frequency
> > to 1GHz on early boot.
> > We can remove this fix when smart hsdk pll driver will be
> > introduced, see discussion:
> > https://www.mail-archive.com/linux-snps-arc@xxxxxxxxxxxxxxxxxxx/msg02689.html
> >
> > Signed-off-by: Eugeniy Paltsev <Eugeniy.Paltsev@xxxxxxxxxxxx>
> > ---
> > Â arch/arc/plat-hsdk/platform.c | 42 ++++++++++++++++++++++++++++++++++++++++++
> > Â 1 file changed, 42 insertions(+)
> >
> > diff --git a/arch/arc/plat-hsdk/platform.c b/arch/arc/plat-hsdk/platform.c
> > index a2e7fd1..744e62e 100644
> > --- a/arch/arc/plat-hsdk/platform.c
> > +++ b/arch/arc/plat-hsdk/platform.c
> > @@ -38,6 +38,42 @@ static void __init hsdk_init_per_cpu(unsigned int cpu)
> > Â #define CREG_PAE (CREG_BASE + 0x180)
> > Â #define CREG_PAE_UPDATE (CREG_BASE + 0x194)
> > ÂÂ
> > +#define CREG_CORE_IF_CLK_DIV (CREG_BASE + 0x4B8)
> > +#define CREG_CORE_IF_CLK_DIV_2 0x1
> > +#define CGU_BASE ARC_PERIPHERAL_BASE
> > +#define CGU_PLL_STATUS (ARC_PERIPHERAL_BASE + 0x4)
> > +#define CGU_PLL_CTRL (ARC_PERIPHERAL_BASE + 0x0)
> > +#define CGU_PLL_STATUS_LOCK BIT(0)
> > +#define CGU_PLL_STATUS_ERR BIT(1)
> > +#define CGU_PLL_CTRL_1GHZ 0x3A10
> > +#define HSDK_PLL_LOCK_TIMEOUT 500
> > +
> > +#define HSDK_PLL_LOCKED() \
> > + !!(ioread32((void __iomem *) CGU_PLL_STATUS) & CGU_PLL_STATUS_LOCK)
> > +
> > +#define HSDK_PLL_ERR() \
> > + !!(ioread32((void __iomem *) CGU_PLL_STATUS) & CGU_PLL_STATUS_ERR)
> > +
> > +static void __init hsdk_set_cpu_freq_1ghz(void)
> > +{
> > + u32 timeout = HSDK_PLL_LOCK_TIMEOUT;
> > +
> > + /*
> > + Â* As we set cpu clock which exceeds 500MHz, the divider for the interface
> > + Â* clock must be programmed to div-by-2.
> > + Â*/
> > + iowrite32(CREG_CORE_IF_CLK_DIV_2, (void __iomem *) CREG_CORE_IF_CLK_DIV);
> > +
> > + /* Set cpu clock to 1GHz */
> > + iowrite32(CGU_PLL_CTRL_1GHZ, (void __iomem *) CGU_PLL_CTRL);
> > +
> > + while (!HSDK_PLL_LOCKED() && timeout--)
> > + cpu_relax();
> > +
> > + if (!HSDK_PLL_LOCKED() || HSDK_PLL_ERR())
> > + pr_err("Failed to setup CPU frequency to 1GHz!");
> > +}
> > +
> > Â static void __init hsdk_init_early(void)
> > Â {
> > ÂÂ /*
> > @@ -52,6 +88,12 @@ static void __init hsdk_init_early(void)
> > ÂÂ
> > ÂÂ /* Really apply settings made above */
> > ÂÂ writel(1, (void __iomem *) CREG_PAE_UPDATE);
> > +
> > + /*
> > + Â* Setup CPU frequency to 1GHz.
> > + Â* TODO: remove it after smart hsdk pll driver will be introduced.
> > + Â*/
> > + hsdk_set_cpu_freq_1ghz();
> > Â }
>
> While this suffices our needs in the interim, is there a way to invoke theÂ
> existing clk driver to achieve the same results ?

There is only one place where we can add call of existing clk driver to set
cpu frequency.
It is in time_init function between early clock initialising and clocksource timer
probing:

------------------arch/arc/kernel/setup.c--------------------------
/*
Â* Called from start_kernel() - boot CPU only
Â*/
void __init time_init(void)
{
of_clk_init(NULL);
/* <--- We can add it here. */
timer_probe();
}
--------------------------->8--------------------------------------

It will be look like that:
https://www.mail-archive.com/linux-snps-arc@xxxxxxxxxxxxxxxxxxx/msg02587.html


> > ÂÂ
> > Â static const char *hsdk_compat[] __initconst = {
> >
>
>
--
ÂEugeniy Paltsev