1c646b347SAmit Nischal // SPDX-License-Identifier: GPL-2.0 2c646b347SAmit Nischal /* 3c646b347SAmit Nischal * Copyright (c) 2018, The Linux Foundation. All rights reserved. 4c646b347SAmit Nischal */ 5c646b347SAmit Nischal 6c646b347SAmit Nischal #include <linux/clk-provider.h> 7c646b347SAmit Nischal #include <linux/module.h> 8c646b347SAmit Nischal #include <linux/platform_device.h> 9c646b347SAmit Nischal #include <linux/regmap.h> 10c646b347SAmit Nischal 11c646b347SAmit Nischal #include <dt-bindings/clock/qcom,videocc-sdm845.h> 12c646b347SAmit Nischal 13c646b347SAmit Nischal #include "common.h" 14c646b347SAmit Nischal #include "clk-alpha-pll.h" 15c646b347SAmit Nischal #include "clk-branch.h" 16c646b347SAmit Nischal #include "clk-rcg.h" 17c646b347SAmit Nischal #include "clk-regmap.h" 18c646b347SAmit Nischal #include "clk-pll.h" 19c646b347SAmit Nischal #include "gdsc.h" 20c646b347SAmit Nischal 21c646b347SAmit Nischal enum { 22c646b347SAmit Nischal P_BI_TCXO, 23c646b347SAmit Nischal P_CORE_BI_PLL_TEST_SE, 24c646b347SAmit Nischal P_VIDEO_PLL0_OUT_MAIN, 25*7f195c06SDmitry Baryshkov /* P_VIDEO_PLL0_OUT_EVEN, */ 26*7f195c06SDmitry Baryshkov /* P_VIDEO_PLL0_OUT_ODD, */ 27c646b347SAmit Nischal }; 28c646b347SAmit Nischal 29c646b347SAmit Nischal static const struct alpha_pll_config video_pll0_config = { 30c646b347SAmit Nischal .l = 0x10, 31c646b347SAmit Nischal .alpha = 0xaaab, 32c646b347SAmit Nischal }; 33c646b347SAmit Nischal 34c646b347SAmit Nischal static struct clk_alpha_pll video_pll0 = { 35c646b347SAmit Nischal .offset = 0x42c, 36c646b347SAmit Nischal .regs = clk_alpha_pll_regs[CLK_ALPHA_PLL_TYPE_FABIA], 37c646b347SAmit Nischal .clkr = { 38c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 39c646b347SAmit Nischal .name = "video_pll0", 40*7f195c06SDmitry Baryshkov .parent_data = &(const struct clk_parent_data){ 41*7f195c06SDmitry Baryshkov .fw_name = "bi_tcxo", .name = "bi_tcxo", 42*7f195c06SDmitry Baryshkov }, 43c646b347SAmit Nischal .num_parents = 1, 44c646b347SAmit Nischal .ops = &clk_alpha_pll_fabia_ops, 45c646b347SAmit Nischal }, 46c646b347SAmit Nischal }, 47c646b347SAmit Nischal }; 48c646b347SAmit Nischal 49*7f195c06SDmitry Baryshkov static const struct parent_map video_cc_parent_map_0[] = { 50*7f195c06SDmitry Baryshkov { P_BI_TCXO, 0 }, 51*7f195c06SDmitry Baryshkov { P_VIDEO_PLL0_OUT_MAIN, 1 }, 52*7f195c06SDmitry Baryshkov /* { P_VIDEO_PLL0_OUT_EVEN, 2 }, */ 53*7f195c06SDmitry Baryshkov /* { P_VIDEO_PLL0_OUT_ODD, 3 }, */ 54*7f195c06SDmitry Baryshkov { P_CORE_BI_PLL_TEST_SE, 4 }, 55*7f195c06SDmitry Baryshkov }; 56*7f195c06SDmitry Baryshkov 57*7f195c06SDmitry Baryshkov static const struct clk_parent_data video_cc_parent_data_0[] = { 58*7f195c06SDmitry Baryshkov { .fw_name = "bi_tcxo", .name = "bi_tcxo" }, 59*7f195c06SDmitry Baryshkov { .hw = &video_pll0.clkr.hw }, 60*7f195c06SDmitry Baryshkov /* { .name = "video_pll0_out_even" }, */ 61*7f195c06SDmitry Baryshkov /* { .name = "video_pll0_out_odd" }, */ 62*7f195c06SDmitry Baryshkov { .fw_name = "core_bi_pll_test_se", .name = "core_bi_pll_test_se" }, 63*7f195c06SDmitry Baryshkov }; 64*7f195c06SDmitry Baryshkov 65c646b347SAmit Nischal static const struct freq_tbl ftbl_video_cc_venus_clk_src[] = { 66c646b347SAmit Nischal F(100000000, P_VIDEO_PLL0_OUT_MAIN, 4, 0, 0), 67c646b347SAmit Nischal F(200000000, P_VIDEO_PLL0_OUT_MAIN, 2, 0, 0), 68c646b347SAmit Nischal F(330000000, P_VIDEO_PLL0_OUT_MAIN, 1, 0, 0), 69c646b347SAmit Nischal F(404000000, P_VIDEO_PLL0_OUT_MAIN, 1, 0, 0), 70c646b347SAmit Nischal F(444000000, P_VIDEO_PLL0_OUT_MAIN, 1, 0, 0), 71c646b347SAmit Nischal F(533000000, P_VIDEO_PLL0_OUT_MAIN, 1, 0, 0), 72c646b347SAmit Nischal { } 73c646b347SAmit Nischal }; 74c646b347SAmit Nischal 75c646b347SAmit Nischal static struct clk_rcg2 video_cc_venus_clk_src = { 76c646b347SAmit Nischal .cmd_rcgr = 0x7f0, 77c646b347SAmit Nischal .mnd_width = 0, 78c646b347SAmit Nischal .hid_width = 5, 79c646b347SAmit Nischal .parent_map = video_cc_parent_map_0, 80c646b347SAmit Nischal .freq_tbl = ftbl_video_cc_venus_clk_src, 81c646b347SAmit Nischal .clkr.hw.init = &(struct clk_init_data){ 82c646b347SAmit Nischal .name = "video_cc_venus_clk_src", 83*7f195c06SDmitry Baryshkov .parent_data = video_cc_parent_data_0, 84*7f195c06SDmitry Baryshkov .num_parents = ARRAY_SIZE(video_cc_parent_data_0), 85c646b347SAmit Nischal .flags = CLK_SET_RATE_PARENT, 86c646b347SAmit Nischal .ops = &clk_rcg2_shared_ops, 87c646b347SAmit Nischal }, 88c646b347SAmit Nischal }; 89c646b347SAmit Nischal 90c646b347SAmit Nischal static struct clk_branch video_cc_apb_clk = { 91c646b347SAmit Nischal .halt_reg = 0x990, 92c646b347SAmit Nischal .halt_check = BRANCH_HALT, 93c646b347SAmit Nischal .clkr = { 94c646b347SAmit Nischal .enable_reg = 0x990, 95c646b347SAmit Nischal .enable_mask = BIT(0), 96c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 97c646b347SAmit Nischal .name = "video_cc_apb_clk", 98c646b347SAmit Nischal .ops = &clk_branch2_ops, 99c646b347SAmit Nischal }, 100c646b347SAmit Nischal }, 101c646b347SAmit Nischal }; 102c646b347SAmit Nischal 103c646b347SAmit Nischal static struct clk_branch video_cc_at_clk = { 104c646b347SAmit Nischal .halt_reg = 0x9f0, 105c646b347SAmit Nischal .halt_check = BRANCH_HALT, 106c646b347SAmit Nischal .clkr = { 107c646b347SAmit Nischal .enable_reg = 0x9f0, 108c646b347SAmit Nischal .enable_mask = BIT(0), 109c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 110c646b347SAmit Nischal .name = "video_cc_at_clk", 111c646b347SAmit Nischal .ops = &clk_branch2_ops, 112c646b347SAmit Nischal }, 113c646b347SAmit Nischal }, 114c646b347SAmit Nischal }; 115c646b347SAmit Nischal 116c646b347SAmit Nischal static struct clk_branch video_cc_qdss_trig_clk = { 117c646b347SAmit Nischal .halt_reg = 0x970, 118c646b347SAmit Nischal .halt_check = BRANCH_HALT, 119c646b347SAmit Nischal .clkr = { 120c646b347SAmit Nischal .enable_reg = 0x970, 121c646b347SAmit Nischal .enable_mask = BIT(0), 122c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 123c646b347SAmit Nischal .name = "video_cc_qdss_trig_clk", 124c646b347SAmit Nischal .ops = &clk_branch2_ops, 125c646b347SAmit Nischal }, 126c646b347SAmit Nischal }, 127c646b347SAmit Nischal }; 128c646b347SAmit Nischal 129c646b347SAmit Nischal static struct clk_branch video_cc_qdss_tsctr_div8_clk = { 130c646b347SAmit Nischal .halt_reg = 0x9d0, 131c646b347SAmit Nischal .halt_check = BRANCH_HALT, 132c646b347SAmit Nischal .clkr = { 133c646b347SAmit Nischal .enable_reg = 0x9d0, 134c646b347SAmit Nischal .enable_mask = BIT(0), 135c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 136c646b347SAmit Nischal .name = "video_cc_qdss_tsctr_div8_clk", 137c646b347SAmit Nischal .ops = &clk_branch2_ops, 138c646b347SAmit Nischal }, 139c646b347SAmit Nischal }, 140c646b347SAmit Nischal }; 141c646b347SAmit Nischal 142c646b347SAmit Nischal static struct clk_branch video_cc_vcodec0_axi_clk = { 143c646b347SAmit Nischal .halt_reg = 0x930, 144c646b347SAmit Nischal .halt_check = BRANCH_HALT, 145c646b347SAmit Nischal .clkr = { 146c646b347SAmit Nischal .enable_reg = 0x930, 147c646b347SAmit Nischal .enable_mask = BIT(0), 148c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 149c646b347SAmit Nischal .name = "video_cc_vcodec0_axi_clk", 150c646b347SAmit Nischal .ops = &clk_branch2_ops, 151c646b347SAmit Nischal }, 152c646b347SAmit Nischal }, 153c646b347SAmit Nischal }; 154c646b347SAmit Nischal 155c646b347SAmit Nischal static struct clk_branch video_cc_vcodec0_core_clk = { 156c646b347SAmit Nischal .halt_reg = 0x890, 157c646b347SAmit Nischal .halt_check = BRANCH_VOTED, 158c646b347SAmit Nischal .clkr = { 159c646b347SAmit Nischal .enable_reg = 0x890, 160c646b347SAmit Nischal .enable_mask = BIT(0), 161c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 162c646b347SAmit Nischal .name = "video_cc_vcodec0_core_clk", 163*7f195c06SDmitry Baryshkov .parent_hws = (const struct clk_hw*[]){ 164*7f195c06SDmitry Baryshkov &video_cc_venus_clk_src.clkr.hw, 165c646b347SAmit Nischal }, 166c646b347SAmit Nischal .num_parents = 1, 167c646b347SAmit Nischal .flags = CLK_SET_RATE_PARENT, 168c646b347SAmit Nischal .ops = &clk_branch2_ops, 169c646b347SAmit Nischal }, 170c646b347SAmit Nischal }, 171c646b347SAmit Nischal }; 172c646b347SAmit Nischal 173c646b347SAmit Nischal static struct clk_branch video_cc_vcodec1_axi_clk = { 174c646b347SAmit Nischal .halt_reg = 0x950, 175c646b347SAmit Nischal .halt_check = BRANCH_HALT, 176c646b347SAmit Nischal .clkr = { 177c646b347SAmit Nischal .enable_reg = 0x950, 178c646b347SAmit Nischal .enable_mask = BIT(0), 179c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 180c646b347SAmit Nischal .name = "video_cc_vcodec1_axi_clk", 181c646b347SAmit Nischal .ops = &clk_branch2_ops, 182c646b347SAmit Nischal }, 183c646b347SAmit Nischal }, 184c646b347SAmit Nischal }; 185c646b347SAmit Nischal 186c646b347SAmit Nischal static struct clk_branch video_cc_vcodec1_core_clk = { 187c646b347SAmit Nischal .halt_reg = 0x8d0, 188c646b347SAmit Nischal .halt_check = BRANCH_VOTED, 189c646b347SAmit Nischal .clkr = { 190c646b347SAmit Nischal .enable_reg = 0x8d0, 191c646b347SAmit Nischal .enable_mask = BIT(0), 192c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 193c646b347SAmit Nischal .name = "video_cc_vcodec1_core_clk", 194*7f195c06SDmitry Baryshkov .parent_hws = (const struct clk_hw*[]){ 195*7f195c06SDmitry Baryshkov &video_cc_venus_clk_src.clkr.hw, 196c646b347SAmit Nischal }, 197c646b347SAmit Nischal .num_parents = 1, 198c646b347SAmit Nischal .flags = CLK_SET_RATE_PARENT, 199c646b347SAmit Nischal .ops = &clk_branch2_ops, 200c646b347SAmit Nischal }, 201c646b347SAmit Nischal }, 202c646b347SAmit Nischal }; 203c646b347SAmit Nischal 204c646b347SAmit Nischal static struct clk_branch video_cc_venus_ahb_clk = { 205c646b347SAmit Nischal .halt_reg = 0x9b0, 206c646b347SAmit Nischal .halt_check = BRANCH_HALT, 207c646b347SAmit Nischal .clkr = { 208c646b347SAmit Nischal .enable_reg = 0x9b0, 209c646b347SAmit Nischal .enable_mask = BIT(0), 210c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 211c646b347SAmit Nischal .name = "video_cc_venus_ahb_clk", 212c646b347SAmit Nischal .ops = &clk_branch2_ops, 213c646b347SAmit Nischal }, 214c646b347SAmit Nischal }, 215c646b347SAmit Nischal }; 216c646b347SAmit Nischal 217c646b347SAmit Nischal static struct clk_branch video_cc_venus_ctl_axi_clk = { 218c646b347SAmit Nischal .halt_reg = 0x910, 219c646b347SAmit Nischal .halt_check = BRANCH_HALT, 220c646b347SAmit Nischal .clkr = { 221c646b347SAmit Nischal .enable_reg = 0x910, 222c646b347SAmit Nischal .enable_mask = BIT(0), 223c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 224c646b347SAmit Nischal .name = "video_cc_venus_ctl_axi_clk", 225c646b347SAmit Nischal .ops = &clk_branch2_ops, 226c646b347SAmit Nischal }, 227c646b347SAmit Nischal }, 228c646b347SAmit Nischal }; 229c646b347SAmit Nischal 230c646b347SAmit Nischal static struct clk_branch video_cc_venus_ctl_core_clk = { 231c646b347SAmit Nischal .halt_reg = 0x850, 232c646b347SAmit Nischal .halt_check = BRANCH_HALT, 233c646b347SAmit Nischal .clkr = { 234c646b347SAmit Nischal .enable_reg = 0x850, 235c646b347SAmit Nischal .enable_mask = BIT(0), 236c646b347SAmit Nischal .hw.init = &(struct clk_init_data){ 237c646b347SAmit Nischal .name = "video_cc_venus_ctl_core_clk", 238*7f195c06SDmitry Baryshkov .parent_hws = (const struct clk_hw*[]){ 239*7f195c06SDmitry Baryshkov &video_cc_venus_clk_src.clkr.hw, 240c646b347SAmit Nischal }, 241c646b347SAmit Nischal .num_parents = 1, 242c646b347SAmit Nischal .flags = CLK_SET_RATE_PARENT, 243c646b347SAmit Nischal .ops = &clk_branch2_ops, 244c646b347SAmit Nischal }, 245c646b347SAmit Nischal }, 246c646b347SAmit Nischal }; 247c646b347SAmit Nischal 248c646b347SAmit Nischal static struct gdsc venus_gdsc = { 249c646b347SAmit Nischal .gdscr = 0x814, 250c646b347SAmit Nischal .pd = { 251c646b347SAmit Nischal .name = "venus_gdsc", 252c646b347SAmit Nischal }, 253c646b347SAmit Nischal .cxcs = (unsigned int []){ 0x850, 0x910 }, 254c646b347SAmit Nischal .cxc_count = 2, 255c646b347SAmit Nischal .pwrsts = PWRSTS_OFF_ON, 256c646b347SAmit Nischal .flags = POLL_CFG_GDSCR, 257c646b347SAmit Nischal }; 258c646b347SAmit Nischal 259c646b347SAmit Nischal static struct gdsc vcodec0_gdsc = { 260c646b347SAmit Nischal .gdscr = 0x874, 261c646b347SAmit Nischal .pd = { 262c646b347SAmit Nischal .name = "vcodec0_gdsc", 263c646b347SAmit Nischal }, 264c646b347SAmit Nischal .cxcs = (unsigned int []){ 0x890, 0x930 }, 265c646b347SAmit Nischal .cxc_count = 2, 266c646b347SAmit Nischal .flags = HW_CTRL | POLL_CFG_GDSCR, 267c646b347SAmit Nischal .pwrsts = PWRSTS_OFF_ON, 268c646b347SAmit Nischal }; 269c646b347SAmit Nischal 270c646b347SAmit Nischal static struct gdsc vcodec1_gdsc = { 271c646b347SAmit Nischal .gdscr = 0x8b4, 272c646b347SAmit Nischal .pd = { 273c646b347SAmit Nischal .name = "vcodec1_gdsc", 274c646b347SAmit Nischal }, 275c646b347SAmit Nischal .cxcs = (unsigned int []){ 0x8d0, 0x950 }, 276c646b347SAmit Nischal .cxc_count = 2, 277c646b347SAmit Nischal .flags = HW_CTRL | POLL_CFG_GDSCR, 278c646b347SAmit Nischal .pwrsts = PWRSTS_OFF_ON, 279c646b347SAmit Nischal }; 280c646b347SAmit Nischal 281c646b347SAmit Nischal static struct clk_regmap *video_cc_sdm845_clocks[] = { 282c646b347SAmit Nischal [VIDEO_CC_APB_CLK] = &video_cc_apb_clk.clkr, 283c646b347SAmit Nischal [VIDEO_CC_AT_CLK] = &video_cc_at_clk.clkr, 284c646b347SAmit Nischal [VIDEO_CC_QDSS_TRIG_CLK] = &video_cc_qdss_trig_clk.clkr, 285c646b347SAmit Nischal [VIDEO_CC_QDSS_TSCTR_DIV8_CLK] = &video_cc_qdss_tsctr_div8_clk.clkr, 286c646b347SAmit Nischal [VIDEO_CC_VCODEC0_AXI_CLK] = &video_cc_vcodec0_axi_clk.clkr, 287c646b347SAmit Nischal [VIDEO_CC_VCODEC0_CORE_CLK] = &video_cc_vcodec0_core_clk.clkr, 288c646b347SAmit Nischal [VIDEO_CC_VCODEC1_AXI_CLK] = &video_cc_vcodec1_axi_clk.clkr, 289c646b347SAmit Nischal [VIDEO_CC_VCODEC1_CORE_CLK] = &video_cc_vcodec1_core_clk.clkr, 290c646b347SAmit Nischal [VIDEO_CC_VENUS_AHB_CLK] = &video_cc_venus_ahb_clk.clkr, 291c646b347SAmit Nischal [VIDEO_CC_VENUS_CLK_SRC] = &video_cc_venus_clk_src.clkr, 292c646b347SAmit Nischal [VIDEO_CC_VENUS_CTL_AXI_CLK] = &video_cc_venus_ctl_axi_clk.clkr, 293c646b347SAmit Nischal [VIDEO_CC_VENUS_CTL_CORE_CLK] = &video_cc_venus_ctl_core_clk.clkr, 294c646b347SAmit Nischal [VIDEO_PLL0] = &video_pll0.clkr, 295c646b347SAmit Nischal }; 296c646b347SAmit Nischal 297c646b347SAmit Nischal static struct gdsc *video_cc_sdm845_gdscs[] = { 298c646b347SAmit Nischal [VENUS_GDSC] = &venus_gdsc, 299c646b347SAmit Nischal [VCODEC0_GDSC] = &vcodec0_gdsc, 300c646b347SAmit Nischal [VCODEC1_GDSC] = &vcodec1_gdsc, 301c646b347SAmit Nischal }; 302c646b347SAmit Nischal 303c646b347SAmit Nischal static const struct regmap_config video_cc_sdm845_regmap_config = { 304c646b347SAmit Nischal .reg_bits = 32, 305c646b347SAmit Nischal .reg_stride = 4, 306c646b347SAmit Nischal .val_bits = 32, 307c646b347SAmit Nischal .max_register = 0xb90, 308c646b347SAmit Nischal .fast_io = true, 309c646b347SAmit Nischal }; 310c646b347SAmit Nischal 311c646b347SAmit Nischal static const struct qcom_cc_desc video_cc_sdm845_desc = { 312c646b347SAmit Nischal .config = &video_cc_sdm845_regmap_config, 313c646b347SAmit Nischal .clks = video_cc_sdm845_clocks, 314c646b347SAmit Nischal .num_clks = ARRAY_SIZE(video_cc_sdm845_clocks), 315c646b347SAmit Nischal .gdscs = video_cc_sdm845_gdscs, 316c646b347SAmit Nischal .num_gdscs = ARRAY_SIZE(video_cc_sdm845_gdscs), 317c646b347SAmit Nischal }; 318c646b347SAmit Nischal 319c646b347SAmit Nischal static const struct of_device_id video_cc_sdm845_match_table[] = { 320c646b347SAmit Nischal { .compatible = "qcom,sdm845-videocc" }, 321c646b347SAmit Nischal { } 322c646b347SAmit Nischal }; 323c646b347SAmit Nischal MODULE_DEVICE_TABLE(of, video_cc_sdm845_match_table); 324c646b347SAmit Nischal 325c646b347SAmit Nischal static int video_cc_sdm845_probe(struct platform_device *pdev) 326c646b347SAmit Nischal { 327c646b347SAmit Nischal struct regmap *regmap; 328c646b347SAmit Nischal 329c646b347SAmit Nischal regmap = qcom_cc_map(pdev, &video_cc_sdm845_desc); 330c646b347SAmit Nischal if (IS_ERR(regmap)) 331c646b347SAmit Nischal return PTR_ERR(regmap); 332c646b347SAmit Nischal 333c646b347SAmit Nischal clk_fabia_pll_configure(&video_pll0, regmap, &video_pll0_config); 334c646b347SAmit Nischal 335c646b347SAmit Nischal return qcom_cc_really_probe(pdev, &video_cc_sdm845_desc, regmap); 336c646b347SAmit Nischal } 337c646b347SAmit Nischal 338c646b347SAmit Nischal static struct platform_driver video_cc_sdm845_driver = { 339c646b347SAmit Nischal .probe = video_cc_sdm845_probe, 340c646b347SAmit Nischal .driver = { 341c646b347SAmit Nischal .name = "sdm845-videocc", 342c646b347SAmit Nischal .of_match_table = video_cc_sdm845_match_table, 343c646b347SAmit Nischal }, 344c646b347SAmit Nischal }; 345c646b347SAmit Nischal 346c646b347SAmit Nischal static int __init video_cc_sdm845_init(void) 347c646b347SAmit Nischal { 348c646b347SAmit Nischal return platform_driver_register(&video_cc_sdm845_driver); 349c646b347SAmit Nischal } 350c646b347SAmit Nischal subsys_initcall(video_cc_sdm845_init); 351c646b347SAmit Nischal 352c646b347SAmit Nischal static void __exit video_cc_sdm845_exit(void) 353c646b347SAmit Nischal { 354c646b347SAmit Nischal platform_driver_unregister(&video_cc_sdm845_driver); 355c646b347SAmit Nischal } 356c646b347SAmit Nischal module_exit(video_cc_sdm845_exit); 357c646b347SAmit Nischal 358c646b347SAmit Nischal MODULE_LICENSE("GPL v2"); 359