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