diff mbox series

[v7,04/19] rockchip: rk3066: add clock driver for rk3066 soc

Message ID 20220111211843.10958-5-jbx6244@gmail.com
State Superseded
Delegated to: Kever Yang
Headers show
Series Add Rikomagic MK808 board | expand

Commit Message

Johan Jonker Jan. 11, 2022, 9:18 p.m. UTC
From: Paweł Jarosz <paweljarosz3691@gmail.com>

Add clock driver for rk3066 platform.

Signed-off-by: Paweł Jarosz <paweljarosz3691@gmail.com>
Signed-off-by: Johan Jonker <jbx6244@gmail.com>
---

Changed V7:
  changed function prefix
  changed #if where possible
  restyle U_BOOT_DRIVER structure
---
 .../include/asm/arch-rockchip/cru_rk3066.h    | 203 +++++
 drivers/clk/rockchip/Makefile                 |   1 +
 drivers/clk/rockchip/clk_rk3066.c             | 700 ++++++++++++++++++
 3 files changed, 904 insertions(+)
 create mode 100644 arch/arm/include/asm/arch-rockchip/cru_rk3066.h
 create mode 100644 drivers/clk/rockchip/clk_rk3066.c

Comments

Sean Anderson Jan. 15, 2022, 5:27 p.m. UTC | #1
On 1/11/22 4:18 PM, Johan Jonker wrote:
> From: Paweł Jarosz <paweljarosz3691@gmail.com>
>
> Add clock driver for rk3066 platform.

Can you comment a bit on what you support? For example, it seems like
there are some clocks which are fixed at particular frequencies. Why did
you choose those? Which clocks can be set freely? A comment on the cpu
frequency would be good as well.

>
> Signed-off-by: Paweł Jarosz <paweljarosz3691@gmail.com>
> Signed-off-by: Johan Jonker <jbx6244@gmail.com>
> ---
>
> Changed V7:
>    changed function prefix
>    changed #if where possible
>    restyle U_BOOT_DRIVER structure
> ---
>   .../include/asm/arch-rockchip/cru_rk3066.h    | 203 +++++
>   drivers/clk/rockchip/Makefile                 |   1 +
>   drivers/clk/rockchip/clk_rk3066.c             | 700 ++++++++++++++++++
>   3 files changed, 904 insertions(+)
>   create mode 100644 arch/arm/include/asm/arch-rockchip/cru_rk3066.h
>   create mode 100644 drivers/clk/rockchip/clk_rk3066.c
>
> diff --git a/arch/arm/include/asm/arch-rockchip/cru_rk3066.h b/arch/arm/include/asm/arch-rockchip/cru_rk3066.h
> new file mode 100644
> index 00000000..711366d5
> --- /dev/null
> +++ b/arch/arm/include/asm/arch-rockchip/cru_rk3066.h
> @@ -0,0 +1,203 @@
> +/* SPDX-License-Identifier: GPL-2.0+ */
> +/*
> + * (C) Copyright 2021 Paweł Jarosz <paweljarosz3691@gmail.com>
> + */
> +
> +#ifndef _ASM_ARCH_CRU_RK3066_H
> +#define _ASM_ARCH_CRU_RK3066_H
> +
> +#define OSC_HZ        (24 * 1000 * 1000)
> +
> +#define APLL_HZ    (1416 * 1000000)
> +#define APLL_SAFE_HZ    (600 * 1000000)
> +#define GPLL_HZ    (594 * 1000000)
> +#define CPLL_HZ    (384 * 1000000)
> +
> +/* The SRAM is clocked off aclk_cpu, so we want to max it out for bootspeed */
> +#define CPU_ACLK_HZ    297000000
> +#define CPU_HCLK_HZ    148500000
> +#define CPU_PCLK_HZ    74250000
> +#define CPU_H2P_HZ    74250000
> +
> +#define PERI_ACLK_HZ    148500000
> +#define PERI_HCLK_HZ    148500000
> +#define PERI_PCLK_HZ    74250000
> +
> +/* Private data for the clock driver - used by rockchip_get_cru() */
> +struct rk3066_clk_priv {
> +    struct rk3066_grf *grf;
> +    struct rk3066_cru *cru;
> +    ulong rate;
> +    bool has_bwadj;
> +};
> +
> +struct rk3066_cru {
> +    struct rk3066_pll {
> +        u32 con0;
> +        u32 con1;
> +        u32 con2;
> +        u32 con3;
> +    } pll[4];
> +    u32 cru_mode_con;
> +    u32 cru_clksel_con[35];
> +    u32 cru_clkgate_con[10];
> +    u32 reserved1[2];
> +    u32 cru_glb_srst_fst_value;
> +    u32 cru_glb_srst_snd_value;
> +    u32 reserved2[2];
> +    u32 cru_softrst_con[9];
> +    u32 cru_misc_con;
> +    u32 reserved3[2];
> +    u32 cru_glb_cnt_th;
> +};
> +
> +check_member(rk3066_cru, cru_glb_cnt_th, 0x0140);
> +
> +/* CRU_CLKSEL0_CON */
> +enum {
> +    /* a9_core_div: core = core_src / (a9_core_div + 1) */
> +    A9_CORE_DIV_SHIFT    = 9,
> +    A9_CORE_DIV_MASK    = 0x1f << A9_CORE_DIV_SHIFT,
> +    CORE_PLL_SHIFT        = 8,
> +    CORE_PLL_MASK        = 1 << CORE_PLL_SHIFT,
> +    CORE_PLL_SELECT_APLL    = 0,
> +    CORE_PLL_SELECT_GPLL,

Can you use GENMASK for this? e.g.

     A9_CORE_DIV_MASK = GENMASK(13, 9),

this will make it easier to see which bits are being used. You can also
do e.g.

     A9_CODE_DIV_SHIFT = __bf_shf(A9_CORE_DIV_MASK),

which could of course be combined to something like

     #define REG(name, h, l) \
         name##_MASK = GENMASK(h, l), \
         name##_SHIFT = __bf_shf(name##_MASK)

if you so desire

> +    /* core peri div: core:core_peri = 2:1, 4:1, 8:1 or 16:1 */
> +    CORE_PERI_DIV_SHIFT    = 6,
> +    CORE_PERI_DIV_MASK    = 3 << CORE_PERI_DIV_SHIFT,
> +
> +    /* aclk_cpu pll selection */
> +    CPU_ACLK_PLL_SHIFT    = 5,
> +    CPU_ACLK_PLL_MASK    = 1 << CPU_ACLK_PLL_SHIFT,
> +    CPU_ACLK_PLL_SELECT_APLL    = 0,
> +    CPU_ACLK_PLL_SELECT_GPLL,
> +
> +    /* a9_cpu_div: aclk_cpu = cpu_src / (a9_cpu_div + 1) */
> +    A9_CPU_DIV_SHIFT    = 0,
> +    A9_CPU_DIV_MASK    = 0x1f << A9_CPU_DIV_SHIFT,
> +};
> +
> +/* CRU_CLKSEL1_CON */
> +enum {
> +    /* ahb2apb_pclk_div: hclk_cpu:pclk_cpu = 1:1, 2:1 or 4:1 */
> +    AHB2APB_DIV_SHIFT    = 14,
> +    AHB2APB_DIV_MASK    = 3 << AHB2APB_DIV_SHIFT,
> +
> +    /* cpu_pclk_div: aclk_cpu:pclk_cpu = 1:1, 2:1, 4:1 or 8:1 */
> +    CPU_PCLK_DIV_SHIFT    = 12,
> +    CPU_PCLK_DIV_MASK    = 3 << CPU_PCLK_DIV_SHIFT,
> +
> +    /* cpu_hclk_div: aclk_cpu:hclk_cpu = 1:1, 2:1 or 4:1 */
> +    CPU_HCLK_DIV_SHIFT    = 8,
> +    CPU_HCLK_DIV_MASK    = 3 << CPU_HCLK_DIV_SHIFT,
> +
> +    /* core_aclk_div: cire:aclk_core = 1:1, 2:1, 3:1, 4:1 or 8:1 */
> +    CORE_ACLK_DIV_SHIFT    = 3,
> +    CORE_ACLK_DIV_MASK    = 7 << CORE_ACLK_DIV_SHIFT,
> +};
> +
> +/* CRU_CLKSEL10_CON */
> +enum {
> +    PERI_SEL_PLL_SHIFT    = 15,
> +    PERI_SEL_PLL_MASK    = 1 << PERI_SEL_PLL_SHIFT,
> +    PERI_SEL_CPLL        = 0,
> +    PERI_SEL_GPLL,
> +
> +    /* peri pclk div: aclk_bus:pclk_bus = 1:1, 2:1, 4:1 or 8:1 */
> +    PERI_PCLK_DIV_SHIFT    = 12,
> +    PERI_PCLK_DIV_MASK    = 3 << PERI_PCLK_DIV_SHIFT,
> +
> +    /* peripheral bus hclk div:aclk_bus: hclk_bus = 1:1, 2:1 or 4:1 */
> +    PERI_HCLK_DIV_SHIFT    = 8,
> +    PERI_HCLK_DIV_MASK    = 3 << PERI_HCLK_DIV_SHIFT,
> +
> +    /* peri aclk div: aclk_peri = periph_src / (peri_aclk_div + 1) */
> +    PERI_ACLK_DIV_SHIFT    = 0,
> +    PERI_ACLK_DIV_MASK    = 0x1f << PERI_ACLK_DIV_SHIFT,
> +};
> +
> +/* CRU_CLKSEL11_CON */
> +enum {
> +    MMC0_DIV_SHIFT        = 0,
> +    MMC0_DIV_MASK        = 0x3f << MMC0_DIV_SHIFT,
> +};
> +
> +/* CRU_CLKSEL12_CON */
> +enum {
> +    UART_PLL_SHIFT        = 15,
> +    UART_PLL_MASK        = 1 << UART_PLL_SHIFT,
> +    UART_PLL_SELECT_GENERAL    = 0,
> +    UART_PLL_SELECT_CODEC,
> +
> +    EMMC_DIV_SHIFT        = 8,
> +    EMMC_DIV_MASK        = 0x3f << EMMC_DIV_SHIFT,
> +
> +    SDIO_DIV_SHIFT        = 0,
> +    SDIO_DIV_MASK        = 0x3f << SDIO_DIV_SHIFT,
> +};
> +
> +/* CRU_CLKSEL24_CON */
> +enum {
> +    SARADC_DIV_SHIFT    = 8,
> +    SARADC_DIV_MASK    = 0xff << SARADC_DIV_SHIFT,
> +};
> +
> +/* CRU_CLKSEL25_CON */
> +enum {
> +    SPI1_DIV_SHIFT        = 8,
> +    SPI1_DIV_MASK        = 0x7f << SPI1_DIV_SHIFT,
> +
> +    SPI0_DIV_SHIFT        = 0,
> +    SPI0_DIV_MASK        = 0x7f << SPI0_DIV_SHIFT,
> +};
> +
> +/* CRU_CLKSEL34_CON */
> +enum {
> +    TSADC_DIV_SHIFT    = 0,
> +    TSADC_DIV_MASK        = 0xffff << TSADC_DIV_SHIFT,
> +};
> +
> +/* CRU_MODE_CON */
> +enum {
> +    GPLL_MODE_SHIFT    = 12,
> +    GPLL_MODE_MASK        = 3 << GPLL_MODE_SHIFT,
> +    GPLL_MODE_SLOW        = 0,
> +    GPLL_MODE_NORMAL,
> +    GPLL_MODE_DEEP,
> +
> +    CPLL_MODE_SHIFT        = 8,
> +    CPLL_MODE_MASK        = 3 << CPLL_MODE_SHIFT,
> +    CPLL_MODE_SLOW        = 0,
> +    CPLL_MODE_NORMAL,
> +    CPLL_MODE_DEEP,
> +
> +    DPLL_MODE_SHIFT        = 4,
> +    DPLL_MODE_MASK        = 3 << DPLL_MODE_SHIFT,
> +    DPLL_MODE_SLOW        = 0,
> +    DPLL_MODE_NORMAL,
> +    DPLL_MODE_DEEP,
> +
> +    APLL_MODE_SHIFT        = 0,
> +    APLL_MODE_MASK        = 3 << APLL_MODE_SHIFT,
> +    APLL_MODE_SLOW        = 0,
> +    APLL_MODE_NORMAL,
> +    APLL_MODE_DEEP,
> +};
> +
> +/* CRU_APLL_CON0 */
> +enum {
> +    CLKR_SHIFT        = 8,
> +    CLKR_MASK        = 0x3f << CLKR_SHIFT,
> +
> +    CLKOD_SHIFT        = 0,
> +    CLKOD_MASK        = 0x3f << CLKOD_SHIFT,
> +};
> +
> +/* CRU_APLL_CON1 */
> +enum {
> +    CLKF_SHIFT        = 0,
> +    CLKF_MASK        = 0x1fff << CLKF_SHIFT,
> +};
> +
> +#endif
> diff --git a/drivers/clk/rockchip/Makefile b/drivers/clk/rockchip/Makefile
> index 913f611a..a72d8fe5 100644
> --- a/drivers/clk/rockchip/Makefile
> +++ b/drivers/clk/rockchip/Makefile
> @@ -6,6 +6,7 @@
>   obj-y += clk_pll.o
>   obj-$(CONFIG_ROCKCHIP_PX30) += clk_px30.o
>   obj-$(CONFIG_ROCKCHIP_RK3036) += clk_rk3036.o
> +obj-$(CONFIG_ROCKCHIP_RK3066) += clk_rk3066.o
>   obj-$(CONFIG_ROCKCHIP_RK3128) += clk_rk3128.o
>   obj-$(CONFIG_ROCKCHIP_RK3188) += clk_rk3188.o
>   obj-$(CONFIG_ROCKCHIP_RK322X) += clk_rk322x.o
> diff --git a/drivers/clk/rockchip/clk_rk3066.c b/drivers/clk/rockchip/clk_rk3066.c
> new file mode 100644
> index 00000000..3b7de51c
> --- /dev/null
> +++ b/drivers/clk/rockchip/clk_rk3066.c
> @@ -0,0 +1,700 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/*
> + * (C) Copyright 2015 Google, Inc
> + * (C) Copyright 2016 Heiko Stuebner <heiko@sntech.de>
> + */
> +
> +#include <bitfield.h>
> +#include <common.h>
> +#include <clk-uclass.h>
> +#include <dm.h>
> +#include <dt-structs.h>
> +#include <errno.h>
> +#include <log.h>
> +#include <malloc.h>
> +#include <mapmem.h>
> +#include <syscon.h>
> +#include <asm/io.h>
> +#include <asm/arch-rockchip/clock.h>
> +#include <asm/arch-rockchip/cru_rk3066.h>
> +#include <asm/arch-rockchip/grf_rk3066.h>
> +#include <asm/arch-rockchip/hardware.h>
> +#include <dt-bindings/clock/rk3066a-cru.h>
> +#include <dm/device-internal.h>
> +#include <dm/lists.h>
> +#include <dm/uclass-internal.h>
> +#include <linux/delay.h>
> +#include <linux/err.h>
> +#include <linux/log2.h>
> +#include <linux/stringify.h>
> +
> +struct rk3066_clk_plat {
> +#if CONFIG_IS_ENABLED(OF_PLATDATA)
> +    struct dtd_rockchip_rk3066a_cru dtd;
> +#endif
> +};
> +
> +struct pll_div {
> +    u32 nr;
> +    u32 nf;
> +    u32 no;
> +};
> +
> +enum {
> +    VCO_MAX_HZ    = 1416U * 1000000,
> +    VCO_MIN_HZ    = 300 * 1000000,
> +    OUTPUT_MAX_HZ    = 1416U * 1000000,
> +    OUTPUT_MIN_HZ    = 30 * 1000000,
> +    FREF_MAX_HZ    = 1416U * 1000000,
> +    FREF_MIN_HZ    = 30 * 1000,
> +};
> +
> +enum {
> +    /* PLL CON0 */
> +    PLL_OD_MASK        = GENMASK(3, 0),
> +
> +    /* PLL CON1 */
> +    PLL_NF_MASK        = GENMASK(12, 0),
> +
> +    /* PLL CON2 */
> +    PLL_BWADJ_MASK        = GENMASK(11, 0),
> +
> +    /* PLL CON3 */
> +    PLL_RESET_SHIFT        = 5,
> +
> +    /* GRF_SOC_STATUS0 */
> +    SOCSTS_DPLL_LOCK    = BIT(4),
> +    SOCSTS_APLL_LOCK    = BIT(5),
> +    SOCSTS_CPLL_LOCK    = BIT(6),
> +    SOCSTS_GPLL_LOCK    = BIT(7),
> +};
> +
> +#define DIV_TO_RATE(input_rate, div)    ((input_rate) / ((div) + 1))
> +
> +#define PLL_DIVISORS(hz, _nr, _no) {\
> +    .nr = _nr, .nf = (u32)((u64)hz * _nr * _no / OSC_HZ), .no = _no};\
> +    _Static_assert(((u64)hz * _nr * _no / OSC_HZ) * OSC_HZ /\
> +               (_nr * _no) == hz, #hz "Hz cannot be hit with PLL "\
> +               "divisors on line " __stringify(__LINE__))
> +
> +/* Keep divisors as low as possible to reduce jitter and power usage */
> +static const struct pll_div gpll_init_cfg = PLL_DIVISORS(GPLL_HZ, 2,2);
> +static const struct pll_div cpll_init_cfg = PLL_DIVISORS(CPLL_HZ, 1,2);
> +
> +static int rk3066_clk_set_pll(struct rk3066_cru *cru, enum rk_clk_id clk_id,
> +                  const struct pll_div *div)
> +{
> +    int pll_id = rk_pll_id(clk_id);
> +    struct rk3066_pll *pll = &cru->pll[pll_id];
> +    /* All PLLs have same VCO and output frequency range restrictions.*/
> +    uint vco_hz = OSC_HZ / 1000 * div->nf / div->nr * 1000;
> +    uint output_hz = vco_hz / div->no;
> +
> +    debug("PLL at %x: nf=%d, nr=%d, no=%d, vco=%u Hz, output=%u Hz\n",
> +          (uint)pll, div->nf, div->nr, div->no, vco_hz, output_hz);
> +    assert(vco_hz >= VCO_MIN_HZ && vco_hz <= VCO_MAX_HZ &&
> +           output_hz >= OUTPUT_MIN_HZ && output_hz <= OUTPUT_MAX_HZ &&
> +           (div->no == 1 || !(div->no % 2)));
> +
> +    /* enter reset */
> +    rk_setreg(&pll->con3, BIT(PLL_RESET_SHIFT));
> +
> +    rk_clrsetreg(&pll->con0,
> +             CLKR_MASK | PLL_OD_MASK,
> +             ((div->nr - 1) << CLKR_SHIFT) | (div->no - 1));
> +    rk_clrsetreg(&pll->con1, CLKF_MASK, div->nf - 1);
> +
> +    rk_clrsetreg(&pll->con2, PLL_BWADJ_MASK, (div->nf >> 1) - 1);
> +
> +    /* return from reset */
> +    rk_clrreg(&pll->con3, BIT(PLL_RESET_SHIFT));
> +
> +    return 0;
> +}
> +
> +static int rk3066_clk_configure_ddr(struct rk3066_cru *cru, struct rk3066_grf *grf,
> +                    unsigned int hz)
> +{
> +    static const struct pll_div dpll_cfg[] = {
> +        {.nf = 25, .nr = 2, .no = 1},
> +        {.nf = 400, .nr = 9, .no = 2},
> +        {.nf = 500, .nr = 9, .no = 2},
> +        {.nf = 100, .nr = 3, .no = 1},
> +    };
> +    int cfg;
> +
> +    switch (hz) {
> +    case 300000000:
> +        cfg = 0;
> +        break;
> +    case 533000000:    /* actually 533.3P MHz */
> +        cfg = 1;
> +        break;
> +    case 666000000:    /* actually 666.6P MHz */
> +        cfg = 2;
> +        break;
> +    case 800000000:
> +        cfg = 3;
> +        break;
> +    default:
> +        debug("Unsupported SDRAM frequency");
> +        return -EINVAL;
> +    }
> +
> +    /* pll enter slow-mode */
> +    rk_clrsetreg(&cru->cru_mode_con, DPLL_MODE_MASK,
> +             DPLL_MODE_SLOW << DPLL_MODE_SHIFT);
> +
> +    rk3066_clk_set_pll(cru, CLK_DDR, &dpll_cfg[cfg]);
> +
> +    /* wait for pll lock */
> +    while (!(readl(&grf->soc_status0) & SOCSTS_DPLL_LOCK))
> +        udelay(1);
> +
> +    /* PLL enter normal-mode */
> +    rk_clrsetreg(&cru->cru_mode_con, DPLL_MODE_MASK,
> +             DPLL_MODE_NORMAL << DPLL_MODE_SHIFT);
> +
> +    return 0;
> +}
> +
> +static int rk3066_clk_configure_cpu(struct rk3066_cru *cru, struct rk3066_grf *grf,
> +                    unsigned int hz)
> +{
> +    static const struct pll_div apll_cfg[] = {
> +        {.nf = 50, .nr = 1, .no = 2},
> +        {.nf = 59, .nr = 1, .no = 1},
> +    };
> +    int div_core_peri, div_aclk_core, cfg;
> +
> +    /*
> +     * We support two possible frequencies, the safe 600MHz
> +     * which will work with default pmic settings and will
> +     * be set to get away from the 24MHz default and
> +     * the maximum of 1.416Ghz, which boards can set if they
> +     * were able to get pmic support for it.
> +     */
> +    switch (hz) {
> +    case APLL_SAFE_HZ:
> +        cfg = 0;
> +        div_core_peri = 1;
> +        div_aclk_core = 3;
> +        break;
> +    case APLL_HZ:
> +        cfg = 1;
> +        div_core_peri = 2;
> +        div_aclk_core = 3;
> +        break;
> +    default:
> +        debug("Unsupported ARMCLK frequency");
> +        return -EINVAL;
> +    }
> +
> +    /* pll enter slow-mode */
> +    rk_clrsetreg(&cru->cru_mode_con, APLL_MODE_MASK,
> +             APLL_MODE_SLOW << APLL_MODE_SHIFT);
> +
> +    rk3066_clk_set_pll(cru, CLK_ARM, &apll_cfg[cfg]);
> +
> +    /* waiting for pll lock */
> +    while (!(readl(&grf->soc_status0) & SOCSTS_APLL_LOCK))
> +        udelay(1);
> +
> +    /* Set divider for peripherals attached to the cpu core. */
> +    rk_clrsetreg(&cru->cru_clksel_con[0],
> +             CORE_PERI_DIV_MASK,
> +             div_core_peri << CORE_PERI_DIV_SHIFT);
> +
> +    /* set up dependent divisor for aclk_core */
> +    rk_clrsetreg(&cru->cru_clksel_con[1],
> +             CORE_ACLK_DIV_MASK,
> +             div_aclk_core << CORE_ACLK_DIV_SHIFT);
> +
> +    /* PLL enter normal-mode */
> +    rk_clrsetreg(&cru->cru_mode_con, APLL_MODE_MASK,
> +             APLL_MODE_NORMAL << APLL_MODE_SHIFT);
> +
> +    return hz;
> +}
> +
> +/* Get pll rate by id */
> +static uint32_t rk3066_clk_pll_get_rate(struct rk3066_cru *cru,
> +                    enum rk_clk_id clk_id)
> +{
> +    u32 nr, no, nf;
> +    u32 con;
> +    int pll_id = rk_pll_id(clk_id);
> +    struct rk3066_pll *pll = &cru->pll[pll_id];
> +    static u8 clk_shift[CLK_COUNT] = {
> +        0xff, APLL_MODE_SHIFT, DPLL_MODE_SHIFT, CPLL_MODE_SHIFT,
> +        GPLL_MODE_SHIFT
> +    };
> +    uint shift;
> +
> +    con = readl(&cru->cru_mode_con);
> +    shift = clk_shift[clk_id];
> +    switch ((con >> shift) & APLL_MODE_MASK >> APLL_MODE_SHIFT) {

Here you can just do

     switch(FIELD_GET(APLL_MODE_MASK, con >> shift)) {

and since the modes are the same for each clock, you may want to do
something like

     PLL_MODE_MASK = GENMASK(1, 0),
     PLL_MODE_SLOW = 0,
     PLL_MODE_NORMAL = 1,
     PLL_MODE_DEEP = 2,
     APLL_MODE_SHIFT = 0,
     DPLL_MODE_SHIFT = 4,
     CPLL_MODE_SHIFT = 8,
     GPLL_MODE_SHIFT = 12,

to make it clearer for readers that each clock has the same fields.

> +    case APLL_MODE_SLOW:
> +        return OSC_HZ;
> +    case APLL_MODE_NORMAL:
> +        /* normal mode */
> +        con = readl(&pll->con0);
> +        no = bitfield_extract_by_mask(con, CLKOD_MASK) + 1;
> +        nr = bitfield_extract_by_mask(con, CLKR_MASK) + 1;
> +        con = readl(&pll->con1);
> +        nf = bitfield_extract_by_mask(con, CLKF_MASK) + 1;
> +
> +        return (OSC_HZ * nf) / (nr * no);
> +    case APLL_MODE_DEEP:
> +    default:

DEEP -> deep sleep mode?

> +        return 32768;
> +    }
> +}
> +
> +static ulong rk3066_clk_mmc_get_clk(struct rk3066_cru *cru, uint gclk_rate,
> +                    int periph)
> +{
> +    uint div;
> +    u32 con;
> +
> +    switch (periph) {
> +    case HCLK_EMMC:
> +    case SCLK_EMMC:
> +        con = readl(&cru->cru_clksel_con[12]);
> +        div = bitfield_extract_by_mask(con, EMMC_DIV_MASK);
> +        break;
> +    case HCLK_SDMMC:
> +    case SCLK_SDMMC:
> +        con = readl(&cru->cru_clksel_con[11]);
> +        div = bitfield_extract_by_mask(con, MMC0_DIV_MASK);
> +        break;
> +    case HCLK_SDIO:
> +    case SCLK_SDIO:
> +        con = readl(&cru->cru_clksel_con[12]);
> +        div = bitfield_extract_by_mask(con, SDIO_DIV_MASK);
> +        break;
> +    default:
> +        return -EINVAL;
> +    }
> +
> +    return DIV_TO_RATE(gclk_rate, div) / 2;
> +}
> +
> +static ulong rk3066_clk_mmc_set_clk(struct rk3066_cru *cru, uint gclk_rate,
> +                    int  periph, uint freq)
> +{
> +    int src_clk_div;
> +
> +    debug("%s: gclk_rate=%u\n", __func__, gclk_rate);
> +    /* mmc clock defaulg div 2 internal, need provide double in cru */
> +    src_clk_div = DIV_ROUND_UP(gclk_rate / 2, freq) - 1;
> +    assert(src_clk_div <= 0x3f);
> +
> +    switch (periph) {
> +    case HCLK_EMMC:
> +    case SCLK_EMMC:
> +        rk_clrsetreg(&cru->cru_clksel_con[12],
> +                 EMMC_DIV_MASK,
> +                 src_clk_div << EMMC_DIV_SHIFT);
> +        break;
> +    case HCLK_SDMMC:
> +    case SCLK_SDMMC:
> +        rk_clrsetreg(&cru->cru_clksel_con[11],
> +                 MMC0_DIV_MASK,
> +                 src_clk_div << MMC0_DIV_SHIFT);
> +        break;
> +    case HCLK_SDIO:
> +    case SCLK_SDIO:
> +        rk_clrsetreg(&cru->cru_clksel_con[12],
> +                 SDIO_DIV_MASK,
> +                 src_clk_div << SDIO_DIV_SHIFT);
> +        break;
> +    default:
> +        return -EINVAL;
> +    }
> +
> +    return rk3066_clk_mmc_get_clk(cru, gclk_rate, periph);
> +}
> +
> +static ulong rk3066_clk_spi_get_clk(struct rk3066_cru *cru, uint gclk_rate,
> +                    int periph)
> +{
> +    uint div;
> +    u32 con;
> +
> +    switch (periph) {
> +    case SCLK_SPI0:
> +        con = readl(&cru->cru_clksel_con[25]);
> +        div = bitfield_extract_by_mask(con, SPI0_DIV_MASK);
> +        break;
> +    case SCLK_SPI1:
> +        con = readl(&cru->cru_clksel_con[25]);
> +        div = bitfield_extract_by_mask(con, SPI1_DIV_MASK);
> +        break;
> +    default:
> +        return -EINVAL;
> +    }
> +
> +    return DIV_TO_RATE(gclk_rate, div);
> +}
> +
> +static ulong rk3066_clk_spi_set_clk(struct rk3066_cru *cru, uint gclk_rate,
> +                    int periph, uint freq)
> +{
> +    int src_clk_div = DIV_ROUND_UP(gclk_rate, freq) - 1;
> +
> +    assert(src_clk_div < 128);
> +    switch (periph) {
> +    case SCLK_SPI0:
> +        assert(src_clk_div <= SPI0_DIV_MASK >> SPI0_DIV_SHIFT);
> +        rk_clrsetreg(&cru->cru_clksel_con[25],
> +                 SPI0_DIV_MASK,
> +                 src_clk_div << SPI0_DIV_SHIFT);
> +        break;
> +    case SCLK_SPI1:
> +        assert(src_clk_div <= SPI1_DIV_MASK >> SPI1_DIV_SHIFT);
> +        rk_clrsetreg(&cru->cru_clksel_con[25],
> +                 SPI1_DIV_MASK,
> +                 src_clk_div << SPI1_DIV_SHIFT);
> +        break;
> +    default:
> +        return -EINVAL;
> +    }
> +
> +    return rk3066_clk_spi_get_clk(cru, gclk_rate, periph);
> +}
> +
> +static ulong rk3066_clk_saradc_get_clk(struct rk3066_cru *cru, int periph)
> +{
> +    u32 div, con;
> +
> +    switch (periph) {
> +    case SCLK_SARADC:
> +        con = readl(&cru->cru_clksel_con[24]);
> +        div = bitfield_extract_by_mask(con, SARADC_DIV_MASK);
> +        break;
> +    case SCLK_TSADC:
> +        con = readl(&cru->cru_clksel_con[34]);
> +        div = bitfield_extract_by_mask(con, TSADC_DIV_MASK);
> +        break;
> +    default:
> +        return -EINVAL;
> +    }
> +    return DIV_TO_RATE(PERI_PCLK_HZ, div);
> +}
> +
> +static ulong rk3066_clk_saradc_set_clk(struct rk3066_cru *cru, uint hz,
> +                       int periph)
> +{
> +    int src_clk_div;
> +
> +    src_clk_div = DIV_ROUND_UP(PERI_PCLK_HZ, hz) - 1;
> +    assert(src_clk_div < 128);
> +
> +    switch (periph) {
> +    case SCLK_SARADC:
> +        rk_clrsetreg(&cru->cru_clksel_con[24],
> +                 SARADC_DIV_MASK,
> +                 src_clk_div << SARADC_DIV_SHIFT);
> +        break;
> +    case SCLK_TSADC:
> +        rk_clrsetreg(&cru->cru_clksel_con[34],
> +                 SARADC_DIV_MASK,
> +                 src_clk_div << SARADC_DIV_SHIFT);
> +        break;
> +    default:
> +        return -EINVAL;
> +    }
> +
> +    return rk3066_clk_saradc_get_clk(cru, periph);
> +}
> +
> +static void rk3066_clk_init(struct rk3066_cru *cru, struct rk3066_grf *grf)
> +{
> +    u32 aclk_div, hclk_div, pclk_div, h2p_div;
> +
> +    /* pll enter slow-mode */
> +    rk_clrsetreg(&cru->cru_mode_con,
> +             GPLL_MODE_MASK |
> +             CPLL_MODE_MASK,
> +             GPLL_MODE_SLOW << GPLL_MODE_SHIFT |
> +             CPLL_MODE_SLOW << CPLL_MODE_SHIFT);
> +
> +    /* init pll */
> +    rk3066_clk_set_pll(cru, CLK_GENERAL, &gpll_init_cfg);
> +    rk3066_clk_set_pll(cru, CLK_CODEC, &cpll_init_cfg);
> +
> +    /* waiting for pll lock */
> +    while ((readl(&grf->soc_status0) &
> +        (SOCSTS_CPLL_LOCK | SOCSTS_GPLL_LOCK)) !=
> +           (SOCSTS_CPLL_LOCK | SOCSTS_GPLL_LOCK))
> +        udelay(1);
> +
> +    /*
> +     * cpu clock pll source selection and
> +     * reparent aclk_cpu_pre from apll to gpll
> +     * set up dependent divisors for PCLK/HCLK and ACLK clocks.
> +     */
> +    aclk_div = DIV_ROUND_UP(GPLL_HZ, CPU_ACLK_HZ) - 1;
> +    assert((aclk_div + 1) * CPU_ACLK_HZ == GPLL_HZ && aclk_div <=0x1f);
> +
> +    rk_clrsetreg(&cru->cru_clksel_con[0],
> +             CPU_ACLK_PLL_MASK |
> +             A9_CPU_DIV_MASK,
> +             CPU_ACLK_PLL_SELECT_GPLL << CPU_ACLK_PLL_SHIFT |
> +             aclk_div << A9_CPU_DIV_SHIFT);
> +
> +    hclk_div = ilog2(CPU_ACLK_HZ / CPU_HCLK_HZ);
> +    assert((1 << hclk_div) * CPU_HCLK_HZ == CPU_ACLK_HZ && hclk_div < 0x3);
> +    pclk_div = ilog2(CPU_ACLK_HZ / CPU_PCLK_HZ);
> +    assert((1 << pclk_div) * CPU_PCLK_HZ == CPU_ACLK_HZ && pclk_div < 0x4);
> +    h2p_div = ilog2(CPU_HCLK_HZ / CPU_H2P_HZ);
> +    assert((1 << h2p_div) * CPU_H2P_HZ == CPU_HCLK_HZ && pclk_div < 0x3);
> +
> +    rk_clrsetreg(&cru->cru_clksel_con[1],
> +             AHB2APB_DIV_MASK |
> +             CPU_PCLK_DIV_MASK |
> +             CPU_HCLK_DIV_MASK,
> +             h2p_div << AHB2APB_DIV_SHIFT |
> +             pclk_div << CPU_PCLK_DIV_SHIFT |
> +             hclk_div << CPU_HCLK_DIV_SHIFT);
> +
> +    /*
> +     * peri clock pll source selection and
> +     * set up dependent divisors for PCLK/HCLK and ACLK clocks.
> +     */
> +    aclk_div = GPLL_HZ / PERI_ACLK_HZ - 1;
> +    assert((aclk_div + 1) * PERI_ACLK_HZ == GPLL_HZ && aclk_div < 0x1f);
> +
> +    hclk_div = ilog2(PERI_ACLK_HZ / PERI_HCLK_HZ);
> +    assert((1 << hclk_div) * PERI_HCLK_HZ ==
> +           PERI_ACLK_HZ && (hclk_div < 0x4));
> +
> +    pclk_div = ilog2(PERI_ACLK_HZ / PERI_PCLK_HZ);
> +    assert((1 << pclk_div) * PERI_PCLK_HZ ==
> +           PERI_ACLK_HZ && (pclk_div < 0x4));
> +
> +    rk_clrsetreg(&cru->cru_clksel_con[10],
> +             PERI_PCLK_DIV_MASK |
> +             PERI_HCLK_DIV_MASK |
> +             PERI_ACLK_DIV_MASK,
> +             PERI_SEL_GPLL << PERI_SEL_PLL_SHIFT |
> +             pclk_div << PERI_PCLK_DIV_SHIFT |
> +             hclk_div << PERI_HCLK_DIV_SHIFT |
> +             aclk_div << PERI_ACLK_DIV_SHIFT);
> +
> +    /* PLL enter normal-mode */
> +    rk_clrsetreg(&cru->cru_mode_con,
> +             GPLL_MODE_MASK |
> +             CPLL_MODE_MASK,
> +             GPLL_MODE_NORMAL << GPLL_MODE_SHIFT |
> +             CPLL_MODE_NORMAL << CPLL_MODE_SHIFT);
> +
> +    rk3066_clk_mmc_set_clk(cru, PERI_HCLK_HZ, HCLK_SDMMC, 16000000);
> +}
> +
> +static ulong rk3066_clk_get_rate(struct clk *clk)
> +{
> +    struct rk3066_clk_priv *priv = dev_get_priv(clk->dev);
> +    ulong new_rate, gclk_rate;
> +
> +    gclk_rate = rk3066_clk_pll_get_rate(priv->cru, CLK_GENERAL);
> +    switch (clk->id) {
> +    case 1 ... 4:
> +        new_rate = rk3066_clk_pll_get_rate(priv->cru, clk->id);
> +        break;
> +    case HCLK_EMMC:
> +    case HCLK_SDMMC:
> +    case HCLK_SDIO:
> +    case SCLK_EMMC:
> +    case SCLK_SDMMC:
> +    case SCLK_SDIO:
> +        new_rate = rk3066_clk_mmc_get_clk(priv->cru, PERI_HCLK_HZ,
> +                          clk->id);
> +        break;
> +    case SCLK_SPI0:
> +    case SCLK_SPI1:
> +        new_rate = rk3066_clk_spi_get_clk(priv->cru, PERI_PCLK_HZ,
> +                          clk->id);
> +        break;
> +    case PCLK_I2C0:
> +    case PCLK_I2C1:
> +    case PCLK_I2C2:
> +    case PCLK_I2C3:
> +    case PCLK_I2C4:
> +        return gclk_rate;
> +    case SCLK_SARADC:
> +    case SCLK_TSADC:
> +        new_rate = rk3066_clk_saradc_get_clk(priv->cru, clk->id);
> +        break;
> +    default:
> +        return -ENOENT;
> +    }
> +
> +    return new_rate;
> +}
> +
> +static ulong rk3066_clk_set_rate(struct clk *clk, ulong rate)
> +{
> +    struct rk3066_clk_priv *priv = dev_get_priv(clk->dev);
> +    struct rk3066_cru *cru = priv->cru;
> +    ulong new_rate;
> +
> +    switch (clk->id) {
> +    case PLL_APLL:
> +        new_rate = rk3066_clk_configure_cpu(priv->cru, priv->grf, rate);
> +        break;
> +    case CLK_DDR:
> +        new_rate = rk3066_clk_configure_ddr(priv->cru, priv->grf, rate);
> +        break;
> +    case HCLK_EMMC:
> +    case HCLK_SDMMC:
> +    case HCLK_SDIO:
> +    case SCLK_EMMC:
> +    case SCLK_SDMMC:
> +    case SCLK_SDIO:
> +        new_rate = rk3066_clk_mmc_set_clk(cru, PERI_HCLK_HZ,
> +                          clk->id, rate);
> +        break;
> +    case SCLK_SPI0:
> +    case SCLK_SPI1:
> +        new_rate = rk3066_clk_spi_set_clk(cru, PERI_PCLK_HZ,
> +                          clk->id, rate);
> +        break;
> +    case SCLK_SARADC:
> +    case SCLK_TSADC:
> +        new_rate = rk3066_clk_saradc_set_clk(cru, rate, clk->id);
> +        break;
> +    default:
> +        return -ENOENT;
> +    }
> +
> +    return new_rate;
> +}
> +
> +static int rk3066_clk_enable(struct clk *clk)
> +{
> +    struct rk3066_clk_priv *priv = dev_get_priv(clk->dev);
> +
> +    switch (clk->id) {
> +    case HCLK_NANDC0:
> +        rk_clrreg(&priv->cru->cru_clkgate_con[5], BIT(9));
> +        break;
> +    case HCLK_SDMMC:
> +        rk_clrreg(&priv->cru->cru_clkgate_con[5], BIT(10));
> +        break;
> +    case HCLK_SDIO:
> +        rk_clrreg(&priv->cru->cru_clkgate_con[5], BIT(11));
> +        break;
> +    }
> +
> +    return 0;
> +}
> +
> +static int rk3066_clk_disable(struct clk *clk)
> +{
> +    struct rk3066_clk_priv *priv = dev_get_priv(clk->dev);
> +
> +    switch (clk->id) {
> +    case HCLK_NANDC0:
> +        rk_setreg(&priv->cru->cru_clkgate_con[5], BIT(9));
> +        break;
> +    case HCLK_SDMMC:
> +        rk_setreg(&priv->cru->cru_clkgate_con[5], BIT(10));
> +        break;
> +    case HCLK_SDIO:
> +        rk_setreg(&priv->cru->cru_clkgate_con[5], BIT(11));
> +        break;
> +    }
> +
> +    return 0;
> +}
> +
> +static struct clk_ops rk3066_clk_ops = {
> +    .disable    = rk3066_clk_disable,
> +    .enable    = rk3066_clk_enable,
> +    .get_rate    = rk3066_clk_get_rate,
> +    .set_rate    = rk3066_clk_set_rate,
> +};
> +
> +static int rk3066_clk_of_to_plat(struct udevice *dev)
> +{
> +    if (CONFIG_IS_ENABLED(OF_REAL)) {
> +        struct rk3066_clk_priv *priv = dev_get_priv(dev);
> +
> +        priv->cru = dev_read_addr_ptr(dev);
> +    }
> +
> +    return 0;
> +}
> +
> +static int rk3066_clk_probe(struct udevice *dev)
> +{
> +    struct rk3066_clk_priv *priv = dev_get_priv(dev);
> +
> +    priv->grf = syscon_get_first_range(ROCKCHIP_SYSCON_GRF);
> +    if (IS_ERR(priv->grf))
> +        return PTR_ERR(priv->grf);
> +
> +#if CONFIG_IS_ENABLED(OF_PLATDATA)
> +    struct rk3066_clk_plat *plat = dev_get_plat(dev);
> +
> +    priv->cru = map_sysmem(plat->dtd.reg[0], plat->dtd.reg[1]);
> +#endif
> +
> +    if (IS_ENABLED(CONFIG_TPL_BUILD)) {
> +        rk3066_clk_init(priv->cru, priv->grf);
> +
> +        /* Init CPU frequency */
> +        rk3066_clk_configure_cpu(priv->cru, priv->grf, APLL_SAFE_HZ);
> +    }
> +
> +    return 0;
> +}
> +
> +static int rk3066_clk_bind(struct udevice *dev)
> +{
> +    int ret;
> +    struct udevice *sys_child;
> +    struct sysreset_reg *priv;
> +
> +    /* The reset driver does not have a device node, so bind it here */
> +    ret = device_bind_driver(dev, "rockchip_sysreset", "sysreset",
> +                 &sys_child);

You can avoid doing a lookup at runtime by doing

     ret = device_bind(dev, DM_DRIVER_GET(sysreset_rockchip), "sysreset",
               NULL, ofnode_null(), &sys_child);

> +    if (ret) {
> +        debug("Warning: No sysreset driver: ret=%d\n", ret);

Use dev_dbg here

> +    } else {
> +        priv = malloc(sizeof(struct sysreset_reg));
> +        priv->glb_srst_fst_value = offsetof(struct rk3066_cru,
> +                            cru_glb_srst_fst_value);
> +        priv->glb_srst_snd_value = offsetof(struct rk3066_cru,
> +                            cru_glb_srst_snd_value);
> +        dev_set_priv(sys_child, priv);

this driver should really use platdata for this... but it is out of
scope for your patch

> +    }
> +
> +    if (CONFIG_IS_ENABLED(RESET_ROCKCHIP)) {
> +        ret = offsetof(struct rk3066_cru, cru_softrst_con[0]);

Please use a different variable name (or just inline this)

> +        ret = rockchip_reset_bind(dev, ret, 9);

Can't we do the same thing here as above?

> +        if (ret)
> +            debug("Warning: software reset driver bind failed\n");
> +    }
> +
> +    return 0;
> +}
> +
> +static const struct udevice_id rk3066_clk_ids[] = {
> +    { .compatible = "rockchip,rk3066a-cru" },
> +    { }
> +};
> +
> +U_BOOT_DRIVER(rockchip_rk3066a_cru) = {
> +    .name        = "rockchip_rk3066a_cru",
> +    .id        = UCLASS_CLK,
> +    .ops        = &rk3066_clk_ops,
> +    .probe        = rk3066_clk_probe,
> +    .bind        = rk3066_clk_bind,
> +    .of_match    = rk3066_clk_ids,
> +    .of_to_plat    = rk3066_clk_of_to_plat,
> +    .priv_auto    = sizeof(struct rk3066_clk_priv),
> +    .plat_auto    = sizeof(struct rk3066_clk_plat),
> +};
>

Other than the above comments, this looks good.

--Sean
diff mbox series

Patch

diff --git a/arch/arm/include/asm/arch-rockchip/cru_rk3066.h b/arch/arm/include/asm/arch-rockchip/cru_rk3066.h
new file mode 100644
index 00000000..711366d5
--- /dev/null
+++ b/arch/arm/include/asm/arch-rockchip/cru_rk3066.h
@@ -0,0 +1,203 @@ 
+/* SPDX-License-Identifier: GPL-2.0+ */
+/*
+ * (C) Copyright 2021 Paweł Jarosz <paweljarosz3691@gmail.com>
+ */
+
+#ifndef _ASM_ARCH_CRU_RK3066_H
+#define _ASM_ARCH_CRU_RK3066_H
+
+#define OSC_HZ		(24 * 1000 * 1000)
+
+#define APLL_HZ	(1416 * 1000000)
+#define APLL_SAFE_HZ	(600 * 1000000)
+#define GPLL_HZ	(594 * 1000000)
+#define CPLL_HZ	(384 * 1000000)
+
+/* The SRAM is clocked off aclk_cpu, so we want to max it out for boot speed */
+#define CPU_ACLK_HZ	297000000
+#define CPU_HCLK_HZ	148500000
+#define CPU_PCLK_HZ	74250000
+#define CPU_H2P_HZ	74250000
+
+#define PERI_ACLK_HZ	148500000
+#define PERI_HCLK_HZ	148500000
+#define PERI_PCLK_HZ	74250000
+
+/* Private data for the clock driver - used by rockchip_get_cru() */
+struct rk3066_clk_priv {
+	struct rk3066_grf *grf;
+	struct rk3066_cru *cru;
+	ulong rate;
+	bool has_bwadj;
+};
+
+struct rk3066_cru {
+	struct rk3066_pll {
+		u32 con0;
+		u32 con1;
+		u32 con2;
+		u32 con3;
+	} pll[4];
+	u32 cru_mode_con;
+	u32 cru_clksel_con[35];
+	u32 cru_clkgate_con[10];
+	u32 reserved1[2];
+	u32 cru_glb_srst_fst_value;
+	u32 cru_glb_srst_snd_value;
+	u32 reserved2[2];
+	u32 cru_softrst_con[9];
+	u32 cru_misc_con;
+	u32 reserved3[2];
+	u32 cru_glb_cnt_th;
+};
+
+check_member(rk3066_cru, cru_glb_cnt_th, 0x0140);
+
+/* CRU_CLKSEL0_CON */
+enum {
+	/* a9_core_div: core = core_src / (a9_core_div + 1) */
+	A9_CORE_DIV_SHIFT	= 9,
+	A9_CORE_DIV_MASK	= 0x1f << A9_CORE_DIV_SHIFT,
+	CORE_PLL_SHIFT		= 8,
+	CORE_PLL_MASK		= 1 << CORE_PLL_SHIFT,
+	CORE_PLL_SELECT_APLL	= 0,
+	CORE_PLL_SELECT_GPLL,
+
+	/* core peri div: core:core_peri = 2:1, 4:1, 8:1 or 16:1 */
+	CORE_PERI_DIV_SHIFT	= 6,
+	CORE_PERI_DIV_MASK	= 3 << CORE_PERI_DIV_SHIFT,
+
+	/* aclk_cpu pll selection */
+	CPU_ACLK_PLL_SHIFT	= 5,
+	CPU_ACLK_PLL_MASK	= 1 << CPU_ACLK_PLL_SHIFT,
+	CPU_ACLK_PLL_SELECT_APLL	= 0,
+	CPU_ACLK_PLL_SELECT_GPLL,
+
+	/* a9_cpu_div: aclk_cpu = cpu_src / (a9_cpu_div + 1) */
+	A9_CPU_DIV_SHIFT	= 0,
+	A9_CPU_DIV_MASK	= 0x1f << A9_CPU_DIV_SHIFT,
+};
+
+/* CRU_CLKSEL1_CON */
+enum {
+	/* ahb2apb_pclk_div: hclk_cpu:pclk_cpu = 1:1, 2:1 or 4:1 */
+	AHB2APB_DIV_SHIFT	= 14,
+	AHB2APB_DIV_MASK	= 3 << AHB2APB_DIV_SHIFT,
+
+	/* cpu_pclk_div: aclk_cpu:pclk_cpu = 1:1, 2:1, 4:1 or 8:1 */
+	CPU_PCLK_DIV_SHIFT	= 12,
+	CPU_PCLK_DIV_MASK	= 3 << CPU_PCLK_DIV_SHIFT,
+
+	/* cpu_hclk_div: aclk_cpu:hclk_cpu = 1:1, 2:1 or 4:1 */
+	CPU_HCLK_DIV_SHIFT	= 8,
+	CPU_HCLK_DIV_MASK	= 3 << CPU_HCLK_DIV_SHIFT,
+
+	/* core_aclk_div: cire:aclk_core = 1:1, 2:1, 3:1, 4:1 or 8:1 */
+	CORE_ACLK_DIV_SHIFT	= 3,
+	CORE_ACLK_DIV_MASK	= 7 << CORE_ACLK_DIV_SHIFT,
+};
+
+/* CRU_CLKSEL10_CON */
+enum {
+	PERI_SEL_PLL_SHIFT	= 15,
+	PERI_SEL_PLL_MASK	= 1 << PERI_SEL_PLL_SHIFT,
+	PERI_SEL_CPLL		= 0,
+	PERI_SEL_GPLL,
+
+	/* peri pclk div: aclk_bus:pclk_bus = 1:1, 2:1, 4:1 or 8:1 */
+	PERI_PCLK_DIV_SHIFT	= 12,
+	PERI_PCLK_DIV_MASK	= 3 << PERI_PCLK_DIV_SHIFT,
+
+	/* peripheral bus hclk div:aclk_bus: hclk_bus = 1:1, 2:1 or 4:1 */
+	PERI_HCLK_DIV_SHIFT	= 8,
+	PERI_HCLK_DIV_MASK	= 3 << PERI_HCLK_DIV_SHIFT,
+
+	/* peri aclk div: aclk_peri = periph_src / (peri_aclk_div + 1) */
+	PERI_ACLK_DIV_SHIFT	= 0,
+	PERI_ACLK_DIV_MASK	= 0x1f << PERI_ACLK_DIV_SHIFT,
+};
+
+/* CRU_CLKSEL11_CON */
+enum {
+	MMC0_DIV_SHIFT		= 0,
+	MMC0_DIV_MASK		= 0x3f << MMC0_DIV_SHIFT,
+};
+
+/* CRU_CLKSEL12_CON */
+enum {
+	UART_PLL_SHIFT		= 15,
+	UART_PLL_MASK		= 1 << UART_PLL_SHIFT,
+	UART_PLL_SELECT_GENERAL	= 0,
+	UART_PLL_SELECT_CODEC,
+
+	EMMC_DIV_SHIFT		= 8,
+	EMMC_DIV_MASK		= 0x3f << EMMC_DIV_SHIFT,
+
+	SDIO_DIV_SHIFT		= 0,
+	SDIO_DIV_MASK		= 0x3f << SDIO_DIV_SHIFT,
+};
+
+/* CRU_CLKSEL24_CON */
+enum {
+	SARADC_DIV_SHIFT	= 8,
+	SARADC_DIV_MASK	= 0xff << SARADC_DIV_SHIFT,
+};
+
+/* CRU_CLKSEL25_CON */
+enum {
+	SPI1_DIV_SHIFT		= 8,
+	SPI1_DIV_MASK		= 0x7f << SPI1_DIV_SHIFT,
+
+	SPI0_DIV_SHIFT		= 0,
+	SPI0_DIV_MASK		= 0x7f << SPI0_DIV_SHIFT,
+};
+
+/* CRU_CLKSEL34_CON */
+enum {
+	TSADC_DIV_SHIFT	= 0,
+	TSADC_DIV_MASK		= 0xffff << TSADC_DIV_SHIFT,
+};
+
+/* CRU_MODE_CON */
+enum {
+	GPLL_MODE_SHIFT	= 12,
+	GPLL_MODE_MASK		= 3 << GPLL_MODE_SHIFT,
+	GPLL_MODE_SLOW		= 0,
+	GPLL_MODE_NORMAL,
+	GPLL_MODE_DEEP,
+
+	CPLL_MODE_SHIFT		= 8,
+	CPLL_MODE_MASK		= 3 << CPLL_MODE_SHIFT,
+	CPLL_MODE_SLOW		= 0,
+	CPLL_MODE_NORMAL,
+	CPLL_MODE_DEEP,
+
+	DPLL_MODE_SHIFT		= 4,
+	DPLL_MODE_MASK		= 3 << DPLL_MODE_SHIFT,
+	DPLL_MODE_SLOW		= 0,
+	DPLL_MODE_NORMAL,
+	DPLL_MODE_DEEP,
+
+	APLL_MODE_SHIFT		= 0,
+	APLL_MODE_MASK		= 3 << APLL_MODE_SHIFT,
+	APLL_MODE_SLOW		= 0,
+	APLL_MODE_NORMAL,
+	APLL_MODE_DEEP,
+};
+
+/* CRU_APLL_CON0 */
+enum {
+	CLKR_SHIFT		= 8,
+	CLKR_MASK		= 0x3f << CLKR_SHIFT,
+
+	CLKOD_SHIFT		= 0,
+	CLKOD_MASK		= 0x3f << CLKOD_SHIFT,
+};
+
+/* CRU_APLL_CON1 */
+enum {
+	CLKF_SHIFT		= 0,
+	CLKF_MASK		= 0x1fff << CLKF_SHIFT,
+};
+
+#endif
diff --git a/drivers/clk/rockchip/Makefile b/drivers/clk/rockchip/Makefile
index 913f611a..a72d8fe5 100644
--- a/drivers/clk/rockchip/Makefile
+++ b/drivers/clk/rockchip/Makefile
@@ -6,6 +6,7 @@ 
 obj-y += clk_pll.o
 obj-$(CONFIG_ROCKCHIP_PX30) += clk_px30.o
 obj-$(CONFIG_ROCKCHIP_RK3036) += clk_rk3036.o
+obj-$(CONFIG_ROCKCHIP_RK3066) += clk_rk3066.o
 obj-$(CONFIG_ROCKCHIP_RK3128) += clk_rk3128.o
 obj-$(CONFIG_ROCKCHIP_RK3188) += clk_rk3188.o
 obj-$(CONFIG_ROCKCHIP_RK322X) += clk_rk322x.o
diff --git a/drivers/clk/rockchip/clk_rk3066.c b/drivers/clk/rockchip/clk_rk3066.c
new file mode 100644
index 00000000..3b7de51c
--- /dev/null
+++ b/drivers/clk/rockchip/clk_rk3066.c
@@ -0,0 +1,700 @@ 
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * (C) Copyright 2015 Google, Inc
+ * (C) Copyright 2016 Heiko Stuebner <heiko@sntech.de>
+ */
+
+#include <bitfield.h>
+#include <common.h>
+#include <clk-uclass.h>
+#include <dm.h>
+#include <dt-structs.h>
+#include <errno.h>
+#include <log.h>
+#include <malloc.h>
+#include <mapmem.h>
+#include <syscon.h>
+#include <asm/io.h>
+#include <asm/arch-rockchip/clock.h>
+#include <asm/arch-rockchip/cru_rk3066.h>
+#include <asm/arch-rockchip/grf_rk3066.h>
+#include <asm/arch-rockchip/hardware.h>
+#include <dt-bindings/clock/rk3066a-cru.h>
+#include <dm/device-internal.h>
+#include <dm/lists.h>
+#include <dm/uclass-internal.h>
+#include <linux/delay.h>
+#include <linux/err.h>
+#include <linux/log2.h>
+#include <linux/stringify.h>
+
+struct rk3066_clk_plat {
+#if CONFIG_IS_ENABLED(OF_PLATDATA)
+	struct dtd_rockchip_rk3066a_cru dtd;
+#endif
+};
+
+struct pll_div {
+	u32 nr;
+	u32 nf;
+	u32 no;
+};
+
+enum {
+	VCO_MAX_HZ	= 1416U * 1000000,
+	VCO_MIN_HZ	= 300 * 1000000,
+	OUTPUT_MAX_HZ	= 1416U * 1000000,
+	OUTPUT_MIN_HZ	= 30 * 1000000,
+	FREF_MAX_HZ	= 1416U * 1000000,
+	FREF_MIN_HZ	= 30 * 1000,
+};
+
+enum {
+	/* PLL CON0 */
+	PLL_OD_MASK		= GENMASK(3, 0),
+
+	/* PLL CON1 */
+	PLL_NF_MASK		= GENMASK(12, 0),
+
+	/* PLL CON2 */
+	PLL_BWADJ_MASK		= GENMASK(11, 0),
+
+	/* PLL CON3 */
+	PLL_RESET_SHIFT		= 5,
+
+	/* GRF_SOC_STATUS0 */
+	SOCSTS_DPLL_LOCK	= BIT(4),
+	SOCSTS_APLL_LOCK	= BIT(5),
+	SOCSTS_CPLL_LOCK	= BIT(6),
+	SOCSTS_GPLL_LOCK	= BIT(7),
+};
+
+#define DIV_TO_RATE(input_rate, div)	((input_rate) / ((div) + 1))
+
+#define PLL_DIVISORS(hz, _nr, _no) {\
+	.nr = _nr, .nf = (u32)((u64)hz * _nr * _no / OSC_HZ), .no = _no};\
+	_Static_assert(((u64)hz * _nr * _no / OSC_HZ) * OSC_HZ /\
+		       (_nr * _no) == hz, #hz "Hz cannot be hit with PLL "\
+		       "divisors on line " __stringify(__LINE__))
+
+/* Keep divisors as low as possible to reduce jitter and power usage */
+static const struct pll_div gpll_init_cfg = PLL_DIVISORS(GPLL_HZ, 2, 2);
+static const struct pll_div cpll_init_cfg = PLL_DIVISORS(CPLL_HZ, 1, 2);
+
+static int rk3066_clk_set_pll(struct rk3066_cru *cru, enum rk_clk_id clk_id,
+			      const struct pll_div *div)
+{
+	int pll_id = rk_pll_id(clk_id);
+	struct rk3066_pll *pll = &cru->pll[pll_id];
+	/* All PLLs have same VCO and output frequency range restrictions. */
+	uint vco_hz = OSC_HZ / 1000 * div->nf / div->nr * 1000;
+	uint output_hz = vco_hz / div->no;
+
+	debug("PLL at %x: nf=%d, nr=%d, no=%d, vco=%u Hz, output=%u Hz\n",
+	      (uint)pll, div->nf, div->nr, div->no, vco_hz, output_hz);
+	assert(vco_hz >= VCO_MIN_HZ && vco_hz <= VCO_MAX_HZ &&
+	       output_hz >= OUTPUT_MIN_HZ && output_hz <= OUTPUT_MAX_HZ &&
+	       (div->no == 1 || !(div->no % 2)));
+
+	/* enter reset */
+	rk_setreg(&pll->con3, BIT(PLL_RESET_SHIFT));
+
+	rk_clrsetreg(&pll->con0,
+		     CLKR_MASK | PLL_OD_MASK,
+		     ((div->nr - 1) << CLKR_SHIFT) | (div->no - 1));
+	rk_clrsetreg(&pll->con1, CLKF_MASK, div->nf - 1);
+
+	rk_clrsetreg(&pll->con2, PLL_BWADJ_MASK, (div->nf >> 1) - 1);
+
+	/* return from reset */
+	rk_clrreg(&pll->con3, BIT(PLL_RESET_SHIFT));
+
+	return 0;
+}
+
+static int rk3066_clk_configure_ddr(struct rk3066_cru *cru, struct rk3066_grf *grf,
+				    unsigned int hz)
+{
+	static const struct pll_div dpll_cfg[] = {
+		{.nf = 25, .nr = 2, .no = 1},
+		{.nf = 400, .nr = 9, .no = 2},
+		{.nf = 500, .nr = 9, .no = 2},
+		{.nf = 100, .nr = 3, .no = 1},
+	};
+	int cfg;
+
+	switch (hz) {
+	case 300000000:
+		cfg = 0;
+		break;
+	case 533000000:	/* actually 533.3P MHz */
+		cfg = 1;
+		break;
+	case 666000000:	/* actually 666.6P MHz */
+		cfg = 2;
+		break;
+	case 800000000:
+		cfg = 3;
+		break;
+	default:
+		debug("Unsupported SDRAM frequency");
+		return -EINVAL;
+	}
+
+	/* pll enter slow-mode */
+	rk_clrsetreg(&cru->cru_mode_con, DPLL_MODE_MASK,
+		     DPLL_MODE_SLOW << DPLL_MODE_SHIFT);
+
+	rk3066_clk_set_pll(cru, CLK_DDR, &dpll_cfg[cfg]);
+
+	/* wait for pll lock */
+	while (!(readl(&grf->soc_status0) & SOCSTS_DPLL_LOCK))
+		udelay(1);
+
+	/* PLL enter normal-mode */
+	rk_clrsetreg(&cru->cru_mode_con, DPLL_MODE_MASK,
+		     DPLL_MODE_NORMAL << DPLL_MODE_SHIFT);
+
+	return 0;
+}
+
+static int rk3066_clk_configure_cpu(struct rk3066_cru *cru, struct rk3066_grf *grf,
+				    unsigned int hz)
+{
+	static const struct pll_div apll_cfg[] = {
+		{.nf = 50, .nr = 1, .no = 2},
+		{.nf = 59, .nr = 1, .no = 1},
+	};
+	int div_core_peri, div_aclk_core, cfg;
+
+	/*
+	 * We support two possible frequencies, the safe 600MHz
+	 * which will work with default pmic settings and will
+	 * be set to get away from the 24MHz default and
+	 * the maximum of 1.416Ghz, which boards can set if they
+	 * were able to get pmic support for it.
+	 */
+	switch (hz) {
+	case APLL_SAFE_HZ:
+		cfg = 0;
+		div_core_peri = 1;
+		div_aclk_core = 3;
+		break;
+	case APLL_HZ:
+		cfg = 1;
+		div_core_peri = 2;
+		div_aclk_core = 3;
+		break;
+	default:
+		debug("Unsupported ARMCLK frequency");
+		return -EINVAL;
+	}
+
+	/* pll enter slow-mode */
+	rk_clrsetreg(&cru->cru_mode_con, APLL_MODE_MASK,
+		     APLL_MODE_SLOW << APLL_MODE_SHIFT);
+
+	rk3066_clk_set_pll(cru, CLK_ARM, &apll_cfg[cfg]);
+
+	/* waiting for pll lock */
+	while (!(readl(&grf->soc_status0) & SOCSTS_APLL_LOCK))
+		udelay(1);
+
+	/* Set divider for peripherals attached to the cpu core. */
+	rk_clrsetreg(&cru->cru_clksel_con[0],
+		     CORE_PERI_DIV_MASK,
+		     div_core_peri << CORE_PERI_DIV_SHIFT);
+
+	/* set up dependent divisor for aclk_core */
+	rk_clrsetreg(&cru->cru_clksel_con[1],
+		     CORE_ACLK_DIV_MASK,
+		     div_aclk_core << CORE_ACLK_DIV_SHIFT);
+
+	/* PLL enter normal-mode */
+	rk_clrsetreg(&cru->cru_mode_con, APLL_MODE_MASK,
+		     APLL_MODE_NORMAL << APLL_MODE_SHIFT);
+
+	return hz;
+}
+
+/* Get pll rate by id */
+static uint32_t rk3066_clk_pll_get_rate(struct rk3066_cru *cru,
+					enum rk_clk_id clk_id)
+{
+	u32 nr, no, nf;
+	u32 con;
+	int pll_id = rk_pll_id(clk_id);
+	struct rk3066_pll *pll = &cru->pll[pll_id];
+	static u8 clk_shift[CLK_COUNT] = {
+		0xff, APLL_MODE_SHIFT, DPLL_MODE_SHIFT, CPLL_MODE_SHIFT,
+		GPLL_MODE_SHIFT
+	};
+	uint shift;
+
+	con = readl(&cru->cru_mode_con);
+	shift = clk_shift[clk_id];
+	switch ((con >> shift) & APLL_MODE_MASK >> APLL_MODE_SHIFT) {
+	case APLL_MODE_SLOW:
+		return OSC_HZ;
+	case APLL_MODE_NORMAL:
+		/* normal mode */
+		con = readl(&pll->con0);
+		no = bitfield_extract_by_mask(con, CLKOD_MASK) + 1;
+		nr = bitfield_extract_by_mask(con, CLKR_MASK) + 1;
+		con = readl(&pll->con1);
+		nf = bitfield_extract_by_mask(con, CLKF_MASK) + 1;
+
+		return (OSC_HZ * nf) / (nr * no);
+	case APLL_MODE_DEEP:
+	default:
+		return 32768;
+	}
+}
+
+static ulong rk3066_clk_mmc_get_clk(struct rk3066_cru *cru, uint gclk_rate,
+				    int periph)
+{
+	uint div;
+	u32 con;
+
+	switch (periph) {
+	case HCLK_EMMC:
+	case SCLK_EMMC:
+		con = readl(&cru->cru_clksel_con[12]);
+		div = bitfield_extract_by_mask(con, EMMC_DIV_MASK);
+		break;
+	case HCLK_SDMMC:
+	case SCLK_SDMMC:
+		con = readl(&cru->cru_clksel_con[11]);
+		div = bitfield_extract_by_mask(con, MMC0_DIV_MASK);
+		break;
+	case HCLK_SDIO:
+	case SCLK_SDIO:
+		con = readl(&cru->cru_clksel_con[12]);
+		div = bitfield_extract_by_mask(con, SDIO_DIV_MASK);
+		break;
+	default:
+		return -EINVAL;
+	}
+
+	return DIV_TO_RATE(gclk_rate, div) / 2;
+}
+
+static ulong rk3066_clk_mmc_set_clk(struct rk3066_cru *cru, uint gclk_rate,
+				    int  periph, uint freq)
+{
+	int src_clk_div;
+
+	debug("%s: gclk_rate=%u\n", __func__, gclk_rate);
+	/* mmc clock defaulg div 2 internal, need provide double in cru */
+	src_clk_div = DIV_ROUND_UP(gclk_rate / 2, freq) - 1;
+	assert(src_clk_div <= 0x3f);
+
+	switch (periph) {
+	case HCLK_EMMC:
+	case SCLK_EMMC:
+		rk_clrsetreg(&cru->cru_clksel_con[12],
+			     EMMC_DIV_MASK,
+			     src_clk_div << EMMC_DIV_SHIFT);
+		break;
+	case HCLK_SDMMC:
+	case SCLK_SDMMC:
+		rk_clrsetreg(&cru->cru_clksel_con[11],
+			     MMC0_DIV_MASK,
+			     src_clk_div << MMC0_DIV_SHIFT);
+		break;
+	case HCLK_SDIO:
+	case SCLK_SDIO:
+		rk_clrsetreg(&cru->cru_clksel_con[12],
+			     SDIO_DIV_MASK,
+			     src_clk_div << SDIO_DIV_SHIFT);
+		break;
+	default:
+		return -EINVAL;
+	}
+
+	return rk3066_clk_mmc_get_clk(cru, gclk_rate, periph);
+}
+
+static ulong rk3066_clk_spi_get_clk(struct rk3066_cru *cru, uint gclk_rate,
+				    int periph)
+{
+	uint div;
+	u32 con;
+
+	switch (periph) {
+	case SCLK_SPI0:
+		con = readl(&cru->cru_clksel_con[25]);
+		div = bitfield_extract_by_mask(con, SPI0_DIV_MASK);
+		break;
+	case SCLK_SPI1:
+		con = readl(&cru->cru_clksel_con[25]);
+		div = bitfield_extract_by_mask(con, SPI1_DIV_MASK);
+		break;
+	default:
+		return -EINVAL;
+	}
+
+	return DIV_TO_RATE(gclk_rate, div);
+}
+
+static ulong rk3066_clk_spi_set_clk(struct rk3066_cru *cru, uint gclk_rate,
+				    int periph, uint freq)
+{
+	int src_clk_div = DIV_ROUND_UP(gclk_rate, freq) - 1;
+
+	assert(src_clk_div < 128);
+	switch (periph) {
+	case SCLK_SPI0:
+		assert(src_clk_div <= SPI0_DIV_MASK >> SPI0_DIV_SHIFT);
+		rk_clrsetreg(&cru->cru_clksel_con[25],
+			     SPI0_DIV_MASK,
+			     src_clk_div << SPI0_DIV_SHIFT);
+		break;
+	case SCLK_SPI1:
+		assert(src_clk_div <= SPI1_DIV_MASK >> SPI1_DIV_SHIFT);
+		rk_clrsetreg(&cru->cru_clksel_con[25],
+			     SPI1_DIV_MASK,
+			     src_clk_div << SPI1_DIV_SHIFT);
+		break;
+	default:
+		return -EINVAL;
+	}
+
+	return rk3066_clk_spi_get_clk(cru, gclk_rate, periph);
+}
+
+static ulong rk3066_clk_saradc_get_clk(struct rk3066_cru *cru, int periph)
+{
+	u32 div, con;
+
+	switch (periph) {
+	case SCLK_SARADC:
+		con = readl(&cru->cru_clksel_con[24]);
+		div = bitfield_extract_by_mask(con, SARADC_DIV_MASK);
+		break;
+	case SCLK_TSADC:
+		con = readl(&cru->cru_clksel_con[34]);
+		div = bitfield_extract_by_mask(con, TSADC_DIV_MASK);
+		break;
+	default:
+		return -EINVAL;
+	}
+	return DIV_TO_RATE(PERI_PCLK_HZ, div);
+}
+
+static ulong rk3066_clk_saradc_set_clk(struct rk3066_cru *cru, uint hz,
+				       int periph)
+{
+	int src_clk_div;
+
+	src_clk_div = DIV_ROUND_UP(PERI_PCLK_HZ, hz) - 1;
+	assert(src_clk_div < 128);
+
+	switch (periph) {
+	case SCLK_SARADC:
+		rk_clrsetreg(&cru->cru_clksel_con[24],
+			     SARADC_DIV_MASK,
+			     src_clk_div << SARADC_DIV_SHIFT);
+		break;
+	case SCLK_TSADC:
+		rk_clrsetreg(&cru->cru_clksel_con[34],
+			     SARADC_DIV_MASK,
+			     src_clk_div << SARADC_DIV_SHIFT);
+		break;
+	default:
+		return -EINVAL;
+	}
+
+	return rk3066_clk_saradc_get_clk(cru, periph);
+}
+
+static void rk3066_clk_init(struct rk3066_cru *cru, struct rk3066_grf *grf)
+{
+	u32 aclk_div, hclk_div, pclk_div, h2p_div;
+
+	/* pll enter slow-mode */
+	rk_clrsetreg(&cru->cru_mode_con,
+		     GPLL_MODE_MASK |
+		     CPLL_MODE_MASK,
+		     GPLL_MODE_SLOW << GPLL_MODE_SHIFT |
+		     CPLL_MODE_SLOW << CPLL_MODE_SHIFT);
+
+	/* init pll */
+	rk3066_clk_set_pll(cru, CLK_GENERAL, &gpll_init_cfg);
+	rk3066_clk_set_pll(cru, CLK_CODEC, &cpll_init_cfg);
+
+	/* waiting for pll lock */
+	while ((readl(&grf->soc_status0) &
+		(SOCSTS_CPLL_LOCK | SOCSTS_GPLL_LOCK)) !=
+	       (SOCSTS_CPLL_LOCK | SOCSTS_GPLL_LOCK))
+		udelay(1);
+
+	/*
+	 * cpu clock pll source selection and
+	 * reparent aclk_cpu_pre from apll to gpll
+	 * set up dependent divisors for PCLK/HCLK and ACLK clocks.
+	 */
+	aclk_div = DIV_ROUND_UP(GPLL_HZ, CPU_ACLK_HZ) - 1;
+	assert((aclk_div + 1) * CPU_ACLK_HZ == GPLL_HZ && aclk_div <= 0x1f);
+
+	rk_clrsetreg(&cru->cru_clksel_con[0],
+		     CPU_ACLK_PLL_MASK |
+		     A9_CPU_DIV_MASK,
+		     CPU_ACLK_PLL_SELECT_GPLL << CPU_ACLK_PLL_SHIFT |
+		     aclk_div << A9_CPU_DIV_SHIFT);
+
+	hclk_div = ilog2(CPU_ACLK_HZ / CPU_HCLK_HZ);
+	assert((1 << hclk_div) * CPU_HCLK_HZ == CPU_ACLK_HZ && hclk_div < 0x3);
+	pclk_div = ilog2(CPU_ACLK_HZ / CPU_PCLK_HZ);
+	assert((1 << pclk_div) * CPU_PCLK_HZ == CPU_ACLK_HZ && pclk_div < 0x4);
+	h2p_div = ilog2(CPU_HCLK_HZ / CPU_H2P_HZ);
+	assert((1 << h2p_div) * CPU_H2P_HZ == CPU_HCLK_HZ && pclk_div < 0x3);
+
+	rk_clrsetreg(&cru->cru_clksel_con[1],
+		     AHB2APB_DIV_MASK |
+		     CPU_PCLK_DIV_MASK |
+		     CPU_HCLK_DIV_MASK,
+		     h2p_div << AHB2APB_DIV_SHIFT |
+		     pclk_div << CPU_PCLK_DIV_SHIFT |
+		     hclk_div << CPU_HCLK_DIV_SHIFT);
+
+	/*
+	 * peri clock pll source selection and
+	 * set up dependent divisors for PCLK/HCLK and ACLK clocks.
+	 */
+	aclk_div = GPLL_HZ / PERI_ACLK_HZ - 1;
+	assert((aclk_div + 1) * PERI_ACLK_HZ == GPLL_HZ && aclk_div < 0x1f);
+
+	hclk_div = ilog2(PERI_ACLK_HZ / PERI_HCLK_HZ);
+	assert((1 << hclk_div) * PERI_HCLK_HZ ==
+	       PERI_ACLK_HZ && (hclk_div < 0x4));
+
+	pclk_div = ilog2(PERI_ACLK_HZ / PERI_PCLK_HZ);
+	assert((1 << pclk_div) * PERI_PCLK_HZ ==
+	       PERI_ACLK_HZ && (pclk_div < 0x4));
+
+	rk_clrsetreg(&cru->cru_clksel_con[10],
+		     PERI_PCLK_DIV_MASK |
+		     PERI_HCLK_DIV_MASK |
+		     PERI_ACLK_DIV_MASK,
+		     PERI_SEL_GPLL << PERI_SEL_PLL_SHIFT |
+		     pclk_div << PERI_PCLK_DIV_SHIFT |
+		     hclk_div << PERI_HCLK_DIV_SHIFT |
+		     aclk_div << PERI_ACLK_DIV_SHIFT);
+
+	/* PLL enter normal-mode */
+	rk_clrsetreg(&cru->cru_mode_con,
+		     GPLL_MODE_MASK |
+		     CPLL_MODE_MASK,
+		     GPLL_MODE_NORMAL << GPLL_MODE_SHIFT |
+		     CPLL_MODE_NORMAL << CPLL_MODE_SHIFT);
+
+	rk3066_clk_mmc_set_clk(cru, PERI_HCLK_HZ, HCLK_SDMMC, 16000000);
+}
+
+static ulong rk3066_clk_get_rate(struct clk *clk)
+{
+	struct rk3066_clk_priv *priv = dev_get_priv(clk->dev);
+	ulong new_rate, gclk_rate;
+
+	gclk_rate = rk3066_clk_pll_get_rate(priv->cru, CLK_GENERAL);
+	switch (clk->id) {
+	case 1 ... 4:
+		new_rate = rk3066_clk_pll_get_rate(priv->cru, clk->id);
+		break;
+	case HCLK_EMMC:
+	case HCLK_SDMMC:
+	case HCLK_SDIO:
+	case SCLK_EMMC:
+	case SCLK_SDMMC:
+	case SCLK_SDIO:
+		new_rate = rk3066_clk_mmc_get_clk(priv->cru, PERI_HCLK_HZ,
+						  clk->id);
+		break;
+	case SCLK_SPI0:
+	case SCLK_SPI1:
+		new_rate = rk3066_clk_spi_get_clk(priv->cru, PERI_PCLK_HZ,
+						  clk->id);
+		break;
+	case PCLK_I2C0:
+	case PCLK_I2C1:
+	case PCLK_I2C2:
+	case PCLK_I2C3:
+	case PCLK_I2C4:
+		return gclk_rate;
+	case SCLK_SARADC:
+	case SCLK_TSADC:
+		new_rate = rk3066_clk_saradc_get_clk(priv->cru, clk->id);
+		break;
+	default:
+		return -ENOENT;
+	}
+
+	return new_rate;
+}
+
+static ulong rk3066_clk_set_rate(struct clk *clk, ulong rate)
+{
+	struct rk3066_clk_priv *priv = dev_get_priv(clk->dev);
+	struct rk3066_cru *cru = priv->cru;
+	ulong new_rate;
+
+	switch (clk->id) {
+	case PLL_APLL:
+		new_rate = rk3066_clk_configure_cpu(priv->cru, priv->grf, rate);
+		break;
+	case CLK_DDR:
+		new_rate = rk3066_clk_configure_ddr(priv->cru, priv->grf, rate);
+		break;
+	case HCLK_EMMC:
+	case HCLK_SDMMC:
+	case HCLK_SDIO:
+	case SCLK_EMMC:
+	case SCLK_SDMMC:
+	case SCLK_SDIO:
+		new_rate = rk3066_clk_mmc_set_clk(cru, PERI_HCLK_HZ,
+						  clk->id, rate);
+		break;
+	case SCLK_SPI0:
+	case SCLK_SPI1:
+		new_rate = rk3066_clk_spi_set_clk(cru, PERI_PCLK_HZ,
+						  clk->id, rate);
+		break;
+	case SCLK_SARADC:
+	case SCLK_TSADC:
+		new_rate = rk3066_clk_saradc_set_clk(cru, rate, clk->id);
+		break;
+	default:
+		return -ENOENT;
+	}
+
+	return new_rate;
+}
+
+static int rk3066_clk_enable(struct clk *clk)
+{
+	struct rk3066_clk_priv *priv = dev_get_priv(clk->dev);
+
+	switch (clk->id) {
+	case HCLK_NANDC0:
+		rk_clrreg(&priv->cru->cru_clkgate_con[5], BIT(9));
+		break;
+	case HCLK_SDMMC:
+		rk_clrreg(&priv->cru->cru_clkgate_con[5], BIT(10));
+		break;
+	case HCLK_SDIO:
+		rk_clrreg(&priv->cru->cru_clkgate_con[5], BIT(11));
+		break;
+	}
+
+	return 0;
+}
+
+static int rk3066_clk_disable(struct clk *clk)
+{
+	struct rk3066_clk_priv *priv = dev_get_priv(clk->dev);
+
+	switch (clk->id) {
+	case HCLK_NANDC0:
+		rk_setreg(&priv->cru->cru_clkgate_con[5], BIT(9));
+		break;
+	case HCLK_SDMMC:
+		rk_setreg(&priv->cru->cru_clkgate_con[5], BIT(10));
+		break;
+	case HCLK_SDIO:
+		rk_setreg(&priv->cru->cru_clkgate_con[5], BIT(11));
+		break;
+	}
+
+	return 0;
+}
+
+static struct clk_ops rk3066_clk_ops = {
+	.disable	= rk3066_clk_disable,
+	.enable	= rk3066_clk_enable,
+	.get_rate	= rk3066_clk_get_rate,
+	.set_rate	= rk3066_clk_set_rate,
+};
+
+static int rk3066_clk_of_to_plat(struct udevice *dev)
+{
+	if (CONFIG_IS_ENABLED(OF_REAL)) {
+		struct rk3066_clk_priv *priv = dev_get_priv(dev);
+
+		priv->cru = dev_read_addr_ptr(dev);
+	}
+
+	return 0;
+}
+
+static int rk3066_clk_probe(struct udevice *dev)
+{
+	struct rk3066_clk_priv *priv = dev_get_priv(dev);
+
+	priv->grf = syscon_get_first_range(ROCKCHIP_SYSCON_GRF);
+	if (IS_ERR(priv->grf))
+		return PTR_ERR(priv->grf);
+
+#if CONFIG_IS_ENABLED(OF_PLATDATA)
+	struct rk3066_clk_plat *plat = dev_get_plat(dev);
+
+	priv->cru = map_sysmem(plat->dtd.reg[0], plat->dtd.reg[1]);
+#endif
+
+	if (IS_ENABLED(CONFIG_TPL_BUILD)) {
+		rk3066_clk_init(priv->cru, priv->grf);
+
+		/* Init CPU frequency */
+		rk3066_clk_configure_cpu(priv->cru, priv->grf, APLL_SAFE_HZ);
+	}
+
+	return 0;
+}
+
+static int rk3066_clk_bind(struct udevice *dev)
+{
+	int ret;
+	struct udevice *sys_child;
+	struct sysreset_reg *priv;
+
+	/* The reset driver does not have a device node, so bind it here */
+	ret = device_bind_driver(dev, "rockchip_sysreset", "sysreset",
+				 &sys_child);
+	if (ret) {
+		debug("Warning: No sysreset driver: ret=%d\n", ret);
+	} else {
+		priv = malloc(sizeof(struct sysreset_reg));
+		priv->glb_srst_fst_value = offsetof(struct rk3066_cru,
+						    cru_glb_srst_fst_value);
+		priv->glb_srst_snd_value = offsetof(struct rk3066_cru,
+						    cru_glb_srst_snd_value);
+		dev_set_priv(sys_child, priv);
+	}
+
+	if (CONFIG_IS_ENABLED(RESET_ROCKCHIP)) {
+		ret = offsetof(struct rk3066_cru, cru_softrst_con[0]);
+		ret = rockchip_reset_bind(dev, ret, 9);
+		if (ret)
+			debug("Warning: software reset driver bind failed\n");
+	}
+
+	return 0;
+}
+
+static const struct udevice_id rk3066_clk_ids[] = {
+	{ .compatible = "rockchip,rk3066a-cru" },
+	{ }
+};
+
+U_BOOT_DRIVER(rockchip_rk3066a_cru) = {
+	.name		= "rockchip_rk3066a_cru",
+	.id		= UCLASS_CLK,
+	.ops		= &rk3066_clk_ops,
+	.probe		= rk3066_clk_probe,
+	.bind		= rk3066_clk_bind,
+	.of_match	= rk3066_clk_ids,
+	.of_to_plat	= rk3066_clk_of_to_plat,
+	.priv_auto	= sizeof(struct rk3066_clk_priv),
+	.plat_auto	= sizeof(struct rk3066_clk_plat),
+};