Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
torvalds
GitHub Repository: torvalds/linux
Path: blob/master/sound/soc/mediatek/mt8186/mt8186-misc-control.c
26488 views
1
// SPDX-License-Identifier: GPL-2.0
2
//
3
// MediaTek ALSA SoC Audio Misc Control
4
//
5
// Copyright (c) 2022 MediaTek Inc.
6
// Author: Jiaxin Yu <[email protected]>
7
8
#include <linux/delay.h>
9
#include <linux/dma-mapping.h>
10
#include <linux/io.h>
11
#include <linux/regmap.h>
12
#include <sound/soc.h>
13
14
#include "../common/mtk-afe-fe-dai.h"
15
#include "../common/mtk-afe-platform-driver.h"
16
#include "mt8186-afe-common.h"
17
18
static const char * const mt8186_sgen_mode_str[] = {
19
"I0I1", "I2", "I3I4", "I5I6",
20
"I7I8", "I9I22", "I10I11", "I12I13",
21
"I14I21", "I15I16", "I17I18", "I19I20",
22
"I23I24", "I25I26", "I27I28", "I33",
23
"I34I35", "I36I37", "I38I39", "I40I41",
24
"I42I43", "I44I45", "I46I47", "I48I49",
25
"I56I57", "I58I59", "I60I61", "I62I63",
26
"O0O1", "O2", "O3O4", "O5O6",
27
"O7O8", "O9O10", "O11", "O12",
28
"O13O14", "O15O16", "O17O18", "O19O20",
29
"O21O22", "O23O24", "O25", "O28O29",
30
"O34", "O35", "O32O33", "O36O37",
31
"O38O39", "O30O31", "O40O41", "O42O43",
32
"O44O45", "O46O47", "O48O49", "O50O51",
33
"O58O59", "O60O61", "O62O63", "O64O65",
34
"O66O67", "O68O69", "O26O27", "OFF",
35
};
36
37
static const int mt8186_sgen_mode_idx[] = {
38
0, 2, 4, 6,
39
8, 22, 10, 12,
40
14, -1, 18, 20,
41
24, 26, 28, 33,
42
34, 36, 38, 40,
43
42, 44, 46, 48,
44
56, 58, 60, 62,
45
128, 130, 132, 134,
46
135, 138, 139, 140,
47
142, 144, 166, 148,
48
150, 152, 153, 156,
49
162, 163, 160, 164,
50
166, -1, 168, 170,
51
172, 174, 176, 178,
52
186, 188, 190, 192,
53
194, 196, -1, -1,
54
};
55
56
static const char * const mt8186_sgen_rate_str[] = {
57
"8K", "11K", "12K", "16K",
58
"22K", "24K", "32K", "44K",
59
"48K", "88k", "96k", "176k",
60
"192k"
61
};
62
63
static const int mt8186_sgen_rate_idx[] = {
64
0, 1, 2, 4,
65
5, 6, 8, 9,
66
10, 11, 12, 13,
67
14
68
};
69
70
/* this order must match reg bit amp_div_ch1/2 */
71
static const char * const mt8186_sgen_amp_str[] = {
72
"1/128", "1/64", "1/32", "1/16", "1/8", "1/4", "1/2", "1" };
73
74
static int mt8186_sgen_get(struct snd_kcontrol *kcontrol,
75
struct snd_ctl_elem_value *ucontrol)
76
{
77
struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol);
78
struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt);
79
struct mt8186_afe_private *afe_priv = afe->platform_priv;
80
81
ucontrol->value.integer.value[0] = afe_priv->sgen_mode;
82
83
return 0;
84
}
85
86
static int mt8186_sgen_set(struct snd_kcontrol *kcontrol,
87
struct snd_ctl_elem_value *ucontrol)
88
{
89
struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol);
90
struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt);
91
struct mt8186_afe_private *afe_priv = afe->platform_priv;
92
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
93
int mode;
94
int mode_idx;
95
96
if (ucontrol->value.enumerated.item[0] >= e->items)
97
return -EINVAL;
98
99
mode = ucontrol->value.integer.value[0];
100
mode_idx = mt8186_sgen_mode_idx[mode];
101
102
dev_dbg(afe->dev, "%s(), mode %d, mode_idx %d\n",
103
__func__, mode, mode_idx);
104
105
if (mode == afe_priv->sgen_mode)
106
return 0;
107
108
if (mode_idx >= 0) {
109
regmap_update_bits(afe->regmap, AFE_SINEGEN_CON2,
110
INNER_LOOP_BACK_MODE_MASK_SFT,
111
mode_idx << INNER_LOOP_BACK_MODE_SFT);
112
regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0,
113
DAC_EN_MASK_SFT, BIT(DAC_EN_SFT));
114
} else {
115
/* disable sgen */
116
regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0,
117
DAC_EN_MASK_SFT, 0);
118
regmap_update_bits(afe->regmap, AFE_SINEGEN_CON2,
119
INNER_LOOP_BACK_MODE_MASK_SFT,
120
0x3f << INNER_LOOP_BACK_MODE_SFT);
121
}
122
123
afe_priv->sgen_mode = mode;
124
125
return 1;
126
}
127
128
static int mt8186_sgen_rate_get(struct snd_kcontrol *kcontrol,
129
struct snd_ctl_elem_value *ucontrol)
130
{
131
struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol);
132
struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt);
133
struct mt8186_afe_private *afe_priv = afe->platform_priv;
134
135
ucontrol->value.integer.value[0] = afe_priv->sgen_rate;
136
137
return 0;
138
}
139
140
static int mt8186_sgen_rate_set(struct snd_kcontrol *kcontrol,
141
struct snd_ctl_elem_value *ucontrol)
142
{
143
struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol);
144
struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt);
145
struct mt8186_afe_private *afe_priv = afe->platform_priv;
146
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
147
int rate;
148
149
if (ucontrol->value.enumerated.item[0] >= e->items)
150
return -EINVAL;
151
152
rate = ucontrol->value.integer.value[0];
153
154
dev_dbg(afe->dev, "%s(), rate %d\n", __func__, rate);
155
156
if (rate == afe_priv->sgen_rate)
157
return 0;
158
159
regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0,
160
SINE_MODE_CH1_MASK_SFT,
161
mt8186_sgen_rate_idx[rate] << SINE_MODE_CH1_SFT);
162
163
regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0,
164
SINE_MODE_CH2_MASK_SFT,
165
mt8186_sgen_rate_idx[rate] << SINE_MODE_CH2_SFT);
166
167
afe_priv->sgen_rate = rate;
168
169
return 1;
170
}
171
172
static int mt8186_sgen_amplitude_get(struct snd_kcontrol *kcontrol,
173
struct snd_ctl_elem_value *ucontrol)
174
{
175
struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol);
176
struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt);
177
struct mt8186_afe_private *afe_priv = afe->platform_priv;
178
179
ucontrol->value.integer.value[0] = afe_priv->sgen_amplitude;
180
return 0;
181
}
182
183
static int mt8186_sgen_amplitude_set(struct snd_kcontrol *kcontrol,
184
struct snd_ctl_elem_value *ucontrol)
185
{
186
struct snd_soc_component *cmpnt = snd_soc_kcontrol_component(kcontrol);
187
struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt);
188
struct mt8186_afe_private *afe_priv = afe->platform_priv;
189
struct soc_enum *e = (struct soc_enum *)kcontrol->private_value;
190
int amplitude;
191
192
if (ucontrol->value.enumerated.item[0] >= e->items)
193
return -EINVAL;
194
195
amplitude = ucontrol->value.integer.value[0];
196
if (amplitude > AMP_DIV_CH1_MASK) {
197
dev_err(afe->dev, "%s(), amplitude %d invalid\n",
198
__func__, amplitude);
199
return -EINVAL;
200
}
201
202
dev_dbg(afe->dev, "%s(), amplitude %d\n", __func__, amplitude);
203
204
if (amplitude == afe_priv->sgen_amplitude)
205
return 0;
206
207
regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0,
208
AMP_DIV_CH1_MASK_SFT,
209
amplitude << AMP_DIV_CH1_SFT);
210
regmap_update_bits(afe->regmap, AFE_SINEGEN_CON0,
211
AMP_DIV_CH2_MASK_SFT,
212
amplitude << AMP_DIV_CH2_SFT);
213
214
afe_priv->sgen_amplitude = amplitude;
215
216
return 1;
217
}
218
219
static const struct soc_enum mt8186_afe_sgen_enum[] = {
220
SOC_ENUM_SINGLE_EXT(ARRAY_SIZE(mt8186_sgen_mode_str),
221
mt8186_sgen_mode_str),
222
SOC_ENUM_SINGLE_EXT(ARRAY_SIZE(mt8186_sgen_rate_str),
223
mt8186_sgen_rate_str),
224
SOC_ENUM_SINGLE_EXT(ARRAY_SIZE(mt8186_sgen_amp_str),
225
mt8186_sgen_amp_str),
226
};
227
228
static const struct snd_kcontrol_new mt8186_afe_sgen_controls[] = {
229
SOC_ENUM_EXT("Audio_SineGen_Switch", mt8186_afe_sgen_enum[0],
230
mt8186_sgen_get, mt8186_sgen_set),
231
SOC_ENUM_EXT("Audio_SineGen_SampleRate", mt8186_afe_sgen_enum[1],
232
mt8186_sgen_rate_get, mt8186_sgen_rate_set),
233
SOC_ENUM_EXT("Audio_SineGen_Amplitude", mt8186_afe_sgen_enum[2],
234
mt8186_sgen_amplitude_get, mt8186_sgen_amplitude_set),
235
SOC_SINGLE("Audio_SineGen_Mute_Ch1", AFE_SINEGEN_CON0,
236
MUTE_SW_CH1_MASK_SFT, MUTE_SW_CH1_MASK, 0),
237
SOC_SINGLE("Audio_SineGen_Mute_Ch2", AFE_SINEGEN_CON0,
238
MUTE_SW_CH2_MASK_SFT, MUTE_SW_CH2_MASK, 0),
239
SOC_SINGLE("Audio_SineGen_Freq_Div_Ch1", AFE_SINEGEN_CON0,
240
FREQ_DIV_CH1_SFT, FREQ_DIV_CH1_MASK, 0),
241
SOC_SINGLE("Audio_SineGen_Freq_Div_Ch2", AFE_SINEGEN_CON0,
242
FREQ_DIV_CH2_SFT, FREQ_DIV_CH2_MASK, 0),
243
};
244
245
int mt8186_add_misc_control(struct snd_soc_component *component)
246
{
247
snd_soc_add_component_controls(component,
248
mt8186_afe_sgen_controls,
249
ARRAY_SIZE(mt8186_afe_sgen_controls));
250
251
return 0;
252
}
253
254