Message ID | 20200228210552.615672-8-seanga2@gmail.com |
---|---|
State | Superseded |
Headers | show |
Series | riscv: Add Sipeed Maix support | expand |
Hi Sean > Due to the large number of clocks, I decided to use the CCF. The overall > structure is modeled after the imx code. Clocks are stored in several > arrays. There are some translation macros (FOOIFY()) which allow for more > dense packing. A possible improvement could be to only store the > parameters we need, instead of the whole CCF struct. > > Signed-off-by: Sean Anderson <seanga2 at gmail.com> > --- Please checkpatch and fix total: 4 errors, 4 warnings, 18 checks, 662 lines checked Thanks Rick > > Changes in v5: > - Don't unmap priv->reg > - Remove comment on APB clocks since it has been clarified by Kendryte > - Add i2s mclks > - Reorder clock ids to be continuous > - Rewrite to statically allocate all clocks. This has helped find several bugs > (since it is easy to see when a clock has the wrong register). > - Fix ACLK sometimes having the wrong parent > - Fix SPI3 having the wrong divider > - Prevent being probed multiple times on failure > > Changes in v4: > - Reparent aclk before configuring pll0 > - Update copyright > - Lint > > Changes in v3: > - Removed sysctl struct, replacing it with defines. This is to have the same > interface to sysctl from C as from the device tree. > - Fixed clocks having the same id > - Fixed clocks not using the correct register/bits > - Aligned the defines in headers > > Changes in v2: > - Add clk.o to obj-y > - Don't probe before relocation > > MAINTAINERS | 7 + > .../mfd/kendryte,k210-sysctl.txt | 33 ++ > drivers/clk/kendryte/Kconfig | 2 +- > drivers/clk/kendryte/Makefile | 2 +- > drivers/clk/kendryte/clk.c | 478 ++++++++++++++++++ > include/dt-bindings/clock/k210-sysctl.h | 56 ++ > include/dt-bindings/mfd/k210-sysctl.h | 38 ++ > include/kendryte/clk.h | 35 ++ > 8 files changed, 649 insertions(+), 2 deletions(-) > create mode 100644 doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt > create mode 100644 drivers/clk/kendryte/clk.c > create mode 100644 include/dt-bindings/clock/k210-sysctl.h > create mode 100644 include/dt-bindings/mfd/k210-sysctl.h > create mode 100644 include/kendryte/clk.h > > diff --git a/MAINTAINERS b/MAINTAINERS > index 82e4159bec..8e9e0569ba 100644 > --- a/MAINTAINERS > +++ b/MAINTAINERS > @@ -805,6 +805,13 @@ F: arch/riscv/ > F: cmd/riscv/ > F: tools/prelink-riscv.c > > +RISC-V KENDRYTE > +M: Sean Anderson <seanga2 at gmail.com> > +S: Maintained > +F: doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt > +F: drivers/clk/kendryte/ > +F: include/kendryte/ > + > RNG > M: Sughosh Ganu <sughosh.ganu at linaro.org> > R: Heinrich Schuchardt <xypron.glpk at gmx.de> > diff --git a/doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt b/doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt > new file mode 100644 > index 0000000000..5b24abcb62 > --- /dev/null > +++ b/doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt > @@ -0,0 +1,33 @@ > +Kendryte K210 Sysctl > + > +This binding describes the K210 sysctl device, which contains many miscellaneous > +registers controlling system functionality. This node is a register map and can > +be reference by other bindings which need a phandle to the K210 sysctl regmap. > + > +Required properties: > +- compatible: should be > + "kendryte,k210-sysctl", "syscon", "simple-mfd" > +- reg: address and length of the sysctl registers > +- reg-io-width: must be <4> > + > +Clock sub-node > + > +This node is a binding for the clock tree driver > + > +Required properties: > +- compatible: should be "kendryte,k210-clk" > +- clocks: phandle to the "in0" external oscillator > +- #clock-cells: must be <1> > + > +Example: > +sysctl: syscon at 50440000 { > + compatible = "kendryte,k210-sysctl", "syscon", "simple-mfd"; > + reg = <0x50440000 0x100>; > + reg-io-width = <4>; > + > + sysclk: clock-controller { > + compatible = "kendryte,k210-clk"; > + clocks = <&in0>; > + #clock-cells = <1>; > + }; > +}; > diff --git a/drivers/clk/kendryte/Kconfig b/drivers/clk/kendryte/Kconfig > index 7b69c8afaf..073fca0781 100644 > --- a/drivers/clk/kendryte/Kconfig > +++ b/drivers/clk/kendryte/Kconfig > @@ -1,6 +1,6 @@ > config CLK_K210 > bool "Clock support for Kendryte K210" > - depends on CLK && CLK_CCF > + depends on CLK && CLK_CCF && CLK_COMPOSITE_CCF > help > This enables support clock driver for Kendryte K210 platforms. > > diff --git a/drivers/clk/kendryte/Makefile b/drivers/clk/kendryte/Makefile > index 47f682fce3..6fb68253ae 100644 > --- a/drivers/clk/kendryte/Makefile > +++ b/drivers/clk/kendryte/Makefile > @@ -1 +1 @@ > -obj-y += bypass.o pll.o > +obj-y += bypass.o clk.o pll.o > diff --git a/drivers/clk/kendryte/clk.c b/drivers/clk/kendryte/clk.c > new file mode 100644 > index 0000000000..b01d246682 > --- /dev/null > +++ b/drivers/clk/kendryte/clk.c > @@ -0,0 +1,478 @@ > +// SPDX-License-Identifier: GPL-2.0+ > +/* > + * Copyright (C) 2019-20 Sean Anderson <seanga2 at gmail.com> > + */ > +#include <kendryte/clk.h> > + > +#include <asm/io.h> > +#include <dt-bindings/clock/k210-sysctl.h> > +#include <dt-bindings/mfd/k210-sysctl.h> > +#include <dm.h> > +#include <log.h> > +#include <mapmem.h> > + > +#include <kendryte/bypass.h> > +#include <kendryte/pll.h> > + > +static ulong k210_clk_get_rate(struct clk *clk) > +{ > + struct clk *c; > + int err = clk_get_by_id(clk->id, &c); > + > + if (err) > + return err; > + return clk_get_rate(c); > +} > + > +static ulong k210_clk_set_rate(struct clk *clk, unsigned long rate) > +{ > + struct clk *c; > + int err = clk_get_by_id(clk->id, &c); > + > + if (err) > + return err; > + return clk_set_rate(c, rate); > +} > + > +static int k210_clk_set_parent(struct clk *clk, struct clk *parent) > +{ > + struct clk *c, *p; > + int err = clk_get_by_id(clk->id, &c); > + > + if (err) > + return err; > + > + err = clk_get_by_id(parent->id, &p); > + if (err) > + return err; > + > + return clk_set_parent(c, p); > +} > + > +static int k210_clk_endisable(struct clk *clk, bool enable) > +{ > + struct clk *c; > + int err = clk_get_by_id(clk->id, &c); > + > + if (err) > + return err; > + return enable ? clk_enable(c) : clk_disable(c); > +} > + > +static int k210_clk_enable(struct clk *clk) > +{ > + return k210_clk_endisable(clk, true); > +} > + > +static int k210_clk_disable(struct clk *clk) > +{ > + return k210_clk_endisable(clk, false); > +} > + > +static const struct clk_ops k210_clk_ops = { > + .set_rate = k210_clk_set_rate, > + .get_rate = k210_clk_get_rate, > + .set_parent = k210_clk_set_parent, > + .enable = k210_clk_enable, > + .disable = k210_clk_disable, > +}; > + > +static const char * const generic_sels[] = { "in0_half", "pll0_half" }; > +/* The first clock is in0, which is filled in by k210_clk_probe */ > +static const char *aclk_sels[] = { NULL, "pll0_half" }; > +static const char *pll2_sels[] = { NULL, "pll0", "pll1" }; > + > +#define DIV(id, reg, shift, width) DIV_FLAGS(id, reg, shift, width, 0) > +#define DIV_LIST \ > + DIV_FLAGS(K210_CLK_ACLK, K210_SYSCTL_SEL0, 1, 2, \ > + CLK_DIVIDER_POWER_OF_TWO) \ > + DIV(K210_CLK_APB0, K210_SYSCTL_SEL0, 3, 3) \ > + DIV(K210_CLK_APB1, K210_SYSCTL_SEL0, 6, 3) \ > + DIV(K210_CLK_APB2, K210_SYSCTL_SEL0, 9, 3) \ > + DIV(K210_CLK_SRAM0, K210_SYSCTL_THR0, 0, 4) \ > + DIV(K210_CLK_SRAM1, K210_SYSCTL_THR0, 4, 4) \ > + DIV(K210_CLK_AI, K210_SYSCTL_THR0, 8, 4) \ > + DIV(K210_CLK_DVP, K210_SYSCTL_THR0, 12, 4) \ > + DIV(K210_CLK_ROM, K210_SYSCTL_THR0, 16, 4) \ > + DIV(K210_CLK_SPI0, K210_SYSCTL_THR1, 0, 8) \ > + DIV(K210_CLK_SPI1, K210_SYSCTL_THR1, 8, 8) \ > + DIV(K210_CLK_SPI2, K210_SYSCTL_THR1, 16, 8) \ > + DIV(K210_CLK_SPI3, K210_SYSCTL_THR1, 24, 8) \ > + DIV(K210_CLK_TIMER0, K210_SYSCTL_THR2, 0, 8) \ > + DIV(K210_CLK_TIMER1, K210_SYSCTL_THR2, 8, 8) \ > + DIV(K210_CLK_TIMER2, K210_SYSCTL_THR2, 16, 8) \ > + DIV(K210_CLK_I2S0, K210_SYSCTL_THR3, 0, 16) \ > + DIV(K210_CLK_I2S1, K210_SYSCTL_THR3, 16, 16) \ > + DIV(K210_CLK_I2S2, K210_SYSCTL_THR4, 0, 16) \ > + DIV(K210_CLK_I2S0_M, K210_SYSCTL_THR4, 16, 8) \ > + DIV(K210_CLK_I2S1_M, K210_SYSCTL_THR4, 24, 8) \ > + DIV(K210_CLK_I2S2_M, K210_SYSCTL_THR4, 0, 8) \ > + DIV(K210_CLK_I2C0, K210_SYSCTL_THR5, 8, 8) \ > + DIV(K210_CLK_I2C1, K210_SYSCTL_THR5, 16, 8) \ > + DIV(K210_CLK_I2C2, K210_SYSCTL_THR5, 24, 8) \ > + DIV(K210_CLK_WDT0, K210_SYSCTL_THR6, 0, 8) \ > + DIV(K210_CLK_WDT1, K210_SYSCTL_THR6, 8, 8) > + > +#define _DIVIFY(id) K210_CLK_DIV_##id > +#define DIVIFY(id) _DIVIFY(id) > +#define DIV_FLAGS(id, ...) DIVIFY(id), > +enum k210_clk_div_ids { > + DIV_LIST > +}; > +#undef DIV_FLAGS > + > +#define DIV_FLAGS(id, _reg, _shift, _width, _flags) [DIVIFY(id)] = { \ > + .reg = (void *)(_reg), \ > + .shift = (_shift), \ > + .width = (_width), \ > + .flags = (_flags), \ > +}, > +static struct clk_divider k210_clk_dividers[] = { > + DIV_LIST > +}; > +#undef DIV_FLAGS > +#undef DIV > +#undef DIV_LIST > + > +#define GATE_LIST \ > + GATE(K210_CLK_CPU, K210_SYSCTL_EN_CENT, 0) \ > + GATE(K210_CLK_SRAM0, K210_SYSCTL_EN_CENT, 1) \ > + GATE(K210_CLK_SRAM1, K210_SYSCTL_EN_CENT, 2) \ > + GATE(K210_CLK_APB0, K210_SYSCTL_EN_CENT, 3) \ > + GATE(K210_CLK_APB1, K210_SYSCTL_EN_CENT, 4) \ > + GATE(K210_CLK_APB2, K210_SYSCTL_EN_CENT, 5) \ > + GATE(K210_CLK_ROM, K210_SYSCTL_EN_PERI, 0) \ > + GATE(K210_CLK_DMA, K210_SYSCTL_EN_PERI, 1) \ > + GATE(K210_CLK_AI, K210_SYSCTL_EN_PERI, 2) \ > + GATE(K210_CLK_DVP, K210_SYSCTL_EN_PERI, 3) \ > + GATE(K210_CLK_FFT, K210_SYSCTL_EN_PERI, 4) \ > + GATE(K210_CLK_GPIO, K210_SYSCTL_EN_PERI, 5) \ > + GATE(K210_CLK_SPI0, K210_SYSCTL_EN_PERI, 6) \ > + GATE(K210_CLK_SPI1, K210_SYSCTL_EN_PERI, 7) \ > + GATE(K210_CLK_SPI2, K210_SYSCTL_EN_PERI, 8) \ > + GATE(K210_CLK_SPI3, K210_SYSCTL_EN_PERI, 9) \ > + GATE(K210_CLK_I2S0, K210_SYSCTL_EN_PERI, 10) \ > + GATE(K210_CLK_I2S1, K210_SYSCTL_EN_PERI, 11) \ > + GATE(K210_CLK_I2S2, K210_SYSCTL_EN_PERI, 12) \ > + GATE(K210_CLK_I2C0, K210_SYSCTL_EN_PERI, 13) \ > + GATE(K210_CLK_I2C1, K210_SYSCTL_EN_PERI, 14) \ > + GATE(K210_CLK_I2C2, K210_SYSCTL_EN_PERI, 15) \ > + GATE(K210_CLK_UART1, K210_SYSCTL_EN_PERI, 16) \ > + GATE(K210_CLK_UART2, K210_SYSCTL_EN_PERI, 17) \ > + GATE(K210_CLK_UART3, K210_SYSCTL_EN_PERI, 18) \ > + GATE(K210_CLK_AES, K210_SYSCTL_EN_PERI, 19) \ > + GATE(K210_CLK_FPIOA, K210_SYSCTL_EN_PERI, 20) \ > + GATE(K210_CLK_TIMER0, K210_SYSCTL_EN_PERI, 21) \ > + GATE(K210_CLK_TIMER1, K210_SYSCTL_EN_PERI, 22) \ > + GATE(K210_CLK_TIMER2, K210_SYSCTL_EN_PERI, 23) \ > + GATE(K210_CLK_WDT0, K210_SYSCTL_EN_PERI, 24) \ > + GATE(K210_CLK_WDT1, K210_SYSCTL_EN_PERI, 25) \ > + GATE(K210_CLK_SHA, K210_SYSCTL_EN_PERI, 26) \ > + GATE(K210_CLK_OTP, K210_SYSCTL_EN_PERI, 27) \ > + GATE(K210_CLK_RTC, K210_SYSCTL_EN_PERI, 29) > + > +#define _GATEIFY(id) K210_CLK_GATE_##id > +#define GATEIFY(id) _GATEIFY(id) > +#define GATE(id, ...) GATEIFY(id), > +enum k210_clk_gate_ids { > + GATE_LIST > +}; > +#undef GATE > + > +#define GATE(id, _reg, _idx) [GATEIFY(id)] = { \ > + .reg = (void *)(_reg), \ > + .bit_idx = (_idx), \ > +}, > +static struct clk_gate k210_clk_gates[] = { > + GATE_LIST > +}; > +#undef GATE > +#undef GATE_LIST > + > +#define MUX(id, reg, shift, width) \ > + MUX_PARENTS(id, generic_sels, reg, shift, width) > +#define MUX_LIST \ > + MUX_PARENTS(K210_CLK_PLL2, pll2_sels, K210_SYSCTL_PLL2, 26, 2) \ > + MUX_PARENTS(K210_CLK_ACLK, aclk_sels, K210_SYSCTL_SEL0, 0, 1) \ > + MUX(K210_CLK_SPI3, K210_SYSCTL_SEL0, 12, 1) \ > + MUX(K210_CLK_TIMER0, K210_SYSCTL_SEL0, 13, 1) \ > + MUX(K210_CLK_TIMER1, K210_SYSCTL_SEL0, 14, 1) \ > + MUX(K210_CLK_TIMER2, K210_SYSCTL_SEL0, 15, 1) > + > +#define _MUXIFY(id) K210_CLK_MUX_##id > +#define MUXIFY(id) _MUXIFY(id) > +#define MUX_PARENTS(id, ...) MUXIFY(id), > +enum k210_clk_mux_ids { > + MUX_LIST > +}; > +#undef MUX_PARENTS > + > +#define MUX_PARENTS(id, parents, _reg, _shift, _width) [MUXIFY(id)] = { \ > + .parent_names = (const char * const *)(parents), \ > + .num_parents = ARRAY_SIZE(parents), \ > + .reg = (void *)(_reg), \ > + .shift = (_shift), \ > + .mask = BIT(_width) - 1, \ > +}, > +static struct clk_mux k210_clk_muxes[] = { > + MUX_LIST > +}; > +#undef MUX_PARENTS > +#undef MUX > +#undef MUX_LIST > + > +#define PLL(_reg, _shift, _width) { \ > + .reg = (void *)(_reg), \ > + .lock = (void *)K210_SYSCTL_PLL_LOCK, \ > + .shift = (_shift), \ > + .width = (_width), \ > +} > +static struct k210_pll k210_clk_plls[] = { > + [0] = PLL(K210_SYSCTL_PLL0, 0, 2), > + [1] = PLL(K210_SYSCTL_PLL1, 8, 1), > + [2] = PLL(K210_SYSCTL_PLL2, 16, 1), > +}; > +#undef PLL > + > +#define COMP(id, mux, div, gate) \ > + COMP_FULL(id, &(mux)->clk, &clk_mux_ops, \ > + &(div)->clk, &clk_divider_ops, \ > + &(gate)->clk, &clk_gate_ops) > +#define COMP_ID(id) \ > + COMP(id, &k210_clk_muxes[MUXIFY(id)], \ > + &k210_clk_dividers[DIVIFY(id)], \ > + &k210_clk_gates[GATEIFY(id)]) > +#define COMP_NOMUX(id, div, gate) \ > + COMP_FULL(id, NULL, NULL, \ > + &(div)->clk, &clk_divider_ops, \ > + &(gate)->clk, &clk_gate_ops) > +#define COMP_NOMUX_ID(id) \ > + COMP_NOMUX(id, &k210_clk_dividers[DIVIFY(id)], \ > + &k210_clk_gates[GATEIFY(id)]) > +#define COMP_LIST \ > + COMP_FULL(K210_CLK_PLL2, \ > + &k210_clk_muxes[MUXIFY(K210_CLK_PLL2)].clk, &clk_mux_ops, \ > + &k210_clk_plls[2].clk, &k210_pll_ops, \ > + &k210_clk_plls[2].clk, &k210_pll_ops) \ > + COMP_FULL(K210_CLK_ACLK, \ > + &k210_clk_muxes[MUXIFY(K210_CLK_ACLK)].clk, &clk_mux_ops, \ > + &k210_clk_dividers[DIVIFY(K210_CLK_ACLK)].clk, \ > + &clk_divider_ops, \ > + NULL, NULL) \ > + COMP_ID(K210_CLK_SPI3) \ > + COMP_ID(K210_CLK_TIMER0) \ > + COMP_ID(K210_CLK_TIMER1) \ > + COMP_ID(K210_CLK_TIMER2) \ > + COMP_NOMUX_ID(K210_CLK_SRAM0) \ > + COMP_NOMUX_ID(K210_CLK_SRAM1) \ > + COMP_NOMUX_ID(K210_CLK_ROM) \ > + COMP_NOMUX_ID(K210_CLK_DVP) \ > + COMP_NOMUX_ID(K210_CLK_APB0) \ > + COMP_NOMUX_ID(K210_CLK_APB1) \ > + COMP_NOMUX_ID(K210_CLK_APB2) \ > + COMP_NOMUX_ID(K210_CLK_AI) \ > + COMP_NOMUX_ID(K210_CLK_I2S0) \ > + COMP_NOMUX_ID(K210_CLK_I2S1) \ > + COMP_NOMUX_ID(K210_CLK_I2S2) \ > + COMP_NOMUX_ID(K210_CLK_WDT0) \ > + COMP_NOMUX_ID(K210_CLK_WDT1) \ > + COMP_NOMUX_ID(K210_CLK_SPI0) \ > + COMP_NOMUX_ID(K210_CLK_SPI1) \ > + COMP_NOMUX_ID(K210_CLK_SPI2) \ > + COMP_NOMUX_ID(K210_CLK_I2C0) \ > + COMP_NOMUX_ID(K210_CLK_I2C1) \ > + COMP_NOMUX_ID(K210_CLK_I2C2) > + > +#define _COMPIFY(id) K210_CLK_COMP_##id > +#define COMPIFY(id) _COMPIFY(id) > +#define COMP_FULL(id, ...) COMPIFY(id), > +enum k210_clk_comp_ids { > + COMP_LIST > +}; > +#undef COMP_FULL > + > +#define COMP_FULL(id, _mux, _mux_ops, _div, _div_ops, _gate, _gate_ops) \ > +[COMPIFY(id)] = { \ > + .mux = (_mux), \ > + .mux_ops = (_mux_ops), \ > + .rate = (_div), \ > + .rate_ops = (_div_ops), \ > + .gate = (_gate), \ > + .gate_ops = (_gate_ops), \ > +}, > +static struct clk_composite k210_clk_comps[] = { > + COMP_LIST > +}; > +#undef COMP_FULL > +#undef COMP > +#undef COMP_ID > +#undef COMP_NOMUX > +#undef COMP_NOMUX_ID > +#undef COMP_LIST > + > +static struct clk *k210_clk_bypass_children = { > + &k210_clk_comps[COMPIFY(K210_CLK_ACLK)].clk, > +}; > + > +static struct clk *k210_clk_bypass_saved_parents = { > + NULL, > +}; > + > +static struct k210_bypass k210_clk_bypass = { > + .bypassee = &k210_clk_plls[0].clk, > + .bypassee_ops = &k210_pll_ops, > + .children = &k210_clk_bypass_children, > + .child_count = 1, > + .saved_parents = &k210_clk_bypass_saved_parents, > +}; > + > +static bool probed = false; > + > +static int k210_clk_probe(struct udevice *dev) > +{ > + int ret, i; > + const char *in0; > + struct clk *in0_clk; > + void *base; > + > + /* Only one instance of this driver allowed */ > + if (READ_ONCE(probed)) > + return -ENOTSUPP; > + > + base = dev_read_addr_ptr(dev_get_parent(dev)); > + if (!base) > + return -EINVAL; > + > + in0_clk = kzalloc(sizeof(*in0_clk), GFP_KERNEL); > + if (!in0_clk) > + return -ENOMEM; > + > + ret = clk_get_by_index(dev, 0, in0_clk); > + if (ret) > + return ret; > + in0 = in0_clk->dev->name; > + > + WRITE_ONCE(probed, true); > + > + aclk_sels[0] = in0; > + pll2_sels[0] = in0; > + > + /* Fixup registers to be absolute, rather than relative */ > +#define FIXUP_REGS(clocks) \ > + for (i = 0; i < ARRAY_SIZE(clocks); i++) \ > + clocks[i].reg += (ulong)base > + FIXUP_REGS(k210_clk_dividers); > + FIXUP_REGS(k210_clk_gates); > + FIXUP_REGS(k210_clk_muxes); > +#undef FIXUP_REGS > + for (i = 0; i < ARRAY_SIZE(k210_clk_plls); i++) { > + k210_clk_plls[i].reg += (ulong)base; > + k210_clk_plls[i].lock += (ulong)base; > + } > + > + /* > + * All PLLs have a broken bypass, but pll0 has the CPU downstream, so we > + * need to manually reparent it whenever we configure pll0 > + */ > + k210_clk_bypass.alt = in0_clk; > + clk_dm(K210_CLK_PLL0, > + k210_register_bypass_struct("pll0", in0, &k210_clk_bypass)); > + clk_dm(K210_CLK_PLL1, > + k210_register_pll_struct("pll1", in0, &k210_clk_plls[1])); > + /* PLL2 is muxed, so set up a composite clock */ > + clk_dm(K210_CLK_PLL2, > + clk_register_composite_struct("pll2", pll2_sels, > + ARRAY_SIZE(pll2_sels), > + &k210_clk_comps[COMPIFY(K210_CLK_PLL2)])); > + > + /* Half-frequency clocks for "even" dividers */ > + k210_clk_half("in0_half", in0); > + k210_clk_half("pll0_half", "pll0"); > + k210_clk_half("pll2_half", "pll2"); > + > + /* ACLK has no gate */ > + clk_dm(K210_CLK_ACLK, > + clk_register_composite_struct("aclk", aclk_sels, > + ARRAY_SIZE(aclk_sels), > + &k210_clk_comps[COMPIFY(K210_CLK_ACLK)])); > + > +#define REGISTER_COMP(id, name) \ > + clk_dm(id, clk_register_composite_struct(name, generic_sels, \ > + ARRAY_SIZE(generic_sels), \ > + &k210_clk_comps[COMPIFY(id)])) > + REGISTER_COMP(K210_CLK_SPI3, "spi3"); > + REGISTER_COMP(K210_CLK_TIMER0, "timer0"); > + REGISTER_COMP(K210_CLK_TIMER1, "timer1"); > + REGISTER_COMP(K210_CLK_TIMER2, "timer2"); > +#undef COMP > + > + /* Dividing clocks, no mux */ > +#define REGISTER_COMP_NOMUX(id, name, _parent) do { \ > + const char *parent = _parent; \ > + clk_dm(id, \ > + clk_register_composite_struct(name, &parent, 1, \ > + &k210_clk_comps[COMPIFY(id)])); \ > +} while (false) > + REGISTER_COMP_NOMUX(K210_CLK_SRAM0, "sram0", "aclk"); > + REGISTER_COMP_NOMUX(K210_CLK_SRAM1, "sram1", "aclk"); > + REGISTER_COMP_NOMUX(K210_CLK_ROM, "rom", "aclk"); > + REGISTER_COMP_NOMUX(K210_CLK_DVP, "dvp", "aclk"); > + REGISTER_COMP_NOMUX(K210_CLK_APB0, "apb0", "aclk"); > + REGISTER_COMP_NOMUX(K210_CLK_APB1, "apb1", "aclk"); > + REGISTER_COMP_NOMUX(K210_CLK_APB2, "apb2", "aclk"); > + REGISTER_COMP_NOMUX(K210_CLK_AI, "ai", "pll1"); > + REGISTER_COMP_NOMUX(K210_CLK_I2S0, "i2s0", "pll2_half"); > + REGISTER_COMP_NOMUX(K210_CLK_I2S1, "i2s1", "pll2_half"); > + REGISTER_COMP_NOMUX(K210_CLK_I2S2, "i2s2", "pll2_half"); > + REGISTER_COMP_NOMUX(K210_CLK_WDT0, "wdt0", "in0_half"); > + REGISTER_COMP_NOMUX(K210_CLK_WDT1, "wdt1", "in0_half"); > + REGISTER_COMP_NOMUX(K210_CLK_SPI0, "spi0", "pll0_half"); > + REGISTER_COMP_NOMUX(K210_CLK_SPI1, "spi1", "pll0_half"); > + REGISTER_COMP_NOMUX(K210_CLK_SPI2, "spi2", "pll0_half"); > + REGISTER_COMP_NOMUX(K210_CLK_I2C0, "i2c0", "pll0_half"); > + REGISTER_COMP_NOMUX(K210_CLK_I2C1, "i2c1", "pll0_half"); > + REGISTER_COMP_NOMUX(K210_CLK_I2C2, "i2c2", "pll0_half"); > +#undef REGISTER_COMP_NOMUX > + > + /* Dividing clocks */ > +#define REGISTER_DIV(id, name, parent) clk_dm(id, \ > + clk_register_divider_struct(name, parent, \ > + &k210_clk_dividers[DIVIFY(id)])) > + REGISTER_DIV(K210_CLK_I2S0_M, "i2s0_m", "pll2_half"); > + REGISTER_DIV(K210_CLK_I2S1_M, "i2s1_m", "pll2_half"); > + REGISTER_DIV(K210_CLK_I2S2_M, "i2s2_m", "pll2_half"); > +#undef REGISTER_DIV > + > + /* Gated clocks */ > +#define REGISTER_GATE(id, name, parent) \ > + clk_dm(id, clk_register_gate_struct(name, parent, \ > + &k210_clk_gates[GATEIFY(id)])) > + REGISTER_GATE(K210_CLK_CPU, "cpu", "aclk"); > + REGISTER_GATE(K210_CLK_DMA, "dma", "aclk"); > + REGISTER_GATE(K210_CLK_FFT, "fft", "aclk"); > + REGISTER_GATE(K210_CLK_GPIO, "gpio", "apb0"); > + REGISTER_GATE(K210_CLK_UART1, "uart1", "apb0"); > + REGISTER_GATE(K210_CLK_UART2, "uart2", "apb0"); > + REGISTER_GATE(K210_CLK_UART3, "uart3", "apb0"); > + REGISTER_GATE(K210_CLK_FPIOA, "fpioa", "apb0"); > + REGISTER_GATE(K210_CLK_SHA, "sha", "apb0"); > + REGISTER_GATE(K210_CLK_AES, "aes", "apb1"); > + REGISTER_GATE(K210_CLK_OTP, "otp", "apb1"); > + REGISTER_GATE(K210_CLK_RTC, "rtc", in0); > +#undef REGISTER_GATE > + > + return 0; > +} > + > +static const struct udevice_id k210_clk_ids[] = { > + { .compatible = "kendryte,k210-clk" }, > + { }, > +}; > + > +U_BOOT_DRIVER(k210_clk) = { > + .name = "k210_clk", > + .id = UCLASS_CLK, > + .of_match = k210_clk_ids, > + .ops = &k210_clk_ops, > + .probe = k210_clk_probe, > +}; > diff --git a/include/dt-bindings/clock/k210-sysctl.h b/include/dt-bindings/clock/k210-sysctl.h > new file mode 100644 > index 0000000000..16d67b282f > --- /dev/null > +++ b/include/dt-bindings/clock/k210-sysctl.h > @@ -0,0 +1,56 @@ > +/* SPDX-License-Identifier: GPL-2.0+ */ > +/* > + * Copyright (C) 2019-20 Sean Anderson <seanga2 at gmail.com> > + */ > + > +#ifndef CLOCK_K210_SYSCTL_H > +#define CLOCK_K210_SYSCTL_H > + > +/* > + * Arbitrary identifiers for clocks. > + */ > +#define K210_CLK_NONE 0 > +#define K210_CLK_PLL0 1 > +#define K210_CLK_PLL1 2 > +#define K210_CLK_PLL2 3 > +#define K210_CLK_CPU 4 > +#define K210_CLK_SRAM0 5 > +#define K210_CLK_SRAM1 6 > +#define K210_CLK_APB0 7 > +#define K210_CLK_APB1 8 > +#define K210_CLK_APB2 9 > +#define K210_CLK_ROM 10 > +#define K210_CLK_DMA 11 > +#define K210_CLK_AI 12 > +#define K210_CLK_DVP 13 > +#define K210_CLK_FFT 14 > +#define K210_CLK_GPIO 15 > +#define K210_CLK_SPI0 16 > +#define K210_CLK_SPI1 17 > +#define K210_CLK_SPI2 18 > +#define K210_CLK_SPI3 19 > +#define K210_CLK_I2S0 20 > +#define K210_CLK_I2S1 21 > +#define K210_CLK_I2S2 22 > +#define K210_CLK_I2S0_M 23 > +#define K210_CLK_I2S1_M 24 > +#define K210_CLK_I2S2_M 25 > +#define K210_CLK_I2C0 26 > +#define K210_CLK_I2C1 27 > +#define K210_CLK_I2C2 28 > +#define K210_CLK_UART1 29 > +#define K210_CLK_UART2 30 > +#define K210_CLK_UART3 31 > +#define K210_CLK_AES 32 > +#define K210_CLK_FPIOA 33 > +#define K210_CLK_TIMER0 34 > +#define K210_CLK_TIMER1 35 > +#define K210_CLK_TIMER2 36 > +#define K210_CLK_WDT0 37 > +#define K210_CLK_WDT1 38 > +#define K210_CLK_SHA 39 > +#define K210_CLK_OTP 40 > +#define K210_CLK_RTC 41 > +#define K210_CLK_ACLK 42 > + > +#endif /* CLOCK_K210_SYSCTL_H */ > diff --git a/include/dt-bindings/mfd/k210-sysctl.h b/include/dt-bindings/mfd/k210-sysctl.h > new file mode 100644 > index 0000000000..e16d7302cd > --- /dev/null > +++ b/include/dt-bindings/mfd/k210-sysctl.h > @@ -0,0 +1,38 @@ > +/* SPDX-License-Identifier: GPL-2.0+ */ > +/* > + * Copyright (C) 2020 Sean Anderson <seanga2 at gmail.com> > + */ > + > +#ifndef K210_SYSCTL_H > +#define K210_SYSCTL_H > + > +/* Taken from kendryte-standalone-sdk/lib/drivers/include/sysctl.h */ > +#define K210_SYSCTL_GIT_ID 0x00 /* Git short commit id */ > +#define K210_SYSCTL_CLK_FREQ 0x04 /* System clock base frequency */ > +#define K210_SYSCTL_PLL0 0x08 /* PLL0 controller */ > +#define K210_SYSCTL_PLL1 0x0C /* PLL1 controller */ > +#define K210_SYSCTL_PLL2 0x10 /* PLL2 controller */ > +#define K210_SYSCTL_PLL_LOCK 0x18 /* PLL lock tester */ > +#define K210_SYSCTL_ROM_ERROR 0x1C /* AXI ROM detector */ > +#define K210_SYSCTL_SEL0 0x20 /* Clock select controller0 */ > +#define K210_SYSCTL_SEL1 0x24 /* Clock select controller1 */ > +#define K210_SYSCTL_EN_CENT 0x28 /* Central clock enable */ > +#define K210_SYSCTL_EN_PERI 0x2C /* Peripheral clock enable */ > +#define K210_SYSCTL_SOFT_RESET 0x30 /* Soft reset ctrl */ > +#define K210_SYSCTL_PERI_RESET 0x34 /* Peripheral reset controller */ > +#define K210_SYSCTL_THR0 0x38 /* Clock threshold controller 0 */ > +#define K210_SYSCTL_THR1 0x3C /* Clock threshold controller 1 */ > +#define K210_SYSCTL_THR2 0x40 /* Clock threshold controller 2 */ > +#define K210_SYSCTL_THR3 0x44 /* Clock threshold controller 3 */ > +#define K210_SYSCTL_THR4 0x48 /* Clock threshold controller 4 */ > +#define K210_SYSCTL_THR5 0x4C /* Clock threshold controller 5 */ > +#define K210_SYSCTL_THR6 0x50 /* Clock threshold controller 6 */ > +#define K210_SYSCTL_MISC 0x54 /* Miscellaneous controller */ > +#define K210_SYSCTL_PERI 0x58 /* Peripheral controller */ > +#define K210_SYSCTL_SPI_SLEEP 0x5C /* SPI sleep controller */ > +#define K210_SYSCTL_RESET_STAT 0x60 /* Reset source status */ > +#define K210_SYSCTL_DMA_SEL0 0x64 /* DMA handshake selector */ > +#define K210_SYSCTL_DMA_SEL1 0x68 /* DMA handshake selector */ > +#define K210_SYSCTL_POWER_SEL 0x6C /* IO Power Mode Select controller */ > + > +#endif /* K210_SYSCTL_H */ > diff --git a/include/kendryte/clk.h b/include/kendryte/clk.h > new file mode 100644 > index 0000000000..9c6245d468 > --- /dev/null > +++ b/include/kendryte/clk.h > @@ -0,0 +1,35 @@ > +/* SPDX-License-Identifier: GPL-2.0+ */ > +/* > + * Copyright (C) 2019-20 Sean Anderson <seanga2 at gmail.com> > + */ > + > +#ifndef K210_CLK_H > +#define K210_CLK_H > + > +#define LOG_CATEGORY UCLASS_CLK > +#include <linux/types.h> > +#include <linux/clk-provider.h> > + > +static inline struct clk *k210_clk_gate(const char *name, > + const char *parent_name, > + void __iomem *reg, u8 bit_idx) > +{ > + return clk_register_gate(NULL, name, parent_name, 0, reg, bit_idx, 0, > + NULL); > +} > + > +static inline struct clk *k210_clk_half(const char *name, > + const char *parent_name) > +{ > + return clk_register_fixed_factor(NULL, name, parent_name, 0, 1, 2); > +} > + > +static inline struct clk *k210_clk_div(const char *name, > + const char *parent_name, > + void __iomem *reg, u8 shift, u8 width) > +{ > + return clk_register_divider(NULL, name, parent_name, 0, reg, shift, > + width, 0); > +} > + > +#endif /* K210_CLK_H */ > -- > 2.25.0 >
On 3/4/20 1:58 AM, Rick Chen wrote: > Hi Sean > >> Due to the large number of clocks, I decided to use the CCF. The overall >> structure is modeled after the imx code. Clocks are stored in several >> arrays. There are some translation macros (FOOIFY()) which allow for more >> dense packing. A possible improvement could be to only store the >> parameters we need, instead of the whole CCF struct. >> >> Signed-off-by: Sean Anderson <seanga2 at gmail.com> >> --- > > Please checkpatch and fix > total: 4 errors, 4 warnings, 18 checks, 662 lines checked > > Thanks > Rick > Here is the output of checkpatch > drivers/clk/kendryte/clk.c:82: warning: static const char * array should probably be static const char * const > drivers/clk/kendryte/clk.c:83: warning: static const char * array should probably be static const char * const These arrays can't have both consts because it needs to have the actual name of the in0 clock added. > drivers/clk/kendryte/clk.c:122: check: Please use a blank line after function/struct/union/enum declarations This is due to using macros in the style #define FOO_LIST \ FOO(bar) \ FOO(baz) #define FOO(x) FOO_##x, enum foo_ids { FOO_LIST }; #undef FOO I think sticking the undefinition of FOO immediately after the closing enum bracket makes it clearer that FOO is only used with that definition during the declaration of that enum. It is closing the scope, so to speak. If you'd like I can add a newline after enums declared this way. > drivers/clk/kendryte/clk.c:124: error: space prohibited before open square bracket '[' This is due to macros declared like #define FOO(x) [FOO_##x] = { \ .y = (x), \ } Where there is clearly a space before the [, but it is necessary for the macro. I could declare it like #define FOO(X) \ [FOO_##x] = { \ .y = (x), \ } but I think that the former style is more elegant. However, this can also be changed if needed. > drivers/clk/kendryte/clk.c:133: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:180: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:182: error: space prohibited before open square bracket '[' > drivers/clk/kendryte/clk.c:189: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:208: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:210: error: space prohibited before open square bracket '[' > drivers/clk/kendryte/clk.c:210: check: Macro argument reuse 'parents' - possible side-effects? No possible side-effects here, since this macro argument doesn't make sense unless it is a compile-time constant. > drivers/clk/kendryte/clk.c:220: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:230: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:235: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:241: check: Macro argument reuse 'id' - possible side-effects? > drivers/clk/kendryte/clk.c:249: check: Macro argument reuse 'id' - possible side-effects? > drivers/clk/kendryte/clk.c:292: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:306: check: Please use a blank line after function/struct/union/enum declarations > drivers/clk/kendryte/clk.c:329: error: do not initialise statics to false > drivers/clk/kendryte/clk.c:361: check: Macro argument reuse 'clocks' - possible side-effects? > drivers/clk/kendryte/clk.c:386: warning: line over 80 characters > drivers/clk/kendryte/clk.c:397: warning: line over 80 characters Unfortunately, I don't see any way to keep these two lines under 80 characters without seriously sacrificing readability. For reference, the lines look like clk_dm(K210_CLK_PLL2, clk_register_composite_struct("pll2", pll2_sels, ARRAY_SIZE(pll2_sels), &k210_clk_comps[COMPIFY(K210_CLK_PLL2)])); The only way to further reduce the length would be to split the array access over two lines, which I think harms readability too much. > drivers/clk/kendryte/clk.c:399: check: Macro argument reuse 'id' - possible side-effects? > drivers/clk/kendryte/clk.c:410: check: Macro argument reuse 'id' - possible side-effects? > drivers/clk/kendryte/clk.c:438: check: Macro argument reuse 'id' - possible side-effects? > drivers/clk/kendryte/clk.c:447: check: Macro argument reuse 'id' - possible side-effects? > <unknown>:0: warning: DT binding docs and includes should be a separate patch. See: Documentation/devicetree/bindings/submitting-patches.txt > <unknown>:0: warning: DT binding docs and includes should be a separate patch. See: Documentation/devicetree/bindings/submitting-patches.txt AFAIK U-Boot has no such policy. --Sean
diff --git a/MAINTAINERS b/MAINTAINERS index 82e4159bec..8e9e0569ba 100644 --- a/MAINTAINERS +++ b/MAINTAINERS @@ -805,6 +805,13 @@ F: arch/riscv/ F: cmd/riscv/ F: tools/prelink-riscv.c +RISC-V KENDRYTE +M: Sean Anderson <seanga2 at gmail.com> +S: Maintained +F: doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt +F: drivers/clk/kendryte/ +F: include/kendryte/ + RNG M: Sughosh Ganu <sughosh.ganu at linaro.org> R: Heinrich Schuchardt <xypron.glpk at gmx.de> diff --git a/doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt b/doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt new file mode 100644 index 0000000000..5b24abcb62 --- /dev/null +++ b/doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt @@ -0,0 +1,33 @@ +Kendryte K210 Sysctl + +This binding describes the K210 sysctl device, which contains many miscellaneous +registers controlling system functionality. This node is a register map and can +be reference by other bindings which need a phandle to the K210 sysctl regmap. + +Required properties: +- compatible: should be + "kendryte,k210-sysctl", "syscon", "simple-mfd" +- reg: address and length of the sysctl registers +- reg-io-width: must be <4> + +Clock sub-node + +This node is a binding for the clock tree driver + +Required properties: +- compatible: should be "kendryte,k210-clk" +- clocks: phandle to the "in0" external oscillator +- #clock-cells: must be <1> + +Example: +sysctl: syscon at 50440000 { + compatible = "kendryte,k210-sysctl", "syscon", "simple-mfd"; + reg = <0x50440000 0x100>; + reg-io-width = <4>; + + sysclk: clock-controller { + compatible = "kendryte,k210-clk"; + clocks = <&in0>; + #clock-cells = <1>; + }; +}; diff --git a/drivers/clk/kendryte/Kconfig b/drivers/clk/kendryte/Kconfig index 7b69c8afaf..073fca0781 100644 --- a/drivers/clk/kendryte/Kconfig +++ b/drivers/clk/kendryte/Kconfig @@ -1,6 +1,6 @@ config CLK_K210 bool "Clock support for Kendryte K210" - depends on CLK && CLK_CCF + depends on CLK && CLK_CCF && CLK_COMPOSITE_CCF help This enables support clock driver for Kendryte K210 platforms. diff --git a/drivers/clk/kendryte/Makefile b/drivers/clk/kendryte/Makefile index 47f682fce3..6fb68253ae 100644 --- a/drivers/clk/kendryte/Makefile +++ b/drivers/clk/kendryte/Makefile @@ -1 +1 @@ -obj-y += bypass.o pll.o +obj-y += bypass.o clk.o pll.o diff --git a/drivers/clk/kendryte/clk.c b/drivers/clk/kendryte/clk.c new file mode 100644 index 0000000000..b01d246682 --- /dev/null +++ b/drivers/clk/kendryte/clk.c @@ -0,0 +1,478 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * Copyright (C) 2019-20 Sean Anderson <seanga2 at gmail.com> + */ +#include <kendryte/clk.h> + +#include <asm/io.h> +#include <dt-bindings/clock/k210-sysctl.h> +#include <dt-bindings/mfd/k210-sysctl.h> +#include <dm.h> +#include <log.h> +#include <mapmem.h> + +#include <kendryte/bypass.h> +#include <kendryte/pll.h> + +static ulong k210_clk_get_rate(struct clk *clk) +{ + struct clk *c; + int err = clk_get_by_id(clk->id, &c); + + if (err) + return err; + return clk_get_rate(c); +} + +static ulong k210_clk_set_rate(struct clk *clk, unsigned long rate) +{ + struct clk *c; + int err = clk_get_by_id(clk->id, &c); + + if (err) + return err; + return clk_set_rate(c, rate); +} + +static int k210_clk_set_parent(struct clk *clk, struct clk *parent) +{ + struct clk *c, *p; + int err = clk_get_by_id(clk->id, &c); + + if (err) + return err; + + err = clk_get_by_id(parent->id, &p); + if (err) + return err; + + return clk_set_parent(c, p); +} + +static int k210_clk_endisable(struct clk *clk, bool enable) +{ + struct clk *c; + int err = clk_get_by_id(clk->id, &c); + + if (err) + return err; + return enable ? clk_enable(c) : clk_disable(c); +} + +static int k210_clk_enable(struct clk *clk) +{ + return k210_clk_endisable(clk, true); +} + +static int k210_clk_disable(struct clk *clk) +{ + return k210_clk_endisable(clk, false); +} + +static const struct clk_ops k210_clk_ops = { + .set_rate = k210_clk_set_rate, + .get_rate = k210_clk_get_rate, + .set_parent = k210_clk_set_parent, + .enable = k210_clk_enable, + .disable = k210_clk_disable, +}; + +static const char * const generic_sels[] = { "in0_half", "pll0_half" }; +/* The first clock is in0, which is filled in by k210_clk_probe */ +static const char *aclk_sels[] = { NULL, "pll0_half" }; +static const char *pll2_sels[] = { NULL, "pll0", "pll1" }; + +#define DIV(id, reg, shift, width) DIV_FLAGS(id, reg, shift, width, 0) +#define DIV_LIST \ + DIV_FLAGS(K210_CLK_ACLK, K210_SYSCTL_SEL0, 1, 2, \ + CLK_DIVIDER_POWER_OF_TWO) \ + DIV(K210_CLK_APB0, K210_SYSCTL_SEL0, 3, 3) \ + DIV(K210_CLK_APB1, K210_SYSCTL_SEL0, 6, 3) \ + DIV(K210_CLK_APB2, K210_SYSCTL_SEL0, 9, 3) \ + DIV(K210_CLK_SRAM0, K210_SYSCTL_THR0, 0, 4) \ + DIV(K210_CLK_SRAM1, K210_SYSCTL_THR0, 4, 4) \ + DIV(K210_CLK_AI, K210_SYSCTL_THR0, 8, 4) \ + DIV(K210_CLK_DVP, K210_SYSCTL_THR0, 12, 4) \ + DIV(K210_CLK_ROM, K210_SYSCTL_THR0, 16, 4) \ + DIV(K210_CLK_SPI0, K210_SYSCTL_THR1, 0, 8) \ + DIV(K210_CLK_SPI1, K210_SYSCTL_THR1, 8, 8) \ + DIV(K210_CLK_SPI2, K210_SYSCTL_THR1, 16, 8) \ + DIV(K210_CLK_SPI3, K210_SYSCTL_THR1, 24, 8) \ + DIV(K210_CLK_TIMER0, K210_SYSCTL_THR2, 0, 8) \ + DIV(K210_CLK_TIMER1, K210_SYSCTL_THR2, 8, 8) \ + DIV(K210_CLK_TIMER2, K210_SYSCTL_THR2, 16, 8) \ + DIV(K210_CLK_I2S0, K210_SYSCTL_THR3, 0, 16) \ + DIV(K210_CLK_I2S1, K210_SYSCTL_THR3, 16, 16) \ + DIV(K210_CLK_I2S2, K210_SYSCTL_THR4, 0, 16) \ + DIV(K210_CLK_I2S0_M, K210_SYSCTL_THR4, 16, 8) \ + DIV(K210_CLK_I2S1_M, K210_SYSCTL_THR4, 24, 8) \ + DIV(K210_CLK_I2S2_M, K210_SYSCTL_THR4, 0, 8) \ + DIV(K210_CLK_I2C0, K210_SYSCTL_THR5, 8, 8) \ + DIV(K210_CLK_I2C1, K210_SYSCTL_THR5, 16, 8) \ + DIV(K210_CLK_I2C2, K210_SYSCTL_THR5, 24, 8) \ + DIV(K210_CLK_WDT0, K210_SYSCTL_THR6, 0, 8) \ + DIV(K210_CLK_WDT1, K210_SYSCTL_THR6, 8, 8) + +#define _DIVIFY(id) K210_CLK_DIV_##id +#define DIVIFY(id) _DIVIFY(id) +#define DIV_FLAGS(id, ...) DIVIFY(id), +enum k210_clk_div_ids { + DIV_LIST +}; +#undef DIV_FLAGS + +#define DIV_FLAGS(id, _reg, _shift, _width, _flags) [DIVIFY(id)] = { \ + .reg = (void *)(_reg), \ + .shift = (_shift), \ + .width = (_width), \ + .flags = (_flags), \ +}, +static struct clk_divider k210_clk_dividers[] = { + DIV_LIST +}; +#undef DIV_FLAGS +#undef DIV +#undef DIV_LIST + +#define GATE_LIST \ + GATE(K210_CLK_CPU, K210_SYSCTL_EN_CENT, 0) \ + GATE(K210_CLK_SRAM0, K210_SYSCTL_EN_CENT, 1) \ + GATE(K210_CLK_SRAM1, K210_SYSCTL_EN_CENT, 2) \ + GATE(K210_CLK_APB0, K210_SYSCTL_EN_CENT, 3) \ + GATE(K210_CLK_APB1, K210_SYSCTL_EN_CENT, 4) \ + GATE(K210_CLK_APB2, K210_SYSCTL_EN_CENT, 5) \ + GATE(K210_CLK_ROM, K210_SYSCTL_EN_PERI, 0) \ + GATE(K210_CLK_DMA, K210_SYSCTL_EN_PERI, 1) \ + GATE(K210_CLK_AI, K210_SYSCTL_EN_PERI, 2) \ + GATE(K210_CLK_DVP, K210_SYSCTL_EN_PERI, 3) \ + GATE(K210_CLK_FFT, K210_SYSCTL_EN_PERI, 4) \ + GATE(K210_CLK_GPIO, K210_SYSCTL_EN_PERI, 5) \ + GATE(K210_CLK_SPI0, K210_SYSCTL_EN_PERI, 6) \ + GATE(K210_CLK_SPI1, K210_SYSCTL_EN_PERI, 7) \ + GATE(K210_CLK_SPI2, K210_SYSCTL_EN_PERI, 8) \ + GATE(K210_CLK_SPI3, K210_SYSCTL_EN_PERI, 9) \ + GATE(K210_CLK_I2S0, K210_SYSCTL_EN_PERI, 10) \ + GATE(K210_CLK_I2S1, K210_SYSCTL_EN_PERI, 11) \ + GATE(K210_CLK_I2S2, K210_SYSCTL_EN_PERI, 12) \ + GATE(K210_CLK_I2C0, K210_SYSCTL_EN_PERI, 13) \ + GATE(K210_CLK_I2C1, K210_SYSCTL_EN_PERI, 14) \ + GATE(K210_CLK_I2C2, K210_SYSCTL_EN_PERI, 15) \ + GATE(K210_CLK_UART1, K210_SYSCTL_EN_PERI, 16) \ + GATE(K210_CLK_UART2, K210_SYSCTL_EN_PERI, 17) \ + GATE(K210_CLK_UART3, K210_SYSCTL_EN_PERI, 18) \ + GATE(K210_CLK_AES, K210_SYSCTL_EN_PERI, 19) \ + GATE(K210_CLK_FPIOA, K210_SYSCTL_EN_PERI, 20) \ + GATE(K210_CLK_TIMER0, K210_SYSCTL_EN_PERI, 21) \ + GATE(K210_CLK_TIMER1, K210_SYSCTL_EN_PERI, 22) \ + GATE(K210_CLK_TIMER2, K210_SYSCTL_EN_PERI, 23) \ + GATE(K210_CLK_WDT0, K210_SYSCTL_EN_PERI, 24) \ + GATE(K210_CLK_WDT1, K210_SYSCTL_EN_PERI, 25) \ + GATE(K210_CLK_SHA, K210_SYSCTL_EN_PERI, 26) \ + GATE(K210_CLK_OTP, K210_SYSCTL_EN_PERI, 27) \ + GATE(K210_CLK_RTC, K210_SYSCTL_EN_PERI, 29) + +#define _GATEIFY(id) K210_CLK_GATE_##id +#define GATEIFY(id) _GATEIFY(id) +#define GATE(id, ...) GATEIFY(id), +enum k210_clk_gate_ids { + GATE_LIST +}; +#undef GATE + +#define GATE(id, _reg, _idx) [GATEIFY(id)] = { \ + .reg = (void *)(_reg), \ + .bit_idx = (_idx), \ +}, +static struct clk_gate k210_clk_gates[] = { + GATE_LIST +}; +#undef GATE +#undef GATE_LIST + +#define MUX(id, reg, shift, width) \ + MUX_PARENTS(id, generic_sels, reg, shift, width) +#define MUX_LIST \ + MUX_PARENTS(K210_CLK_PLL2, pll2_sels, K210_SYSCTL_PLL2, 26, 2) \ + MUX_PARENTS(K210_CLK_ACLK, aclk_sels, K210_SYSCTL_SEL0, 0, 1) \ + MUX(K210_CLK_SPI3, K210_SYSCTL_SEL0, 12, 1) \ + MUX(K210_CLK_TIMER0, K210_SYSCTL_SEL0, 13, 1) \ + MUX(K210_CLK_TIMER1, K210_SYSCTL_SEL0, 14, 1) \ + MUX(K210_CLK_TIMER2, K210_SYSCTL_SEL0, 15, 1) + +#define _MUXIFY(id) K210_CLK_MUX_##id +#define MUXIFY(id) _MUXIFY(id) +#define MUX_PARENTS(id, ...) MUXIFY(id), +enum k210_clk_mux_ids { + MUX_LIST +}; +#undef MUX_PARENTS + +#define MUX_PARENTS(id, parents, _reg, _shift, _width) [MUXIFY(id)] = { \ + .parent_names = (const char * const *)(parents), \ + .num_parents = ARRAY_SIZE(parents), \ + .reg = (void *)(_reg), \ + .shift = (_shift), \ + .mask = BIT(_width) - 1, \ +}, +static struct clk_mux k210_clk_muxes[] = { + MUX_LIST +}; +#undef MUX_PARENTS +#undef MUX +#undef MUX_LIST + +#define PLL(_reg, _shift, _width) { \ + .reg = (void *)(_reg), \ + .lock = (void *)K210_SYSCTL_PLL_LOCK, \ + .shift = (_shift), \ + .width = (_width), \ +} +static struct k210_pll k210_clk_plls[] = { + [0] = PLL(K210_SYSCTL_PLL0, 0, 2), + [1] = PLL(K210_SYSCTL_PLL1, 8, 1), + [2] = PLL(K210_SYSCTL_PLL2, 16, 1), +}; +#undef PLL + +#define COMP(id, mux, div, gate) \ + COMP_FULL(id, &(mux)->clk, &clk_mux_ops, \ + &(div)->clk, &clk_divider_ops, \ + &(gate)->clk, &clk_gate_ops) +#define COMP_ID(id) \ + COMP(id, &k210_clk_muxes[MUXIFY(id)], \ + &k210_clk_dividers[DIVIFY(id)], \ + &k210_clk_gates[GATEIFY(id)]) +#define COMP_NOMUX(id, div, gate) \ + COMP_FULL(id, NULL, NULL, \ + &(div)->clk, &clk_divider_ops, \ + &(gate)->clk, &clk_gate_ops) +#define COMP_NOMUX_ID(id) \ + COMP_NOMUX(id, &k210_clk_dividers[DIVIFY(id)], \ + &k210_clk_gates[GATEIFY(id)]) +#define COMP_LIST \ + COMP_FULL(K210_CLK_PLL2, \ + &k210_clk_muxes[MUXIFY(K210_CLK_PLL2)].clk, &clk_mux_ops, \ + &k210_clk_plls[2].clk, &k210_pll_ops, \ + &k210_clk_plls[2].clk, &k210_pll_ops) \ + COMP_FULL(K210_CLK_ACLK, \ + &k210_clk_muxes[MUXIFY(K210_CLK_ACLK)].clk, &clk_mux_ops, \ + &k210_clk_dividers[DIVIFY(K210_CLK_ACLK)].clk, \ + &clk_divider_ops, \ + NULL, NULL) \ + COMP_ID(K210_CLK_SPI3) \ + COMP_ID(K210_CLK_TIMER0) \ + COMP_ID(K210_CLK_TIMER1) \ + COMP_ID(K210_CLK_TIMER2) \ + COMP_NOMUX_ID(K210_CLK_SRAM0) \ + COMP_NOMUX_ID(K210_CLK_SRAM1) \ + COMP_NOMUX_ID(K210_CLK_ROM) \ + COMP_NOMUX_ID(K210_CLK_DVP) \ + COMP_NOMUX_ID(K210_CLK_APB0) \ + COMP_NOMUX_ID(K210_CLK_APB1) \ + COMP_NOMUX_ID(K210_CLK_APB2) \ + COMP_NOMUX_ID(K210_CLK_AI) \ + COMP_NOMUX_ID(K210_CLK_I2S0) \ + COMP_NOMUX_ID(K210_CLK_I2S1) \ + COMP_NOMUX_ID(K210_CLK_I2S2) \ + COMP_NOMUX_ID(K210_CLK_WDT0) \ + COMP_NOMUX_ID(K210_CLK_WDT1) \ + COMP_NOMUX_ID(K210_CLK_SPI0) \ + COMP_NOMUX_ID(K210_CLK_SPI1) \ + COMP_NOMUX_ID(K210_CLK_SPI2) \ + COMP_NOMUX_ID(K210_CLK_I2C0) \ + COMP_NOMUX_ID(K210_CLK_I2C1) \ + COMP_NOMUX_ID(K210_CLK_I2C2) + +#define _COMPIFY(id) K210_CLK_COMP_##id +#define COMPIFY(id) _COMPIFY(id) +#define COMP_FULL(id, ...) COMPIFY(id), +enum k210_clk_comp_ids { + COMP_LIST +}; +#undef COMP_FULL + +#define COMP_FULL(id, _mux, _mux_ops, _div, _div_ops, _gate, _gate_ops) \ +[COMPIFY(id)] = { \ + .mux = (_mux), \ + .mux_ops = (_mux_ops), \ + .rate = (_div), \ + .rate_ops = (_div_ops), \ + .gate = (_gate), \ + .gate_ops = (_gate_ops), \ +}, +static struct clk_composite k210_clk_comps[] = { + COMP_LIST +}; +#undef COMP_FULL +#undef COMP +#undef COMP_ID +#undef COMP_NOMUX +#undef COMP_NOMUX_ID +#undef COMP_LIST + +static struct clk *k210_clk_bypass_children = { + &k210_clk_comps[COMPIFY(K210_CLK_ACLK)].clk, +}; + +static struct clk *k210_clk_bypass_saved_parents = { + NULL, +}; + +static struct k210_bypass k210_clk_bypass = { + .bypassee = &k210_clk_plls[0].clk, + .bypassee_ops = &k210_pll_ops, + .children = &k210_clk_bypass_children, + .child_count = 1, + .saved_parents = &k210_clk_bypass_saved_parents, +}; + +static bool probed = false; + +static int k210_clk_probe(struct udevice *dev) +{ + int ret, i; + const char *in0; + struct clk *in0_clk; + void *base; + + /* Only one instance of this driver allowed */ + if (READ_ONCE(probed)) + return -ENOTSUPP; + + base = dev_read_addr_ptr(dev_get_parent(dev)); + if (!base) + return -EINVAL; + + in0_clk = kzalloc(sizeof(*in0_clk), GFP_KERNEL); + if (!in0_clk) + return -ENOMEM; + + ret = clk_get_by_index(dev, 0, in0_clk); + if (ret) + return ret; + in0 = in0_clk->dev->name; + + WRITE_ONCE(probed, true); + + aclk_sels[0] = in0; + pll2_sels[0] = in0; + + /* Fixup registers to be absolute, rather than relative */ +#define FIXUP_REGS(clocks) \ + for (i = 0; i < ARRAY_SIZE(clocks); i++) \ + clocks[i].reg += (ulong)base + FIXUP_REGS(k210_clk_dividers); + FIXUP_REGS(k210_clk_gates); + FIXUP_REGS(k210_clk_muxes); +#undef FIXUP_REGS + for (i = 0; i < ARRAY_SIZE(k210_clk_plls); i++) { + k210_clk_plls[i].reg += (ulong)base; + k210_clk_plls[i].lock += (ulong)base; + } + + /* + * All PLLs have a broken bypass, but pll0 has the CPU downstream, so we + * need to manually reparent it whenever we configure pll0 + */ + k210_clk_bypass.alt = in0_clk; + clk_dm(K210_CLK_PLL0, + k210_register_bypass_struct("pll0", in0, &k210_clk_bypass)); + clk_dm(K210_CLK_PLL1, + k210_register_pll_struct("pll1", in0, &k210_clk_plls[1])); + /* PLL2 is muxed, so set up a composite clock */ + clk_dm(K210_CLK_PLL2, + clk_register_composite_struct("pll2", pll2_sels, + ARRAY_SIZE(pll2_sels), + &k210_clk_comps[COMPIFY(K210_CLK_PLL2)])); + + /* Half-frequency clocks for "even" dividers */ + k210_clk_half("in0_half", in0); + k210_clk_half("pll0_half", "pll0"); + k210_clk_half("pll2_half", "pll2"); + + /* ACLK has no gate */ + clk_dm(K210_CLK_ACLK, + clk_register_composite_struct("aclk", aclk_sels, + ARRAY_SIZE(aclk_sels), + &k210_clk_comps[COMPIFY(K210_CLK_ACLK)])); + +#define REGISTER_COMP(id, name) \ + clk_dm(id, clk_register_composite_struct(name, generic_sels, \ + ARRAY_SIZE(generic_sels), \ + &k210_clk_comps[COMPIFY(id)])) + REGISTER_COMP(K210_CLK_SPI3, "spi3"); + REGISTER_COMP(K210_CLK_TIMER0, "timer0"); + REGISTER_COMP(K210_CLK_TIMER1, "timer1"); + REGISTER_COMP(K210_CLK_TIMER2, "timer2"); +#undef COMP + + /* Dividing clocks, no mux */ +#define REGISTER_COMP_NOMUX(id, name, _parent) do { \ + const char *parent = _parent; \ + clk_dm(id, \ + clk_register_composite_struct(name, &parent, 1, \ + &k210_clk_comps[COMPIFY(id)])); \ +} while (false) + REGISTER_COMP_NOMUX(K210_CLK_SRAM0, "sram0", "aclk"); + REGISTER_COMP_NOMUX(K210_CLK_SRAM1, "sram1", "aclk"); + REGISTER_COMP_NOMUX(K210_CLK_ROM, "rom", "aclk"); + REGISTER_COMP_NOMUX(K210_CLK_DVP, "dvp", "aclk"); + REGISTER_COMP_NOMUX(K210_CLK_APB0, "apb0", "aclk"); + REGISTER_COMP_NOMUX(K210_CLK_APB1, "apb1", "aclk"); + REGISTER_COMP_NOMUX(K210_CLK_APB2, "apb2", "aclk"); + REGISTER_COMP_NOMUX(K210_CLK_AI, "ai", "pll1"); + REGISTER_COMP_NOMUX(K210_CLK_I2S0, "i2s0", "pll2_half"); + REGISTER_COMP_NOMUX(K210_CLK_I2S1, "i2s1", "pll2_half"); + REGISTER_COMP_NOMUX(K210_CLK_I2S2, "i2s2", "pll2_half"); + REGISTER_COMP_NOMUX(K210_CLK_WDT0, "wdt0", "in0_half"); + REGISTER_COMP_NOMUX(K210_CLK_WDT1, "wdt1", "in0_half"); + REGISTER_COMP_NOMUX(K210_CLK_SPI0, "spi0", "pll0_half"); + REGISTER_COMP_NOMUX(K210_CLK_SPI1, "spi1", "pll0_half"); + REGISTER_COMP_NOMUX(K210_CLK_SPI2, "spi2", "pll0_half"); + REGISTER_COMP_NOMUX(K210_CLK_I2C0, "i2c0", "pll0_half"); + REGISTER_COMP_NOMUX(K210_CLK_I2C1, "i2c1", "pll0_half"); + REGISTER_COMP_NOMUX(K210_CLK_I2C2, "i2c2", "pll0_half"); +#undef REGISTER_COMP_NOMUX + + /* Dividing clocks */ +#define REGISTER_DIV(id, name, parent) clk_dm(id, \ + clk_register_divider_struct(name, parent, \ + &k210_clk_dividers[DIVIFY(id)])) + REGISTER_DIV(K210_CLK_I2S0_M, "i2s0_m", "pll2_half"); + REGISTER_DIV(K210_CLK_I2S1_M, "i2s1_m", "pll2_half"); + REGISTER_DIV(K210_CLK_I2S2_M, "i2s2_m", "pll2_half"); +#undef REGISTER_DIV + + /* Gated clocks */ +#define REGISTER_GATE(id, name, parent) \ + clk_dm(id, clk_register_gate_struct(name, parent, \ + &k210_clk_gates[GATEIFY(id)])) + REGISTER_GATE(K210_CLK_CPU, "cpu", "aclk"); + REGISTER_GATE(K210_CLK_DMA, "dma", "aclk"); + REGISTER_GATE(K210_CLK_FFT, "fft", "aclk"); + REGISTER_GATE(K210_CLK_GPIO, "gpio", "apb0"); + REGISTER_GATE(K210_CLK_UART1, "uart1", "apb0"); + REGISTER_GATE(K210_CLK_UART2, "uart2", "apb0"); + REGISTER_GATE(K210_CLK_UART3, "uart3", "apb0"); + REGISTER_GATE(K210_CLK_FPIOA, "fpioa", "apb0"); + REGISTER_GATE(K210_CLK_SHA, "sha", "apb0"); + REGISTER_GATE(K210_CLK_AES, "aes", "apb1"); + REGISTER_GATE(K210_CLK_OTP, "otp", "apb1"); + REGISTER_GATE(K210_CLK_RTC, "rtc", in0); +#undef REGISTER_GATE + + return 0; +} + +static const struct udevice_id k210_clk_ids[] = { + { .compatible = "kendryte,k210-clk" }, + { }, +}; + +U_BOOT_DRIVER(k210_clk) = { + .name = "k210_clk", + .id = UCLASS_CLK, + .of_match = k210_clk_ids, + .ops = &k210_clk_ops, + .probe = k210_clk_probe, +}; diff --git a/include/dt-bindings/clock/k210-sysctl.h b/include/dt-bindings/clock/k210-sysctl.h new file mode 100644 index 0000000000..16d67b282f --- /dev/null +++ b/include/dt-bindings/clock/k210-sysctl.h @@ -0,0 +1,56 @@ +/* SPDX-License-Identifier: GPL-2.0+ */ +/* + * Copyright (C) 2019-20 Sean Anderson <seanga2 at gmail.com> + */ + +#ifndef CLOCK_K210_SYSCTL_H +#define CLOCK_K210_SYSCTL_H + +/* + * Arbitrary identifiers for clocks. + */ +#define K210_CLK_NONE 0 +#define K210_CLK_PLL0 1 +#define K210_CLK_PLL1 2 +#define K210_CLK_PLL2 3 +#define K210_CLK_CPU 4 +#define K210_CLK_SRAM0 5 +#define K210_CLK_SRAM1 6 +#define K210_CLK_APB0 7 +#define K210_CLK_APB1 8 +#define K210_CLK_APB2 9 +#define K210_CLK_ROM 10 +#define K210_CLK_DMA 11 +#define K210_CLK_AI 12 +#define K210_CLK_DVP 13 +#define K210_CLK_FFT 14 +#define K210_CLK_GPIO 15 +#define K210_CLK_SPI0 16 +#define K210_CLK_SPI1 17 +#define K210_CLK_SPI2 18 +#define K210_CLK_SPI3 19 +#define K210_CLK_I2S0 20 +#define K210_CLK_I2S1 21 +#define K210_CLK_I2S2 22 +#define K210_CLK_I2S0_M 23 +#define K210_CLK_I2S1_M 24 +#define K210_CLK_I2S2_M 25 +#define K210_CLK_I2C0 26 +#define K210_CLK_I2C1 27 +#define K210_CLK_I2C2 28 +#define K210_CLK_UART1 29 +#define K210_CLK_UART2 30 +#define K210_CLK_UART3 31 +#define K210_CLK_AES 32 +#define K210_CLK_FPIOA 33 +#define K210_CLK_TIMER0 34 +#define K210_CLK_TIMER1 35 +#define K210_CLK_TIMER2 36 +#define K210_CLK_WDT0 37 +#define K210_CLK_WDT1 38 +#define K210_CLK_SHA 39 +#define K210_CLK_OTP 40 +#define K210_CLK_RTC 41 +#define K210_CLK_ACLK 42 + +#endif /* CLOCK_K210_SYSCTL_H */ diff --git a/include/dt-bindings/mfd/k210-sysctl.h b/include/dt-bindings/mfd/k210-sysctl.h new file mode 100644 index 0000000000..e16d7302cd --- /dev/null +++ b/include/dt-bindings/mfd/k210-sysctl.h @@ -0,0 +1,38 @@ +/* SPDX-License-Identifier: GPL-2.0+ */ +/* + * Copyright (C) 2020 Sean Anderson <seanga2 at gmail.com> + */ + +#ifndef K210_SYSCTL_H +#define K210_SYSCTL_H + +/* Taken from kendryte-standalone-sdk/lib/drivers/include/sysctl.h */ +#define K210_SYSCTL_GIT_ID 0x00 /* Git short commit id */ +#define K210_SYSCTL_CLK_FREQ 0x04 /* System clock base frequency */ +#define K210_SYSCTL_PLL0 0x08 /* PLL0 controller */ +#define K210_SYSCTL_PLL1 0x0C /* PLL1 controller */ +#define K210_SYSCTL_PLL2 0x10 /* PLL2 controller */ +#define K210_SYSCTL_PLL_LOCK 0x18 /* PLL lock tester */ +#define K210_SYSCTL_ROM_ERROR 0x1C /* AXI ROM detector */ +#define K210_SYSCTL_SEL0 0x20 /* Clock select controller0 */ +#define K210_SYSCTL_SEL1 0x24 /* Clock select controller1 */ +#define K210_SYSCTL_EN_CENT 0x28 /* Central clock enable */ +#define K210_SYSCTL_EN_PERI 0x2C /* Peripheral clock enable */ +#define K210_SYSCTL_SOFT_RESET 0x30 /* Soft reset ctrl */ +#define K210_SYSCTL_PERI_RESET 0x34 /* Peripheral reset controller */ +#define K210_SYSCTL_THR0 0x38 /* Clock threshold controller 0 */ +#define K210_SYSCTL_THR1 0x3C /* Clock threshold controller 1 */ +#define K210_SYSCTL_THR2 0x40 /* Clock threshold controller 2 */ +#define K210_SYSCTL_THR3 0x44 /* Clock threshold controller 3 */ +#define K210_SYSCTL_THR4 0x48 /* Clock threshold controller 4 */ +#define K210_SYSCTL_THR5 0x4C /* Clock threshold controller 5 */ +#define K210_SYSCTL_THR6 0x50 /* Clock threshold controller 6 */ +#define K210_SYSCTL_MISC 0x54 /* Miscellaneous controller */ +#define K210_SYSCTL_PERI 0x58 /* Peripheral controller */ +#define K210_SYSCTL_SPI_SLEEP 0x5C /* SPI sleep controller */ +#define K210_SYSCTL_RESET_STAT 0x60 /* Reset source status */ +#define K210_SYSCTL_DMA_SEL0 0x64 /* DMA handshake selector */ +#define K210_SYSCTL_DMA_SEL1 0x68 /* DMA handshake selector */ +#define K210_SYSCTL_POWER_SEL 0x6C /* IO Power Mode Select controller */ + +#endif /* K210_SYSCTL_H */ diff --git a/include/kendryte/clk.h b/include/kendryte/clk.h new file mode 100644 index 0000000000..9c6245d468 --- /dev/null +++ b/include/kendryte/clk.h @@ -0,0 +1,35 @@ +/* SPDX-License-Identifier: GPL-2.0+ */ +/* + * Copyright (C) 2019-20 Sean Anderson <seanga2 at gmail.com> + */ + +#ifndef K210_CLK_H +#define K210_CLK_H + +#define LOG_CATEGORY UCLASS_CLK +#include <linux/types.h> +#include <linux/clk-provider.h> + +static inline struct clk *k210_clk_gate(const char *name, + const char *parent_name, + void __iomem *reg, u8 bit_idx) +{ + return clk_register_gate(NULL, name, parent_name, 0, reg, bit_idx, 0, + NULL); +} + +static inline struct clk *k210_clk_half(const char *name, + const char *parent_name) +{ + return clk_register_fixed_factor(NULL, name, parent_name, 0, 1, 2); +} + +static inline struct clk *k210_clk_div(const char *name, + const char *parent_name, + void __iomem *reg, u8 shift, u8 width) +{ + return clk_register_divider(NULL, name, parent_name, 0, reg, shift, + width, 0); +} + +#endif /* K210_CLK_H */
Due to the large number of clocks, I decided to use the CCF. The overall structure is modeled after the imx code. Clocks are stored in several arrays. There are some translation macros (FOOIFY()) which allow for more dense packing. A possible improvement could be to only store the parameters we need, instead of the whole CCF struct. Signed-off-by: Sean Anderson <seanga2 at gmail.com> --- Changes in v5: - Don't unmap priv->reg - Remove comment on APB clocks since it has been clarified by Kendryte - Add i2s mclks - Reorder clock ids to be continuous - Rewrite to statically allocate all clocks. This has helped find several bugs (since it is easy to see when a clock has the wrong register). - Fix ACLK sometimes having the wrong parent - Fix SPI3 having the wrong divider - Prevent being probed multiple times on failure Changes in v4: - Reparent aclk before configuring pll0 - Update copyright - Lint Changes in v3: - Removed sysctl struct, replacing it with defines. This is to have the same interface to sysctl from C as from the device tree. - Fixed clocks having the same id - Fixed clocks not using the correct register/bits - Aligned the defines in headers Changes in v2: - Add clk.o to obj-y - Don't probe before relocation MAINTAINERS | 7 + .../mfd/kendryte,k210-sysctl.txt | 33 ++ drivers/clk/kendryte/Kconfig | 2 +- drivers/clk/kendryte/Makefile | 2 +- drivers/clk/kendryte/clk.c | 478 ++++++++++++++++++ include/dt-bindings/clock/k210-sysctl.h | 56 ++ include/dt-bindings/mfd/k210-sysctl.h | 38 ++ include/kendryte/clk.h | 35 ++ 8 files changed, 649 insertions(+), 2 deletions(-) create mode 100644 doc/device-tree-bindings/mfd/kendryte,k210-sysctl.txt create mode 100644 drivers/clk/kendryte/clk.c create mode 100644 include/dt-bindings/clock/k210-sysctl.h create mode 100644 include/dt-bindings/mfd/k210-sysctl.h create mode 100644 include/kendryte/clk.h