162306a36Sopenharmony_ci// SPDX-License-Identifier: GPL-2.0 262306a36Sopenharmony_ci// 362306a36Sopenharmony_ci// MediaTek ALSA SoC Audio Misc Control 462306a36Sopenharmony_ci// 562306a36Sopenharmony_ci// Copyright (c) 2022 MediaTek Inc. 662306a36Sopenharmony_ci// Author: Jiaxin Yu <jiaxin.yu@mediatek.com> 762306a36Sopenharmony_ci 862306a36Sopenharmony_ci#include <linux/delay.h> 962306a36Sopenharmony_ci#include <linux/dma-mapping.h> 1062306a36Sopenharmony_ci#include <linux/io.h> 1162306a36Sopenharmony_ci#include <linux/regmap.h> 1262306a36Sopenharmony_ci#include <sound/soc.h> 1362306a36Sopenharmony_ci 1462306a36Sopenharmony_ci#include "../common/mtk-afe-fe-dai.h" 1562306a36Sopenharmony_ci#include "../common/mtk-afe-platform-driver.h" 1662306a36Sopenharmony_ci#include "mt8186-afe-common.h" 1762306a36Sopenharmony_ci 1862306a36Sopenharmony_cistatic const char * const mt8186_sgen_mode_str[] = { 1962306a36Sopenharmony_ci "I0I1", "I2", "I3I4", "I5I6", 2062306a36Sopenharmony_ci "I7I8", "I9I22", "I10I11", "I12I13", 2162306a36Sopenharmony_ci "I14I21", "I15I16", "I17I18", "I19I20", 2262306a36Sopenharmony_ci "I23I24", "I25I26", "I27I28", "I33", 2362306a36Sopenharmony_ci "I34I35", "I36I37", "I38I39", "I40I41", 2462306a36Sopenharmony_ci "I42I43", "I44I45", "I46I47", "I48I49", 2562306a36Sopenharmony_ci "I56I57", "I58I59", "I60I61", "I62I63", 2662306a36Sopenharmony_ci "O0O1", "O2", "O3O4", "O5O6", 2762306a36Sopenharmony_ci "O7O8", "O9O10", "O11", "O12", 2862306a36Sopenharmony_ci "O13O14", "O15O16", "O17O18", "O19O20", 2962306a36Sopenharmony_ci "O21O22", "O23O24", "O25", "O28O29", 3062306a36Sopenharmony_ci "O34", "O35", "O32O33", "O36O37", 3162306a36Sopenharmony_ci "O38O39", "O30O31", "O40O41", "O42O43", 3262306a36Sopenharmony_ci "O44O45", "O46O47", "O48O49", "O50O51", 3362306a36Sopenharmony_ci "O58O59", "O60O61", "O62O63", "O64O65", 3462306a36Sopenharmony_ci "O66O67", "O68O69", "O26O27", "OFF", 3562306a36Sopenharmony_ci}; 3662306a36Sopenharmony_ci 3762306a36Sopenharmony_cistatic const int mt8186_sgen_mode_idx[] = { 3862306a36Sopenharmony_ci 0, 2, 4, 6, 3962306a36Sopenharmony_ci 8, 22, 10, 12, 4062306a36Sopenharmony_ci 14, -1, 18, 20, 4162306a36Sopenharmony_ci 24, 26, 28, 33, 4262306a36Sopenharmony_ci 34, 36, 38, 40, 4362306a36Sopenharmony_ci 42, 44, 46, 48, 4462306a36Sopenharmony_ci 56, 58, 60, 62, 4562306a36Sopenharmony_ci 128, 130, 132, 134, 4662306a36Sopenharmony_ci 135, 138, 139, 140, 4762306a36Sopenharmony_ci 142, 144, 166, 148, 4862306a36Sopenharmony_ci 150, 152, 153, 156, 4962306a36Sopenharmony_ci 162, 163, 160, 164, 5062306a36Sopenharmony_ci 166, -1, 168, 170, 5162306a36Sopenharmony_ci 172, 174, 176, 178, 5262306a36Sopenharmony_ci 186, 188, 190, 192, 5362306a36Sopenharmony_ci 194, 196, -1, -1, 5462306a36Sopenharmony_ci}; 5562306a36Sopenharmony_ci 5662306a36Sopenharmony_cistatic const char * const mt8186_sgen_rate_str[] = { 5762306a36Sopenharmony_ci "8K", "11K", "12K", "16K", 5862306a36Sopenharmony_ci "22K", "24K", "32K", "44K", 5962306a36Sopenharmony_ci "48K", "88k", "96k", "176k", 6062306a36Sopenharmony_ci "192k" 6162306a36Sopenharmony_ci}; 6262306a36Sopenharmony_ci 6362306a36Sopenharmony_cistatic const int mt8186_sgen_rate_idx[] = { 6462306a36Sopenharmony_ci 0, 1, 2, 4, 6562306a36Sopenharmony_ci 5, 6, 8, 9, 6662306a36Sopenharmony_ci 10, 11, 12, 13, 6762306a36Sopenharmony_ci 14 6862306a36Sopenharmony_ci}; 6962306a36Sopenharmony_ci 7062306a36Sopenharmony_ci/* this order must match reg bit amp_div_ch1/2 */ 7162306a36Sopenharmony_cistatic const char * const mt8186_sgen_amp_str[] = { 7262306a36Sopenharmony_ci "1/128", "1/64", "1/32", "1/16", "1/8", "1/4", "1/2", "1" }; 7362306a36Sopenharmony_ci 7462306a36Sopenharmony_cistatic int mt8186_sgen_get(struct snd_kcontrol *kcontrol, 7562306a36Sopenharmony_ci struct snd_ctl_elem_value *ucontrol) 7662306a36Sopenharmony_ci{ 7762306a36Sopenharmony_ci struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol); 7862306a36Sopenharmony_ci struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt); 7962306a36Sopenharmony_ci struct mt8186_afe_private *afe_priv = afe->platform_priv; 8062306a36Sopenharmony_ci 8162306a36Sopenharmony_ci ucontrol->value.integer.value[0] = afe_priv->sgen_mode; 8262306a36Sopenharmony_ci 8362306a36Sopenharmony_ci return 0; 8462306a36Sopenharmony_ci} 8562306a36Sopenharmony_ci 8662306a36Sopenharmony_cistatic int mt8186_sgen_set(struct snd_kcontrol *kcontrol, 8762306a36Sopenharmony_ci struct snd_ctl_elem_value *ucontrol) 8862306a36Sopenharmony_ci{ 8962306a36Sopenharmony_ci struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol); 9062306a36Sopenharmony_ci struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt); 9162306a36Sopenharmony_ci struct mt8186_afe_private *afe_priv = afe->platform_priv; 9262306a36Sopenharmony_ci struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; 9362306a36Sopenharmony_ci int mode; 9462306a36Sopenharmony_ci int mode_idx; 9562306a36Sopenharmony_ci 9662306a36Sopenharmony_ci if (ucontrol->value.enumerated.item[0] >= e->items) 9762306a36Sopenharmony_ci return -EINVAL; 9862306a36Sopenharmony_ci 9962306a36Sopenharmony_ci mode = ucontrol->value.integer.value[0]; 10062306a36Sopenharmony_ci mode_idx = mt8186_sgen_mode_idx[mode]; 10162306a36Sopenharmony_ci 10262306a36Sopenharmony_ci dev_dbg(afe->dev, "%s(), mode %d, mode_idx %d\n", 10362306a36Sopenharmony_ci __func__, mode, mode_idx); 10462306a36Sopenharmony_ci 10562306a36Sopenharmony_ci if (mode == afe_priv->sgen_mode) 10662306a36Sopenharmony_ci return 0; 10762306a36Sopenharmony_ci 10862306a36Sopenharmony_ci if (mode_idx >= 0) { 10962306a36Sopenharmony_ci regmap_update_bits(afe->regmap, AFE_SINEGEN_CON2, 11062306a36Sopenharmony_ci INNER_LOOP_BACK_MODE_MASK_SFT, 11162306a36Sopenharmony_ci mode_idx << INNER_LOOP_BACK_MODE_SFT); 11262306a36Sopenharmony_ci regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0, 11362306a36Sopenharmony_ci DAC_EN_MASK_SFT, BIT(DAC_EN_SFT)); 11462306a36Sopenharmony_ci } else { 11562306a36Sopenharmony_ci /* disable sgen */ 11662306a36Sopenharmony_ci regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0, 11762306a36Sopenharmony_ci DAC_EN_MASK_SFT, 0); 11862306a36Sopenharmony_ci regmap_update_bits(afe->regmap, AFE_SINEGEN_CON2, 11962306a36Sopenharmony_ci INNER_LOOP_BACK_MODE_MASK_SFT, 12062306a36Sopenharmony_ci 0x3f << INNER_LOOP_BACK_MODE_SFT); 12162306a36Sopenharmony_ci } 12262306a36Sopenharmony_ci 12362306a36Sopenharmony_ci afe_priv->sgen_mode = mode; 12462306a36Sopenharmony_ci 12562306a36Sopenharmony_ci return 1; 12662306a36Sopenharmony_ci} 12762306a36Sopenharmony_ci 12862306a36Sopenharmony_cistatic int mt8186_sgen_rate_get(struct snd_kcontrol *kcontrol, 12962306a36Sopenharmony_ci struct snd_ctl_elem_value *ucontrol) 13062306a36Sopenharmony_ci{ 13162306a36Sopenharmony_ci struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol); 13262306a36Sopenharmony_ci struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt); 13362306a36Sopenharmony_ci struct mt8186_afe_private *afe_priv = afe->platform_priv; 13462306a36Sopenharmony_ci 13562306a36Sopenharmony_ci ucontrol->value.integer.value[0] = afe_priv->sgen_rate; 13662306a36Sopenharmony_ci 13762306a36Sopenharmony_ci return 0; 13862306a36Sopenharmony_ci} 13962306a36Sopenharmony_ci 14062306a36Sopenharmony_cistatic int mt8186_sgen_rate_set(struct snd_kcontrol *kcontrol, 14162306a36Sopenharmony_ci struct snd_ctl_elem_value *ucontrol) 14262306a36Sopenharmony_ci{ 14362306a36Sopenharmony_ci struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol); 14462306a36Sopenharmony_ci struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt); 14562306a36Sopenharmony_ci struct mt8186_afe_private *afe_priv = afe->platform_priv; 14662306a36Sopenharmony_ci struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; 14762306a36Sopenharmony_ci int rate; 14862306a36Sopenharmony_ci 14962306a36Sopenharmony_ci if (ucontrol->value.enumerated.item[0] >= e->items) 15062306a36Sopenharmony_ci return -EINVAL; 15162306a36Sopenharmony_ci 15262306a36Sopenharmony_ci rate = ucontrol->value.integer.value[0]; 15362306a36Sopenharmony_ci 15462306a36Sopenharmony_ci dev_dbg(afe->dev, "%s(), rate %d\n", __func__, rate); 15562306a36Sopenharmony_ci 15662306a36Sopenharmony_ci if (rate == afe_priv->sgen_rate) 15762306a36Sopenharmony_ci return 0; 15862306a36Sopenharmony_ci 15962306a36Sopenharmony_ci regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0, 16062306a36Sopenharmony_ci SINE_MODE_CH1_MASK_SFT, 16162306a36Sopenharmony_ci mt8186_sgen_rate_idx[rate] << SINE_MODE_CH1_SFT); 16262306a36Sopenharmony_ci 16362306a36Sopenharmony_ci regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0, 16462306a36Sopenharmony_ci SINE_MODE_CH2_MASK_SFT, 16562306a36Sopenharmony_ci mt8186_sgen_rate_idx[rate] << SINE_MODE_CH2_SFT); 16662306a36Sopenharmony_ci 16762306a36Sopenharmony_ci afe_priv->sgen_rate = rate; 16862306a36Sopenharmony_ci 16962306a36Sopenharmony_ci return 1; 17062306a36Sopenharmony_ci} 17162306a36Sopenharmony_ci 17262306a36Sopenharmony_cistatic int mt8186_sgen_amplitude_get(struct snd_kcontrol *kcontrol, 17362306a36Sopenharmony_ci struct snd_ctl_elem_value *ucontrol) 17462306a36Sopenharmony_ci{ 17562306a36Sopenharmony_ci struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol); 17662306a36Sopenharmony_ci struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt); 17762306a36Sopenharmony_ci struct mt8186_afe_private *afe_priv = afe->platform_priv; 17862306a36Sopenharmony_ci 17962306a36Sopenharmony_ci ucontrol->value.integer.value[0] = afe_priv->sgen_amplitude; 18062306a36Sopenharmony_ci return 0; 18162306a36Sopenharmony_ci} 18262306a36Sopenharmony_ci 18362306a36Sopenharmony_cistatic int mt8186_sgen_amplitude_set(struct snd_kcontrol *kcontrol, 18462306a36Sopenharmony_ci struct snd_ctl_elem_value *ucontrol) 18562306a36Sopenharmony_ci{ 18662306a36Sopenharmony_ci struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol); 18762306a36Sopenharmony_ci struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt); 18862306a36Sopenharmony_ci struct mt8186_afe_private *afe_priv = afe->platform_priv; 18962306a36Sopenharmony_ci struct soc_enum *e = (struct soc_enum *)kcontrol->private_value; 19062306a36Sopenharmony_ci int amplitude; 19162306a36Sopenharmony_ci 19262306a36Sopenharmony_ci if (ucontrol->value.enumerated.item[0] >= e->items) 19362306a36Sopenharmony_ci return -EINVAL; 19462306a36Sopenharmony_ci 19562306a36Sopenharmony_ci amplitude = ucontrol->value.integer.value[0]; 19662306a36Sopenharmony_ci if (amplitude > AMP_DIV_CH1_MASK) { 19762306a36Sopenharmony_ci dev_err(afe->dev, "%s(), amplitude %d invalid\n", 19862306a36Sopenharmony_ci __func__, amplitude); 19962306a36Sopenharmony_ci return -EINVAL; 20062306a36Sopenharmony_ci } 20162306a36Sopenharmony_ci 20262306a36Sopenharmony_ci dev_dbg(afe->dev, "%s(), amplitude %d\n", __func__, amplitude); 20362306a36Sopenharmony_ci 20462306a36Sopenharmony_ci if (amplitude == afe_priv->sgen_amplitude) 20562306a36Sopenharmony_ci return 0; 20662306a36Sopenharmony_ci 20762306a36Sopenharmony_ci regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0, 20862306a36Sopenharmony_ci AMP_DIV_CH1_MASK_SFT, 20962306a36Sopenharmony_ci amplitude << AMP_DIV_CH1_SFT); 21062306a36Sopenharmony_ci regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0, 21162306a36Sopenharmony_ci AMP_DIV_CH2_MASK_SFT, 21262306a36Sopenharmony_ci amplitude << AMP_DIV_CH2_SFT); 21362306a36Sopenharmony_ci 21462306a36Sopenharmony_ci afe_priv->sgen_amplitude = amplitude; 21562306a36Sopenharmony_ci 21662306a36Sopenharmony_ci return 1; 21762306a36Sopenharmony_ci} 21862306a36Sopenharmony_ci 21962306a36Sopenharmony_cistatic const struct soc_enum mt8186_afe_sgen_enum[] = { 22062306a36Sopenharmony_ci SOC_ENUM_SINGLE_EXT(ARRAY_SIZE(mt8186_sgen_mode_str), 22162306a36Sopenharmony_ci mt8186_sgen_mode_str), 22262306a36Sopenharmony_ci SOC_ENUM_SINGLE_EXT(ARRAY_SIZE(mt8186_sgen_rate_str), 22362306a36Sopenharmony_ci mt8186_sgen_rate_str), 22462306a36Sopenharmony_ci SOC_ENUM_SINGLE_EXT(ARRAY_SIZE(mt8186_sgen_amp_str), 22562306a36Sopenharmony_ci mt8186_sgen_amp_str), 22662306a36Sopenharmony_ci}; 22762306a36Sopenharmony_ci 22862306a36Sopenharmony_cistatic const struct snd_kcontrol_new mt8186_afe_sgen_controls[] = { 22962306a36Sopenharmony_ci SOC_ENUM_EXT("Audio_SineGen_Switch", mt8186_afe_sgen_enum[0], 23062306a36Sopenharmony_ci mt8186_sgen_get, mt8186_sgen_set), 23162306a36Sopenharmony_ci SOC_ENUM_EXT("Audio_SineGen_SampleRate", mt8186_afe_sgen_enum[1], 23262306a36Sopenharmony_ci mt8186_sgen_rate_get, mt8186_sgen_rate_set), 23362306a36Sopenharmony_ci SOC_ENUM_EXT("Audio_SineGen_Amplitude", mt8186_afe_sgen_enum[2], 23462306a36Sopenharmony_ci mt8186_sgen_amplitude_get, mt8186_sgen_amplitude_set), 23562306a36Sopenharmony_ci SOC_SINGLE("Audio_SineGen_Mute_Ch1", AFE_SINEGEN_CON0, 23662306a36Sopenharmony_ci MUTE_SW_CH1_MASK_SFT, MUTE_SW_CH1_MASK, 0), 23762306a36Sopenharmony_ci SOC_SINGLE("Audio_SineGen_Mute_Ch2", AFE_SINEGEN_CON0, 23862306a36Sopenharmony_ci MUTE_SW_CH2_MASK_SFT, MUTE_SW_CH2_MASK, 0), 23962306a36Sopenharmony_ci SOC_SINGLE("Audio_SineGen_Freq_Div_Ch1", AFE_SINEGEN_CON0, 24062306a36Sopenharmony_ci FREQ_DIV_CH1_SFT, FREQ_DIV_CH1_MASK, 0), 24162306a36Sopenharmony_ci SOC_SINGLE("Audio_SineGen_Freq_Div_Ch2", AFE_SINEGEN_CON0, 24262306a36Sopenharmony_ci FREQ_DIV_CH2_SFT, FREQ_DIV_CH2_MASK, 0), 24362306a36Sopenharmony_ci}; 24462306a36Sopenharmony_ci 24562306a36Sopenharmony_ciint mt8186_add_misc_control(struct snd_soc_component *component) 24662306a36Sopenharmony_ci{ 24762306a36Sopenharmony_ci snd_soc_add_component_controls(component, 24862306a36Sopenharmony_ci mt8186_afe_sgen_controls, 24962306a36Sopenharmony_ci ARRAY_SIZE(mt8186_afe_sgen_controls)); 25062306a36Sopenharmony_ci 25162306a36Sopenharmony_ci return 0; 25262306a36Sopenharmony_ci} 253