root/drivers/clk/meson/axg.c
// SPDX-License-Identifier: GPL-2.0+
/*
 * AmLogic Meson-AXG Clock Controller Driver
 *
 * Copyright (c) 2016 Baylibre SAS.
 * Author: Michael Turquette <mturquette@baylibre.com>
 *
 * Copyright (c) 2017 Amlogic, inc.
 * Author: Qiufang Dai <qiufang.dai@amlogic.com>
 */

#include <linux/clk-provider.h>
#include <linux/init.h>
#include <linux/mod_devicetable.h>
#include <linux/platform_device.h>
#include <linux/module.h>

#include "clk-regmap.h"
#include "clk-pll.h"
#include "clk-mpll.h"
#include "meson-clkc-utils.h"

#include <dt-bindings/clock/axg-clkc.h>

#define HHI_GP0_PLL_CNTL                0x40
#define HHI_GP0_PLL_CNTL2               0x44
#define HHI_GP0_PLL_CNTL3               0x48
#define HHI_GP0_PLL_CNTL4               0x4c
#define HHI_GP0_PLL_CNTL5               0x50
#define HHI_GP0_PLL_STS                 0x54
#define HHI_GP0_PLL_CNTL1               0x58
#define HHI_HIFI_PLL_CNTL               0x80
#define HHI_HIFI_PLL_CNTL2              0x84
#define HHI_HIFI_PLL_CNTL3              0x88
#define HHI_HIFI_PLL_CNTL4              0x8C
#define HHI_HIFI_PLL_CNTL5              0x90
#define HHI_HIFI_PLL_STS                0x94
#define HHI_HIFI_PLL_CNTL1              0x98

#define HHI_XTAL_DIVN_CNTL              0xbc
#define HHI_GCLK2_MPEG0                 0xc0
#define HHI_GCLK2_MPEG1                 0xc4
#define HHI_GCLK2_MPEG2                 0xc8
#define HHI_GCLK2_OTHER                 0xd0
#define HHI_GCLK2_AO                    0xd4
#define HHI_PCIE_PLL_CNTL               0xd8
#define HHI_PCIE_PLL_CNTL1              0xdC
#define HHI_PCIE_PLL_CNTL2              0xe0
#define HHI_PCIE_PLL_CNTL3              0xe4
#define HHI_PCIE_PLL_CNTL4              0xe8
#define HHI_PCIE_PLL_CNTL5              0xec
#define HHI_PCIE_PLL_CNTL6              0xf0
#define HHI_PCIE_PLL_STS                0xf4

#define HHI_MEM_PD_REG0                 0x100
#define HHI_VPU_MEM_PD_REG0             0x104
#define HHI_VIID_CLK_DIV                0x128
#define HHI_VIID_CLK_CNTL               0x12c

#define HHI_GCLK_MPEG0                  0x140
#define HHI_GCLK_MPEG1                  0x144
#define HHI_GCLK_MPEG2                  0x148
#define HHI_GCLK_OTHER                  0x150
#define HHI_GCLK_AO                     0x154
#define HHI_SYS_CPU_CLK_CNTL1           0x15c
#define HHI_SYS_CPU_RESET_CNTL          0x160
#define HHI_VID_CLK_DIV                 0x164
#define HHI_SPICC_HCLK_CNTL             0x168

#define HHI_MPEG_CLK_CNTL               0x174
#define HHI_VID_CLK_CNTL                0x17c
#define HHI_TS_CLK_CNTL                 0x190
#define HHI_VID_CLK_CNTL2               0x194
#define HHI_SYS_CPU_CLK_CNTL0           0x19c
#define HHI_VID_PLL_CLK_DIV             0x1a0
#define HHI_VPU_CLK_CNTL                0x1bC

#define HHI_VAPBCLK_CNTL                0x1F4

#define HHI_GEN_CLK_CNTL                0x228

#define HHI_VDIN_MEAS_CLK_CNTL          0x250
#define HHI_NAND_CLK_CNTL               0x25C
#define HHI_SD_EMMC_CLK_CNTL            0x264

#define HHI_MPLL_CNTL                   0x280
#define HHI_MPLL_CNTL2                  0x284
#define HHI_MPLL_CNTL3                  0x288
#define HHI_MPLL_CNTL4                  0x28C
#define HHI_MPLL_CNTL5                  0x290
#define HHI_MPLL_CNTL6                  0x294
#define HHI_MPLL_CNTL7                  0x298
#define HHI_MPLL_CNTL8                  0x29C
#define HHI_MPLL_CNTL9                  0x2A0
#define HHI_MPLL_CNTL10                 0x2A4

#define HHI_MPLL3_CNTL0                 0x2E0
#define HHI_MPLL3_CNTL1                 0x2E4
#define HHI_PLL_TOP_MISC                0x2E8

#define HHI_SYS_PLL_CNTL1               0x2FC
#define HHI_SYS_PLL_CNTL                0x300
#define HHI_SYS_PLL_CNTL2               0x304
#define HHI_SYS_PLL_CNTL3               0x308
#define HHI_SYS_PLL_CNTL4               0x30c
#define HHI_SYS_PLL_CNTL5               0x310
#define HHI_SYS_PLL_STS                 0x314
#define HHI_DPLL_TOP_I                  0x318
#define HHI_DPLL_TOP2_I                 0x31C

static struct clk_regmap axg_fixed_pll_dco = {
        .data = &(struct meson_clk_pll_data){
                .en = {
                        .reg_off = HHI_MPLL_CNTL,
                        .shift   = 30,
                        .width   = 1,
                },
                .m = {
                        .reg_off = HHI_MPLL_CNTL,
                        .shift   = 0,
                        .width   = 9,
                },
                .n = {
                        .reg_off = HHI_MPLL_CNTL,
                        .shift   = 9,
                        .width   = 5,
                },
                .frac = {
                        .reg_off = HHI_MPLL_CNTL2,
                        .shift   = 0,
                        .width   = 12,
                },
                .l = {
                        .reg_off = HHI_MPLL_CNTL,
                        .shift   = 31,
                        .width   = 1,
                },
                .rst = {
                        .reg_off = HHI_MPLL_CNTL,
                        .shift   = 29,
                        .width   = 1,
                },
        },
        .hw.init = &(struct clk_init_data){
                .name = "fixed_pll_dco",
                .ops = &meson_clk_pll_ro_ops,
                .parent_data = &(const struct clk_parent_data) {
                        .fw_name = "xtal",
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_fixed_pll = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_MPLL_CNTL,
                .shift = 16,
                .width = 2,
                .flags = CLK_DIVIDER_POWER_OF_TWO,
        },
        .hw.init = &(struct clk_init_data){
                .name = "fixed_pll",
                .ops = &clk_regmap_divider_ro_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_fixed_pll_dco.hw
                },
                .num_parents = 1,
                /*
                 * This clock won't ever change at runtime so
                 * CLK_SET_RATE_PARENT is not required
                 */
        },
};

static struct clk_regmap axg_sys_pll_dco = {
        .data = &(struct meson_clk_pll_data){
                .en = {
                        .reg_off = HHI_SYS_PLL_CNTL,
                        .shift   = 30,
                        .width   = 1,
                },
                .m = {
                        .reg_off = HHI_SYS_PLL_CNTL,
                        .shift   = 0,
                        .width   = 9,
                },
                .n = {
                        .reg_off = HHI_SYS_PLL_CNTL,
                        .shift   = 9,
                        .width   = 5,
                },
                .l = {
                        .reg_off = HHI_SYS_PLL_CNTL,
                        .shift   = 31,
                        .width   = 1,
                },
                .rst = {
                        .reg_off = HHI_SYS_PLL_CNTL,
                        .shift   = 29,
                        .width   = 1,
                },
        },
        .hw.init = &(struct clk_init_data){
                .name = "sys_pll_dco",
                .ops = &meson_clk_pll_ro_ops,
                .parent_data = &(const struct clk_parent_data) {
                        .fw_name = "xtal",
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_sys_pll = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_SYS_PLL_CNTL,
                .shift = 16,
                .width = 2,
                .flags = CLK_DIVIDER_POWER_OF_TWO,
        },
        .hw.init = &(struct clk_init_data){
                .name = "sys_pll",
                .ops = &clk_regmap_divider_ro_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_sys_pll_dco.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static const struct pll_params_table axg_gp0_pll_params_table[] = {
        PLL_PARAMS(40, 1),
        PLL_PARAMS(41, 1),
        PLL_PARAMS(42, 1),
        PLL_PARAMS(43, 1),
        PLL_PARAMS(44, 1),
        PLL_PARAMS(45, 1),
        PLL_PARAMS(46, 1),
        PLL_PARAMS(47, 1),
        PLL_PARAMS(48, 1),
        PLL_PARAMS(49, 1),
        PLL_PARAMS(50, 1),
        PLL_PARAMS(51, 1),
        PLL_PARAMS(52, 1),
        PLL_PARAMS(53, 1),
        PLL_PARAMS(54, 1),
        PLL_PARAMS(55, 1),
        PLL_PARAMS(56, 1),
        PLL_PARAMS(57, 1),
        PLL_PARAMS(58, 1),
        PLL_PARAMS(59, 1),
        PLL_PARAMS(60, 1),
        PLL_PARAMS(61, 1),
        PLL_PARAMS(62, 1),
        PLL_PARAMS(63, 1),
        PLL_PARAMS(64, 1),
        PLL_PARAMS(65, 1),
        PLL_PARAMS(66, 1),
        PLL_PARAMS(67, 1),
        PLL_PARAMS(68, 1),
        { /* sentinel */ },
};

static const struct reg_sequence axg_gp0_init_regs[] = {
        { .reg = HHI_GP0_PLL_CNTL1,     .def = 0xc084b000 },
        { .reg = HHI_GP0_PLL_CNTL2,     .def = 0xb75020be },
        { .reg = HHI_GP0_PLL_CNTL3,     .def = 0x0a59a288 },
        { .reg = HHI_GP0_PLL_CNTL4,     .def = 0xc000004d },
        { .reg = HHI_GP0_PLL_CNTL5,     .def = 0x00078000 },
};

static struct clk_regmap axg_gp0_pll_dco = {
        .data = &(struct meson_clk_pll_data){
                .en = {
                        .reg_off = HHI_GP0_PLL_CNTL,
                        .shift   = 30,
                        .width   = 1,
                },
                .m = {
                        .reg_off = HHI_GP0_PLL_CNTL,
                        .shift   = 0,
                        .width   = 9,
                },
                .n = {
                        .reg_off = HHI_GP0_PLL_CNTL,
                        .shift   = 9,
                        .width   = 5,
                },
                .frac = {
                        .reg_off = HHI_GP0_PLL_CNTL1,
                        .shift   = 0,
                        .width   = 10,
                },
                .l = {
                        .reg_off = HHI_GP0_PLL_CNTL,
                        .shift   = 31,
                        .width   = 1,
                },
                .rst = {
                        .reg_off = HHI_GP0_PLL_CNTL,
                        .shift   = 29,
                        .width   = 1,
                },
                .table = axg_gp0_pll_params_table,
                .init_regs = axg_gp0_init_regs,
                .init_count = ARRAY_SIZE(axg_gp0_init_regs),
        },
        .hw.init = &(struct clk_init_data){
                .name = "gp0_pll_dco",
                .ops = &meson_clk_pll_ops,
                .parent_data = &(const struct clk_parent_data) {
                        .fw_name = "xtal",
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_gp0_pll = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_GP0_PLL_CNTL,
                .shift = 16,
                .width = 2,
                .flags = CLK_DIVIDER_POWER_OF_TWO,
        },
        .hw.init = &(struct clk_init_data){
                .name = "gp0_pll",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_gp0_pll_dco.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static const struct reg_sequence axg_hifi_pll_init_regs[] = {
        { .reg = HHI_HIFI_PLL_CNTL1,    .def = 0xc084b000 },
        { .reg = HHI_HIFI_PLL_CNTL2,    .def = 0xb75020be },
        { .reg = HHI_HIFI_PLL_CNTL3,    .def = 0x0a6a3a88 },
        { .reg = HHI_HIFI_PLL_CNTL4,    .def = 0xc000004d },
        { .reg = HHI_HIFI_PLL_CNTL5,    .def = 0x00058000 },
};

static struct clk_regmap axg_hifi_pll_dco = {
        .data = &(struct meson_clk_pll_data){
                .en = {
                        .reg_off = HHI_HIFI_PLL_CNTL,
                        .shift   = 30,
                        .width   = 1,
                },
                .m = {
                        .reg_off = HHI_HIFI_PLL_CNTL,
                        .shift   = 0,
                        .width   = 9,
                },
                .n = {
                        .reg_off = HHI_HIFI_PLL_CNTL,
                        .shift   = 9,
                        .width   = 5,
                },
                .frac = {
                        .reg_off = HHI_HIFI_PLL_CNTL5,
                        .shift   = 0,
                        .width   = 13,
                },
                .l = {
                        .reg_off = HHI_HIFI_PLL_CNTL,
                        .shift   = 31,
                        .width   = 1,
                },
                .rst = {
                        .reg_off = HHI_HIFI_PLL_CNTL,
                        .shift   = 29,
                        .width   = 1,
                },
                .table = axg_gp0_pll_params_table,
                .init_regs = axg_hifi_pll_init_regs,
                .init_count = ARRAY_SIZE(axg_hifi_pll_init_regs),
                .flags = CLK_MESON_PLL_ROUND_CLOSEST,
        },
        .hw.init = &(struct clk_init_data){
                .name = "hifi_pll_dco",
                .ops = &meson_clk_pll_ops,
                .parent_data = &(const struct clk_parent_data) {
                        .fw_name = "xtal",
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_hifi_pll = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_HIFI_PLL_CNTL,
                .shift = 16,
                .width = 2,
                .flags = CLK_DIVIDER_POWER_OF_TWO,
        },
        .hw.init = &(struct clk_init_data){
                .name = "hifi_pll",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_hifi_pll_dco.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_fixed_factor axg_fclk_div2_div = {
        .mult = 1,
        .div = 2,
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div2_div",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_fixed_pll.hw },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_fclk_div2 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL_CNTL6,
                .bit_idx = 27,
        },
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div2",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_fclk_div2_div.hw
                },
                .num_parents = 1,
                .flags = CLK_IS_CRITICAL,
        },
};

static struct clk_fixed_factor axg_fclk_div3_div = {
        .mult = 1,
        .div = 3,
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div3_div",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_fixed_pll.hw },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_fclk_div3 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL_CNTL6,
                .bit_idx = 28,
        },
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div3",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_fclk_div3_div.hw
                },
                .num_parents = 1,
                /*
                 * FIXME:
                 * This clock, as fdiv2, is used by the SCPI FW and is required
                 * by the platform to operate correctly.
                 * Until the following condition are met, we need this clock to
                 * be marked as critical:
                 * a) The SCPI generic driver claims and enable all the clocks
                 *    it needs
                 * b) CCF has a clock hand-off mechanism to make the sure the
                 *    clock stays on until the proper driver comes along
                 */
                .flags = CLK_IS_CRITICAL,
        },
};

static struct clk_fixed_factor axg_fclk_div4_div = {
        .mult = 1,
        .div = 4,
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div4_div",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_fixed_pll.hw },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_fclk_div4 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL_CNTL6,
                .bit_idx = 29,
        },
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div4",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_fclk_div4_div.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_fclk_div5_div = {
        .mult = 1,
        .div = 5,
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div5_div",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_fixed_pll.hw },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_fclk_div5 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL_CNTL6,
                .bit_idx = 30,
        },
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div5",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_fclk_div5_div.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_fclk_div7_div = {
        .mult = 1,
        .div = 7,
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div7_div",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_fixed_pll.hw
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_fclk_div7 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL_CNTL6,
                .bit_idx = 31,
        },
        .hw.init = &(struct clk_init_data){
                .name = "fclk_div7",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_fclk_div7_div.hw
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_mpll_prediv = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_MPLL_CNTL5,
                .shift = 12,
                .width = 1,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll_prediv",
                .ops = &clk_regmap_divider_ro_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_fixed_pll.hw
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_mpll0_div = {
        .data = &(struct meson_clk_mpll_data){
                .sdm = {
                        .reg_off = HHI_MPLL_CNTL7,
                        .shift   = 0,
                        .width   = 14,
                },
                .sdm_en = {
                        .reg_off = HHI_MPLL_CNTL7,
                        .shift   = 15,
                        .width   = 1,
                },
                .n2 = {
                        .reg_off = HHI_MPLL_CNTL7,
                        .shift   = 16,
                        .width   = 9,
                },
                .misc = {
                        .reg_off = HHI_PLL_TOP_MISC,
                        .shift   = 0,
                        .width   = 1,
                },
                .flags = CLK_MESON_MPLL_ROUND_CLOSEST,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll0_div",
                .ops = &meson_clk_mpll_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_mpll_prediv.hw
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_mpll0 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL_CNTL7,
                .bit_idx = 14,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll0",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_mpll0_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_mpll1_div = {
        .data = &(struct meson_clk_mpll_data){
                .sdm = {
                        .reg_off = HHI_MPLL_CNTL8,
                        .shift   = 0,
                        .width   = 14,
                },
                .sdm_en = {
                        .reg_off = HHI_MPLL_CNTL8,
                        .shift   = 15,
                        .width   = 1,
                },
                .n2 = {
                        .reg_off = HHI_MPLL_CNTL8,
                        .shift   = 16,
                        .width   = 9,
                },
                .misc = {
                        .reg_off = HHI_PLL_TOP_MISC,
                        .shift   = 1,
                        .width   = 1,
                },
                .flags = CLK_MESON_MPLL_ROUND_CLOSEST,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll1_div",
                .ops = &meson_clk_mpll_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_mpll_prediv.hw
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_mpll1 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL_CNTL8,
                .bit_idx = 14,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll1",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_mpll1_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_mpll2_div = {
        .data = &(struct meson_clk_mpll_data){
                .sdm = {
                        .reg_off = HHI_MPLL_CNTL9,
                        .shift   = 0,
                        .width   = 14,
                },
                .sdm_en = {
                        .reg_off = HHI_MPLL_CNTL9,
                        .shift   = 15,
                        .width   = 1,
                },
                .n2 = {
                        .reg_off = HHI_MPLL_CNTL9,
                        .shift   = 16,
                        .width   = 9,
                },
                .ssen = {
                        .reg_off = HHI_MPLL_CNTL,
                        .shift   = 25,
                        .width   = 1,
                },
                .misc = {
                        .reg_off = HHI_PLL_TOP_MISC,
                        .shift   = 2,
                        .width   = 1,
                },
                .flags = CLK_MESON_MPLL_ROUND_CLOSEST,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll2_div",
                .ops = &meson_clk_mpll_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_mpll_prediv.hw
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_mpll2 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL_CNTL9,
                .bit_idx = 14,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll2",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_mpll2_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_mpll3_div = {
        .data = &(struct meson_clk_mpll_data){
                .sdm = {
                        .reg_off = HHI_MPLL3_CNTL0,
                        .shift   = 12,
                        .width   = 14,
                },
                .sdm_en = {
                        .reg_off = HHI_MPLL3_CNTL0,
                        .shift   = 11,
                        .width   = 1,
                },
                .n2 = {
                        .reg_off = HHI_MPLL3_CNTL0,
                        .shift   = 2,
                        .width   = 9,
                },
                .misc = {
                        .reg_off = HHI_PLL_TOP_MISC,
                        .shift   = 3,
                        .width   = 1,
                },
                .flags = CLK_MESON_MPLL_ROUND_CLOSEST,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll3_div",
                .ops = &meson_clk_mpll_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_mpll_prediv.hw
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_mpll3 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPLL3_CNTL0,
                .bit_idx = 0,
        },
        .hw.init = &(struct clk_init_data){
                .name = "mpll3",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_mpll3_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static const struct pll_params_table axg_pcie_pll_params_table[] = {
        {
                .m = 200,
                .n = 3,
        },
        { /* sentinel */ },
};

static const struct reg_sequence axg_pcie_pll_init_regs[] = {
        { .reg = HHI_PCIE_PLL_CNTL1,    .def = 0x0084a2aa },
        { .reg = HHI_PCIE_PLL_CNTL2,    .def = 0xb75020be },
        { .reg = HHI_PCIE_PLL_CNTL3,    .def = 0x0a47488e },
        { .reg = HHI_PCIE_PLL_CNTL4,    .def = 0xc000004d },
        { .reg = HHI_PCIE_PLL_CNTL5,    .def = 0x00078000 },
        { .reg = HHI_PCIE_PLL_CNTL6,    .def = 0x002323c6 },
        { .reg = HHI_PCIE_PLL_CNTL,     .def = 0x400106c8 },
};

static struct clk_regmap axg_pcie_pll_dco = {
        .data = &(struct meson_clk_pll_data){
                .en = {
                        .reg_off = HHI_PCIE_PLL_CNTL,
                        .shift   = 30,
                        .width   = 1,
                },
                .m = {
                        .reg_off = HHI_PCIE_PLL_CNTL,
                        .shift   = 0,
                        .width   = 9,
                },
                .n = {
                        .reg_off = HHI_PCIE_PLL_CNTL,
                        .shift   = 9,
                        .width   = 5,
                },
                .frac = {
                        .reg_off = HHI_PCIE_PLL_CNTL1,
                        .shift   = 0,
                        .width   = 12,
                },
                .l = {
                        .reg_off = HHI_PCIE_PLL_CNTL,
                        .shift   = 31,
                        .width   = 1,
                },
                .rst = {
                        .reg_off = HHI_PCIE_PLL_CNTL,
                        .shift   = 29,
                        .width   = 1,
                },
                .table = axg_pcie_pll_params_table,
                .init_regs = axg_pcie_pll_init_regs,
                .init_count = ARRAY_SIZE(axg_pcie_pll_init_regs),
        },
        .hw.init = &(struct clk_init_data){
                .name = "pcie_pll_dco",
                .ops = &meson_clk_pll_ops,
                .parent_data = &(const struct clk_parent_data) {
                        .fw_name = "xtal",
                },
                .num_parents = 1,
        },
};

static struct clk_regmap axg_pcie_pll_od = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_PCIE_PLL_CNTL,
                .shift = 16,
                .width = 2,
                .flags = CLK_DIVIDER_POWER_OF_TWO,
        },
        .hw.init = &(struct clk_init_data){
                .name = "pcie_pll_od",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_pcie_pll_dco.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_pcie_pll = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_PCIE_PLL_CNTL6,
                .shift = 6,
                .width = 2,
                .flags = CLK_DIVIDER_POWER_OF_TWO,
        },
        .hw.init = &(struct clk_init_data){
                .name = "pcie_pll",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_pcie_pll_od.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_pcie_mux = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_PCIE_PLL_CNTL6,
                .mask = 0x1,
                .shift = 2,
                /* skip the parent mpll3, reserved for debug */
                .table = (u32[]){ 1 },
        },
        .hw.init = &(struct clk_init_data){
                .name = "pcie_mux",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_pcie_pll.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_pcie_ref = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_PCIE_PLL_CNTL6,
                .mask = 0x1,
                .shift = 1,
                /* skip the parent 0, reserved for debug */
                .table = (u32[]){ 1 },
        },
        .hw.init = &(struct clk_init_data){
                .name = "pcie_ref",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_pcie_mux.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_pcie_cml_en0 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_PCIE_PLL_CNTL6,
                .bit_idx = 4,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "pcie_cml_en0",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_pcie_ref.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,

        },
};

static struct clk_regmap axg_pcie_cml_en1 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_PCIE_PLL_CNTL6,
                .bit_idx = 3,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "pcie_cml_en1",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_pcie_ref.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

/* clk81 is often referred as "mpeg_clk" */
static u32 clk81_parents_val_table[] = { 0, 2, 3, 4, 5, 6, 7 };
static const struct clk_parent_data clk81_parents[] = {
        { .fw_name = "xtal", },
        { .hw = &axg_fclk_div7.hw },
        { .hw = &axg_mpll1.hw },
        { .hw = &axg_mpll2.hw },
        { .hw = &axg_fclk_div4.hw },
        { .hw = &axg_fclk_div3.hw },
        { .hw = &axg_fclk_div5.hw },
};

static struct clk_regmap axg_clk81_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_MPEG_CLK_CNTL,
                .mask = 0x7,
                .shift = 12,
                .table = clk81_parents_val_table,
        },
        .hw.init = &(struct clk_init_data){
                .name = "clk81_sel",
                .ops = &clk_regmap_mux_ro_ops,
                .parent_data = clk81_parents,
                .num_parents = ARRAY_SIZE(clk81_parents),
        },
};

static struct clk_regmap axg_clk81_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_MPEG_CLK_CNTL,
                .shift = 0,
                .width = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "clk81_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_clk81_sel.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_clk81 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_MPEG_CLK_CNTL,
                .bit_idx = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "clk81",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_clk81_div.hw
                },
                .num_parents = 1,
                .flags = (CLK_SET_RATE_PARENT | CLK_IS_CRITICAL),
        },
};

static const struct clk_parent_data axg_sd_emmc_clk0_parents[] = {
        { .fw_name = "xtal", },
        { .hw = &axg_fclk_div2.hw },
        { .hw = &axg_fclk_div3.hw },
        { .hw = &axg_fclk_div5.hw },
        { .hw = &axg_fclk_div7.hw },
        /*
         * Following these parent clocks, we should also have had mpll2, mpll3
         * and gp0_pll but these clocks are too precious to be used here. All
         * the necessary rates for MMC and NAND operation can be achieved using
         * xtal or fclk_div clocks
         */
};

/* SDcard clock */
static struct clk_regmap axg_sd_emmc_b_clk0_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_SD_EMMC_CLK_CNTL,
                .mask = 0x7,
                .shift = 25,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "sd_emmc_b_clk0_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_data = axg_sd_emmc_clk0_parents,
                .num_parents = ARRAY_SIZE(axg_sd_emmc_clk0_parents),
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_sd_emmc_b_clk0_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_SD_EMMC_CLK_CNTL,
                .shift = 16,
                .width = 7,
                .flags = CLK_DIVIDER_ROUND_CLOSEST,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "sd_emmc_b_clk0_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_sd_emmc_b_clk0_sel.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_sd_emmc_b_clk0 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_SD_EMMC_CLK_CNTL,
                .bit_idx = 23,
        },
        .hw.init = &(struct clk_init_data){
                .name = "sd_emmc_b_clk0",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_sd_emmc_b_clk0_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

/* EMMC/NAND clock */
static struct clk_regmap axg_sd_emmc_c_clk0_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_NAND_CLK_CNTL,
                .mask = 0x7,
                .shift = 9,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "sd_emmc_c_clk0_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_data = axg_sd_emmc_clk0_parents,
                .num_parents = ARRAY_SIZE(axg_sd_emmc_clk0_parents),
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_sd_emmc_c_clk0_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_NAND_CLK_CNTL,
                .shift = 0,
                .width = 7,
                .flags = CLK_DIVIDER_ROUND_CLOSEST,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "sd_emmc_c_clk0_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_sd_emmc_c_clk0_sel.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_sd_emmc_c_clk0 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_NAND_CLK_CNTL,
                .bit_idx = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "sd_emmc_c_clk0",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_sd_emmc_c_clk0_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

/* VPU Clock */

static const struct clk_hw *axg_vpu_parents[] = {
        &axg_fclk_div4.hw,
        &axg_fclk_div3.hw,
        &axg_fclk_div5.hw,
        &axg_fclk_div7.hw,
};

static struct clk_regmap axg_vpu_0_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VPU_CLK_CNTL,
                .mask = 0x3,
                .shift = 9,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vpu_0_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = axg_vpu_parents,
                .num_parents = ARRAY_SIZE(axg_vpu_parents),
                /* We need a specific parent for VPU clock source, let it be set in DT */
                .flags = CLK_SET_RATE_NO_REPARENT,
        },
};

static struct clk_regmap axg_vpu_0_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_VPU_CLK_CNTL,
                .shift = 0,
                .width = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vpu_0_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vpu_0_sel.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_vpu_0 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VPU_CLK_CNTL,
                .bit_idx = 8,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vpu_0",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vpu_0_div.hw },
                .num_parents = 1,
                /*
                 * We want to avoid CCF to disable the VPU clock if
                 * display has been set by Bootloader
                 */
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vpu_1_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VPU_CLK_CNTL,
                .mask = 0x3,
                .shift = 25,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vpu_1_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = axg_vpu_parents,
                .num_parents = ARRAY_SIZE(axg_vpu_parents),
                /* We need a specific parent for VPU clock source, let it be set in DT */
                .flags = CLK_SET_RATE_NO_REPARENT,
        },
};

static struct clk_regmap axg_vpu_1_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_VPU_CLK_CNTL,
                .shift = 16,
                .width = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vpu_1_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vpu_1_sel.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_vpu_1 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VPU_CLK_CNTL,
                .bit_idx = 24,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vpu_1",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vpu_1_div.hw },
                .num_parents = 1,
                /*
                 * We want to avoid CCF to disable the VPU clock if
                 * display has been set by Bootloader
                 */
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vpu = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VPU_CLK_CNTL,
                .mask = 1,
                .shift = 31,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vpu",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vpu_0.hw,
                        &axg_vpu_1.hw
                },
                .num_parents = 2,
                .flags = CLK_SET_RATE_NO_REPARENT,
        },
};

/* VAPB Clock */

static struct clk_regmap axg_vapb_0_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VAPBCLK_CNTL,
                .mask = 0x3,
                .shift = 9,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vapb_0_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = axg_vpu_parents,
                .num_parents = ARRAY_SIZE(axg_vpu_parents),
                .flags = CLK_SET_RATE_NO_REPARENT,
        },
};

static struct clk_regmap axg_vapb_0_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_VAPBCLK_CNTL,
                .shift = 0,
                .width = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vapb_0_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vapb_0_sel.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_vapb_0 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VAPBCLK_CNTL,
                .bit_idx = 8,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vapb_0",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vapb_0_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vapb_1_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VAPBCLK_CNTL,
                .mask = 0x3,
                .shift = 25,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vapb_1_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = axg_vpu_parents,
                .num_parents = ARRAY_SIZE(axg_vpu_parents),
                .flags = CLK_SET_RATE_NO_REPARENT,
        },
};

static struct clk_regmap axg_vapb_1_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_VAPBCLK_CNTL,
                .shift = 16,
                .width = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vapb_1_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vapb_1_sel.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_vapb_1 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VAPBCLK_CNTL,
                .bit_idx = 24,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vapb_1",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vapb_1_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vapb_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VAPBCLK_CNTL,
                .mask = 1,
                .shift = 31,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vapb_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vapb_0.hw,
                        &axg_vapb_1.hw
                },
                .num_parents = 2,
                .flags = CLK_SET_RATE_NO_REPARENT,
        },
};

static struct clk_regmap axg_vapb = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VAPBCLK_CNTL,
                .bit_idx = 30,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vapb",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vapb_sel.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

/* Video Clocks */

static const struct clk_hw *axg_vclk_parents[] = {
        &axg_gp0_pll.hw,
        &axg_fclk_div4.hw,
        &axg_fclk_div3.hw,
        &axg_fclk_div5.hw,
        &axg_fclk_div2.hw,
        &axg_fclk_div7.hw,
        &axg_mpll1.hw,
};

static struct clk_regmap axg_vclk_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VID_CLK_CNTL,
                .mask = 0x7,
                .shift = 16,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vclk_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = axg_vclk_parents,
                .num_parents = ARRAY_SIZE(axg_vclk_parents),
                .flags = CLK_SET_RATE_NO_REPARENT | CLK_GET_RATE_NOCACHE,
        },
};

static struct clk_regmap axg_vclk2_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VIID_CLK_CNTL,
                .mask = 0x7,
                .shift = 16,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vclk2_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = axg_vclk_parents,
                .num_parents = ARRAY_SIZE(axg_vclk_parents),
                .flags = CLK_SET_RATE_NO_REPARENT | CLK_GET_RATE_NOCACHE,
        },
};

static struct clk_regmap axg_vclk_input = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VID_CLK_DIV,
                .bit_idx = 16,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk_input",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk_sel.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk2_input = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VIID_CLK_DIV,
                .bit_idx = 16,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk2_input",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk2_sel.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_VID_CLK_DIV,
                .shift = 0,
                .width = 8,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vclk_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk_input.hw
                },
                .num_parents = 1,
                .flags = CLK_GET_RATE_NOCACHE,
        },
};

static struct clk_regmap axg_vclk2_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_VIID_CLK_DIV,
                .shift = 0,
                .width = 8,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vclk2_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk2_input.hw
                },
                .num_parents = 1,
                .flags = CLK_GET_RATE_NOCACHE,
        },
};

static struct clk_regmap axg_vclk = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VID_CLK_CNTL,
                .bit_idx = 19,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk_div.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk2 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VIID_CLK_CNTL,
                .bit_idx = 19,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk2",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk2_div.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk_div1 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VID_CLK_CNTL,
                .bit_idx = 0,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk_div1",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk_div2_en = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VID_CLK_CNTL,
                .bit_idx = 1,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk_div2_en",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk_div4_en = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VID_CLK_CNTL,
                .bit_idx = 2,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk_div4_en",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk_div6_en = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VID_CLK_CNTL,
                .bit_idx = 3,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk_div6_en",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk_div12_en = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VID_CLK_CNTL,
                .bit_idx = 4,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk_div12_en",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk2_div1 = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VIID_CLK_CNTL,
                .bit_idx = 0,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk2_div1",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk2.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk2_div2_en = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VIID_CLK_CNTL,
                .bit_idx = 1,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk2_div2_en",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk2.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk2_div4_en = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VIID_CLK_CNTL,
                .bit_idx = 2,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk2_div4_en",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk2.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk2_div6_en = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VIID_CLK_CNTL,
                .bit_idx = 3,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk2_div6_en",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk2.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_regmap axg_vclk2_div12_en = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VIID_CLK_CNTL,
                .bit_idx = 4,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vclk2_div12_en",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) { &axg_vclk2.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

static struct clk_fixed_factor axg_vclk_div2 = {
        .mult = 1,
        .div = 2,
        .hw.init = &(struct clk_init_data){
                .name = "vclk_div2",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk_div2_en.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_vclk_div4 = {
        .mult = 1,
        .div = 4,
        .hw.init = &(struct clk_init_data){
                .name = "vclk_div4",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk_div4_en.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_vclk_div6 = {
        .mult = 1,
        .div = 6,
        .hw.init = &(struct clk_init_data){
                .name = "vclk_div6",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk_div6_en.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_vclk_div12 = {
        .mult = 1,
        .div = 12,
        .hw.init = &(struct clk_init_data){
                .name = "vclk_div12",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk_div12_en.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_vclk2_div2 = {
        .mult = 1,
        .div = 2,
        .hw.init = &(struct clk_init_data){
                .name = "vclk2_div2",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk2_div2_en.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_vclk2_div4 = {
        .mult = 1,
        .div = 4,
        .hw.init = &(struct clk_init_data){
                .name = "vclk2_div4",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk2_div4_en.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_vclk2_div6 = {
        .mult = 1,
        .div = 6,
        .hw.init = &(struct clk_init_data){
                .name = "vclk2_div6",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk2_div6_en.hw
                },
                .num_parents = 1,
        },
};

static struct clk_fixed_factor axg_vclk2_div12 = {
        .mult = 1,
        .div = 12,
        .hw.init = &(struct clk_init_data){
                .name = "vclk2_div12",
                .ops = &clk_fixed_factor_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vclk2_div12_en.hw
                },
                .num_parents = 1,
        },
};

static u32 axg_cts_encl_parents_val_table[] = { 0, 1, 2, 3, 4, 8, 9, 10, 11, 12 };
static const struct clk_hw *axg_cts_encl_parents[] = {
        &axg_vclk_div1.hw,
        &axg_vclk_div2.hw,
        &axg_vclk_div4.hw,
        &axg_vclk_div6.hw,
        &axg_vclk_div12.hw,
        &axg_vclk2_div1.hw,
        &axg_vclk2_div2.hw,
        &axg_vclk2_div4.hw,
        &axg_vclk2_div6.hw,
        &axg_vclk2_div12.hw,
};

static struct clk_regmap axg_cts_encl_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VIID_CLK_DIV,
                .mask = 0xf,
                .shift = 12,
                .table = axg_cts_encl_parents_val_table,
        },
        .hw.init = &(struct clk_init_data){
                .name = "cts_encl_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_hws = axg_cts_encl_parents,
                .num_parents = ARRAY_SIZE(axg_cts_encl_parents),
                .flags = CLK_SET_RATE_NO_REPARENT | CLK_GET_RATE_NOCACHE,
        },
};

static struct clk_regmap axg_cts_encl = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VID_CLK_CNTL2,
                .bit_idx = 3,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "cts_encl",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_cts_encl_sel.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT | CLK_IGNORE_UNUSED,
        },
};

/* MIPI DSI Host Clock */

static u32 axg_vdin_meas_parents_val_table[] = { 0, 1, 2, 3, 6, 7 };
static const struct clk_parent_data axg_vdin_meas_parents[] = {
        { .fw_name = "xtal", },
        { .hw = &axg_fclk_div4.hw },
        { .hw = &axg_fclk_div3.hw },
        { .hw = &axg_fclk_div5.hw },
        { .hw = &axg_fclk_div2.hw },
        { .hw = &axg_fclk_div7.hw },
};

static struct clk_regmap axg_vdin_meas_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_VDIN_MEAS_CLK_CNTL,
                .mask = 0x7,
                .shift = 21,
                .flags = CLK_MUX_ROUND_CLOSEST,
                .table = axg_vdin_meas_parents_val_table,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vdin_meas_sel",
                .ops = &clk_regmap_mux_ops,
                .parent_data = axg_vdin_meas_parents,
                .num_parents = ARRAY_SIZE(axg_vdin_meas_parents),
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_vdin_meas_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_VDIN_MEAS_CLK_CNTL,
                .shift = 12,
                .width = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "vdin_meas_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vdin_meas_sel.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_vdin_meas = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_VDIN_MEAS_CLK_CNTL,
                .bit_idx = 20,
        },
        .hw.init = &(struct clk_init_data) {
                .name = "vdin_meas",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_vdin_meas_div.hw },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static u32 gen_clk_parents_val_table[] = { 0, 4, 5, 6, 7, 8, 9, 10, 11, 13, 14, };
static const struct clk_parent_data gen_clk_parents[] = {
        { .fw_name = "xtal", },
        { .hw = &axg_hifi_pll.hw },
        { .hw = &axg_mpll0.hw },
        { .hw = &axg_mpll1.hw },
        { .hw = &axg_mpll2.hw },
        { .hw = &axg_mpll3.hw },
        { .hw = &axg_fclk_div4.hw },
        { .hw = &axg_fclk_div3.hw },
        { .hw = &axg_fclk_div5.hw },
        { .hw = &axg_fclk_div7.hw },
        { .hw = &axg_gp0_pll.hw },
};

static struct clk_regmap axg_gen_clk_sel = {
        .data = &(struct clk_regmap_mux_data){
                .offset = HHI_GEN_CLK_CNTL,
                .mask = 0xf,
                .shift = 12,
                .table = gen_clk_parents_val_table,
        },
        .hw.init = &(struct clk_init_data){
                .name = "gen_clk_sel",
                .ops = &clk_regmap_mux_ops,
                /*
                 * bits 15:12 selects from 14 possible parents:
                 * xtal, [rtc_oscin_i], [sys_cpu_div16], [ddr_dpll_pt],
                 * hifi_pll, mpll0, mpll1, mpll2, mpll3, fdiv4,
                 * fdiv3, fdiv5, [cts_msr_clk], fdiv7, gp0_pll
                 */
                .parent_data = gen_clk_parents,
                .num_parents = ARRAY_SIZE(gen_clk_parents),
        },
};

static struct clk_regmap axg_gen_clk_div = {
        .data = &(struct clk_regmap_div_data){
                .offset = HHI_GEN_CLK_CNTL,
                .shift = 0,
                .width = 11,
        },
        .hw.init = &(struct clk_init_data){
                .name = "gen_clk_div",
                .ops = &clk_regmap_divider_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_gen_clk_sel.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static struct clk_regmap axg_gen_clk = {
        .data = &(struct clk_regmap_gate_data){
                .offset = HHI_GEN_CLK_CNTL,
                .bit_idx = 7,
        },
        .hw.init = &(struct clk_init_data){
                .name = "gen_clk",
                .ops = &clk_regmap_gate_ops,
                .parent_hws = (const struct clk_hw *[]) {
                        &axg_gen_clk_div.hw
                },
                .num_parents = 1,
                .flags = CLK_SET_RATE_PARENT,
        },
};

static const struct clk_parent_data axg_pclk_parents = { .hw = &axg_clk81.hw };

#define AXG_PCLK(_name, _reg, _bit, _flags) \
        MESON_PCLK(axg_##_name, _reg, _bit, &axg_pclk_parents, _flags)

/*
 * Everything Else (EE) domain gates
 *
 * NOTE: The gates below are marked with CLK_IGNORE_UNUSED for historic reasons
 * Users are encouraged to test without it and submit changes to:
 *  - remove the flag if not necessary
 *  - replace the flag with something more adequate, such as CLK_IS_CRITICAL,
 *    if appropriate.
 *  - add a comment explaining why the use of CLK_IGNORE_UNUSED is desirable
 *    for a particular clock.
 */
static AXG_PCLK(ddr,                    HHI_GCLK_MPEG0,  0, CLK_IGNORE_UNUSED);
static AXG_PCLK(audio_locker,           HHI_GCLK_MPEG0,  2, CLK_IGNORE_UNUSED);
static AXG_PCLK(mipi_dsi_host,          HHI_GCLK_MPEG0,  3, CLK_IGNORE_UNUSED);
static AXG_PCLK(isa,                    HHI_GCLK_MPEG0,  5, CLK_IGNORE_UNUSED);
static AXG_PCLK(pl301,                  HHI_GCLK_MPEG0,  6, CLK_IGNORE_UNUSED);
static AXG_PCLK(periphs,                HHI_GCLK_MPEG0,  7, CLK_IGNORE_UNUSED);
static AXG_PCLK(spicc_0,                HHI_GCLK_MPEG0,  8, CLK_IGNORE_UNUSED);
static AXG_PCLK(i2c,                    HHI_GCLK_MPEG0,  9, CLK_IGNORE_UNUSED);
static AXG_PCLK(rng0,                   HHI_GCLK_MPEG0, 12, CLK_IGNORE_UNUSED);
static AXG_PCLK(uart0,                  HHI_GCLK_MPEG0, 13, CLK_IGNORE_UNUSED);
static AXG_PCLK(mipi_dsi_phy,           HHI_GCLK_MPEG0, 14, CLK_IGNORE_UNUSED);
static AXG_PCLK(spicc_1,                HHI_GCLK_MPEG0, 15, CLK_IGNORE_UNUSED);
static AXG_PCLK(pcie_a,                 HHI_GCLK_MPEG0, 16, CLK_IGNORE_UNUSED);
static AXG_PCLK(pcie_b,                 HHI_GCLK_MPEG0, 17, CLK_IGNORE_UNUSED);
static AXG_PCLK(hiu_reg,                HHI_GCLK_MPEG0, 19, CLK_IGNORE_UNUSED);
static AXG_PCLK(assist_misc,            HHI_GCLK_MPEG0, 23, CLK_IGNORE_UNUSED);
static AXG_PCLK(emmc_b,                 HHI_GCLK_MPEG0, 25, CLK_IGNORE_UNUSED);
static AXG_PCLK(emmc_c,                 HHI_GCLK_MPEG0, 26, CLK_IGNORE_UNUSED);
static AXG_PCLK(dma,                    HHI_GCLK_MPEG0, 27, CLK_IGNORE_UNUSED);
static AXG_PCLK(spi,                    HHI_GCLK_MPEG0, 30, CLK_IGNORE_UNUSED);

static AXG_PCLK(audio,                  HHI_GCLK_MPEG1,  0, CLK_IGNORE_UNUSED);
static AXG_PCLK(eth_core,               HHI_GCLK_MPEG1,  3, CLK_IGNORE_UNUSED);
static AXG_PCLK(uart1,                  HHI_GCLK_MPEG1, 16, CLK_IGNORE_UNUSED);
static AXG_PCLK(g2d,                    HHI_GCLK_MPEG1, 20, CLK_IGNORE_UNUSED);
static AXG_PCLK(usb0,                   HHI_GCLK_MPEG1, 21, CLK_IGNORE_UNUSED);
static AXG_PCLK(usb1,                   HHI_GCLK_MPEG1, 22, CLK_IGNORE_UNUSED);
static AXG_PCLK(reset,                  HHI_GCLK_MPEG1, 23, CLK_IGNORE_UNUSED);
static AXG_PCLK(usb_general,            HHI_GCLK_MPEG1, 26, CLK_IGNORE_UNUSED);
static AXG_PCLK(ahb_arb0,               HHI_GCLK_MPEG1, 29, CLK_IGNORE_UNUSED);
static AXG_PCLK(efuse,                  HHI_GCLK_MPEG1, 30, CLK_IGNORE_UNUSED);
static AXG_PCLK(boot_rom,               HHI_GCLK_MPEG1, 31, CLK_IGNORE_UNUSED);

static AXG_PCLK(ahb_data_bus,           HHI_GCLK_MPEG2,  1, CLK_IGNORE_UNUSED);
static AXG_PCLK(ahb_ctrl_bus,           HHI_GCLK_MPEG2,  2, CLK_IGNORE_UNUSED);
static AXG_PCLK(usb1_to_ddr,            HHI_GCLK_MPEG2,  8, CLK_IGNORE_UNUSED);
static AXG_PCLK(usb0_to_ddr,            HHI_GCLK_MPEG2,  9, CLK_IGNORE_UNUSED);
static AXG_PCLK(mmc_pclk,               HHI_GCLK_MPEG2, 11, CLK_IGNORE_UNUSED);
static AXG_PCLK(vpu_intr,               HHI_GCLK_MPEG2, 25, CLK_IGNORE_UNUSED);
static AXG_PCLK(sec_ahb_ahb3_bridge,    HHI_GCLK_MPEG2, 26, CLK_IGNORE_UNUSED);
static AXG_PCLK(gic,                    HHI_GCLK_MPEG2, 30, CLK_IGNORE_UNUSED);

/* Always On (AO) domain gates */

static AXG_PCLK(ao_media_cpu,           HHI_GCLK_AO, 0, CLK_IGNORE_UNUSED);
static AXG_PCLK(ao_ahb_sram,            HHI_GCLK_AO, 1, CLK_IGNORE_UNUSED);
static AXG_PCLK(ao_ahb_bus,             HHI_GCLK_AO, 2, CLK_IGNORE_UNUSED);
static AXG_PCLK(ao_iface,               HHI_GCLK_AO, 3, CLK_IGNORE_UNUSED);
static AXG_PCLK(ao_i2c,                 HHI_GCLK_AO, 4, CLK_IGNORE_UNUSED);

/* Array of all clocks provided by this provider */

static struct clk_hw *axg_hw_clks[] = {
        [CLKID_SYS_PLL]                 = &axg_sys_pll.hw,
        [CLKID_FIXED_PLL]               = &axg_fixed_pll.hw,
        [CLKID_FCLK_DIV2]               = &axg_fclk_div2.hw,
        [CLKID_FCLK_DIV3]               = &axg_fclk_div3.hw,
        [CLKID_FCLK_DIV4]               = &axg_fclk_div4.hw,
        [CLKID_FCLK_DIV5]               = &axg_fclk_div5.hw,
        [CLKID_FCLK_DIV7]               = &axg_fclk_div7.hw,
        [CLKID_GP0_PLL]                 = &axg_gp0_pll.hw,
        [CLKID_MPEG_SEL]                = &axg_clk81_sel.hw,
        [CLKID_MPEG_DIV]                = &axg_clk81_div.hw,
        [CLKID_CLK81]                   = &axg_clk81.hw,
        [CLKID_MPLL0]                   = &axg_mpll0.hw,
        [CLKID_MPLL1]                   = &axg_mpll1.hw,
        [CLKID_MPLL2]                   = &axg_mpll2.hw,
        [CLKID_MPLL3]                   = &axg_mpll3.hw,
        [CLKID_DDR]                     = &axg_ddr.hw,
        [CLKID_AUDIO_LOCKER]            = &axg_audio_locker.hw,
        [CLKID_MIPI_DSI_HOST]           = &axg_mipi_dsi_host.hw,
        [CLKID_ISA]                     = &axg_isa.hw,
        [CLKID_PL301]                   = &axg_pl301.hw,
        [CLKID_PERIPHS]                 = &axg_periphs.hw,
        [CLKID_SPICC0]                  = &axg_spicc_0.hw,
        [CLKID_I2C]                     = &axg_i2c.hw,
        [CLKID_RNG0]                    = &axg_rng0.hw,
        [CLKID_UART0]                   = &axg_uart0.hw,
        [CLKID_MIPI_DSI_PHY]            = &axg_mipi_dsi_phy.hw,
        [CLKID_SPICC1]                  = &axg_spicc_1.hw,
        [CLKID_PCIE_A]                  = &axg_pcie_a.hw,
        [CLKID_PCIE_B]                  = &axg_pcie_b.hw,
        [CLKID_HIU_IFACE]               = &axg_hiu_reg.hw,
        [CLKID_ASSIST_MISC]             = &axg_assist_misc.hw,
        [CLKID_SD_EMMC_B]               = &axg_emmc_b.hw,
        [CLKID_SD_EMMC_C]               = &axg_emmc_c.hw,
        [CLKID_DMA]                     = &axg_dma.hw,
        [CLKID_SPI]                     = &axg_spi.hw,
        [CLKID_AUDIO]                   = &axg_audio.hw,
        [CLKID_ETH]                     = &axg_eth_core.hw,
        [CLKID_UART1]                   = &axg_uart1.hw,
        [CLKID_G2D]                     = &axg_g2d.hw,
        [CLKID_USB0]                    = &axg_usb0.hw,
        [CLKID_USB1]                    = &axg_usb1.hw,
        [CLKID_RESET]                   = &axg_reset.hw,
        [CLKID_USB]                     = &axg_usb_general.hw,
        [CLKID_AHB_ARB0]                = &axg_ahb_arb0.hw,
        [CLKID_EFUSE]                   = &axg_efuse.hw,
        [CLKID_BOOT_ROM]                = &axg_boot_rom.hw,
        [CLKID_AHB_DATA_BUS]            = &axg_ahb_data_bus.hw,
        [CLKID_AHB_CTRL_BUS]            = &axg_ahb_ctrl_bus.hw,
        [CLKID_USB1_DDR_BRIDGE]         = &axg_usb1_to_ddr.hw,
        [CLKID_USB0_DDR_BRIDGE]         = &axg_usb0_to_ddr.hw,
        [CLKID_MMC_PCLK]                = &axg_mmc_pclk.hw,
        [CLKID_VPU_INTR]                = &axg_vpu_intr.hw,
        [CLKID_SEC_AHB_AHB3_BRIDGE]     = &axg_sec_ahb_ahb3_bridge.hw,
        [CLKID_GIC]                     = &axg_gic.hw,
        [CLKID_AO_MEDIA_CPU]            = &axg_ao_media_cpu.hw,
        [CLKID_AO_AHB_SRAM]             = &axg_ao_ahb_sram.hw,
        [CLKID_AO_AHB_BUS]              = &axg_ao_ahb_bus.hw,
        [CLKID_AO_IFACE]                = &axg_ao_iface.hw,
        [CLKID_AO_I2C]                  = &axg_ao_i2c.hw,
        [CLKID_SD_EMMC_B_CLK0_SEL]      = &axg_sd_emmc_b_clk0_sel.hw,
        [CLKID_SD_EMMC_B_CLK0_DIV]      = &axg_sd_emmc_b_clk0_div.hw,
        [CLKID_SD_EMMC_B_CLK0]          = &axg_sd_emmc_b_clk0.hw,
        [CLKID_SD_EMMC_C_CLK0_SEL]      = &axg_sd_emmc_c_clk0_sel.hw,
        [CLKID_SD_EMMC_C_CLK0_DIV]      = &axg_sd_emmc_c_clk0_div.hw,
        [CLKID_SD_EMMC_C_CLK0]          = &axg_sd_emmc_c_clk0.hw,
        [CLKID_MPLL0_DIV]               = &axg_mpll0_div.hw,
        [CLKID_MPLL1_DIV]               = &axg_mpll1_div.hw,
        [CLKID_MPLL2_DIV]               = &axg_mpll2_div.hw,
        [CLKID_MPLL3_DIV]               = &axg_mpll3_div.hw,
        [CLKID_HIFI_PLL]                = &axg_hifi_pll.hw,
        [CLKID_MPLL_PREDIV]             = &axg_mpll_prediv.hw,
        [CLKID_FCLK_DIV2_DIV]           = &axg_fclk_div2_div.hw,
        [CLKID_FCLK_DIV3_DIV]           = &axg_fclk_div3_div.hw,
        [CLKID_FCLK_DIV4_DIV]           = &axg_fclk_div4_div.hw,
        [CLKID_FCLK_DIV5_DIV]           = &axg_fclk_div5_div.hw,
        [CLKID_FCLK_DIV7_DIV]           = &axg_fclk_div7_div.hw,
        [CLKID_PCIE_PLL]                = &axg_pcie_pll.hw,
        [CLKID_PCIE_MUX]                = &axg_pcie_mux.hw,
        [CLKID_PCIE_REF]                = &axg_pcie_ref.hw,
        [CLKID_PCIE_CML_EN0]            = &axg_pcie_cml_en0.hw,
        [CLKID_PCIE_CML_EN1]            = &axg_pcie_cml_en1.hw,
        [CLKID_GEN_CLK_SEL]             = &axg_gen_clk_sel.hw,
        [CLKID_GEN_CLK_DIV]             = &axg_gen_clk_div.hw,
        [CLKID_GEN_CLK]                 = &axg_gen_clk.hw,
        [CLKID_SYS_PLL_DCO]             = &axg_sys_pll_dco.hw,
        [CLKID_FIXED_PLL_DCO]           = &axg_fixed_pll_dco.hw,
        [CLKID_GP0_PLL_DCO]             = &axg_gp0_pll_dco.hw,
        [CLKID_HIFI_PLL_DCO]            = &axg_hifi_pll_dco.hw,
        [CLKID_PCIE_PLL_DCO]            = &axg_pcie_pll_dco.hw,
        [CLKID_PCIE_PLL_OD]             = &axg_pcie_pll_od.hw,
        [CLKID_VPU_0_DIV]               = &axg_vpu_0_div.hw,
        [CLKID_VPU_0_SEL]               = &axg_vpu_0_sel.hw,
        [CLKID_VPU_0]                   = &axg_vpu_0.hw,
        [CLKID_VPU_1_DIV]               = &axg_vpu_1_div.hw,
        [CLKID_VPU_1_SEL]               = &axg_vpu_1_sel.hw,
        [CLKID_VPU_1]                   = &axg_vpu_1.hw,
        [CLKID_VPU]                     = &axg_vpu.hw,
        [CLKID_VAPB_0_DIV]              = &axg_vapb_0_div.hw,
        [CLKID_VAPB_0_SEL]              = &axg_vapb_0_sel.hw,
        [CLKID_VAPB_0]                  = &axg_vapb_0.hw,
        [CLKID_VAPB_1_DIV]              = &axg_vapb_1_div.hw,
        [CLKID_VAPB_1_SEL]              = &axg_vapb_1_sel.hw,
        [CLKID_VAPB_1]                  = &axg_vapb_1.hw,
        [CLKID_VAPB_SEL]                = &axg_vapb_sel.hw,
        [CLKID_VAPB]                    = &axg_vapb.hw,
        [CLKID_VCLK]                    = &axg_vclk.hw,
        [CLKID_VCLK2]                   = &axg_vclk2.hw,
        [CLKID_VCLK_SEL]                = &axg_vclk_sel.hw,
        [CLKID_VCLK2_SEL]               = &axg_vclk2_sel.hw,
        [CLKID_VCLK_INPUT]              = &axg_vclk_input.hw,
        [CLKID_VCLK2_INPUT]             = &axg_vclk2_input.hw,
        [CLKID_VCLK_DIV]                = &axg_vclk_div.hw,
        [CLKID_VCLK2_DIV]               = &axg_vclk2_div.hw,
        [CLKID_VCLK_DIV2_EN]            = &axg_vclk_div2_en.hw,
        [CLKID_VCLK_DIV4_EN]            = &axg_vclk_div4_en.hw,
        [CLKID_VCLK_DIV6_EN]            = &axg_vclk_div6_en.hw,
        [CLKID_VCLK_DIV12_EN]           = &axg_vclk_div12_en.hw,
        [CLKID_VCLK2_DIV2_EN]           = &axg_vclk2_div2_en.hw,
        [CLKID_VCLK2_DIV4_EN]           = &axg_vclk2_div4_en.hw,
        [CLKID_VCLK2_DIV6_EN]           = &axg_vclk2_div6_en.hw,
        [CLKID_VCLK2_DIV12_EN]          = &axg_vclk2_div12_en.hw,
        [CLKID_VCLK_DIV1]               = &axg_vclk_div1.hw,
        [CLKID_VCLK_DIV2]               = &axg_vclk_div2.hw,
        [CLKID_VCLK_DIV4]               = &axg_vclk_div4.hw,
        [CLKID_VCLK_DIV6]               = &axg_vclk_div6.hw,
        [CLKID_VCLK_DIV12]              = &axg_vclk_div12.hw,
        [CLKID_VCLK2_DIV1]              = &axg_vclk2_div1.hw,
        [CLKID_VCLK2_DIV2]              = &axg_vclk2_div2.hw,
        [CLKID_VCLK2_DIV4]              = &axg_vclk2_div4.hw,
        [CLKID_VCLK2_DIV6]              = &axg_vclk2_div6.hw,
        [CLKID_VCLK2_DIV12]             = &axg_vclk2_div12.hw,
        [CLKID_CTS_ENCL_SEL]            = &axg_cts_encl_sel.hw,
        [CLKID_CTS_ENCL]                = &axg_cts_encl.hw,
        [CLKID_VDIN_MEAS_SEL]           = &axg_vdin_meas_sel.hw,
        [CLKID_VDIN_MEAS_DIV]           = &axg_vdin_meas_div.hw,
        [CLKID_VDIN_MEAS]               = &axg_vdin_meas.hw,
};

static const struct meson_clkc_data axg_clkc_data = {
        .hw_clks = {
                .hws = axg_hw_clks,
                .num = ARRAY_SIZE(axg_hw_clks),
        },
};

static const struct of_device_id axg_clkc_match_table[] = {
        { .compatible = "amlogic,axg-clkc", .data = &axg_clkc_data },
        {}
};
MODULE_DEVICE_TABLE(of, axg_clkc_match_table);

static struct platform_driver axg_clkc_driver = {
        .probe          = meson_clkc_syscon_probe,
        .driver         = {
                .name   = "axg-clkc",
                .of_match_table = axg_clkc_match_table,
        },
};
module_platform_driver(axg_clkc_driver);

MODULE_DESCRIPTION("Amlogic AXG Main Clock Controller driver");
MODULE_LICENSE("GPL");
MODULE_IMPORT_NS("CLK_MESON");