[PATCH v1 2/9] clk: nuvoton: Add support for NPCM845
Sean Anderson
seanga2 at gmail.com
Thu Dec 16 05:32:20 AEDT 2021
On 12/14/21 9:57 PM, Stanley Chu wrote:
> Add clock controller driver for NPCM845
>
> Signed-off-by: Stanley Chu <yschu at nuvoton.com>
> ---
> drivers/clk/Makefile | 1 +
> drivers/clk/nuvoton/Makefile | 1 +
> drivers/clk/nuvoton/clk_npcm8xx.c | 213 ++++++++++++++++++++++
> include/dt-bindings/clock/npcm845-clock.h | 17 ++
> 4 files changed, 232 insertions(+)
> create mode 100644 drivers/clk/nuvoton/Makefile
> create mode 100644 drivers/clk/nuvoton/clk_npcm8xx.c
> create mode 100644 include/dt-bindings/clock/npcm845-clock.h
>
> diff --git a/drivers/clk/Makefile b/drivers/clk/Makefile
> index 711ae5bc29..a3b64b73c2 100644
> --- a/drivers/clk/Makefile
> +++ b/drivers/clk/Makefile
> @@ -53,3 +53,4 @@ obj-$(CONFIG_STM32H7) += clk_stm32h7.o
> obj-$(CONFIG_CLK_VERSAL) += clk_versal.o
> obj-$(CONFIG_CLK_CDCE9XX) += clk-cdce9xx.o
> obj-$(CONFIG_CLK_VERSACLOCK) += clk_versaclock.o
> +obj-$(CONFIG_ARCH_NPCM) += nuvoton/
Please keep this in alphabetical order (I know the file isn't perfect).
> diff --git a/drivers/clk/nuvoton/Makefile b/drivers/clk/nuvoton/Makefile
> new file mode 100644
> index 0000000000..998e5329bb
> --- /dev/null
> +++ b/drivers/clk/nuvoton/Makefile
> @@ -0,0 +1 @@
> +obj-$(CONFIG_ARCH_NPCM8XX) += clk_npcm8xx.o
> diff --git a/drivers/clk/nuvoton/clk_npcm8xx.c b/drivers/clk/nuvoton/clk_npcm8xx.c
> new file mode 100644
> index 0000000000..c547c47e82
> --- /dev/null
> +++ b/drivers/clk/nuvoton/clk_npcm8xx.c
> @@ -0,0 +1,213 @@
> +// SPDX-License-Identifier: GPL-2.0+
> +/*
> + * Copyright (c) 2021 Nuvoton Technology.
> + */
> +
> +#include <common.h>
> +#include <dm.h>
> +#include <clk-uclass.h>
> +#include <asm/types.h>
> +#include <asm/arch/clock.h>
Please add this include file in this patch.
> +#include <asm/io.h>
> +#include <linux/delay.h>
> +#include <dt-bindings/clock/npcm845-clock.h>
Please order these correctly. See https://www.denx.de/wiki/U-Boot/CodingStyle#Include_files
> +
> +struct npcm_clk_priv {
> + struct clk_ctl *regs;
> +};
> +
> +enum regss {
perhaps "pll_id" or similar?
> + PLL_0,
> + PLL_1,
> + PLL_2,
> + PLL_CLKREF,
> +};
> +
> +static u32 clk_get_pll_freq(struct clk_ctl *regs, enum regss pll)
> +{
> + u32 pllval;
> + u32 fin = EXT_CLOCK_FREQUENCY_KHZ; /* 25KHz */
Please get this from the device tree.
> + u32 fout, nr, nf, no;
> +
> + switch (pll) {
> + case PLL_0:
> + pllval = readl(®s->pllcon0);
> + break;
> + case PLL_1:
> + pllval = readl(®s->pllcon1);
> + break;
> + case PLL_2:
> + pllval = readl(®s->pllcon2);
> + break;
> + case PLL_CLKREF:
This is not used.
> + default:
> + return 0;
> + }
> +
> + /* PLL Input Clock Divider */
> + nr = (pllval >> PLLCON_INDV) & 0x1f;
With
#define PLLCON_INDV GENMASK(6, 0)
you can do
nr = FIELD_GET(PLLCON_INDV, pllval);
This applies to all your other register accesses.
> + /* PLL VCO Output Clock Feedback Divider */
> + nf = (pllval >> PLLCON_FBDV) & 0xfff;
> + /* PLL Output Clock Divider 1 */
> + no = ((pllval >> PLLCON_OTDV1) & 0x7) *
> + ((pllval >> PLLCON_OTDV2) & 0x7);
> +
> + fout = ((10 * fin * nf) / (no * nr));
Can this overflow? Can you add a comment about that?
> +
> + return fout * 100;
Where do these 10 and 100 factors come from? Please combine them.
> +}
> +
> +static u32 npcm_mmc_set_rate(struct clk_ctl *regs, ulong rate)
> +{
> + u32 pll0_freq, div, sdhci_clk;
> +
> + /* To acquire PLL0 frequency. */
> + pll0_freq = clk_get_pll_freq(regs, PLL_0);
> +
> + /* Calculate rounded up div to produce closest to
> + * target output clock
> + */
> + div = (pll0_freq % rate == 0) ? (pll0_freq / rate) :
> + (pll0_freq / rate) + 1;
div = DIV_ROUND_UP(pll0_freq, rate);
> +
> + writel((readl(®s->clkdiv1) & ~(0x1f << CLKDIV1_MMCCKDIV))
> + | (div - 1) << CLKDIV1_MMCCKDIV, ®s->clkdiv1);
example of FIELD_PREP:
clkdiv1 = readl(®s->clkdiv1);
clkdiv1 &= ~CLKDIV1_MMCCKDIV;
clkdiv1 |= FIELD_PREP(CLKDIV1_MMCCKDIV, div - 1);
writel(clkdiv1, ®s->clkdiv1);
You don't have to break out each line, but please apply this to your
register writes.
> +
> + /* Wait to the div to stabilize */
> + udelay(100);
> +
> + /* Select PLL0 as source */
> + writel((readl(®s->clksel) & ~(0x3 << CLKSEL_SDCKSEL))
> + | (CLKSEL_SDCKSEL_PLL0 << CLKSEL_SDCKSEL),
> + ®s->clksel);
> +
> + sdhci_clk = pll0_freq / div;
> +
> + return sdhci_clk;
> +}
> +
> +static u32 npcm_uart_set_rate(struct clk_ctl *regs, ulong rate)
> +{
> + u32 src_freq, div;
> +
> + src_freq = clk_get_pll_freq(regs, PLL_2) / 2;
> + div = (src_freq % rate == 0) ? (src_freq / rate) :
> + (src_freq / rate) + 1;
> + writel((readl(®s->clkdiv1) & ~(0x1f << CLKDIV1_UARTDIV))
> + | (div - 1) << CLKDIV1_UARTDIV, ®s->clkdiv1);
> + writel((readl(®s->clksel) & ~(3 << CLKSEL_UARTCKSEL))
> + | (CLKSEL_UARTCKSEL_PLL2 << CLKSEL_UARTCKSEL),
> + ®s->clksel);
> +
> + return (src_freq / div);
> +}
> +
> +static ulong npcm_get_cpu_freq(struct clk_ctl *regs)
> +{
> + ulong fout = 0;
> + u32 clksel = readl(®s->clksel) & (0x3 << CLKSEL_CPUCKSEL);
> +
> + if (clksel == CLKSEL_CPUCKSEL_PLL0)
Use a switch here.
> + fout = (ulong)clk_get_pll_freq(regs, PLL_0);
> + else if (clksel == CLKSEL_CPUCKSEL_PLL1)
> + fout = (ulong)clk_get_pll_freq(regs, PLL_1);
> + else if (clksel == CLKSEL_CPUCKSEL_CLKREF)
> + fout = EXT_CLOCK_FREQUENCY_MHZ; /* FOUT is specified in MHz */
> + else
> + fout = EXT_CLOCK_FREQUENCY_MHZ; /* FOUT is specified in MHz */
> +
> + return fout;
> +}
> +
> +static u32 npcm_get_apb_divisor(struct clk_ctl *regs, u32 apb)
> +{
> + u32 apb_divisor = 2;
Just inline this. E.g.
return 2 << ...;
> +
> + /* AHBn div */
> + apb_divisor = apb_divisor * (1 << ((readl(®s->clkdiv1)
> + >> CLKDIV1_CLK4DIV) & 0x3));
> +
> + switch (apb) {
> + case APB2: /* APB divisor */
> + apb_divisor = apb_divisor *
> + (1 << ((readl(®s->clkdiv2)
> + >> CLKDIV2_APB2CKDIV) & 0x3));
> + break;
> + case APB5: /* APB divisor */
> + apb_divisor = apb_divisor *
> + (1 << ((readl(®s->clkdiv2)
> + >> CLKDIV2_APB5CKDIV) & 0x3));
> + break;
> + default:
> + apb_divisor = 0xFFFFFFFF;
Isn't getting here a bug?
> + break;
> + }
> +
> + return apb_divisor;
> +}
> +
> +static ulong npcm_clk_get_rate(struct clk *clk)
> +{
> + struct npcm_clk_priv *priv = dev_get_priv(clk->dev);
> +
> + switch (clk->id) {
> + case CLK_APB2:
> + return npcm_get_cpu_freq(priv->regs) /
> + npcm_get_apb_divisor(priv->regs, APB2);
> + case CLK_APB5:
> + return npcm_get_cpu_freq(priv->regs) /
> + npcm_get_apb_divisor(priv->regs, APB5);
I think you can use a more modular approach here:
struct clk parent;
switch (clk->id) {
case CLK_APB2:
parent.id = CLK_AHB;
ret = clk_request(clk->dev, &parent);
if (ret)
return ret;
fin = clk_get_rate(&parent);
if (IS_ERR_VALUE(fin))
return fin;
return fin / FIELD_GET(CLKDIV2_APB2CKDIV, readl(®s->clkdiv2));
}
And of course you can go further and create some arrays to hold those
parameters if you like.
This switch statement should also return -ENOSYS in the default case.
> + }
> +
> + return 0;
> +}
> +
> +static ulong npcm_clk_set_rate(struct clk *clk, ulong rate)
> +{
> + struct npcm_clk_priv *priv = dev_get_priv(clk->dev);
> +
> + switch (clk->id) {
> + case CLK_EMMC:
> + return npcm_mmc_set_rate(priv->regs, rate);
> +
> + case CLK_UART:
> + return npcm_uart_set_rate(priv->regs, rate);
> + default:
> + return -ENOENT;
-ENOSYS
> + }
> +
> + return 0;
> +}
> +
> +static int npcm_clk_probe(struct udevice *dev)
> +{
> + struct npcm_clk_priv *priv = dev_get_priv(dev);
> + void *base;
> +
> + base = dev_read_addr_ptr(dev);
> + if (!base)
> + return -ENOENT;
> +
> + priv->regs = (struct clk_ctl *)base;
You can directly assign to regs. And there is no need to cast here,
since base is a void pointer.
> +
> + return 0;
> +}
> +
> +static struct clk_ops npcm_clk_ops = {
> + .get_rate = npcm_clk_get_rate,
> + .set_rate = npcm_clk_set_rate,
Please add a .request callback which enforces the max clock id.
--Sean
> +};
> +
> +static const struct udevice_id npcm_clk_ids[] = {
> + { .compatible = "nuvoton,npcm845-clock" },
> + { }
> +};
> +
> +U_BOOT_DRIVER(clk_npcm) = {
> + .name = "clk_npcm",
> + .id = UCLASS_CLK,
> + .of_match = npcm_clk_ids,
> + .ops = &npcm_clk_ops,
> + .priv_auto = sizeof(struct npcm_clk_priv),
> + .probe = npcm_clk_probe,
> +};
> diff --git a/include/dt-bindings/clock/npcm845-clock.h b/include/dt-bindings/clock/npcm845-clock.h
> new file mode 100644
> index 0000000000..fca10d39c8
> --- /dev/null
> +++ b/include/dt-bindings/clock/npcm845-clock.h
> @@ -0,0 +1,17 @@
> +/* SPDX-License-Identifier: GPL-2.0+ */
> +
> +#ifndef _DT_BINDINGS_NPCM845_CLOCK_H_
> +#define _DT_BINDINGS_NPCM845_CLOCK_H_
> +
> +#define CLK_TIMER 0
> +#define CLK_UART 1
> +#define CLK_SD 2
> +#define CLK_EMMC 3
> +#define CLK_APB1 4
> +#define CLK_APB2 5
> +#define CLK_APB3 6
> +#define CLK_APB4 7
> +#define CLK_APB5 8
> +#define CLK_AHB 9
> +
> +#endif
>
More information about the openbmc
mailing list