162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci/* 362306a36Sopenharmony_ci * LPASS Audio CC and Always ON CC Glitch Free Mux clock driver 462306a36Sopenharmony_ci * 562306a36Sopenharmony_ci * Copyright (c) 2020 Linaro Ltd. 662306a36Sopenharmony_ci * Author: Srinivas Kandagatla <srinivas.kandagatla@linaro.org> 762306a36Sopenharmony_ci */ 862306a36Sopenharmony_ci 962306a36Sopenharmony_ci#include <linux/kernel.h> 1062306a36Sopenharmony_ci#include <linux/module.h> 1162306a36Sopenharmony_ci#include <linux/clk-provider.h> 1262306a36Sopenharmony_ci#include <linux/io.h> 1362306a36Sopenharmony_ci#include <linux/slab.h> 1462306a36Sopenharmony_ci#include <linux/err.h> 1562306a36Sopenharmony_ci#include <linux/pm_clock.h> 1662306a36Sopenharmony_ci#include <linux/pm_runtime.h> 1762306a36Sopenharmony_ci#include <linux/device.h> 1862306a36Sopenharmony_ci#include <linux/of.h> 1962306a36Sopenharmony_ci#include <linux/platform_device.h> 2062306a36Sopenharmony_ci#include <dt-bindings/clock/qcom,sm8250-lpass-audiocc.h> 2162306a36Sopenharmony_ci#include <dt-bindings/clock/qcom,sm8250-lpass-aoncc.h> 2262306a36Sopenharmony_ci 2362306a36Sopenharmony_cistruct lpass_gfm { 2462306a36Sopenharmony_ci struct device *dev; 2562306a36Sopenharmony_ci void __iomem *base; 2662306a36Sopenharmony_ci}; 2762306a36Sopenharmony_ci 2862306a36Sopenharmony_cistruct clk_gfm { 2962306a36Sopenharmony_ci unsigned int mux_reg; 3062306a36Sopenharmony_ci unsigned int mux_mask; 3162306a36Sopenharmony_ci struct clk_hw hw; 3262306a36Sopenharmony_ci struct lpass_gfm *priv; 3362306a36Sopenharmony_ci void __iomem *gfm_mux; 3462306a36Sopenharmony_ci}; 3562306a36Sopenharmony_ci 3662306a36Sopenharmony_ci#define to_clk_gfm(_hw) container_of(_hw, struct clk_gfm, hw) 3762306a36Sopenharmony_ci 3862306a36Sopenharmony_cistatic u8 clk_gfm_get_parent(struct clk_hw *hw) 3962306a36Sopenharmony_ci{ 4062306a36Sopenharmony_ci struct clk_gfm *clk = to_clk_gfm(hw); 4162306a36Sopenharmony_ci 4262306a36Sopenharmony_ci return readl(clk->gfm_mux) & clk->mux_mask; 4362306a36Sopenharmony_ci} 4462306a36Sopenharmony_ci 4562306a36Sopenharmony_cistatic int clk_gfm_set_parent(struct clk_hw *hw, u8 index) 4662306a36Sopenharmony_ci{ 4762306a36Sopenharmony_ci struct clk_gfm *clk = to_clk_gfm(hw); 4862306a36Sopenharmony_ci unsigned int val; 4962306a36Sopenharmony_ci 5062306a36Sopenharmony_ci val = readl(clk->gfm_mux); 5162306a36Sopenharmony_ci 5262306a36Sopenharmony_ci if (index) 5362306a36Sopenharmony_ci val |= clk->mux_mask; 5462306a36Sopenharmony_ci else 5562306a36Sopenharmony_ci val &= ~clk->mux_mask; 5662306a36Sopenharmony_ci 5762306a36Sopenharmony_ci 5862306a36Sopenharmony_ci writel(val, clk->gfm_mux); 5962306a36Sopenharmony_ci 6062306a36Sopenharmony_ci return 0; 6162306a36Sopenharmony_ci} 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_cistatic const struct clk_ops clk_gfm_ops = { 6462306a36Sopenharmony_ci .get_parent = clk_gfm_get_parent, 6562306a36Sopenharmony_ci .set_parent = clk_gfm_set_parent, 6662306a36Sopenharmony_ci .determine_rate = __clk_mux_determine_rate, 6762306a36Sopenharmony_ci}; 6862306a36Sopenharmony_ci 6962306a36Sopenharmony_cistatic struct clk_gfm lpass_gfm_va_mclk = { 7062306a36Sopenharmony_ci .mux_reg = 0x20000, 7162306a36Sopenharmony_ci .mux_mask = BIT(0), 7262306a36Sopenharmony_ci .hw.init = &(struct clk_init_data) { 7362306a36Sopenharmony_ci .name = "VA_MCLK", 7462306a36Sopenharmony_ci .ops = &clk_gfm_ops, 7562306a36Sopenharmony_ci .flags = CLK_SET_RATE_PARENT | CLK_OPS_PARENT_ENABLE, 7662306a36Sopenharmony_ci .num_parents = 2, 7762306a36Sopenharmony_ci .parent_data = (const struct clk_parent_data[]){ 7862306a36Sopenharmony_ci { 7962306a36Sopenharmony_ci .index = 0, 8062306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_TX_CORE_MCLK", 8162306a36Sopenharmony_ci }, { 8262306a36Sopenharmony_ci .index = 1, 8362306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_VA_CORE_MCLK", 8462306a36Sopenharmony_ci }, 8562306a36Sopenharmony_ci }, 8662306a36Sopenharmony_ci }, 8762306a36Sopenharmony_ci}; 8862306a36Sopenharmony_ci 8962306a36Sopenharmony_cistatic struct clk_gfm lpass_gfm_tx_npl = { 9062306a36Sopenharmony_ci .mux_reg = 0x20000, 9162306a36Sopenharmony_ci .mux_mask = BIT(0), 9262306a36Sopenharmony_ci .hw.init = &(struct clk_init_data) { 9362306a36Sopenharmony_ci .name = "TX_NPL", 9462306a36Sopenharmony_ci .ops = &clk_gfm_ops, 9562306a36Sopenharmony_ci .flags = CLK_SET_RATE_PARENT | CLK_OPS_PARENT_ENABLE, 9662306a36Sopenharmony_ci .parent_data = (const struct clk_parent_data[]){ 9762306a36Sopenharmony_ci { 9862306a36Sopenharmony_ci .index = 0, 9962306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_TX_CORE_NPL_MCLK", 10062306a36Sopenharmony_ci }, { 10162306a36Sopenharmony_ci .index = 1, 10262306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_VA_CORE_2X_MCLK", 10362306a36Sopenharmony_ci }, 10462306a36Sopenharmony_ci }, 10562306a36Sopenharmony_ci .num_parents = 2, 10662306a36Sopenharmony_ci }, 10762306a36Sopenharmony_ci}; 10862306a36Sopenharmony_ci 10962306a36Sopenharmony_cistatic struct clk_gfm lpass_gfm_wsa_mclk = { 11062306a36Sopenharmony_ci .mux_reg = 0x220d8, 11162306a36Sopenharmony_ci .mux_mask = BIT(0), 11262306a36Sopenharmony_ci .hw.init = &(struct clk_init_data) { 11362306a36Sopenharmony_ci .name = "WSA_MCLK", 11462306a36Sopenharmony_ci .ops = &clk_gfm_ops, 11562306a36Sopenharmony_ci .flags = CLK_SET_RATE_PARENT | CLK_OPS_PARENT_ENABLE, 11662306a36Sopenharmony_ci .parent_data = (const struct clk_parent_data[]){ 11762306a36Sopenharmony_ci { 11862306a36Sopenharmony_ci .index = 0, 11962306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_TX_CORE_MCLK", 12062306a36Sopenharmony_ci }, { 12162306a36Sopenharmony_ci .index = 1, 12262306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_WSA_CORE_MCLK", 12362306a36Sopenharmony_ci }, 12462306a36Sopenharmony_ci }, 12562306a36Sopenharmony_ci .num_parents = 2, 12662306a36Sopenharmony_ci }, 12762306a36Sopenharmony_ci}; 12862306a36Sopenharmony_ci 12962306a36Sopenharmony_cistatic struct clk_gfm lpass_gfm_wsa_npl = { 13062306a36Sopenharmony_ci .mux_reg = 0x220d8, 13162306a36Sopenharmony_ci .mux_mask = BIT(0), 13262306a36Sopenharmony_ci .hw.init = &(struct clk_init_data) { 13362306a36Sopenharmony_ci .name = "WSA_NPL", 13462306a36Sopenharmony_ci .ops = &clk_gfm_ops, 13562306a36Sopenharmony_ci .flags = CLK_SET_RATE_PARENT | CLK_OPS_PARENT_ENABLE, 13662306a36Sopenharmony_ci .parent_data = (const struct clk_parent_data[]){ 13762306a36Sopenharmony_ci { 13862306a36Sopenharmony_ci .index = 0, 13962306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_TX_CORE_NPL_MCLK", 14062306a36Sopenharmony_ci }, { 14162306a36Sopenharmony_ci .index = 1, 14262306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_WSA_CORE_NPL_MCLK", 14362306a36Sopenharmony_ci }, 14462306a36Sopenharmony_ci }, 14562306a36Sopenharmony_ci .num_parents = 2, 14662306a36Sopenharmony_ci }, 14762306a36Sopenharmony_ci}; 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_cistatic struct clk_gfm lpass_gfm_rx_mclk_mclk2 = { 15062306a36Sopenharmony_ci .mux_reg = 0x240d8, 15162306a36Sopenharmony_ci .mux_mask = BIT(0), 15262306a36Sopenharmony_ci .hw.init = &(struct clk_init_data) { 15362306a36Sopenharmony_ci .name = "RX_MCLK_MCLK2", 15462306a36Sopenharmony_ci .ops = &clk_gfm_ops, 15562306a36Sopenharmony_ci .flags = CLK_SET_RATE_PARENT | CLK_OPS_PARENT_ENABLE, 15662306a36Sopenharmony_ci .parent_data = (const struct clk_parent_data[]){ 15762306a36Sopenharmony_ci { 15862306a36Sopenharmony_ci .index = 0, 15962306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_TX_CORE_MCLK", 16062306a36Sopenharmony_ci }, { 16162306a36Sopenharmony_ci .index = 1, 16262306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_RX_CORE_MCLK", 16362306a36Sopenharmony_ci }, 16462306a36Sopenharmony_ci }, 16562306a36Sopenharmony_ci .num_parents = 2, 16662306a36Sopenharmony_ci }, 16762306a36Sopenharmony_ci}; 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_cistatic struct clk_gfm lpass_gfm_rx_npl = { 17062306a36Sopenharmony_ci .mux_reg = 0x240d8, 17162306a36Sopenharmony_ci .mux_mask = BIT(0), 17262306a36Sopenharmony_ci .hw.init = &(struct clk_init_data) { 17362306a36Sopenharmony_ci .name = "RX_NPL", 17462306a36Sopenharmony_ci .ops = &clk_gfm_ops, 17562306a36Sopenharmony_ci .flags = CLK_SET_RATE_PARENT | CLK_OPS_PARENT_ENABLE, 17662306a36Sopenharmony_ci .parent_data = (const struct clk_parent_data[]){ 17762306a36Sopenharmony_ci { 17862306a36Sopenharmony_ci .index = 0, 17962306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_TX_CORE_NPL_MCLK", 18062306a36Sopenharmony_ci }, { 18162306a36Sopenharmony_ci .index = 1, 18262306a36Sopenharmony_ci .fw_name = "LPASS_CLK_ID_RX_CORE_NPL_MCLK", 18362306a36Sopenharmony_ci }, 18462306a36Sopenharmony_ci }, 18562306a36Sopenharmony_ci .num_parents = 2, 18662306a36Sopenharmony_ci }, 18762306a36Sopenharmony_ci}; 18862306a36Sopenharmony_ci 18962306a36Sopenharmony_cistatic struct clk_gfm *aoncc_gfm_clks[] = { 19062306a36Sopenharmony_ci [LPASS_CDC_VA_MCLK] = &lpass_gfm_va_mclk, 19162306a36Sopenharmony_ci [LPASS_CDC_TX_NPL] = &lpass_gfm_tx_npl, 19262306a36Sopenharmony_ci}; 19362306a36Sopenharmony_ci 19462306a36Sopenharmony_cistatic struct clk_hw_onecell_data aoncc_hw_onecell_data = { 19562306a36Sopenharmony_ci .hws = { 19662306a36Sopenharmony_ci [LPASS_CDC_VA_MCLK] = &lpass_gfm_va_mclk.hw, 19762306a36Sopenharmony_ci [LPASS_CDC_TX_NPL] = &lpass_gfm_tx_npl.hw, 19862306a36Sopenharmony_ci }, 19962306a36Sopenharmony_ci .num = ARRAY_SIZE(aoncc_gfm_clks), 20062306a36Sopenharmony_ci}; 20162306a36Sopenharmony_ci 20262306a36Sopenharmony_cistatic struct clk_gfm *audiocc_gfm_clks[] = { 20362306a36Sopenharmony_ci [LPASS_CDC_WSA_NPL] = &lpass_gfm_wsa_npl, 20462306a36Sopenharmony_ci [LPASS_CDC_WSA_MCLK] = &lpass_gfm_wsa_mclk, 20562306a36Sopenharmony_ci [LPASS_CDC_RX_NPL] = &lpass_gfm_rx_npl, 20662306a36Sopenharmony_ci [LPASS_CDC_RX_MCLK_MCLK2] = &lpass_gfm_rx_mclk_mclk2, 20762306a36Sopenharmony_ci}; 20862306a36Sopenharmony_ci 20962306a36Sopenharmony_cistatic struct clk_hw_onecell_data audiocc_hw_onecell_data = { 21062306a36Sopenharmony_ci .hws = { 21162306a36Sopenharmony_ci [LPASS_CDC_WSA_NPL] = &lpass_gfm_wsa_npl.hw, 21262306a36Sopenharmony_ci [LPASS_CDC_WSA_MCLK] = &lpass_gfm_wsa_mclk.hw, 21362306a36Sopenharmony_ci [LPASS_CDC_RX_NPL] = &lpass_gfm_rx_npl.hw, 21462306a36Sopenharmony_ci [LPASS_CDC_RX_MCLK_MCLK2] = &lpass_gfm_rx_mclk_mclk2.hw, 21562306a36Sopenharmony_ci }, 21662306a36Sopenharmony_ci .num = ARRAY_SIZE(audiocc_gfm_clks), 21762306a36Sopenharmony_ci}; 21862306a36Sopenharmony_ci 21962306a36Sopenharmony_cistruct lpass_gfm_data { 22062306a36Sopenharmony_ci struct clk_hw_onecell_data *onecell_data; 22162306a36Sopenharmony_ci struct clk_gfm **gfm_clks; 22262306a36Sopenharmony_ci}; 22362306a36Sopenharmony_ci 22462306a36Sopenharmony_cistatic struct lpass_gfm_data audiocc_data = { 22562306a36Sopenharmony_ci .onecell_data = &audiocc_hw_onecell_data, 22662306a36Sopenharmony_ci .gfm_clks = audiocc_gfm_clks, 22762306a36Sopenharmony_ci}; 22862306a36Sopenharmony_ci 22962306a36Sopenharmony_cistatic struct lpass_gfm_data aoncc_data = { 23062306a36Sopenharmony_ci .onecell_data = &aoncc_hw_onecell_data, 23162306a36Sopenharmony_ci .gfm_clks = aoncc_gfm_clks, 23262306a36Sopenharmony_ci}; 23362306a36Sopenharmony_ci 23462306a36Sopenharmony_cistatic int lpass_gfm_clk_driver_probe(struct platform_device *pdev) 23562306a36Sopenharmony_ci{ 23662306a36Sopenharmony_ci const struct lpass_gfm_data *data; 23762306a36Sopenharmony_ci struct device *dev = &pdev->dev; 23862306a36Sopenharmony_ci struct clk_gfm *gfm; 23962306a36Sopenharmony_ci struct lpass_gfm *cc; 24062306a36Sopenharmony_ci int err, i; 24162306a36Sopenharmony_ci 24262306a36Sopenharmony_ci data = of_device_get_match_data(dev); 24362306a36Sopenharmony_ci if (!data) 24462306a36Sopenharmony_ci return -EINVAL; 24562306a36Sopenharmony_ci 24662306a36Sopenharmony_ci cc = devm_kzalloc(dev, sizeof(*cc), GFP_KERNEL); 24762306a36Sopenharmony_ci if (!cc) 24862306a36Sopenharmony_ci return -ENOMEM; 24962306a36Sopenharmony_ci 25062306a36Sopenharmony_ci cc->base = devm_platform_ioremap_resource(pdev, 0); 25162306a36Sopenharmony_ci if (IS_ERR(cc->base)) 25262306a36Sopenharmony_ci return PTR_ERR(cc->base); 25362306a36Sopenharmony_ci 25462306a36Sopenharmony_ci err = devm_pm_runtime_enable(dev); 25562306a36Sopenharmony_ci if (err) 25662306a36Sopenharmony_ci return err; 25762306a36Sopenharmony_ci 25862306a36Sopenharmony_ci err = devm_pm_clk_create(dev); 25962306a36Sopenharmony_ci if (err) 26062306a36Sopenharmony_ci return err; 26162306a36Sopenharmony_ci 26262306a36Sopenharmony_ci err = of_pm_clk_add_clks(dev); 26362306a36Sopenharmony_ci if (err < 0) { 26462306a36Sopenharmony_ci dev_dbg(dev, "Failed to get lpass core voting clocks\n"); 26562306a36Sopenharmony_ci return err; 26662306a36Sopenharmony_ci } 26762306a36Sopenharmony_ci 26862306a36Sopenharmony_ci for (i = 0; i < data->onecell_data->num; i++) { 26962306a36Sopenharmony_ci if (!data->gfm_clks[i]) 27062306a36Sopenharmony_ci continue; 27162306a36Sopenharmony_ci 27262306a36Sopenharmony_ci gfm = data->gfm_clks[i]; 27362306a36Sopenharmony_ci gfm->priv = cc; 27462306a36Sopenharmony_ci gfm->gfm_mux = cc->base; 27562306a36Sopenharmony_ci gfm->gfm_mux = gfm->gfm_mux + data->gfm_clks[i]->mux_reg; 27662306a36Sopenharmony_ci 27762306a36Sopenharmony_ci err = devm_clk_hw_register(dev, &data->gfm_clks[i]->hw); 27862306a36Sopenharmony_ci if (err) 27962306a36Sopenharmony_ci return err; 28062306a36Sopenharmony_ci 28162306a36Sopenharmony_ci } 28262306a36Sopenharmony_ci 28362306a36Sopenharmony_ci err = devm_of_clk_add_hw_provider(dev, of_clk_hw_onecell_get, 28462306a36Sopenharmony_ci data->onecell_data); 28562306a36Sopenharmony_ci if (err) 28662306a36Sopenharmony_ci return err; 28762306a36Sopenharmony_ci 28862306a36Sopenharmony_ci return 0; 28962306a36Sopenharmony_ci} 29062306a36Sopenharmony_ci 29162306a36Sopenharmony_cistatic const struct of_device_id lpass_gfm_clk_match_table[] = { 29262306a36Sopenharmony_ci { 29362306a36Sopenharmony_ci .compatible = "qcom,sm8250-lpass-aoncc", 29462306a36Sopenharmony_ci .data = &aoncc_data, 29562306a36Sopenharmony_ci }, 29662306a36Sopenharmony_ci { 29762306a36Sopenharmony_ci .compatible = "qcom,sm8250-lpass-audiocc", 29862306a36Sopenharmony_ci .data = &audiocc_data, 29962306a36Sopenharmony_ci }, 30062306a36Sopenharmony_ci { } 30162306a36Sopenharmony_ci}; 30262306a36Sopenharmony_ciMODULE_DEVICE_TABLE(of, lpass_gfm_clk_match_table); 30362306a36Sopenharmony_ci 30462306a36Sopenharmony_cistatic const struct dev_pm_ops lpass_gfm_pm_ops = { 30562306a36Sopenharmony_ci SET_RUNTIME_PM_OPS(pm_clk_suspend, pm_clk_resume, NULL) 30662306a36Sopenharmony_ci}; 30762306a36Sopenharmony_ci 30862306a36Sopenharmony_cistatic struct platform_driver lpass_gfm_clk_driver = { 30962306a36Sopenharmony_ci .probe = lpass_gfm_clk_driver_probe, 31062306a36Sopenharmony_ci .driver = { 31162306a36Sopenharmony_ci .name = "lpass-gfm-clk", 31262306a36Sopenharmony_ci .of_match_table = lpass_gfm_clk_match_table, 31362306a36Sopenharmony_ci .pm = &lpass_gfm_pm_ops, 31462306a36Sopenharmony_ci }, 31562306a36Sopenharmony_ci}; 31662306a36Sopenharmony_cimodule_platform_driver(lpass_gfm_clk_driver); 31762306a36Sopenharmony_ciMODULE_LICENSE("GPL v2"); 318