// SPDX-License-Identifier: GPL-2.0 /* * Copyright 2024 NXP */ #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #include enum { CLK_GATE, CLK_DIVIDER, CLK_MUX, }; struct imx95_blk_ctl { struct device *dev; spinlock_t lock; struct clk *clk_apb; void __iomem *base; /* clock gate register */ u32 clk_reg_restore; }; struct imx95_blk_ctl_clk_dev_data { const char *name; const char * const *parent_names; u32 num_parents; u32 reg; u32 bit_idx; u32 bit_width; u32 clk_type; u32 flags; u32 flags2; u32 type; }; struct imx95_blk_ctl_dev_data { const struct imx95_blk_ctl_clk_dev_data *clk_dev_data; u32 num_clks; bool rpm_enabled; u32 clk_reg_offset; }; static const struct imx95_blk_ctl_clk_dev_data vpublk_clk_dev_data[] = { [IMX95_CLK_VPUBLK_WAVE] = { .name = "vpublk_wave_vpu", .parent_names = (const char *[]){ "vpu", }, .num_parents = 1, .reg = 8, .bit_idx = 0, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_VPUBLK_JPEG_ENC] = { .name = "vpublk_jpeg_enc", .parent_names = (const char *[]){ "vpujpeg", }, .num_parents = 1, .reg = 8, .bit_idx = 1, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_VPUBLK_JPEG_DEC] = { .name = "vpublk_jpeg_dec", .parent_names = (const char *[]){ "vpujpeg", }, .num_parents = 1, .reg = 8, .bit_idx = 2, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, } }; static const struct imx95_blk_ctl_dev_data vpublk_dev_data = { .num_clks = ARRAY_SIZE(vpublk_clk_dev_data), .clk_dev_data = vpublk_clk_dev_data, .rpm_enabled = true, .clk_reg_offset = 8, }; static const struct imx95_blk_ctl_clk_dev_data camblk_clk_dev_data[] = { [IMX95_CLK_CAMBLK_CSI2_FOR0] = { .name = "camblk_csi2_for0", .parent_names = (const char *[]){ "camisi", }, .num_parents = 1, .reg = 0, .bit_idx = 0, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_CAMBLK_CSI2_FOR1] = { .name = "camblk_csi2_for1", .parent_names = (const char *[]){ "camisi", }, .num_parents = 1, .reg = 0, .bit_idx = 1, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_CAMBLK_ISP_AXI] = { .name = "camblk_isp_axi", .parent_names = (const char *[]){ "camaxi", }, .num_parents = 1, .reg = 0, .bit_idx = 4, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_CAMBLK_ISP_PIXEL] = { .name = "camblk_isp_pixel", .parent_names = (const char *[]){ "camisi", }, .num_parents = 1, .reg = 0, .bit_idx = 5, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_CAMBLK_ISP] = { .name = "camblk_isp", .parent_names = (const char *[]){ "camisi", }, .num_parents = 1, .reg = 0, .bit_idx = 6, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, } }; static const struct imx95_blk_ctl_dev_data camblk_dev_data = { .num_clks = ARRAY_SIZE(camblk_clk_dev_data), .clk_dev_data = camblk_clk_dev_data, .clk_reg_offset = 0, }; static const struct imx95_blk_ctl_clk_dev_data lvds_clk_dev_data[] = { [IMX95_CLK_DISPMIX_LVDS_PHY_DIV] = { .name = "ldb_phy_div", .parent_names = (const char *[]){ "ldbpll", }, .num_parents = 1, .reg = 0, .bit_idx = 0, .bit_width = 1, .type = CLK_DIVIDER, .flags2 = CLK_DIVIDER_POWER_OF_TWO, }, [IMX95_CLK_DISPMIX_LVDS_CH0_GATE] = { .name = "lvds_ch0_gate", .parent_names = (const char *[]){ "ldb_phy_div", }, .num_parents = 1, .reg = 0, .bit_idx = 1, .bit_width = 1, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_DISPMIX_LVDS_CH1_GATE] = { .name = "lvds_ch1_gate", .parent_names = (const char *[]){ "ldb_phy_div", }, .num_parents = 1, .reg = 0, .bit_idx = 2, .bit_width = 1, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_DISPMIX_PIX_DI0_GATE] = { .name = "lvds_di0_gate", .parent_names = (const char *[]){ "ldb_pll_div7", }, .num_parents = 1, .reg = 0, .bit_idx = 3, .bit_width = 1, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, [IMX95_CLK_DISPMIX_PIX_DI1_GATE] = { .name = "lvds_di1_gate", .parent_names = (const char *[]){ "ldb_pll_div7", }, .num_parents = 1, .reg = 0, .bit_idx = 4, .bit_width = 1, .type = CLK_GATE, .flags = CLK_SET_RATE_PARENT, .flags2 = CLK_GATE_SET_TO_DISABLE, }, }; static const struct imx95_blk_ctl_dev_data lvds_csr_dev_data = { .num_clks = ARRAY_SIZE(lvds_clk_dev_data), .clk_dev_data = lvds_clk_dev_data, .clk_reg_offset = 0, }; static const struct imx95_blk_ctl_clk_dev_data dispmix_csr_clk_dev_data[] = { [IMX95_CLK_DISPMIX_ENG0_SEL] = { .name = "disp_engine0_sel", .parent_names = (const char *[]){"videopll1", "dsi_pll", "ldb_pll_div7", }, .num_parents = 4, .reg = 0, .bit_idx = 0, .bit_width = 2, .type = CLK_MUX, .flags = CLK_SET_RATE_NO_REPARENT | CLK_SET_RATE_PARENT, }, [IMX95_CLK_DISPMIX_ENG1_SEL] = { .name = "disp_engine1_sel", .parent_names = (const char *[]){"videopll1", "dsi_pll", "ldb_pll_div7", }, .num_parents = 4, .reg = 0, .bit_idx = 2, .bit_width = 2, .type = CLK_MUX, .flags = CLK_SET_RATE_NO_REPARENT | CLK_SET_RATE_PARENT, } }; static const struct imx95_blk_ctl_dev_data dispmix_csr_dev_data = { .num_clks = ARRAY_SIZE(dispmix_csr_clk_dev_data), .clk_dev_data = dispmix_csr_clk_dev_data, .clk_reg_offset = 0, }; static int imx95_bc_probe(struct platform_device *pdev) { struct device *dev = &pdev->dev; const struct imx95_blk_ctl_dev_data *bc_data; struct imx95_blk_ctl *bc; struct clk_hw_onecell_data *clk_hw_data; struct clk_hw **hws; void __iomem *base; int i, ret; bc = devm_kzalloc(dev, sizeof(*bc), GFP_KERNEL); if (!bc) return -ENOMEM; bc->dev = dev; dev_set_drvdata(&pdev->dev, bc); spin_lock_init(&bc->lock); base = devm_platform_ioremap_resource(pdev, 0); if (IS_ERR(base)) return PTR_ERR(base); bc->base = base; bc->clk_apb = devm_clk_get(dev, NULL); if (IS_ERR(bc->clk_apb)) return dev_err_probe(dev, PTR_ERR(bc->clk_apb), "failed to get APB clock\n"); ret = clk_prepare_enable(bc->clk_apb); if (ret) { dev_err(dev, "failed to enable apb clock: %d\n", ret); return ret; } bc_data = of_device_get_match_data(dev); if (!bc_data) return devm_of_platform_populate(dev); clk_hw_data = devm_kzalloc(dev, struct_size(clk_hw_data, hws, bc_data->num_clks), GFP_KERNEL); if (!clk_hw_data) return -ENOMEM; if (bc_data->rpm_enabled) pm_runtime_enable(&pdev->dev); clk_hw_data->num = bc_data->num_clks; hws = clk_hw_data->hws; for (i = 0; i < bc_data->num_clks; i++) { const struct imx95_blk_ctl_clk_dev_data *data = &bc_data->clk_dev_data[i]; void __iomem *reg = base + data->reg; if (data->type == CLK_MUX) { hws[i] = clk_hw_register_mux(dev, data->name, data->parent_names, data->num_parents, data->flags, reg, data->bit_idx, data->bit_width, data->flags2, &bc->lock); } else if (data->type == CLK_DIVIDER) { hws[i] = clk_hw_register_divider(dev, data->name, data->parent_names[0], data->flags, reg, data->bit_idx, data->bit_width, data->flags2, &bc->lock); } else { hws[i] = clk_hw_register_gate(dev, data->name, data->parent_names[0], data->flags, reg, data->bit_idx, data->flags2, &bc->lock); } if (IS_ERR(hws[i])) { ret = PTR_ERR(hws[i]); dev_err(dev, "failed to register: %s:%d\n", data->name, ret); goto cleanup; } } ret = of_clk_add_hw_provider(dev->of_node, of_clk_hw_onecell_get, clk_hw_data); if (ret) goto cleanup; ret = devm_of_platform_populate(dev); if (ret) { of_clk_del_provider(dev->of_node); goto cleanup; } if (pm_runtime_enabled(bc->dev)) clk_disable_unprepare(bc->clk_apb); return 0; cleanup: for (i = 0; i < bc_data->num_clks; i++) { if (IS_ERR_OR_NULL(hws[i])) continue; clk_hw_unregister(hws[i]); } if (bc_data->rpm_enabled) pm_runtime_disable(&pdev->dev); return ret; } #ifdef CONFIG_PM static int imx95_bc_runtime_suspend(struct device *dev) { struct imx95_blk_ctl *bc = dev_get_drvdata(dev); clk_disable_unprepare(bc->clk_apb); return 0; } static int imx95_bc_runtime_resume(struct device *dev) { struct imx95_blk_ctl *bc = dev_get_drvdata(dev); return clk_prepare_enable(bc->clk_apb); } #endif #ifdef CONFIG_PM_SLEEP static int imx95_bc_suspend(struct device *dev) { struct imx95_blk_ctl *bc = dev_get_drvdata(dev); const struct imx95_blk_ctl_dev_data *bc_data; int ret; bc_data = of_device_get_match_data(dev); if (!bc_data) return 0; if (bc_data->rpm_enabled) { ret = pm_runtime_get_sync(bc->dev); if (ret < 0) { pm_runtime_put_noidle(bc->dev); return ret; } } bc->clk_reg_restore = readl(bc->base + bc_data->clk_reg_offset); return 0; } static int imx95_bc_resume(struct device *dev) { struct imx95_blk_ctl *bc = dev_get_drvdata(dev); const struct imx95_blk_ctl_dev_data *bc_data; bc_data = of_device_get_match_data(dev); if (!bc_data) return 0; writel(bc->clk_reg_restore, bc->base + bc_data->clk_reg_offset); if (bc_data->rpm_enabled) pm_runtime_put(bc->dev); return 0; } #endif static const struct dev_pm_ops imx95_bc_pm_ops = { SET_RUNTIME_PM_OPS(imx95_bc_runtime_suspend, imx95_bc_runtime_resume, NULL) SET_SYSTEM_SLEEP_PM_OPS(imx95_bc_suspend, imx95_bc_resume) }; static const struct of_device_id imx95_bc_of_match[] = { { .compatible = "nxp,imx95-camera-csr", .data = &camblk_dev_data }, { .compatible = "nxp,imx95-display-master-csr", }, { .compatible = "nxp,imx95-lvds-csr", .data = &lvds_csr_dev_data }, { .compatible = "nxp,imx95-display-csr", .data = &dispmix_csr_dev_data }, { .compatible = "nxp,imx95-vpu-csr", .data = &vpublk_dev_data }, { /* Sentinel */ }, }; MODULE_DEVICE_TABLE(of, imx95_bc_of_match); static struct platform_driver imx95_bc_driver = { .probe = imx95_bc_probe, .driver = { .name = "imx95-blk-ctl", .of_match_table = imx95_bc_of_match, .pm = &imx95_bc_pm_ops, }, }; module_platform_driver(imx95_bc_driver); MODULE_DESCRIPTION("NXP i.MX95 blk ctl driver"); MODULE_AUTHOR("Peng Fan "); MODULE_LICENSE("GPL");