Rtwo/kernel/motorola/sm8550/drivers/clk/qcom/cmn-blk-pll.c

199 lines
4.5 KiB
C
Raw Normal View History

2025-09-30 20:22:48 -04:00
// SPDX-License-Identifier: GPL-2.0-only
/*
* Copyright (c) 2019, 2021, The Linux Foundation. All rights reserved.
* Copyright (c) 2023, Qualcomm Innovation Center, Inc. All rights reserved.
*/
#include <linux/clk.h>
#include <linux/clk-provider.h>
#include <linux/err.h>
#include <linux/kernel.h>
#include <linux/module.h>
#include <linux/of.h>
#include <linux/of_device.h>
#include <linux/platform_device.h>
#include <linux/regmap.h>
#include <linux/reset.h>
#include <dt-bindings/clock/qcom,cmn-blk-pll.h>
#include "common.h"
#define SW_RESET_LOGIC_MASK BIT(6)
struct cmn_blk_pll {
struct regmap *regmap;
struct clk *misc_reset;
struct clk *ahb_clk;
struct clk *aon_clk;
struct reset_control *reset;
};
static struct cmn_blk_pll pll;
#define to_clk_fixed_rate(_hw) container_of(_hw, struct clk_fixed_rate, hw)
static unsigned long clk_cmn_blk_recalc_rate(struct clk_hw *hw,
unsigned long parent_rate)
{
return to_clk_fixed_rate(hw)->fixed_rate;
}
static int clk_cmn_blk_pll_enable(struct clk_hw *hw)
{
u32 val, ret;
ret = reset_control_reset(pll.reset);
if (ret)
return ret;
ret = clk_prepare_enable(pll.misc_reset);
if (ret)
return ret;
ret = clk_prepare_enable(pll.aon_clk);
if (ret)
return ret;
ret = clk_prepare_enable(pll.ahb_clk);
if (ret)
return ret;
ret = regmap_read(pll.regmap, 0, &val);
if (ret)
return ret;
if (val & SW_RESET_LOGIC_MASK) {
val &= ~SW_RESET_LOGIC_MASK;
regmap_write(pll.regmap, 0, val);
}
val |= SW_RESET_LOGIC_MASK;
regmap_write(pll.regmap, 0, val);
return 0;
}
static void clk_cmn_blk_pll_disable(struct clk_hw *hw)
{
u32 val;
regmap_read(pll.regmap, 0, &val);
val &= ~SW_RESET_LOGIC_MASK;
regmap_write(pll.regmap, 0, val);
clk_disable_unprepare(pll.misc_reset);
clk_disable_unprepare(pll.aon_clk);
clk_disable_unprepare(pll.ahb_clk);
}
const struct clk_ops clk_cmn_blk_ops = {
.enable = clk_cmn_blk_pll_enable,
.disable = clk_cmn_blk_pll_disable,
.recalc_rate = clk_cmn_blk_recalc_rate,
};
static struct clk_fixed_rate cmn_blk_pll = {
.fixed_rate = 100000000,
.hw.init = &(struct clk_init_data){
.name = "cmn_blk_pll",
.parent_data = &(const struct clk_parent_data){
.fw_name = "bi_tcxo",
},
.num_parents = 1,
.ops = &clk_cmn_blk_ops,
},
};
struct clk_hw *cmn_blk_pll_hws[] = {
[CMN_BLK_PLL] = &cmn_blk_pll.hw,
};
static const struct regmap_config cmn_blk_pll_regmap_config = {
.reg_bits = 32,
.reg_stride = 4,
.val_bits = 32,
.max_register = 0x8,
.fast_io = true,
};
static const struct qcom_cc_desc cmn_blk_pll_desc = {
.config = &cmn_blk_pll_regmap_config,
.clk_hws = cmn_blk_pll_hws,
.num_clk_hws = ARRAY_SIZE(cmn_blk_pll_hws),
};
static const struct of_device_id cmn_blk_pll_match_table[] = {
{ .compatible = "qcom,cmn_blk_pll" },
{ }
};
MODULE_DEVICE_TABLE(of, cmn_blk_pll_match_table);
static int cmn_blk_pll_probe(struct platform_device *pdev)
{
int ret;
pll.regmap = qcom_cc_map(pdev, &cmn_blk_pll_desc);
if (IS_ERR(pll.regmap))
return PTR_ERR(pll.regmap);
pll.misc_reset = devm_clk_get(&pdev->dev, "misc_reset_clk");
if (IS_ERR(pll.misc_reset)) {
if (PTR_ERR(pll.misc_reset) != -EPROBE_DEFER)
dev_err(&pdev->dev, "Unable to get misc_reset clock\n");
return PTR_ERR(pll.misc_reset);
}
pll.ahb_clk = devm_clk_get(&pdev->dev, "ahb_clk");
if (IS_ERR(pll.ahb_clk)) {
if (PTR_ERR(pll.ahb_clk) != -EPROBE_DEFER)
dev_err(&pdev->dev, "Unable to get ahb_clk clock\n");
return PTR_ERR(pll.ahb_clk);
}
pll.aon_clk = devm_clk_get(&pdev->dev, "aon_clk");
if (IS_ERR(pll.aon_clk)) {
if (PTR_ERR(pll.aon_clk) != -EPROBE_DEFER)
dev_err(&pdev->dev, "Unable to get aon_clk clock\n");
return PTR_ERR(pll.aon_clk);
}
pll.reset = devm_reset_control_get(&pdev->dev, "cmn_blk_pll_reset");
if (IS_ERR(pll.reset)) {
if (PTR_ERR(pll.reset) != -EPROBE_DEFER)
dev_err(&pdev->dev, "Unable to get cmn_blk_pll_reset\n");
return PTR_ERR(pll.reset);
}
ret = qcom_cc_really_probe(pdev, &cmn_blk_pll_desc, pll.regmap);
if (ret) {
dev_err(&pdev->dev, "Failed to register CMN BLK PLL\n");
return ret;
}
dev_info(&pdev->dev, "Registered CMN BLK PLL\n");
return 0;
}
static struct platform_driver cmn_blk_pll_driver = {
.probe = cmn_blk_pll_probe,
.driver = {
.name = "cmn_blk_pll",
.of_match_table = cmn_blk_pll_match_table,
},
};
static int __init cmn_blk_pll_init(void)
{
return platform_driver_register(&cmn_blk_pll_driver);
}
subsys_initcall(cmn_blk_pll_init);
static void __exit cmn_blk_pll_exit(void)
{
platform_driver_unregister(&cmn_blk_pll_driver);
}
module_exit(cmn_blk_pll_exit);
MODULE_DESCRIPTION("QTI CMN BLK PLL Driver");
MODULE_LICENSE("GPL v2");