[PATCH v1 2/9] clk: nuvoton: Add support for NPCM845
Stanley Chu
stanley.chuys at gmail.com
Wed Dec 15 03:57:53 CET 2021
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/
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>
+#include <asm/io.h>
+#include <linux/delay.h>
+#include <dt-bindings/clock/npcm845-clock.h>
+
+struct npcm_clk_priv {
+ struct clk_ctl *regs;
+};
+
+enum regss {
+ 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 */
+ 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:
+ default:
+ return 0;
+ }
+
+ /* PLL Input Clock Divider */
+ nr = (pllval >> PLLCON_INDV) & 0x1f;
+ /* 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));
+
+ return fout * 100;
+}
+
+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;
+
+ writel((readl(®s->clkdiv1) & ~(0x1f << CLKDIV1_MMCCKDIV))
+ | (div - 1) << CLKDIV1_MMCCKDIV, ®s->clkdiv1);
+
+ /* 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)
+ 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;
+
+ /* 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;
+ 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);
+ }
+
+ 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;
+ }
+
+ 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;
+
+ return 0;
+}
+
+static struct clk_ops npcm_clk_ops = {
+ .get_rate = npcm_clk_get_rate,
+ .set_rate = npcm_clk_set_rate,
+};
+
+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
--
2.17.1
More information about the U-Boot
mailing list