clk: Add rp1 clock driver
authorPhil Elwell <phil@raspberrypi.com>
Mon, 10 Oct 2022 13:25:38 +0000 (14:25 +0100)
committerDom Cobley <popcornmix@gmail.com>
Mon, 19 Feb 2024 11:34:46 +0000 (11:34 +0000)
RP1 contains various PLLs and clocks for driving the hardware
blocks, so add a driver to configure these.

Signed-off-by: Phil Elwell <phil@raspberrypi.com>
drivers/clk/Kconfig
drivers/clk/Makefile
drivers/clk/clk-rp1.c [new file with mode: 0644]
include/dt-bindings/clock/rp1.h

index a592c58..3ca6061 100644 (file)
@@ -88,6 +88,13 @@ config COMMON_CLK_RK808
          These multi-function devices have two fixed-rate oscillators, clocked at 32KHz each.
          Clkout1 is always on, Clkout2 can off by control register.
 
+config COMMON_CLK_RP1
+       tristate "Raspberry Pi RP1-based clock support"
+       depends on PCI || COMPILE_TEST
+       depends on COMMON_CLK
+       help
+         Enable common clock framework support for Raspberry Pi RP1
+
 config COMMON_CLK_HI655X
        tristate "Clock driver for Hi655x" if EXPERT
        depends on (MFD_HI655X_PMIC || COMPILE_TEST)
index b5253c9..3730887 100644 (file)
@@ -59,6 +59,7 @@ obj-$(CONFIG_CLK_LS1028A_PLLDIG)      += clk-plldig.o
 obj-$(CONFIG_COMMON_CLK_PWM)           += clk-pwm.o
 obj-$(CONFIG_CLK_QORIQ)                        += clk-qoriq.o
 obj-$(CONFIG_COMMON_CLK_RK808)         += clk-rk808.o
+obj-$(CONFIG_COMMON_CLK_RP1)           += clk-rp1.o
 obj-$(CONFIG_COMMON_CLK_HI655X)                += clk-hi655x.o
 obj-$(CONFIG_COMMON_CLK_S2MPS11)       += clk-s2mps11.o
 obj-$(CONFIG_COMMON_CLK_SCMI)           += clk-scmi.o
diff --git a/drivers/clk/clk-rp1.c b/drivers/clk/clk-rp1.c
new file mode 100644 (file)
index 0000000..f61f752
--- /dev/null
@@ -0,0 +1,2085 @@
+// SPDX-License-Identifier: GPL-2.0+
+/*
+ * Copyright (C) 2023 Raspberry Pi Ltd.
+ *
+ * Clock driver for RP1 PCIe multifunction chip.
+ */
+
+#include <linux/clk-provider.h>
+#include <linux/clkdev.h>
+#include <linux/clk.h>
+#include <linux/debugfs.h>
+#include <linux/delay.h>
+#include <linux/io.h>
+#include <linux/math64.h>
+#include <linux/module.h>
+#include <linux/of_device.h>
+#include <linux/platform_device.h>
+#include <linux/rp1_platform.h>
+#include <linux/slab.h>
+
+#include <asm/div64.h>
+
+#include <dt-bindings/clock/rp1.h>
+
+#define PLL_SYS_CS                     0x08000
+#define PLL_SYS_PWR                    0x08004
+#define PLL_SYS_FBDIV_INT              0x08008
+#define PLL_SYS_FBDIV_FRAC             0x0800c
+#define PLL_SYS_PRIM                   0x08010
+#define PLL_SYS_SEC                    0x08014
+
+#define PLL_AUDIO_CS                   0x0c000
+#define PLL_AUDIO_PWR                  0x0c004
+#define PLL_AUDIO_FBDIV_INT            0x0c008
+#define PLL_AUDIO_FBDIV_FRAC           0x0c00c
+#define PLL_AUDIO_PRIM                 0x0c010
+#define PLL_AUDIO_SEC                  0x0c014
+
+#define PLL_VIDEO_CS                   0x10000
+#define PLL_VIDEO_PWR                  0x10004
+#define PLL_VIDEO_FBDIV_INT            0x10008
+#define PLL_VIDEO_FBDIV_FRAC           0x1000c
+#define PLL_VIDEO_PRIM                 0x10010
+#define PLL_VIDEO_SEC                  0x10014
+
+#define CLK_SYS_CTRL                   0x00014
+#define CLK_SYS_DIV_INT                        0x00018
+#define CLK_SYS_SEL                    0x00020
+
+#define CLK_SLOW_SYS_CTRL              0x00024
+#define CLK_SLOW_SYS_DIV_INT           0x00028
+#define CLK_SLOW_SYS_SEL               0x00030
+
+#define CLK_DMA_CTRL                   0x00044
+#define CLK_DMA_DIV_INT                        0x00048
+#define CLK_DMA_SEL                    0x00050
+
+#define CLK_UART_CTRL                  0x00054
+#define CLK_UART_DIV_INT               0x00058
+#define CLK_UART_SEL                   0x00060
+
+#define CLK_ETH_CTRL                   0x00064
+#define CLK_ETH_DIV_INT                        0x00068
+#define CLK_ETH_SEL                    0x00070
+
+#define CLK_PWM0_CTRL                  0x00074
+#define CLK_PWM0_DIV_INT               0x00078
+#define CLK_PWM0_DIV_FRAC              0x0007c
+#define CLK_PWM0_SEL                   0x00080
+
+#define CLK_PWM1_CTRL                  0x00084
+#define CLK_PWM1_DIV_INT               0x00088
+#define CLK_PWM1_DIV_FRAC              0x0008c
+#define CLK_PWM1_SEL                   0x00090
+
+#define CLK_AUDIO_IN_CTRL              0x00094
+#define CLK_AUDIO_IN_DIV_INT           0x00098
+#define CLK_AUDIO_IN_SEL               0x000a0
+
+#define CLK_AUDIO_OUT_CTRL             0x000a4
+#define CLK_AUDIO_OUT_DIV_INT          0x000a8
+#define CLK_AUDIO_OUT_SEL              0x000b0
+
+#define CLK_I2S_CTRL                   0x000b4
+#define CLK_I2S_DIV_INT                        0x000b8
+#define CLK_I2S_SEL                    0x000c0
+
+#define CLK_MIPI0_CFG_CTRL             0x000c4
+#define CLK_MIPI0_CFG_DIV_INT          0x000c8
+#define CLK_MIPI0_CFG_SEL              0x000d0
+
+#define CLK_MIPI1_CFG_CTRL             0x000d4
+#define CLK_MIPI1_CFG_DIV_INT          0x000d8
+#define CLK_MIPI1_CFG_SEL              0x000e0
+
+#define CLK_PCIE_AUX_CTRL              0x000e4
+#define CLK_PCIE_AUX_DIV_INT           0x000e8
+#define CLK_PCIE_AUX_SEL               0x000f0
+
+#define CLK_USBH0_MICROFRAME_CTRL      0x000f4
+#define CLK_USBH0_MICROFRAME_DIV_INT   0x000f8
+#define CLK_USBH0_MICROFRAME_SEL       0x00100
+
+#define CLK_USBH1_MICROFRAME_CTRL      0x00104
+#define CLK_USBH1_MICROFRAME_DIV_INT   0x00108
+#define CLK_USBH1_MICROFRAME_SEL       0x00110
+
+#define CLK_USBH0_SUSPEND_CTRL         0x00114
+#define CLK_USBH0_SUSPEND_DIV_INT      0x00118
+#define CLK_USBH0_SUSPEND_SEL          0x00120
+
+#define CLK_USBH1_SUSPEND_CTRL         0x00124
+#define CLK_USBH1_SUSPEND_DIV_INT      0x00128
+#define CLK_USBH1_SUSPEND_SEL          0x00130
+
+#define CLK_ETH_TSU_CTRL               0x00134
+#define CLK_ETH_TSU_DIV_INT            0x00138
+#define CLK_ETH_TSU_SEL                        0x00140
+
+#define CLK_ADC_CTRL                   0x00144
+#define CLK_ADC_DIV_INT                        0x00148
+#define CLK_ADC_SEL                    0x00150
+
+#define CLK_SDIO_TIMER_CTRL            0x00154
+#define CLK_SDIO_TIMER_DIV_INT         0x00158
+#define CLK_SDIO_TIMER_SEL             0x00160
+
+#define CLK_SDIO_ALT_SRC_CTRL          0x00164
+#define CLK_SDIO_ALT_SRC_DIV_INT       0x00168
+#define CLK_SDIO_ALT_SRC_SEL           0x00170
+
+#define CLK_GP0_CTRL                   0x00174
+#define CLK_GP0_DIV_INT                        0x00178
+#define CLK_GP0_DIV_FRAC               0x0017c
+#define CLK_GP0_SEL                    0x00180
+
+#define CLK_GP1_CTRL                   0x00184
+#define CLK_GP1_DIV_INT                        0x00188
+#define CLK_GP1_DIV_FRAC               0x0018c
+#define CLK_GP1_SEL                    0x00190
+
+#define CLK_GP2_CTRL                   0x00194
+#define CLK_GP2_DIV_INT                        0x00198
+#define CLK_GP2_DIV_FRAC               0x0019c
+#define CLK_GP2_SEL                    0x001a0
+
+#define CLK_GP3_CTRL                   0x001a4
+#define CLK_GP3_DIV_INT                        0x001a8
+#define CLK_GP3_DIV_FRAC               0x001ac
+#define CLK_GP3_SEL                    0x001b0
+
+#define CLK_GP4_CTRL                   0x001b4
+#define CLK_GP4_DIV_INT                        0x001b8
+#define CLK_GP4_DIV_FRAC               0x001bc
+#define CLK_GP4_SEL                    0x001c0
+
+#define CLK_GP5_CTRL                   0x001c4
+#define CLK_GP5_DIV_INT                        0x001c8
+#define CLK_GP5_DIV_FRAC               0x001cc
+#define CLK_GP5_SEL                    0x001d0
+
+#define CLK_SYS_RESUS_CTRL             0x0020c
+
+#define CLK_SLOW_SYS_RESUS_CTRL                0x00214
+
+#define FC0_REF_KHZ                    0x0021c
+#define FC0_MIN_KHZ                    0x00220
+#define FC0_MAX_KHZ                    0x00224
+#define FC0_DELAY                      0x00228
+#define FC0_INTERVAL                   0x0022c
+#define FC0_SRC                                0x00230
+#define FC0_STATUS                     0x00234
+#define FC0_RESULT                     0x00238
+#define FC_SIZE                                0x20
+#define FC_COUNT                       8
+#define FC_NUM(idx, off)               ((idx) * 32 + (off))
+
+#define AUX_SEL                                1
+
+#define VIDEO_CLOCKS_OFFSET            0x4000
+#define VIDEO_CLK_VEC_CTRL             (VIDEO_CLOCKS_OFFSET + 0x0000)
+#define VIDEO_CLK_VEC_DIV_INT          (VIDEO_CLOCKS_OFFSET + 0x0004)
+#define VIDEO_CLK_VEC_SEL              (VIDEO_CLOCKS_OFFSET + 0x000c)
+#define VIDEO_CLK_DPI_CTRL             (VIDEO_CLOCKS_OFFSET + 0x0010)
+#define VIDEO_CLK_DPI_DIV_INT          (VIDEO_CLOCKS_OFFSET + 0x0014)
+#define VIDEO_CLK_DPI_SEL              (VIDEO_CLOCKS_OFFSET + 0x001c)
+#define VIDEO_CLK_MIPI0_DPI_CTRL       (VIDEO_CLOCKS_OFFSET + 0x0020)
+#define VIDEO_CLK_MIPI0_DPI_DIV_INT    (VIDEO_CLOCKS_OFFSET + 0x0024)
+#define VIDEO_CLK_MIPI0_DPI_DIV_FRAC   (VIDEO_CLOCKS_OFFSET + 0x0028)
+#define VIDEO_CLK_MIPI0_DPI_SEL                (VIDEO_CLOCKS_OFFSET + 0x002c)
+#define VIDEO_CLK_MIPI1_DPI_CTRL       (VIDEO_CLOCKS_OFFSET + 0x0030)
+#define VIDEO_CLK_MIPI1_DPI_DIV_INT    (VIDEO_CLOCKS_OFFSET + 0x0034)
+#define VIDEO_CLK_MIPI1_DPI_DIV_FRAC   (VIDEO_CLOCKS_OFFSET + 0x0038)
+#define VIDEO_CLK_MIPI1_DPI_SEL                (VIDEO_CLOCKS_OFFSET + 0x003c)
+
+#define DIV_INT_8BIT_MAX               0x000000ffu /* max divide for most clocks */
+#define DIV_INT_16BIT_MAX              0x0000ffffu /* max divide for GPx, PWM */
+#define DIV_INT_24BIT_MAX               0x00ffffffu /* max divide for CLK_SYS */
+
+#define FC0_STATUS_DONE                        BIT(4)
+#define FC0_STATUS_RUNNING             BIT(8)
+#define FC0_RESULT_FRAC_SHIFT          5
+
+#define PLL_PRIM_DIV1_SHIFT            16
+#define PLL_PRIM_DIV1_MASK             0x00070000
+#define PLL_PRIM_DIV2_SHIFT            12
+#define PLL_PRIM_DIV2_MASK             0x00007000
+
+#define PLL_SEC_DIV_SHIFT              8
+#define PLL_SEC_DIV_WIDTH              5
+#define PLL_SEC_DIV_MASK               0x00001f00
+
+#define PLL_CS_LOCK                    BIT(31)
+#define PLL_CS_REFDIV_SHIFT            0
+
+#define PLL_PWR_PD                     BIT(0)
+#define PLL_PWR_DACPD                  BIT(1)
+#define PLL_PWR_DSMPD                  BIT(2)
+#define PLL_PWR_POSTDIVPD              BIT(3)
+#define PLL_PWR_4PHASEPD               BIT(4)
+#define PLL_PWR_VCOPD                  BIT(5)
+#define PLL_PWR_MASK                   0x0000003f
+
+#define PLL_SEC_RST                    BIT(16)
+#define PLL_SEC_IMPL                   BIT(31)
+
+/* PLL phase output for both PRI and SEC */
+#define PLL_PH_EN                      BIT(4)
+#define PLL_PH_PHASE_SHIFT             0
+
+#define RP1_PLL_PHASE_0                        0
+#define RP1_PLL_PHASE_90               1
+#define RP1_PLL_PHASE_180              2
+#define RP1_PLL_PHASE_270              3
+
+/* Clock fields for all clocks */
+#define CLK_CTRL_ENABLE                        BIT(11)
+#define CLK_CTRL_AUXSRC_MASK           0x000003e0
+#define CLK_CTRL_AUXSRC_SHIFT          5
+#define CLK_CTRL_SRC_SHIFT             0
+#define CLK_DIV_FRAC_BITS              16
+
+#define KHz                            1000
+#define MHz                            (KHz * KHz)
+#define LOCK_TIMEOUT_NS                        100000000
+#define FC_TIMEOUT_NS                  100000000
+
+#define MAX_CLK_PARENTS        8
+
+#define MEASURE_CLOCK_RATE
+const char * const fc0_ref_clk_name = "clk_slow_sys";
+
+#define ABS_DIFF(a, b) ((a) > (b) ? (a) - (b) : (b) - (a))
+#define DIV_U64_NEAREST(a, b) div_u64(((a) + ((b) >> 1)), (b))
+
+/*
+ * Names of the reference clock for the pll cores.  This name must match
+ * the DT reference clock-output-name.
+ */
+static const char *const ref_clock = "xosc";
+
+/*
+ * Secondary PLL channel output divider table.
+ * Divider values range from 8 to 19.
+ * Invalid values default to 19
+ */
+static const struct clk_div_table pll_sec_div_table[] = {
+       { 0x00, 19 },
+       { 0x01, 19 },
+       { 0x02, 19 },
+       { 0x03, 19 },
+       { 0x04, 19 },
+       { 0x05, 19 },
+       { 0x06, 19 },
+       { 0x07, 19 },
+       { 0x08,  8 },
+       { 0x09,  9 },
+       { 0x0a, 10 },
+       { 0x0b, 11 },
+       { 0x0c, 12 },
+       { 0x0d, 13 },
+       { 0x0e, 14 },
+       { 0x0f, 15 },
+       { 0x10, 16 },
+       { 0x11, 17 },
+       { 0x12, 18 },
+       { 0x13, 19 },
+       { 0x14, 19 },
+       { 0x15, 19 },
+       { 0x16, 19 },
+       { 0x17, 19 },
+       { 0x18, 19 },
+       { 0x19, 19 },
+       { 0x1a, 19 },
+       { 0x1b, 19 },
+       { 0x1c, 19 },
+       { 0x1d, 19 },
+       { 0x1e, 19 },
+       { 0x1f, 19 },
+       { 0 }
+};
+
+struct rp1_clockman {
+       struct device *dev;
+       void __iomem *regs;
+       spinlock_t regs_lock; /* spinlock for all clocks */
+
+       /* Must be last */
+       struct clk_hw_onecell_data onecell;
+};
+
+struct rp1_pll_core_data {
+       const char *name;
+       u32 cs_reg;
+       u32 pwr_reg;
+       u32 fbdiv_int_reg;
+       u32 fbdiv_frac_reg;
+       unsigned long flags;
+       u32 fc0_src;
+};
+
+struct rp1_pll_data {
+       const char *name;
+       const char *source_pll;
+       u32 ctrl_reg;
+       unsigned long flags;
+       u32 fc0_src;
+};
+
+struct rp1_pll_ph_data {
+       const char *name;
+       const char *source_pll;
+       unsigned int phase;
+       unsigned int fixed_divider;
+       u32 ph_reg;
+       unsigned long flags;
+       u32 fc0_src;
+};
+
+struct rp1_pll_divider_data {
+       const char *name;
+       const char *source_pll;
+       u32 sec_reg;
+       unsigned long flags;
+       u32 fc0_src;
+};
+
+struct rp1_clock_data {
+       const char *name;
+       const char *const parents[MAX_CLK_PARENTS];
+       int num_std_parents;
+       int num_aux_parents;
+       unsigned long flags;
+       u32 clk_src_mask;
+       u32 ctrl_reg;
+       u32 div_int_reg;
+       u32 div_frac_reg;
+       u32 sel_reg;
+       u32 div_int_max;
+       u32 fc0_src;
+};
+
+struct rp1_pll_core {
+       struct clk_hw hw;
+       struct rp1_clockman *clockman;
+       const struct rp1_pll_core_data *data;
+       unsigned long cached_rate;
+};
+
+struct rp1_pll {
+       struct clk_hw hw;
+       struct clk_divider div;
+       struct rp1_clockman *clockman;
+       const struct rp1_pll_data *data;
+       unsigned long cached_rate;
+};
+
+struct rp1_pll_ph {
+       struct clk_hw hw;
+       struct rp1_clockman *clockman;
+       const struct rp1_pll_ph_data *data;
+};
+
+struct rp1_clock {
+       struct clk_hw hw;
+       struct rp1_clockman *clockman;
+       const struct rp1_clock_data *data;
+       unsigned long cached_rate;
+};
+
+static void rp1_debugfs_regset(struct rp1_clockman *clockman, u32 base,
+                              const struct debugfs_reg32 *regs,
+                              size_t nregs, struct dentry *dentry)
+{
+       struct debugfs_regset32 *regset;
+
+       regset = devm_kzalloc(clockman->dev, sizeof(*regset), GFP_KERNEL);
+       if (!regset)
+               return;
+
+       regset->regs = regs;
+       regset->nregs = nregs;
+       regset->base = clockman->regs + base;
+
+       debugfs_create_regset32("regdump", 0444, dentry, regset);
+}
+
+static inline u32 set_register_field(u32 reg, u32 val, u32 mask, u32 shift)
+{
+       reg &= ~mask;
+       reg |= (val << shift) & mask;
+       return reg;
+}
+
+static inline
+void clockman_write(struct rp1_clockman *clockman, u32 reg, u32 val)
+{
+       writel(val, clockman->regs + reg);
+}
+
+static inline u32 clockman_read(struct rp1_clockman *clockman, u32 reg)
+{
+       return readl(clockman->regs + reg);
+}
+
+#ifdef MEASURE_CLOCK_RATE
+static unsigned long clockman_measure_clock(struct rp1_clockman *clockman,
+                                           const char *clk_name,
+                                           unsigned int fc0_src)
+{
+       struct clk *ref_clk = __clk_lookup(fc0_ref_clk_name);
+       unsigned long result;
+       ktime_t timeout;
+       unsigned int fc_idx, fc_offset, fc_src;
+
+       fc_idx = fc0_src / 32;
+       fc_src = fc0_src % 32;
+
+       /* fc_src == 0 is invalid. */
+       if (!fc_src || fc_idx >= FC_COUNT)
+               return 0;
+
+       fc_offset = fc_idx * FC_SIZE;
+
+       /* Ensure the frequency counter is idle. */
+       timeout = ktime_add_ns(ktime_get(), FC_TIMEOUT_NS);
+       while (clockman_read(clockman, fc_offset + FC0_STATUS) & FC0_STATUS_RUNNING) {
+               if (ktime_after(ktime_get(), timeout)) {
+                       dev_err(clockman->dev, "%s: FC0 busy timeout\n",
+                               clk_name);
+                       return 0;
+               }
+               cpu_relax();
+       }
+
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, fc_offset + FC0_REF_KHZ,
+                      clk_get_rate(ref_clk) / KHz);
+       clockman_write(clockman, fc_offset + FC0_MIN_KHZ, 0);
+       clockman_write(clockman, fc_offset + FC0_MAX_KHZ, 0x1ffffff);
+       clockman_write(clockman, fc_offset + FC0_INTERVAL, 8);
+       clockman_write(clockman, fc_offset + FC0_DELAY, 7);
+       clockman_write(clockman, fc_offset + FC0_SRC, fc_src);
+       spin_unlock(&clockman->regs_lock);
+
+       /* Ensure the frequency counter is idle. */
+       timeout = ktime_add_ns(ktime_get(), FC_TIMEOUT_NS);
+       while (!(clockman_read(clockman, fc_offset + FC0_STATUS) & FC0_STATUS_DONE)) {
+               if (ktime_after(ktime_get(), timeout)) {
+                       dev_err(clockman->dev, "%s: FC0 wait timeout\n",
+                               clk_name);
+                       return 0;
+               }
+               cpu_relax();
+       }
+
+       result = clockman_read(clockman, fc_offset + FC0_RESULT);
+
+       /* Disable FC0 */
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, fc_offset + FC0_SRC, 0);
+       spin_unlock(&clockman->regs_lock);
+
+       return result;
+}
+#endif
+
+static int rp1_pll_core_is_on(struct clk_hw *hw)
+{
+       struct rp1_pll_core *pll_core = container_of(hw, struct rp1_pll_core, hw);
+       struct rp1_clockman *clockman = pll_core->clockman;
+       const struct rp1_pll_core_data *data = pll_core->data;
+       u32 pwr = clockman_read(clockman, data->pwr_reg);
+
+       return (pwr & PLL_PWR_PD) || (pwr & PLL_PWR_POSTDIVPD);
+}
+
+static int rp1_pll_core_on(struct clk_hw *hw)
+{
+       struct rp1_pll_core *pll_core = container_of(hw, struct rp1_pll_core, hw);
+       struct rp1_clockman *clockman = pll_core->clockman;
+       const struct rp1_pll_core_data *data = pll_core->data;
+       u32 fbdiv_frac;
+       ktime_t timeout;
+
+       spin_lock(&clockman->regs_lock);
+
+       if (!(clockman_read(clockman, data->cs_reg) & PLL_CS_LOCK)) {
+               /* Reset to a known state. */
+               clockman_write(clockman, data->pwr_reg, PLL_PWR_MASK);
+               clockman_write(clockman, data->fbdiv_int_reg, 20);
+               clockman_write(clockman, data->fbdiv_frac_reg, 0);
+               clockman_write(clockman, data->cs_reg, 1 << PLL_CS_REFDIV_SHIFT);
+       }
+
+       /* Come out of reset. */
+       fbdiv_frac = clockman_read(clockman, data->fbdiv_frac_reg);
+       clockman_write(clockman, data->pwr_reg, fbdiv_frac ? 0 : PLL_PWR_DSMPD);
+       spin_unlock(&clockman->regs_lock);
+
+       /* Wait for the PLL to lock. */
+       timeout = ktime_add_ns(ktime_get(), LOCK_TIMEOUT_NS);
+       while (!(clockman_read(clockman, data->cs_reg) & PLL_CS_LOCK)) {
+               if (ktime_after(ktime_get(), timeout)) {
+                       dev_err(clockman->dev, "%s: can't lock PLL\n",
+                               clk_hw_get_name(hw));
+                       return -ETIMEDOUT;
+               }
+               cpu_relax();
+       }
+
+       return 0;
+}
+
+static void rp1_pll_core_off(struct clk_hw *hw)
+{
+       struct rp1_pll_core *pll_core = container_of(hw, struct rp1_pll_core, hw);
+       struct rp1_clockman *clockman = pll_core->clockman;
+       const struct rp1_pll_core_data *data = pll_core->data;
+
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, data->pwr_reg, 0);
+       spin_unlock(&clockman->regs_lock);
+}
+
+static inline unsigned long get_pll_core_divider(struct clk_hw *hw,
+                                                unsigned long rate,
+                                                unsigned long parent_rate,
+                                                u32 *div_int, u32 *div_frac)
+{
+       unsigned long calc_rate;
+       u32 fbdiv_int, fbdiv_frac;
+       u64 div_fp64; /* 32.32 fixed point fraction. */
+
+       /* Factor of reference clock to VCO frequency. */
+       div_fp64 = (u64)(rate) << 32;
+       div_fp64 = DIV_U64_NEAREST(div_fp64, parent_rate);
+
+       /* Round the fractional component at 24 bits. */
+       div_fp64 += 1 << (32 - 24 - 1);
+
+       fbdiv_int = div_fp64 >> 32;
+       fbdiv_frac = (div_fp64 >> (32 - 24)) & 0xffffff;
+
+       calc_rate =
+               ((u64)parent_rate * (((u64)fbdiv_int << 24) + fbdiv_frac) + (1 << 23)) >> 24;
+
+       *div_int = fbdiv_int;
+       *div_frac = fbdiv_frac;
+
+       return calc_rate;
+}
+
+static int rp1_pll_core_set_rate(struct clk_hw *hw,
+                                unsigned long rate, unsigned long parent_rate)
+{
+       struct rp1_pll_core *pll_core = container_of(hw, struct rp1_pll_core, hw);
+       struct rp1_clockman *clockman = pll_core->clockman;
+       const struct rp1_pll_core_data *data = pll_core->data;
+       unsigned long calc_rate;
+       u32 fbdiv_int, fbdiv_frac;
+
+       // todo: is this needed??
+       //rp1_pll_off(hw);
+
+       /* Disable dividers to start with. */
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, data->fbdiv_int_reg, 0);
+       clockman_write(clockman, data->fbdiv_frac_reg, 0);
+       spin_unlock(&clockman->regs_lock);
+
+       calc_rate = get_pll_core_divider(hw, rate, parent_rate,
+                                        &fbdiv_int, &fbdiv_frac);
+
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, data->pwr_reg, fbdiv_frac ? 0 : PLL_PWR_DSMPD);
+       clockman_write(clockman, data->fbdiv_int_reg, fbdiv_int);
+       clockman_write(clockman, data->fbdiv_frac_reg, fbdiv_frac);
+       spin_unlock(&clockman->regs_lock);
+
+       /* Check that reference frequency is no greater than VCO / 16. */
+       BUG_ON(parent_rate > (rate / 16));
+
+       pll_core->cached_rate = calc_rate;
+
+       spin_lock(&clockman->regs_lock);
+       /* Don't need to divide ref unless parent_rate > (output freq / 16) */
+       clockman_write(clockman, data->cs_reg,
+                      clockman_read(clockman, data->cs_reg) |
+                                    (1 << PLL_CS_REFDIV_SHIFT));
+       spin_unlock(&clockman->regs_lock);
+
+       return 0;
+}
+
+static unsigned long rp1_pll_core_recalc_rate(struct clk_hw *hw,
+                                             unsigned long parent_rate)
+{
+       struct rp1_pll_core *pll_core = container_of(hw, struct rp1_pll_core, hw);
+       struct rp1_clockman *clockman = pll_core->clockman;
+       const struct rp1_pll_core_data *data = pll_core->data;
+       u32 fbdiv_int, fbdiv_frac;
+       unsigned long calc_rate;
+
+       fbdiv_int = clockman_read(clockman, data->fbdiv_int_reg);
+       fbdiv_frac = clockman_read(clockman, data->fbdiv_frac_reg);
+       calc_rate =
+               ((u64)parent_rate * (((u64)fbdiv_int << 24) + fbdiv_frac) + (1 << 23)) >> 24;
+
+       return calc_rate;
+}
+
+static long rp1_pll_core_round_rate(struct clk_hw *hw, unsigned long rate,
+                                   unsigned long *parent_rate)
+{
+       u32 fbdiv_int, fbdiv_frac;
+       long calc_rate;
+
+       calc_rate = get_pll_core_divider(hw, rate, *parent_rate,
+                                        &fbdiv_int, &fbdiv_frac);
+       return calc_rate;
+}
+
+static void rp1_pll_core_debug_init(struct clk_hw *hw, struct dentry *dentry)
+{
+       struct rp1_pll_core *pll_core = container_of(hw, struct rp1_pll_core, hw);
+       struct rp1_clockman *clockman = pll_core->clockman;
+       const struct rp1_pll_core_data *data = pll_core->data;
+       struct debugfs_reg32 *regs;
+
+       regs = devm_kcalloc(clockman->dev, 4, sizeof(*regs), GFP_KERNEL);
+       if (!regs)
+               return;
+
+       regs[0].name = "cs";
+       regs[0].offset = data->cs_reg;
+       regs[1].name = "pwr";
+       regs[1].offset = data->pwr_reg;
+       regs[2].name = "fbdiv_int";
+       regs[2].offset = data->fbdiv_int_reg;
+       regs[3].name = "fbdiv_frac";
+       regs[3].offset = data->fbdiv_frac_reg;
+
+       rp1_debugfs_regset(clockman, 0, regs, 4, dentry);
+}
+
+static void get_pll_prim_dividers(unsigned long rate, unsigned long parent_rate,
+                                 u32 *divider1, u32 *divider2)
+{
+       unsigned int div1, div2;
+       unsigned int best_div1 = 7, best_div2 = 7;
+       unsigned long best_rate_diff =
+               ABS_DIFF(DIV_ROUND_CLOSEST(parent_rate, best_div1 * best_div2), rate);
+       long rate_diff, calc_rate;
+
+       for (div1 = 1; div1 <= 7; div1++) {
+               for (div2 = 1; div2 <= div1; div2++) {
+                       calc_rate = DIV_ROUND_CLOSEST(parent_rate, div1 * div2);
+                       rate_diff = ABS_DIFF(calc_rate, rate);
+
+                       if (calc_rate == rate) {
+                               best_div1 = div1;
+                               best_div2 = div2;
+                               goto done;
+                       } else if (rate_diff < best_rate_diff) {
+                               best_div1 = div1;
+                               best_div2 = div2;
+                               best_rate_diff = rate_diff;
+                       }
+               }
+       }
+
+done:
+       *divider1 = best_div1;
+       *divider2 = best_div2;
+}
+
+static int rp1_pll_set_rate(struct clk_hw *hw,
+                           unsigned long rate, unsigned long parent_rate)
+{
+       struct rp1_pll *pll = container_of(hw, struct rp1_pll, hw);
+       struct rp1_clockman *clockman = pll->clockman;
+       const struct rp1_pll_data *data = pll->data;
+       u32 prim, prim_div1, prim_div2;
+
+       get_pll_prim_dividers(rate, parent_rate, &prim_div1, &prim_div2);
+
+       spin_lock(&clockman->regs_lock);
+       prim = clockman_read(clockman, data->ctrl_reg);
+       prim = set_register_field(prim, prim_div1, PLL_PRIM_DIV1_MASK,
+                                 PLL_PRIM_DIV1_SHIFT);
+       prim = set_register_field(prim, prim_div2, PLL_PRIM_DIV2_MASK,
+                                 PLL_PRIM_DIV2_SHIFT);
+       clockman_write(clockman, data->ctrl_reg, prim);
+       spin_unlock(&clockman->regs_lock);
+
+#ifdef MEASURE_CLOCK_RATE
+       clockman_measure_clock(clockman, data->name, data->fc0_src);
+#endif
+       return 0;
+}
+
+static unsigned long rp1_pll_recalc_rate(struct clk_hw *hw,
+                                        unsigned long parent_rate)
+{
+       struct rp1_pll *pll = container_of(hw, struct rp1_pll, hw);
+       struct rp1_clockman *clockman = pll->clockman;
+       const struct rp1_pll_data *data = pll->data;
+       u32 prim, prim_div1, prim_div2;
+
+       prim = clockman_read(clockman, data->ctrl_reg);
+       prim_div1 = (prim & PLL_PRIM_DIV1_MASK) >> PLL_PRIM_DIV1_SHIFT;
+       prim_div2 = (prim & PLL_PRIM_DIV2_MASK) >> PLL_PRIM_DIV2_SHIFT;
+
+       if (!prim_div1 || !prim_div2) {
+               dev_err(clockman->dev, "%s: (%s) zero divider value\n",
+                       __func__, data->name);
+               return 0;
+       }
+
+       return DIV_ROUND_CLOSEST(parent_rate, prim_div1 * prim_div2);
+}
+
+static long rp1_pll_round_rate(struct clk_hw *hw, unsigned long rate,
+                              unsigned long *parent_rate)
+{
+       u32 div1, div2;
+
+       get_pll_prim_dividers(rate, *parent_rate, &div1, &div2);
+
+       return DIV_ROUND_CLOSEST(*parent_rate, div1 * div2);
+}
+
+static void rp1_pll_debug_init(struct clk_hw *hw,
+                              struct dentry *dentry)
+{
+       struct rp1_pll *pll = container_of(hw, struct rp1_pll, hw);
+       struct rp1_clockman *clockman = pll->clockman;
+       const struct rp1_pll_data *data = pll->data;
+       struct debugfs_reg32 *regs;
+
+       regs = devm_kcalloc(clockman->dev, 1, sizeof(*regs), GFP_KERNEL);
+       if (!regs)
+               return;
+
+       regs[0].name = "prim";
+       regs[0].offset = data->ctrl_reg;
+
+       rp1_debugfs_regset(clockman, 0, regs, 1, dentry);
+}
+
+static int rp1_pll_ph_is_on(struct clk_hw *hw)
+{
+       struct rp1_pll_ph *pll = container_of(hw, struct rp1_pll_ph, hw);
+       struct rp1_clockman *clockman = pll->clockman;
+       const struct rp1_pll_ph_data *data = pll->data;
+
+       return !!(clockman_read(clockman, data->ph_reg) & PLL_PH_EN);
+}
+
+static int rp1_pll_ph_on(struct clk_hw *hw)
+{
+       struct rp1_pll_ph *pll_ph = container_of(hw, struct rp1_pll_ph, hw);
+       struct rp1_clockman *clockman = pll_ph->clockman;
+       const struct rp1_pll_ph_data *data = pll_ph->data;
+       u32 ph_reg;
+
+       /* todo: ensure pri/sec is enabled! */
+       spin_lock(&clockman->regs_lock);
+       ph_reg = clockman_read(clockman, data->ph_reg);
+       ph_reg |= data->phase << PLL_PH_PHASE_SHIFT;
+       ph_reg |= PLL_PH_EN;
+       clockman_write(clockman, data->ph_reg, ph_reg);
+       spin_unlock(&clockman->regs_lock);
+
+#ifdef MEASURE_CLOCK_RATE
+       clockman_measure_clock(clockman, data->name, data->fc0_src);
+#endif
+       return 0;
+}
+
+static void rp1_pll_ph_off(struct clk_hw *hw)
+{
+       struct rp1_pll_ph *pll_ph = container_of(hw, struct rp1_pll_ph, hw);
+       struct rp1_clockman *clockman = pll_ph->clockman;
+       const struct rp1_pll_ph_data *data = pll_ph->data;
+
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, data->ph_reg,
+                      clockman_read(clockman, data->ph_reg) & ~PLL_PH_EN);
+       spin_unlock(&clockman->regs_lock);
+}
+
+static int rp1_pll_ph_set_rate(struct clk_hw *hw,
+                              unsigned long rate, unsigned long parent_rate)
+{
+       struct rp1_pll_ph *pll_ph = container_of(hw, struct rp1_pll_ph, hw);
+       const struct rp1_pll_ph_data *data = pll_ph->data;
+       struct rp1_clockman *clockman = pll_ph->clockman;
+
+       /* Nothing really to do here! */
+       WARN_ON(data->fixed_divider != 1 && data->fixed_divider != 2);
+       WARN_ON(rate != parent_rate / data->fixed_divider);
+
+#ifdef MEASURE_CLOCK_RATE
+       if (rp1_pll_ph_is_on(hw))
+               clockman_measure_clock(clockman, data->name, data->fc0_src);
+#endif
+       return 0;
+}
+
+static unsigned long rp1_pll_ph_recalc_rate(struct clk_hw *hw,
+                                           unsigned long parent_rate)
+{
+       struct rp1_pll_ph *pll_ph = container_of(hw, struct rp1_pll_ph, hw);
+       const struct rp1_pll_ph_data *data = pll_ph->data;
+
+       return parent_rate / data->fixed_divider;
+}
+
+static long rp1_pll_ph_round_rate(struct clk_hw *hw, unsigned long rate,
+                                 unsigned long *parent_rate)
+{
+       struct rp1_pll_ph *pll_ph = container_of(hw, struct rp1_pll_ph, hw);
+       const struct rp1_pll_ph_data *data = pll_ph->data;
+
+       return *parent_rate / data->fixed_divider;
+}
+
+static void rp1_pll_ph_debug_init(struct clk_hw *hw,
+                                 struct dentry *dentry)
+{
+       struct rp1_pll_ph *pll_ph = container_of(hw, struct rp1_pll_ph, hw);
+       const struct rp1_pll_ph_data *data = pll_ph->data;
+       struct rp1_clockman *clockman = pll_ph->clockman;
+       struct debugfs_reg32 *regs;
+
+       regs = devm_kcalloc(clockman->dev, 1, sizeof(*regs), GFP_KERNEL);
+       if (!regs)
+               return;
+
+       regs[0].name = "ph_reg";
+       regs[0].offset = data->ph_reg;
+
+       rp1_debugfs_regset(clockman, 0, regs, 1, dentry);
+}
+
+static int rp1_pll_divider_is_on(struct clk_hw *hw)
+{
+       struct rp1_pll *divider = container_of(hw, struct rp1_pll, div.hw);
+       struct rp1_clockman *clockman = divider->clockman;
+       const struct rp1_pll_data *data = divider->data;
+
+       return !(clockman_read(clockman, data->ctrl_reg) & PLL_SEC_RST);
+}
+
+static int rp1_pll_divider_on(struct clk_hw *hw)
+{
+       struct rp1_pll *divider = container_of(hw, struct rp1_pll, div.hw);
+       struct rp1_clockman *clockman = divider->clockman;
+       const struct rp1_pll_data *data = divider->data;
+
+       spin_lock(&clockman->regs_lock);
+       /* Check the implementation bit is set! */
+       WARN_ON(!(clockman_read(clockman, data->ctrl_reg) & PLL_SEC_IMPL));
+       clockman_write(clockman, data->ctrl_reg,
+                      clockman_read(clockman, data->ctrl_reg) & ~PLL_SEC_RST);
+       spin_unlock(&clockman->regs_lock);
+
+#ifdef MEASURE_CLOCK_RATE
+       clockman_measure_clock(clockman, data->name, data->fc0_src);
+#endif
+       return 0;
+}
+
+static void rp1_pll_divider_off(struct clk_hw *hw)
+{
+       struct rp1_pll *divider = container_of(hw, struct rp1_pll, div.hw);
+       struct rp1_clockman *clockman = divider->clockman;
+       const struct rp1_pll_data *data = divider->data;
+
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, data->ctrl_reg, PLL_SEC_RST);
+       spin_unlock(&clockman->regs_lock);
+}
+
+static int rp1_pll_divider_set_rate(struct clk_hw *hw,
+                                   unsigned long rate,
+                                   unsigned long parent_rate)
+{
+       struct rp1_pll *divider = container_of(hw, struct rp1_pll, div.hw);
+       struct rp1_clockman *clockman = divider->clockman;
+       const struct rp1_pll_data *data = divider->data;
+       u32 div, sec;
+
+       div = DIV_ROUND_UP_ULL(parent_rate, rate);
+       div = clamp(div, 8u, 19u);
+
+       spin_lock(&clockman->regs_lock);
+       sec = clockman_read(clockman, data->ctrl_reg);
+       sec = set_register_field(sec, div, PLL_SEC_DIV_MASK, PLL_SEC_DIV_SHIFT);
+
+       /* Must keep the divider in reset to change the value. */
+       sec |= PLL_SEC_RST;
+       clockman_write(clockman, data->ctrl_reg, sec);
+
+       // todo: must sleep 10 pll vco cycles
+       sec &= ~PLL_SEC_RST;
+       clockman_write(clockman, data->ctrl_reg, sec);
+       spin_unlock(&clockman->regs_lock);
+
+#ifdef MEASURE_CLOCK_RATE
+       if (rp1_pll_divider_is_on(hw))
+               clockman_measure_clock(clockman, data->name, data->fc0_src);
+#endif
+       return 0;
+}
+
+static unsigned long rp1_pll_divider_recalc_rate(struct clk_hw *hw,
+                                                unsigned long parent_rate)
+{
+       return clk_divider_ops.recalc_rate(hw, parent_rate);
+}
+
+static long rp1_pll_divider_round_rate(struct clk_hw *hw,
+                                      unsigned long rate,
+                                      unsigned long *parent_rate)
+{
+       return clk_divider_ops.round_rate(hw, rate, parent_rate);
+}
+
+static void rp1_pll_divider_debug_init(struct clk_hw *hw, struct dentry *dentry)
+{
+       struct rp1_pll *divider = container_of(hw, struct rp1_pll, div.hw);
+       struct rp1_clockman *clockman = divider->clockman;
+       const struct rp1_pll_data *data = divider->data;
+       struct debugfs_reg32 *regs;
+
+       regs = devm_kcalloc(clockman->dev, 1, sizeof(*regs), GFP_KERNEL);
+       if (!regs)
+               return;
+
+       regs[0].name = "sec";
+       regs[0].offset = data->ctrl_reg;
+
+       rp1_debugfs_regset(clockman, 0, regs, 1, dentry);
+}
+
+static int rp1_clock_is_on(struct clk_hw *hw)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       struct rp1_clockman *clockman = clock->clockman;
+       const struct rp1_clock_data *data = clock->data;
+
+       return !!(clockman_read(clockman, data->ctrl_reg) & CLK_CTRL_ENABLE);
+}
+
+static unsigned long rp1_clock_recalc_rate(struct clk_hw *hw,
+                                          unsigned long parent_rate)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       struct rp1_clockman *clockman = clock->clockman;
+       const struct rp1_clock_data *data = clock->data;
+       u64 calc_rate;
+       u64 div;
+
+       u32 frac;
+
+       div = clockman_read(clockman, data->div_int_reg);
+       frac = (data->div_frac_reg != 0) ?
+               clockman_read(clockman, data->div_frac_reg) : 0;
+
+       /* If the integer portion of the divider is 0, treat it as 2^16 */
+       if (!div)
+               div = 1 << 16;
+
+       div = (div << CLK_DIV_FRAC_BITS) | (frac >> (32 - CLK_DIV_FRAC_BITS));
+
+       calc_rate = (u64)parent_rate << CLK_DIV_FRAC_BITS;
+       calc_rate = div64_u64(calc_rate, div);
+
+       return calc_rate;
+}
+
+static int rp1_clock_on(struct clk_hw *hw)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       struct rp1_clockman *clockman = clock->clockman;
+       const struct rp1_clock_data *data = clock->data;
+
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, data->ctrl_reg,
+                      clockman_read(clockman, data->ctrl_reg) | CLK_CTRL_ENABLE);
+       spin_unlock(&clockman->regs_lock);
+
+#ifdef MEASURE_CLOCK_RATE
+       clockman_measure_clock(clockman, data->name, data->fc0_src);
+#endif
+       return 0;
+}
+
+static void rp1_clock_off(struct clk_hw *hw)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       struct rp1_clockman *clockman = clock->clockman;
+       const struct rp1_clock_data *data = clock->data;
+
+       spin_lock(&clockman->regs_lock);
+       clockman_write(clockman, data->ctrl_reg,
+                      clockman_read(clockman, data->ctrl_reg) & ~CLK_CTRL_ENABLE);
+       spin_unlock(&clockman->regs_lock);
+}
+
+static u32 rp1_clock_choose_div(unsigned long rate, unsigned long parent_rate,
+                               const struct rp1_clock_data *data)
+{
+       u64 div;
+
+       /*
+        * Due to earlier rounding, calculated parent_rate may differ from
+        * expected value. Don't fail on a small discrepancy near unity divide.
+        */
+       if (!rate || rate > parent_rate + (parent_rate >> CLK_DIV_FRAC_BITS))
+               return 0;
+
+       /*
+        * Always express div in fixed-point format for fractional division;
+        * If no fractional divider is present, the fraction part will be zero.
+        */
+       if (data->div_frac_reg) {
+               div = (u64)parent_rate << CLK_DIV_FRAC_BITS;
+               div = DIV_U64_NEAREST(div, rate);
+       } else {
+               div = DIV_U64_NEAREST(parent_rate, rate);
+               div <<= CLK_DIV_FRAC_BITS;
+       }
+
+       div = clamp(div,
+                   1ull << CLK_DIV_FRAC_BITS,
+                   (u64)data->div_int_max << CLK_DIV_FRAC_BITS);
+
+       return div;
+}
+
+static u8 rp1_clock_get_parent(struct clk_hw *hw)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       struct rp1_clockman *clockman = clock->clockman;
+       const struct rp1_clock_data *data = clock->data;
+       u32 sel, ctrl;
+       u8 parent;
+
+       /* Sel is one-hot, so find the first bit set */
+       sel = clockman_read(clockman, data->sel_reg);
+       parent = ffs(sel) - 1;
+
+       /* sel == 0 implies the parent clock is not enabled yet. */
+       if (!sel) {
+               /* Read the clock src from the CTRL register instead */
+               ctrl = clockman_read(clockman, data->ctrl_reg);
+               parent = (ctrl & data->clk_src_mask) >> CLK_CTRL_SRC_SHIFT;
+       }
+
+       if (parent >= data->num_std_parents)
+               parent = AUX_SEL;
+
+       if (parent == AUX_SEL) {
+               /*
+                * Clock parent is an auxiliary source, so get the parent from
+                * the AUXSRC register field.
+                */
+               ctrl = clockman_read(clockman, data->ctrl_reg);
+               parent = (ctrl & CLK_CTRL_AUXSRC_MASK) >> CLK_CTRL_AUXSRC_SHIFT;
+               parent += data->num_std_parents;
+       }
+
+       return parent;
+}
+
+static int rp1_clock_set_parent(struct clk_hw *hw, u8 index)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       struct rp1_clockman *clockman = clock->clockman;
+       const struct rp1_clock_data *data = clock->data;
+       u32 ctrl, sel;
+
+       spin_lock(&clockman->regs_lock);
+       ctrl = clockman_read(clockman, data->ctrl_reg);
+
+       if (index >= data->num_std_parents) {
+               /* This is an aux source request */
+               if (index >= data->num_std_parents + data->num_aux_parents)
+                       return -EINVAL;
+
+               /* Select parent from aux list */
+               ctrl = set_register_field(ctrl, index - data->num_std_parents,
+                                         CLK_CTRL_AUXSRC_MASK,
+                                         CLK_CTRL_AUXSRC_SHIFT);
+               /* Set src to aux list */
+               ctrl = set_register_field(ctrl, AUX_SEL, data->clk_src_mask,
+                                         CLK_CTRL_SRC_SHIFT);
+       } else {
+               ctrl = set_register_field(ctrl, index, data->clk_src_mask,
+                                         CLK_CTRL_SRC_SHIFT);
+       }
+
+       clockman_write(clockman, data->ctrl_reg, ctrl);
+       spin_unlock(&clockman->regs_lock);
+
+       sel = rp1_clock_get_parent(hw);
+       WARN(sel != index, "(%s): Parent index req %u returned back %u\n",
+            data->name, index, sel);
+
+       return 0;
+}
+
+static int rp1_clock_set_rate_and_parent(struct clk_hw *hw,
+                                        unsigned long rate,
+                                        unsigned long parent_rate,
+                                        u8 parent)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       struct rp1_clockman *clockman = clock->clockman;
+       const struct rp1_clock_data *data = clock->data;
+       u32 div = rp1_clock_choose_div(rate, parent_rate, data);
+
+       WARN(rate > 4000000000ll, "rate is -ve (%d)\n", (int)rate);
+
+       if (WARN(!div,
+                "clk divider calculated as 0! (%s, rate %ld, parent rate %ld)\n",
+                data->name, rate, parent_rate))
+               div = 1 << CLK_DIV_FRAC_BITS;
+
+       spin_lock(&clockman->regs_lock);
+
+       clockman_write(clockman, data->div_int_reg, div >> CLK_DIV_FRAC_BITS);
+       if (data->div_frac_reg)
+               clockman_write(clockman, data->div_frac_reg, div << (32 - CLK_DIV_FRAC_BITS));
+
+       spin_unlock(&clockman->regs_lock);
+
+       if (parent != 0xff)
+               rp1_clock_set_parent(hw, parent);
+
+#ifdef MEASURE_CLOCK_RATE
+       if (rp1_clock_is_on(hw))
+               clockman_measure_clock(clockman, data->name, data->fc0_src);
+#endif
+       return 0;
+}
+
+static int rp1_clock_set_rate(struct clk_hw *hw, unsigned long rate,
+                             unsigned long parent_rate)
+{
+       return rp1_clock_set_rate_and_parent(hw, rate, parent_rate, 0xff);
+}
+
+static void rp1_clock_choose_div_and_prate(struct clk_hw *hw,
+                                          int parent_idx,
+                                          unsigned long rate,
+                                          unsigned long *prate,
+                                          unsigned long *calc_rate)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       const struct rp1_clock_data *data = clock->data;
+       struct clk_hw *parent;
+       u32 div;
+       u64 tmp;
+
+       parent = clk_hw_get_parent_by_index(hw, parent_idx);
+       *prate = clk_hw_get_rate(parent);
+       div = rp1_clock_choose_div(rate, *prate, data);
+
+       if (!div) {
+               *calc_rate = 0;
+               return;
+       }
+
+       /* Recalculate to account for rounding errors */
+       tmp = (u64)*prate << CLK_DIV_FRAC_BITS;
+       tmp = div_u64(tmp, div);
+       *calc_rate = tmp;
+}
+
+static int rp1_clock_determine_rate(struct clk_hw *hw,
+                                   struct clk_rate_request *req)
+{
+       struct clk_hw *parent, *best_parent = NULL;
+       unsigned long best_rate = 0;
+       unsigned long best_prate = 0;
+       unsigned long best_rate_diff = ULONG_MAX;
+       unsigned long prate, calc_rate;
+       size_t i;
+
+       /*
+        * If the NO_REPARENT flag is set, try to use existing parent.
+        */
+       if ((clk_hw_get_flags(hw) & CLK_SET_RATE_NO_REPARENT)) {
+               i = rp1_clock_get_parent(hw);
+               parent = clk_hw_get_parent_by_index(hw, i);
+               if (parent) {
+                       rp1_clock_choose_div_and_prate(hw, i, req->rate, &prate,
+                                                      &calc_rate);
+                       if (calc_rate > 0) {
+                               req->best_parent_hw = parent;
+                               req->best_parent_rate = prate;
+                               req->rate = calc_rate;
+                               return 0;
+                       }
+               }
+       }
+
+       /*
+        * Select parent clock that results in the closest rate (lower or
+        * higher)
+        */
+       for (i = 0; i < clk_hw_get_num_parents(hw); i++) {
+               parent = clk_hw_get_parent_by_index(hw, i);
+               if (!parent)
+                       continue;
+
+               rp1_clock_choose_div_and_prate(hw, i, req->rate, &prate,
+                                              &calc_rate);
+
+               if (ABS_DIFF(calc_rate, req->rate) < best_rate_diff) {
+                       best_parent = parent;
+                       best_prate = prate;
+                       best_rate = calc_rate;
+                       best_rate_diff = ABS_DIFF(calc_rate, req->rate);
+
+                       if (best_rate_diff == 0)
+                               break;
+               }
+       }
+
+       if (best_rate == 0)
+               return -EINVAL;
+
+       req->best_parent_hw = best_parent;
+       req->best_parent_rate = best_prate;
+       req->rate = best_rate;
+
+       return 0;
+}
+
+static void rp1_clk_debug_init(struct clk_hw *hw, struct dentry *dentry)
+{
+       struct rp1_clock *clock = container_of(hw, struct rp1_clock, hw);
+       struct rp1_clockman *clockman = clock->clockman;
+       const struct rp1_clock_data *data = clock->data;
+       struct debugfs_reg32 *regs;
+       int i;
+
+       regs = devm_kcalloc(clockman->dev, 4, sizeof(*regs), GFP_KERNEL);
+       if (!regs)
+               return;
+
+       i = 0;
+       regs[i].name = "ctrl";
+       regs[i++].offset = data->ctrl_reg;
+       regs[i].name = "div_int";
+       regs[i++].offset = data->div_int_reg;
+       regs[i].name = "div_frac";
+       regs[i++].offset = data->div_frac_reg;
+       regs[i].name = "sel";
+       regs[i++].offset = data->sel_reg;
+
+       rp1_debugfs_regset(clockman, 0, regs, i, dentry);
+}
+
+static const struct clk_ops rp1_pll_core_ops = {
+       .is_prepared = rp1_pll_core_is_on,
+       .prepare = rp1_pll_core_on,
+       .unprepare = rp1_pll_core_off,
+       .set_rate = rp1_pll_core_set_rate,
+       .recalc_rate = rp1_pll_core_recalc_rate,
+       .round_rate = rp1_pll_core_round_rate,
+       .debug_init = rp1_pll_core_debug_init,
+};
+
+static const struct clk_ops rp1_pll_ops = {
+       .set_rate = rp1_pll_set_rate,
+       .recalc_rate = rp1_pll_recalc_rate,
+       .round_rate = rp1_pll_round_rate,
+       .debug_init = rp1_pll_debug_init,
+};
+
+static const struct clk_ops rp1_pll_ph_ops = {
+       .is_prepared = rp1_pll_ph_is_on,
+       .prepare = rp1_pll_ph_on,
+       .unprepare = rp1_pll_ph_off,
+       .set_rate = rp1_pll_ph_set_rate,
+       .recalc_rate = rp1_pll_ph_recalc_rate,
+       .round_rate = rp1_pll_ph_round_rate,
+       .debug_init = rp1_pll_ph_debug_init,
+};
+
+static const struct clk_ops rp1_pll_divider_ops = {
+       .is_prepared = rp1_pll_divider_is_on,
+       .prepare = rp1_pll_divider_on,
+       .unprepare = rp1_pll_divider_off,
+       .set_rate = rp1_pll_divider_set_rate,
+       .recalc_rate = rp1_pll_divider_recalc_rate,
+       .round_rate = rp1_pll_divider_round_rate,
+       .debug_init = rp1_pll_divider_debug_init,
+};
+
+static const struct clk_ops rp1_clk_ops = {
+       .is_prepared = rp1_clock_is_on,
+       .prepare = rp1_clock_on,
+       .unprepare = rp1_clock_off,
+       .recalc_rate = rp1_clock_recalc_rate,
+       .get_parent = rp1_clock_get_parent,
+       .set_parent = rp1_clock_set_parent,
+       .set_rate_and_parent = rp1_clock_set_rate_and_parent,
+       .set_rate = rp1_clock_set_rate,
+       .determine_rate = rp1_clock_determine_rate,
+       .debug_init = rp1_clk_debug_init,
+};
+
+static bool rp1_clk_is_claimed(const char *name);
+
+static struct clk_hw *rp1_register_pll_core(struct rp1_clockman *clockman,
+                                           const void *data)
+{
+       const struct rp1_pll_core_data *pll_core_data = data;
+       struct rp1_pll_core *pll_core;
+       struct clk_init_data init;
+       int ret;
+
+       memset(&init, 0, sizeof(init));
+
+       /* All of the PLL cores derive from the external oscillator. */
+       init.parent_names = &ref_clock;
+       init.num_parents = 1;
+       init.name = pll_core_data->name;
+       init.ops = &rp1_pll_core_ops;
+       init.flags = pll_core_data->flags | CLK_IGNORE_UNUSED | CLK_IS_CRITICAL;
+
+       pll_core = kzalloc(sizeof(*pll_core), GFP_KERNEL);
+       if (!pll_core)
+               return NULL;
+
+       pll_core->clockman = clockman;
+       pll_core->data = pll_core_data;
+       pll_core->hw.init = &init;
+
+       ret = devm_clk_hw_register(clockman->dev, &pll_core->hw);
+       if (ret) {
+               kfree(pll_core);
+               return NULL;
+       }
+
+       return &pll_core->hw;
+}
+
+static struct clk_hw *rp1_register_pll(struct rp1_clockman *clockman,
+                                      const void *data)
+{
+       const struct rp1_pll_data *pll_data = data;
+       struct rp1_pll *pll;
+       struct clk_init_data init;
+       int ret;
+
+       memset(&init, 0, sizeof(init));
+
+       init.parent_names = &pll_data->source_pll;
+       init.num_parents = 1;
+       init.name = pll_data->name;
+       init.ops = &rp1_pll_ops;
+       init.flags = pll_data->flags | CLK_IGNORE_UNUSED | CLK_IS_CRITICAL;
+
+       pll = kzalloc(sizeof(*pll), GFP_KERNEL);
+       if (!pll)
+               return NULL;
+
+       pll->clockman = clockman;
+       pll->data = pll_data;
+       pll->hw.init = &init;
+
+       ret = devm_clk_hw_register(clockman->dev, &pll->hw);
+       if (ret) {
+               kfree(pll);
+               return NULL;
+       }
+
+       return &pll->hw;
+}
+
+static struct clk_hw *rp1_register_pll_ph(struct rp1_clockman *clockman,
+                                         const void *data)
+{
+       const struct rp1_pll_ph_data *ph_data = data;
+       struct rp1_pll_ph *ph;
+       struct clk_init_data init;
+       int ret;
+
+       memset(&init, 0, sizeof(init));
+
+       /* All of the PLLs derive from the external oscillator. */
+       init.parent_names = &ph_data->source_pll;
+       init.num_parents = 1;
+       init.name = ph_data->name;
+       init.ops = &rp1_pll_ph_ops;
+       init.flags = ph_data->flags | CLK_IGNORE_UNUSED;
+
+       ph = kzalloc(sizeof(*ph), GFP_KERNEL);
+       if (!ph)
+               return NULL;
+
+       ph->clockman = clockman;
+       ph->data = ph_data;
+       ph->hw.init = &init;
+
+       ret = devm_clk_hw_register(clockman->dev, &ph->hw);
+       if (ret) {
+               kfree(ph);
+               return NULL;
+       }
+
+       return &ph->hw;
+}
+
+static struct clk_hw *rp1_register_pll_divider(struct rp1_clockman *clockman,
+                                              const void *data)
+{
+       const struct rp1_pll_data *divider_data = data;
+       struct rp1_pll *divider;
+       struct clk_init_data init;
+       int ret;
+
+       memset(&init, 0, sizeof(init));
+
+       init.parent_names = &divider_data->source_pll;
+       init.num_parents = 1;
+       init.name = divider_data->name;
+       init.ops = &rp1_pll_divider_ops;
+       init.flags = divider_data->flags | CLK_IGNORE_UNUSED;
+
+       divider = devm_kzalloc(clockman->dev, sizeof(*divider), GFP_KERNEL);
+       if (!divider)
+               return NULL;
+
+       divider->div.reg = clockman->regs + divider_data->ctrl_reg;
+       divider->div.shift = PLL_SEC_DIV_SHIFT;
+       divider->div.width = PLL_SEC_DIV_WIDTH;
+       divider->div.flags = CLK_DIVIDER_ROUND_CLOSEST;
+       divider->div.lock = &clockman->regs_lock;
+       divider->div.hw.init = &init;
+       divider->div.table = pll_sec_div_table;
+
+       if (!rp1_clk_is_claimed(divider_data->source_pll))
+               init.flags |= CLK_IS_CRITICAL;
+       if (!rp1_clk_is_claimed(divider_data->name))
+               divider->div.flags |= CLK_IS_CRITICAL;
+
+       divider->clockman = clockman;
+       divider->data = divider_data;
+
+       ret = devm_clk_hw_register(clockman->dev, &divider->div.hw);
+       if (ret)
+               return ERR_PTR(ret);
+
+       return &divider->div.hw;
+}
+
+static struct clk_hw *rp1_register_clock(struct rp1_clockman *clockman,
+                                        const void *data)
+{
+       const struct rp1_clock_data *clock_data = data;
+       struct rp1_clock *clock;
+       struct clk_init_data init;
+       int ret;
+
+       BUG_ON(MAX_CLK_PARENTS <
+              clock_data->num_std_parents + clock_data->num_aux_parents);
+       /* There must be a gap for the AUX selector */
+       BUG_ON((clock_data->num_std_parents > AUX_SEL) &&
+              strcmp("-", clock_data->parents[AUX_SEL]));
+
+       memset(&init, 0, sizeof(init));
+       init.parent_names = clock_data->parents;
+       init.num_parents =
+               clock_data->num_std_parents + clock_data->num_aux_parents;
+       init.name = clock_data->name;
+       init.flags = clock_data->flags | CLK_IGNORE_UNUSED;
+       init.ops = &rp1_clk_ops;
+
+       clock = devm_kzalloc(clockman->dev, sizeof(*clock), GFP_KERNEL);
+       if (!clock)
+               return NULL;
+
+       clock->clockman = clockman;
+       clock->data = clock_data;
+       clock->hw.init = &init;
+
+       ret = devm_clk_hw_register(clockman->dev, &clock->hw);
+       if (ret)
+               return ERR_PTR(ret);
+
+       return &clock->hw;
+}
+
+struct rp1_clk_desc {
+       struct clk_hw *(*clk_register)(struct rp1_clockman *clockman,
+                                      const void *data);
+       const void *data;
+};
+
+/* Assignment helper macros for different clock types. */
+#define _REGISTER(f, ...) { .clk_register = f, .data = __VA_ARGS__ }
+
+#define REGISTER_PLL_CORE(...) _REGISTER(&rp1_register_pll_core,       \
+                                         &(struct rp1_pll_core_data)   \
+                                         {__VA_ARGS__})
+
+#define REGISTER_PLL(...)      _REGISTER(&rp1_register_pll,            \
+                                         &(struct rp1_pll_data)                \
+                                         {__VA_ARGS__})
+
+#define REGISTER_PLL_PH(...)   _REGISTER(&rp1_register_pll_ph,         \
+                                         &(struct rp1_pll_ph_data)     \
+                                         {__VA_ARGS__})
+
+#define REGISTER_PLL_DIV(...)  _REGISTER(&rp1_register_pll_divider,    \
+                                         &(struct rp1_pll_data)        \
+                                         {__VA_ARGS__})
+
+#define REGISTER_CLK(...)      _REGISTER(&rp1_register_clock,          \
+                                         &(struct rp1_clock_data)      \
+                                         {__VA_ARGS__})
+
+static const struct rp1_clk_desc clk_desc_array[] = {
+       [RP1_PLL_SYS_CORE] = REGISTER_PLL_CORE(
+                               .name = "pll_sys_core",
+                               .cs_reg = PLL_SYS_CS,
+                               .pwr_reg = PLL_SYS_PWR,
+                               .fbdiv_int_reg = PLL_SYS_FBDIV_INT,
+                               .fbdiv_frac_reg = PLL_SYS_FBDIV_FRAC,
+                               ),
+
+       [RP1_PLL_AUDIO_CORE] = REGISTER_PLL_CORE(
+                               .name = "pll_audio_core",
+                               .cs_reg = PLL_AUDIO_CS,
+                               .pwr_reg = PLL_AUDIO_PWR,
+                               .fbdiv_int_reg = PLL_AUDIO_FBDIV_INT,
+                               .fbdiv_frac_reg = PLL_AUDIO_FBDIV_FRAC,
+                               ),
+
+       [RP1_PLL_VIDEO_CORE] = REGISTER_PLL_CORE(
+                               .name = "pll_video_core",
+                               .cs_reg = PLL_VIDEO_CS,
+                               .pwr_reg = PLL_VIDEO_PWR,
+                               .fbdiv_int_reg = PLL_VIDEO_FBDIV_INT,
+                               .fbdiv_frac_reg = PLL_VIDEO_FBDIV_FRAC,
+                               ),
+
+       [RP1_PLL_SYS] = REGISTER_PLL(
+                               .name = "pll_sys",
+                               .source_pll = "pll_sys_core",
+                               .ctrl_reg = PLL_SYS_PRIM,
+                               .fc0_src = FC_NUM(0, 2),
+                               ),
+
+       [RP1_PLL_AUDIO] = REGISTER_PLL(
+                               .name = "pll_audio",
+                               .source_pll = "pll_audio_core",
+                               .ctrl_reg = PLL_AUDIO_PRIM,
+                               .fc0_src = FC_NUM(4, 2),
+                               ),
+
+       [RP1_PLL_VIDEO] = REGISTER_PLL(
+                               .name = "pll_video",
+                               .source_pll = "pll_video_core",
+                               .ctrl_reg = PLL_VIDEO_PRIM,
+                               .fc0_src = FC_NUM(3, 2),
+                               ),
+
+       [RP1_PLL_SYS_PRI_PH] = REGISTER_PLL_PH(
+                               .name = "pll_sys_pri_ph",
+                               .source_pll = "pll_sys",
+                               .ph_reg = PLL_SYS_PRIM,
+                               .fixed_divider = 2,
+                               .phase = RP1_PLL_PHASE_0,
+                               .fc0_src = FC_NUM(1, 2),
+                               ),
+
+       [RP1_PLL_AUDIO_PRI_PH] = REGISTER_PLL_PH(
+                               .name = "pll_audio_pri_ph",
+                               .source_pll = "pll_audio",
+                               .ph_reg = PLL_AUDIO_PRIM,
+                               .fixed_divider = 2,
+                               .phase = RP1_PLL_PHASE_0,
+                               .fc0_src = FC_NUM(5, 1),
+                               ),
+
+       [RP1_PLL_SYS_SEC] = REGISTER_PLL_DIV(
+                               .name = "pll_sys_sec",
+                               .source_pll = "pll_sys_core",
+                               .ctrl_reg = PLL_SYS_SEC,
+                               .fc0_src = FC_NUM(2, 2),
+                               ),
+
+       [RP1_PLL_AUDIO_SEC] = REGISTER_PLL_DIV(
+                               .name = "pll_audio_sec",
+                               .source_pll = "pll_audio_core",
+                               .ctrl_reg = PLL_AUDIO_SEC,
+                               .fc0_src = FC_NUM(6, 2),
+                               ),
+
+       [RP1_PLL_VIDEO_SEC] = REGISTER_PLL_DIV(
+                               .name = "pll_video_sec",
+                               .source_pll = "pll_video_core",
+                               .ctrl_reg = PLL_VIDEO_SEC,
+                               .fc0_src = FC_NUM(5, 3),
+                               ),
+
+       [RP1_CLK_SYS] = REGISTER_CLK(
+                               .name = "clk_sys",
+                               .parents = {"xosc", "-", "pll_sys"},
+                               .num_std_parents = 3,
+                               .num_aux_parents = 0,
+                               .ctrl_reg = CLK_SYS_CTRL,
+                               .div_int_reg = CLK_SYS_DIV_INT,
+                               .sel_reg = CLK_SYS_SEL,
+                               .div_int_max = DIV_INT_24BIT_MAX,
+                               .fc0_src = FC_NUM(0, 4),
+                               .clk_src_mask = 0x3,
+                               ),
+
+       [RP1_CLK_SLOW_SYS] = REGISTER_CLK(
+                               .name = "clk_slow_sys",
+                               .parents = {"xosc"},
+                               .num_std_parents = 1,
+                               .num_aux_parents = 0,
+                               .ctrl_reg = CLK_SLOW_SYS_CTRL,
+                               .div_int_reg = CLK_SLOW_SYS_DIV_INT,
+                               .sel_reg = CLK_SLOW_SYS_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(1, 4),
+                               .clk_src_mask = 0x1,
+                               ),
+
+       [RP1_CLK_UART] = REGISTER_CLK(
+                               .name = "clk_uart",
+                               .parents = {"pll_sys_pri_ph",
+                                           "pll_video",
+                                           "xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 3,
+                               .ctrl_reg = CLK_UART_CTRL,
+                               .div_int_reg = CLK_UART_DIV_INT,
+                               .sel_reg = CLK_UART_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(6, 7),
+                               ),
+
+       [RP1_CLK_ETH] = REGISTER_CLK(
+                               .name = "clk_eth",
+                               .parents = {"-"},
+                               .num_std_parents = 1,
+                               .num_aux_parents = 0,
+                               .ctrl_reg = CLK_ETH_CTRL,
+                               .div_int_reg = CLK_ETH_DIV_INT,
+                               .sel_reg = CLK_ETH_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(4, 6),
+                               ),
+
+       [RP1_CLK_PWM0] = REGISTER_CLK(
+                               .name = "clk_pwm0",
+                               .parents = {"pll_audio_pri_ph",
+                                           "pll_video_sec",
+                                           "xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 3,
+                               .ctrl_reg = CLK_PWM0_CTRL,
+                               .div_int_reg = CLK_PWM0_DIV_INT,
+                               .div_frac_reg = CLK_PWM0_DIV_FRAC,
+                               .sel_reg = CLK_PWM0_SEL,
+                               .div_int_max = DIV_INT_16BIT_MAX,
+                               .fc0_src = FC_NUM(0, 5),
+                               ),
+
+       [RP1_CLK_PWM1] = REGISTER_CLK(
+                               .name = "clk_pwm1",
+                               .parents = {"pll_audio_pri_ph",
+                                           "pll_video_sec",
+                                           "xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 3,
+                               .ctrl_reg = CLK_PWM1_CTRL,
+                               .div_int_reg = CLK_PWM1_DIV_INT,
+                               .div_frac_reg = CLK_PWM1_DIV_FRAC,
+                               .sel_reg = CLK_PWM1_SEL,
+                               .div_int_max = DIV_INT_16BIT_MAX,
+                               .fc0_src = FC_NUM(1, 5),
+                               ),
+
+       [RP1_CLK_AUDIO_IN] = REGISTER_CLK(
+                               .name = "clk_audio_in",
+                               .parents = {"-"},
+                               .num_std_parents = 1,
+                               .num_aux_parents = 0,
+                               .ctrl_reg = CLK_AUDIO_IN_CTRL,
+                               .div_int_reg = CLK_AUDIO_IN_DIV_INT,
+                               .sel_reg = CLK_AUDIO_IN_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(2, 5),
+                               ),
+
+       [RP1_CLK_AUDIO_OUT] = REGISTER_CLK(
+                               .name = "clk_audio_out",
+                               .parents = {"-"},
+                               .num_std_parents = 1,
+                               .num_aux_parents = 0,
+                               .ctrl_reg = CLK_AUDIO_OUT_CTRL,
+                               .div_int_reg = CLK_AUDIO_OUT_DIV_INT,
+                               .sel_reg = CLK_AUDIO_OUT_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(3, 5),
+                               ),
+
+       [RP1_CLK_I2S] = REGISTER_CLK(
+                               .name = "clk_i2s",
+                               .parents = {"xosc",
+                                           "pll_audio",
+                                           "pll_audio_sec"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 3,
+                               .ctrl_reg = CLK_I2S_CTRL,
+                               .div_int_reg = CLK_I2S_DIV_INT,
+                               .sel_reg = CLK_I2S_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(4, 4),
+                               ),
+
+       [RP1_CLK_MIPI0_CFG] = REGISTER_CLK(
+                               .name = "clk_mipi0_cfg",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_MIPI0_CFG_CTRL,
+                               .div_int_reg = CLK_MIPI0_CFG_DIV_INT,
+                               .sel_reg = CLK_MIPI0_CFG_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(4, 5),
+                               ),
+
+       [RP1_CLK_MIPI1_CFG] = REGISTER_CLK(
+                               .name = "clk_mipi1_cfg",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_MIPI1_CFG_CTRL,
+                               .div_int_reg = CLK_MIPI1_CFG_DIV_INT,
+                               .sel_reg = CLK_MIPI1_CFG_SEL,
+                               .clk_src_mask = 1,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(5, 6),
+                               ),
+
+       [RP1_CLK_ETH_TSU] = REGISTER_CLK(
+                               .name = "clk_eth_tsu",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_ETH_TSU_CTRL,
+                               .div_int_reg = CLK_ETH_TSU_DIV_INT,
+                               .sel_reg = CLK_ETH_TSU_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(5, 7),
+                               ),
+
+       [RP1_CLK_ADC] = REGISTER_CLK(
+                               .name = "clk_adc",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_ADC_CTRL,
+                               .div_int_reg = CLK_ADC_DIV_INT,
+                               .sel_reg = CLK_ADC_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(5, 5),
+                               ),
+
+       [RP1_CLK_SDIO_TIMER] = REGISTER_CLK(
+                               .name = "clk_sdio_timer",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_SDIO_TIMER_CTRL,
+                               .div_int_reg = CLK_SDIO_TIMER_DIV_INT,
+                               .sel_reg = CLK_SDIO_TIMER_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(3, 4),
+                               ),
+
+       [RP1_CLK_SDIO_ALT_SRC] = REGISTER_CLK(
+                               .name = "clk_sdio_alt_src",
+                               .parents = {"pll_sys"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_SDIO_ALT_SRC_CTRL,
+                               .div_int_reg = CLK_SDIO_ALT_SRC_DIV_INT,
+                               .sel_reg = CLK_SDIO_ALT_SRC_SEL,
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(5, 4),
+                               ),
+
+       [RP1_CLK_GP0] = REGISTER_CLK(
+                               .name = "clk_gp0",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_GP0_CTRL,
+                               .div_int_reg = CLK_GP0_DIV_INT,
+                               .div_frac_reg = CLK_GP0_DIV_FRAC,
+                               .sel_reg = CLK_GP0_SEL,
+                               .div_int_max = DIV_INT_16BIT_MAX,
+                               .fc0_src = FC_NUM(0, 1),
+                               ),
+
+       [RP1_CLK_GP1] = REGISTER_CLK(
+                               .name = "clk_gp1",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_GP1_CTRL,
+                               .div_int_reg = CLK_GP1_DIV_INT,
+                               .div_frac_reg = CLK_GP1_DIV_FRAC,
+                               .sel_reg = CLK_GP1_SEL,
+                               .div_int_max = DIV_INT_16BIT_MAX,
+                               .fc0_src = FC_NUM(1, 1),
+                               ),
+
+       [RP1_CLK_GP2] = REGISTER_CLK(
+                               .name = "clk_gp2",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_GP2_CTRL,
+                               .div_int_reg = CLK_GP2_DIV_INT,
+                               .div_frac_reg = CLK_GP2_DIV_FRAC,
+                               .sel_reg = CLK_GP2_SEL,
+                               .div_int_max = DIV_INT_16BIT_MAX,
+                               .fc0_src = FC_NUM(2, 1),
+                               ),
+
+       [RP1_CLK_GP3] = REGISTER_CLK(
+                               .name = "clk_gp3",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_GP3_CTRL,
+                               .div_int_reg = CLK_GP3_DIV_INT,
+                               .div_frac_reg = CLK_GP3_DIV_FRAC,
+                               .sel_reg = CLK_GP3_SEL,
+                               .div_int_max = DIV_INT_16BIT_MAX,
+                               .fc0_src = FC_NUM(3, 1),
+                               ),
+
+       [RP1_CLK_GP4] = REGISTER_CLK(
+                               .name = "clk_gp4",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_GP4_CTRL,
+                               .div_int_reg = CLK_GP4_DIV_INT,
+                               .div_frac_reg = CLK_GP4_DIV_FRAC,
+                               .sel_reg = CLK_GP4_SEL,
+                               .div_int_max = DIV_INT_16BIT_MAX,
+                               .fc0_src = FC_NUM(4, 1),
+                               ),
+
+       [RP1_CLK_GP5] = REGISTER_CLK(
+                               .name = "clk_gp5",
+                               .parents = {"xosc"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 1,
+                               .ctrl_reg = CLK_GP5_CTRL,
+                               .div_int_reg = CLK_GP5_DIV_INT,
+                               .div_frac_reg = CLK_GP5_DIV_FRAC,
+                               .sel_reg = CLK_GP5_SEL,
+                               .div_int_max = DIV_INT_16BIT_MAX,
+                               .fc0_src = FC_NUM(5, 1),
+                               ),
+
+       [RP1_CLK_VEC] = REGISTER_CLK(
+                               .name = "clk_vec",
+                               .parents = {"pll_sys_pri_ph",
+                                           "pll_video_sec",
+                                           "pll_video",
+                                           "clk_gp0",
+                                           "clk_gp1",
+                                           "clk_gp2",
+                                           "clk_gp3",
+                                           "clk_gp4"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 8, /* XXX in fact there are more than 8 */
+                               .ctrl_reg = VIDEO_CLK_VEC_CTRL,
+                               .div_int_reg = VIDEO_CLK_VEC_DIV_INT,
+                               .sel_reg = VIDEO_CLK_VEC_SEL,
+                               .flags = CLK_SET_RATE_NO_REPARENT, /* Let VEC driver set parent */
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(0, 6),
+                               ),
+
+       [RP1_CLK_DPI] = REGISTER_CLK(
+                               .name = "clk_dpi",
+                               .parents = {"pll_sys",
+                                           "pll_video_sec",
+                                           "pll_video",
+                                           "clk_gp0",
+                                           "clk_gp1",
+                                           "clk_gp2",
+                                           "clk_gp3",
+                                           "clk_gp4"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 8, /* XXX in fact there are more than 8 */
+                               .ctrl_reg = VIDEO_CLK_DPI_CTRL,
+                               .div_int_reg = VIDEO_CLK_DPI_DIV_INT,
+                               .sel_reg = VIDEO_CLK_DPI_SEL,
+                               .flags = CLK_SET_RATE_NO_REPARENT, /* Let DPI driver set parent */
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(1, 6),
+                               ),
+
+       [RP1_CLK_MIPI0_DPI] = REGISTER_CLK(
+                               .name = "clk_mipi0_dpi",
+                               .parents = {"pll_sys",
+                                           "pll_video_sec",
+                                           "pll_video",
+                                           "clksrc_mipi0_dsi_byteclk",
+                                           "clk_gp0",
+                                           "clk_gp1",
+                                           "clk_gp2",
+                                           "clk_gp3"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 8, /* XXX in fact there are more than 8 */
+                               .ctrl_reg = VIDEO_CLK_MIPI0_DPI_CTRL,
+                               .div_int_reg = VIDEO_CLK_MIPI0_DPI_DIV_INT,
+                               .div_frac_reg = VIDEO_CLK_MIPI0_DPI_DIV_FRAC,
+                               .sel_reg = VIDEO_CLK_MIPI0_DPI_SEL,
+                               .flags = CLK_SET_RATE_NO_REPARENT, /* Let DSI driver set parent */
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(2, 6),
+                               ),
+
+       [RP1_CLK_MIPI1_DPI] = REGISTER_CLK(
+                               .name = "clk_mipi1_dpi",
+                               .parents = {"pll_sys",
+                                           "pll_video_sec",
+                                           "pll_video",
+                                           "clksrc_mipi1_dsi_byteclk",
+                                           "clk_gp0",
+                                           "clk_gp1",
+                                           "clk_gp2",
+                                           "clk_gp3"},
+                               .num_std_parents = 0,
+                               .num_aux_parents = 8, /* XXX in fact there are more than 8 */
+                               .ctrl_reg = VIDEO_CLK_MIPI1_DPI_CTRL,
+                               .div_int_reg = VIDEO_CLK_MIPI1_DPI_DIV_INT,
+                               .div_frac_reg = VIDEO_CLK_MIPI1_DPI_DIV_FRAC,
+                               .sel_reg = VIDEO_CLK_MIPI1_DPI_SEL,
+                               .flags = CLK_SET_RATE_NO_REPARENT, /* Let DSI driver set parent */
+                               .div_int_max = DIV_INT_8BIT_MAX,
+                               .fc0_src = FC_NUM(3, 6),
+                               ),
+};
+
+static bool rp1_clk_claimed[ARRAY_SIZE(clk_desc_array)];
+
+static bool rp1_clk_is_claimed(const char *name)
+{
+       unsigned int i;
+
+       for (i = 0; i < ARRAY_SIZE(clk_desc_array); i++) {
+               if (clk_desc_array[i].data) {
+                       const char *clk_name = *(const char **)(clk_desc_array[i].data);
+
+                       if (!strcmp(name, clk_name))
+                               return rp1_clk_claimed[i];
+               }
+       }
+
+       return false;
+}
+
+static int rp1_clk_probe(struct platform_device *pdev)
+{
+       const struct rp1_clk_desc *desc;
+       struct device *dev = &pdev->dev;
+       struct rp1_clockman *clockman;
+       struct resource *res;
+       struct clk_hw **hws;
+       const size_t asize = ARRAY_SIZE(clk_desc_array);
+       u32 chip_id, platform;
+       unsigned int i;
+       u32 clk_id;
+       int ret;
+
+       clockman = devm_kzalloc(dev, struct_size(clockman, onecell.hws, asize),
+                               GFP_KERNEL);
+       if (!clockman)
+               return -ENOMEM;
+
+       rp1_get_platform(&chip_id, &platform);
+
+       spin_lock_init(&clockman->regs_lock);
+       clockman->dev = dev;
+       res = platform_get_resource(pdev, IORESOURCE_MEM, 0);
+       clockman->regs = devm_ioremap_resource(&pdev->dev, res);
+       if (IS_ERR(clockman->regs))
+               return PTR_ERR(clockman->regs);
+
+       memset(rp1_clk_claimed, 0, sizeof(rp1_clk_claimed));
+       for (i = 0;
+            !of_property_read_u32_index(pdev->dev.of_node, "claim-clocks",
+                                        i, &clk_id);
+            i++)
+               rp1_clk_claimed[clk_id] = true;
+
+       platform_set_drvdata(pdev, clockman);
+
+       clockman->onecell.num = asize;
+       hws = clockman->onecell.hws;
+
+       for (i = 0; i < asize; i++) {
+               desc = &clk_desc_array[i];
+               if (desc->clk_register && desc->data)
+                       hws[i] = desc->clk_register(clockman, desc->data);
+       }
+
+       ret = of_clk_add_hw_provider(dev->of_node, of_clk_hw_onecell_get,
+                                    &clockman->onecell);
+       if (ret)
+               return ret;
+
+       return 0;
+}
+
+static const struct of_device_id rp1_clk_of_match[] = {
+       { .compatible = "raspberrypi,rp1-clocks" },
+       {}
+};
+MODULE_DEVICE_TABLE(of, rp1_clk_of_match);
+
+static struct platform_driver rp1_clk_driver = {
+       .driver = {
+               .name = "rp1-clk",
+               .of_match_table = rp1_clk_of_match,
+       },
+       .probe = rp1_clk_probe,
+};
+
+static int __init __rp1_clk_driver_init(void)
+{
+       return platform_driver_register(&rp1_clk_driver);
+}
+postcore_initcall(__rp1_clk_driver_init);
+
+MODULE_AUTHOR("Naushir Patuck <naush@raspberrypi.com>");
+MODULE_DESCRIPTION("RP1 clock driver");
+MODULE_LICENSE("GPL");
index 5fc04d4..1305bf7 100644 (file)
 
 #define RP1_PLL_SYS_PRI_PH             6
 #define RP1_PLL_SYS_SEC_PH             7
+#define RP1_PLL_AUDIO_PRI_PH           8
 
-#define RP1_PLL_SYS_SEC                        8
-#define RP1_PLL_AUDIO_SEC              9
-#define RP1_PLL_VIDEO_SEC              10
+#define RP1_PLL_SYS_SEC                        9
+#define RP1_PLL_AUDIO_SEC              10
+#define RP1_PLL_VIDEO_SEC              11
 
-#define RP1_CLK_SYS                    11
-#define RP1_CLK_SLOW_SYS               12
-#define RP1_CLK_DMA                    13
-#define RP1_CLK_UART                   14
-#define RP1_CLK_ETH                    15
-#define RP1_CLK_PWM0                   16
-#define RP1_CLK_PWM1                   17
-#define RP1_CLK_AUDIO_IN               18
-#define RP1_CLK_AUDIO_OUT              19
-#define RP1_CLK_I2S                    20
-#define RP1_CLK_MIPI0_CFG              21
-#define RP1_CLK_MIPI1_CFG              22
-#define RP1_CLK_PCIE_AUX               23
-#define RP1_CLK_USBH0_MICROFRAME       24
-#define RP1_CLK_USBH1_MICROFRAME       25
-#define RP1_CLK_USBH0_SUSPEND          26
-#define RP1_CLK_USBH1_SUSPEND          27
-#define RP1_CLK_ETH_TSU                        28
-#define RP1_CLK_ADC                    29
-#define RP1_CLK_SDIO_TIMER             30
-#define RP1_CLK_SDIO_ALT_SRC           31
-#define RP1_CLK_GP0                    32
-#define RP1_CLK_GP1                    33
-#define RP1_CLK_GP2                    34
-#define RP1_CLK_GP3                    35
-#define RP1_CLK_GP4                    36
-#define RP1_CLK_GP5                    37
-#define RP1_CLK_VEC                    38
-#define RP1_CLK_DPI                    39
-#define RP1_CLK_MIPI0_DPI              40
-#define RP1_CLK_MIPI1_DPI              41
+#define RP1_CLK_SYS                    12
+#define RP1_CLK_SLOW_SYS               13
+#define RP1_CLK_DMA                    14
+#define RP1_CLK_UART                   15
+#define RP1_CLK_ETH                    16
+#define RP1_CLK_PWM0                   17
+#define RP1_CLK_PWM1                   18
+#define RP1_CLK_AUDIO_IN               19
+#define RP1_CLK_AUDIO_OUT              20
+#define RP1_CLK_I2S                    21
+#define RP1_CLK_MIPI0_CFG              22
+#define RP1_CLK_MIPI1_CFG              23
+#define RP1_CLK_PCIE_AUX               24
+#define RP1_CLK_USBH0_MICROFRAME       25
+#define RP1_CLK_USBH1_MICROFRAME       26
+#define RP1_CLK_USBH0_SUSPEND          27
+#define RP1_CLK_USBH1_SUSPEND          28
+#define RP1_CLK_ETH_TSU                        29
+#define RP1_CLK_ADC                    30
+#define RP1_CLK_SDIO_TIMER             31
+#define RP1_CLK_SDIO_ALT_SRC           32
+#define RP1_CLK_GP0                    33
+#define RP1_CLK_GP1                    34
+#define RP1_CLK_GP2                    35
+#define RP1_CLK_GP3                    36
+#define RP1_CLK_GP4                    37
+#define RP1_CLK_GP5                    38
+#define RP1_CLK_VEC                    39
+#define RP1_CLK_DPI                    40
+#define RP1_CLK_MIPI0_DPI              41
+#define RP1_CLK_MIPI1_DPI              42