Path: blob/master/sound/soc/mediatek/mt8365/mt8365-afe-clk.c
26488 views
// SPDX-License-Identifier: GPL-2.01/*2* MediaTek 8365 AFE clock control3*4* Copyright (c) 2024 MediaTek Inc.5* Authors: Jia Zeng <[email protected]>6* Alexandre Mergnat <[email protected]>7*/89#include "mt8365-afe-clk.h"10#include "mt8365-afe-common.h"11#include "mt8365-reg.h"12#include "../common/mtk-base-afe.h"13#include <linux/device.h>14#include <linux/mfd/syscon.h>1516static const char *aud_clks[MT8365_CLK_NUM] = {17[MT8365_CLK_TOP_AUD_SEL] = "top_audio_sel",18[MT8365_CLK_AUD_I2S0_M] = "audio_i2s0_m",19[MT8365_CLK_AUD_I2S1_M] = "audio_i2s1_m",20[MT8365_CLK_AUD_I2S2_M] = "audio_i2s2_m",21[MT8365_CLK_AUD_I2S3_M] = "audio_i2s3_m",22[MT8365_CLK_ENGEN1] = "engen1",23[MT8365_CLK_ENGEN2] = "engen2",24[MT8365_CLK_AUD1] = "aud1",25[MT8365_CLK_AUD2] = "aud2",26[MT8365_CLK_I2S0_M_SEL] = "i2s0_m_sel",27[MT8365_CLK_I2S1_M_SEL] = "i2s1_m_sel",28[MT8365_CLK_I2S2_M_SEL] = "i2s2_m_sel",29[MT8365_CLK_I2S3_M_SEL] = "i2s3_m_sel",30[MT8365_CLK_CLK26M] = "top_clk26m_clk",31};3233int mt8365_afe_init_audio_clk(struct mtk_base_afe *afe)34{35size_t i;36struct mt8365_afe_private *afe_priv = afe->platform_priv;3738for (i = 0; i < ARRAY_SIZE(aud_clks); i++) {39afe_priv->clocks[i] = devm_clk_get(afe->dev, aud_clks[i]);40if (IS_ERR(afe_priv->clocks[i])) {41dev_err(afe->dev, "%s devm_clk_get %s fail\n",42__func__, aud_clks[i]);43return PTR_ERR(afe_priv->clocks[i]);44}45}46return 0;47}4849void mt8365_afe_disable_clk(struct mtk_base_afe *afe, struct clk *clk)50{51clk_disable_unprepare(clk);52}5354int mt8365_afe_set_clk_rate(struct mtk_base_afe *afe, struct clk *clk,55unsigned int rate)56{57int ret;5859if (clk) {60ret = clk_set_rate(clk, rate);61if (ret) {62dev_err(afe->dev, "Failed to set rate\n");63return ret;64}65}66return 0;67}6869int mt8365_afe_set_clk_parent(struct mtk_base_afe *afe, struct clk *clk,70struct clk *parent)71{72int ret;7374if (clk && parent) {75ret = clk_set_parent(clk, parent);76if (ret) {77dev_err(afe->dev, "Failed to set parent\n");78return ret;79}80}81return 0;82}8384static unsigned int get_top_cg_reg(unsigned int cg_type)85{86switch (cg_type) {87case MT8365_TOP_CG_AFE:88case MT8365_TOP_CG_I2S_IN:89case MT8365_TOP_CG_22M:90case MT8365_TOP_CG_24M:91case MT8365_TOP_CG_INTDIR_CK:92case MT8365_TOP_CG_APLL2_TUNER:93case MT8365_TOP_CG_APLL_TUNER:94case MT8365_TOP_CG_SPDIF:95case MT8365_TOP_CG_TDM_OUT:96case MT8365_TOP_CG_TDM_IN:97case MT8365_TOP_CG_ADC:98case MT8365_TOP_CG_DAC:99case MT8365_TOP_CG_DAC_PREDIS:100case MT8365_TOP_CG_TML:101return AUDIO_TOP_CON0;102case MT8365_TOP_CG_I2S1_BCLK:103case MT8365_TOP_CG_I2S2_BCLK:104case MT8365_TOP_CG_I2S3_BCLK:105case MT8365_TOP_CG_I2S4_BCLK:106case MT8365_TOP_CG_DMIC0_ADC:107case MT8365_TOP_CG_DMIC1_ADC:108case MT8365_TOP_CG_DMIC2_ADC:109case MT8365_TOP_CG_DMIC3_ADC:110case MT8365_TOP_CG_CONNSYS_I2S_ASRC:111case MT8365_TOP_CG_GENERAL1_ASRC:112case MT8365_TOP_CG_GENERAL2_ASRC:113case MT8365_TOP_CG_TDM_ASRC:114return AUDIO_TOP_CON1;115default:116return 0;117}118}119120static unsigned int get_top_cg_mask(unsigned int cg_type)121{122switch (cg_type) {123case MT8365_TOP_CG_AFE:124return AUD_TCON0_PDN_AFE;125case MT8365_TOP_CG_I2S_IN:126return AUD_TCON0_PDN_I2S_IN;127case MT8365_TOP_CG_22M:128return AUD_TCON0_PDN_22M;129case MT8365_TOP_CG_24M:130return AUD_TCON0_PDN_24M;131case MT8365_TOP_CG_INTDIR_CK:132return AUD_TCON0_PDN_INTDIR;133case MT8365_TOP_CG_APLL2_TUNER:134return AUD_TCON0_PDN_APLL2_TUNER;135case MT8365_TOP_CG_APLL_TUNER:136return AUD_TCON0_PDN_APLL_TUNER;137case MT8365_TOP_CG_SPDIF:138return AUD_TCON0_PDN_SPDIF;139case MT8365_TOP_CG_TDM_OUT:140return AUD_TCON0_PDN_TDM_OUT;141case MT8365_TOP_CG_TDM_IN:142return AUD_TCON0_PDN_TDM_IN;143case MT8365_TOP_CG_ADC:144return AUD_TCON0_PDN_ADC;145case MT8365_TOP_CG_DAC:146return AUD_TCON0_PDN_DAC;147case MT8365_TOP_CG_DAC_PREDIS:148return AUD_TCON0_PDN_DAC_PREDIS;149case MT8365_TOP_CG_TML:150return AUD_TCON0_PDN_TML;151case MT8365_TOP_CG_I2S1_BCLK:152return AUD_TCON1_PDN_I2S1_BCLK;153case MT8365_TOP_CG_I2S2_BCLK:154return AUD_TCON1_PDN_I2S2_BCLK;155case MT8365_TOP_CG_I2S3_BCLK:156return AUD_TCON1_PDN_I2S3_BCLK;157case MT8365_TOP_CG_I2S4_BCLK:158return AUD_TCON1_PDN_I2S4_BCLK;159case MT8365_TOP_CG_DMIC0_ADC:160return AUD_TCON1_PDN_DMIC0_ADC;161case MT8365_TOP_CG_DMIC1_ADC:162return AUD_TCON1_PDN_DMIC1_ADC;163case MT8365_TOP_CG_DMIC2_ADC:164return AUD_TCON1_PDN_DMIC2_ADC;165case MT8365_TOP_CG_DMIC3_ADC:166return AUD_TCON1_PDN_DMIC3_ADC;167case MT8365_TOP_CG_CONNSYS_I2S_ASRC:168return AUD_TCON1_PDN_CONNSYS_I2S_ASRC;169case MT8365_TOP_CG_GENERAL1_ASRC:170return AUD_TCON1_PDN_GENERAL1_ASRC;171case MT8365_TOP_CG_GENERAL2_ASRC:172return AUD_TCON1_PDN_GENERAL2_ASRC;173case MT8365_TOP_CG_TDM_ASRC:174return AUD_TCON1_PDN_TDM_ASRC;175default:176return 0;177}178}179180static unsigned int get_top_cg_on_val(unsigned int cg_type)181{182return 0;183}184185static unsigned int get_top_cg_off_val(unsigned int cg_type)186{187return get_top_cg_mask(cg_type);188}189190int mt8365_afe_enable_top_cg(struct mtk_base_afe *afe, unsigned int cg_type)191{192struct mt8365_afe_private *afe_priv = afe->platform_priv;193unsigned int reg = get_top_cg_reg(cg_type);194unsigned int mask = get_top_cg_mask(cg_type);195unsigned int val = get_top_cg_on_val(cg_type);196unsigned long flags;197198spin_lock_irqsave(&afe_priv->afe_ctrl_lock, flags);199200afe_priv->top_cg_ref_cnt[cg_type]++;201if (afe_priv->top_cg_ref_cnt[cg_type] == 1)202regmap_update_bits(afe->regmap, reg, mask, val);203204spin_unlock_irqrestore(&afe_priv->afe_ctrl_lock, flags);205206return 0;207}208209int mt8365_afe_disable_top_cg(struct mtk_base_afe *afe, unsigned int cg_type)210{211struct mt8365_afe_private *afe_priv = afe->platform_priv;212unsigned int reg = get_top_cg_reg(cg_type);213unsigned int mask = get_top_cg_mask(cg_type);214unsigned int val = get_top_cg_off_val(cg_type);215unsigned long flags;216217spin_lock_irqsave(&afe_priv->afe_ctrl_lock, flags);218219afe_priv->top_cg_ref_cnt[cg_type]--;220if (afe_priv->top_cg_ref_cnt[cg_type] == 0)221regmap_update_bits(afe->regmap, reg, mask, val);222else if (afe_priv->top_cg_ref_cnt[cg_type] < 0)223afe_priv->top_cg_ref_cnt[cg_type] = 0;224225spin_unlock_irqrestore(&afe_priv->afe_ctrl_lock, flags);226227return 0;228}229230int mt8365_afe_enable_main_clk(struct mtk_base_afe *afe)231{232struct mt8365_afe_private *afe_priv = afe->platform_priv;233234clk_prepare_enable(afe_priv->clocks[MT8365_CLK_TOP_AUD_SEL]);235mt8365_afe_enable_top_cg(afe, MT8365_TOP_CG_AFE);236mt8365_afe_enable_afe_on(afe);237238return 0;239}240241int mt8365_afe_disable_main_clk(struct mtk_base_afe *afe)242{243struct mt8365_afe_private *afe_priv = afe->platform_priv;244245mt8365_afe_disable_afe_on(afe);246mt8365_afe_disable_top_cg(afe, MT8365_TOP_CG_AFE);247mt8365_afe_disable_clk(afe, afe_priv->clocks[MT8365_CLK_TOP_AUD_SEL]);248249return 0;250}251252int mt8365_afe_emi_clk_on(struct mtk_base_afe *afe)253{254return 0;255}256257int mt8365_afe_emi_clk_off(struct mtk_base_afe *afe)258{259return 0;260}261262int mt8365_afe_enable_afe_on(struct mtk_base_afe *afe)263{264struct mt8365_afe_private *afe_priv = afe->platform_priv;265unsigned long flags;266267spin_lock_irqsave(&afe_priv->afe_ctrl_lock, flags);268269afe_priv->afe_on_ref_cnt++;270if (afe_priv->afe_on_ref_cnt == 1)271regmap_update_bits(afe->regmap, AFE_DAC_CON0, 0x1, 0x1);272273spin_unlock_irqrestore(&afe_priv->afe_ctrl_lock, flags);274275return 0;276}277278int mt8365_afe_disable_afe_on(struct mtk_base_afe *afe)279{280struct mt8365_afe_private *afe_priv = afe->platform_priv;281unsigned long flags;282283spin_lock_irqsave(&afe_priv->afe_ctrl_lock, flags);284285afe_priv->afe_on_ref_cnt--;286if (afe_priv->afe_on_ref_cnt == 0)287regmap_update_bits(afe->regmap, AFE_DAC_CON0, 0x1, 0x0);288else if (afe_priv->afe_on_ref_cnt < 0)289afe_priv->afe_on_ref_cnt = 0;290291spin_unlock_irqrestore(&afe_priv->afe_ctrl_lock, flags);292293return 0;294}295296static int mt8365_afe_hd_engen_enable(struct mtk_base_afe *afe, bool apll1)297{298if (apll1)299regmap_update_bits(afe->regmap, AFE_HD_ENGEN_ENABLE,300AFE_22M_PLL_EN, AFE_22M_PLL_EN);301else302regmap_update_bits(afe->regmap, AFE_HD_ENGEN_ENABLE,303AFE_24M_PLL_EN, AFE_24M_PLL_EN);304305return 0;306}307308static int mt8365_afe_hd_engen_disable(struct mtk_base_afe *afe, bool apll1)309{310if (apll1)311regmap_update_bits(afe->regmap, AFE_HD_ENGEN_ENABLE,312AFE_22M_PLL_EN, ~AFE_22M_PLL_EN);313else314regmap_update_bits(afe->regmap, AFE_HD_ENGEN_ENABLE,315AFE_24M_PLL_EN, ~AFE_24M_PLL_EN);316317return 0;318}319320int mt8365_afe_enable_apll_tuner_cfg(struct mtk_base_afe *afe, unsigned int apll)321{322struct mt8365_afe_private *afe_priv = afe->platform_priv;323324mutex_lock(&afe_priv->afe_clk_mutex);325326afe_priv->apll_tuner_ref_cnt[apll]++;327if (afe_priv->apll_tuner_ref_cnt[apll] != 1) {328mutex_unlock(&afe_priv->afe_clk_mutex);329return 0;330}331332if (apll == MT8365_AFE_APLL1) {333regmap_update_bits(afe->regmap, AFE_APLL_TUNER_CFG,334AFE_APLL_TUNER_CFG_MASK, 0x432);335regmap_update_bits(afe->regmap, AFE_APLL_TUNER_CFG,336AFE_APLL_TUNER_CFG_EN_MASK, 0x1);337} else {338regmap_update_bits(afe->regmap, AFE_APLL_TUNER_CFG1,339AFE_APLL_TUNER_CFG1_MASK, 0x434);340regmap_update_bits(afe->regmap, AFE_APLL_TUNER_CFG1,341AFE_APLL_TUNER_CFG1_EN_MASK, 0x1);342}343344mutex_unlock(&afe_priv->afe_clk_mutex);345return 0;346}347348int mt8365_afe_disable_apll_tuner_cfg(struct mtk_base_afe *afe, unsigned int apll)349{350struct mt8365_afe_private *afe_priv = afe->platform_priv;351352mutex_lock(&afe_priv->afe_clk_mutex);353354afe_priv->apll_tuner_ref_cnt[apll]--;355if (afe_priv->apll_tuner_ref_cnt[apll] == 0) {356if (apll == MT8365_AFE_APLL1)357regmap_update_bits(afe->regmap, AFE_APLL_TUNER_CFG,358AFE_APLL_TUNER_CFG_EN_MASK, 0x0);359else360regmap_update_bits(afe->regmap, AFE_APLL_TUNER_CFG1,361AFE_APLL_TUNER_CFG1_EN_MASK, 0x0);362363} else if (afe_priv->apll_tuner_ref_cnt[apll] < 0) {364afe_priv->apll_tuner_ref_cnt[apll] = 0;365}366367mutex_unlock(&afe_priv->afe_clk_mutex);368return 0;369}370371int mt8365_afe_enable_apll_associated_cfg(struct mtk_base_afe *afe, unsigned int apll)372{373struct mt8365_afe_private *afe_priv = afe->platform_priv;374375if (apll == MT8365_AFE_APLL1) {376if (clk_prepare_enable(afe_priv->clocks[MT8365_CLK_ENGEN1])) {377dev_info(afe->dev, "%s Failed to enable ENGEN1 clk\n",378__func__);379return 0;380}381mt8365_afe_enable_top_cg(afe, MT8365_TOP_CG_22M);382mt8365_afe_hd_engen_enable(afe, true);383mt8365_afe_enable_top_cg(afe, MT8365_TOP_CG_APLL_TUNER);384mt8365_afe_enable_apll_tuner_cfg(afe, MT8365_AFE_APLL1);385} else {386if (clk_prepare_enable(afe_priv->clocks[MT8365_CLK_ENGEN2])) {387dev_info(afe->dev, "%s Failed to enable ENGEN2 clk\n",388__func__);389return 0;390}391mt8365_afe_enable_top_cg(afe, MT8365_TOP_CG_24M);392mt8365_afe_hd_engen_enable(afe, false);393mt8365_afe_enable_top_cg(afe, MT8365_TOP_CG_APLL2_TUNER);394mt8365_afe_enable_apll_tuner_cfg(afe, MT8365_AFE_APLL2);395}396397return 0;398}399400int mt8365_afe_disable_apll_associated_cfg(struct mtk_base_afe *afe, unsigned int apll)401{402struct mt8365_afe_private *afe_priv = afe->platform_priv;403404if (apll == MT8365_AFE_APLL1) {405mt8365_afe_disable_apll_tuner_cfg(afe, MT8365_AFE_APLL1);406mt8365_afe_disable_top_cg(afe, MT8365_TOP_CG_APLL_TUNER);407mt8365_afe_hd_engen_disable(afe, true);408mt8365_afe_disable_top_cg(afe, MT8365_TOP_CG_22M);409clk_disable_unprepare(afe_priv->clocks[MT8365_CLK_ENGEN1]);410} else {411mt8365_afe_disable_apll_tuner_cfg(afe, MT8365_AFE_APLL2);412mt8365_afe_disable_top_cg(afe, MT8365_TOP_CG_APLL2_TUNER);413mt8365_afe_hd_engen_disable(afe, false);414mt8365_afe_disable_top_cg(afe, MT8365_TOP_CG_24M);415clk_disable_unprepare(afe_priv->clocks[MT8365_CLK_ENGEN2]);416}417418return 0;419}420421422