~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

TOMOYO Linux Cross Reference
Linux/sound/soc/mediatek/mt8186/mt8186-misc-control.c

Version: ~ [ linux-6.11-rc3 ] ~ [ linux-6.10.4 ] ~ [ linux-6.9.12 ] ~ [ linux-6.8.12 ] ~ [ linux-6.7.12 ] ~ [ linux-6.6.45 ] ~ [ linux-6.5.13 ] ~ [ linux-6.4.16 ] ~ [ linux-6.3.13 ] ~ [ linux-6.2.16 ] ~ [ linux-6.1.104 ] ~ [ linux-6.0.19 ] ~ [ linux-5.19.17 ] ~ [ linux-5.18.19 ] ~ [ linux-5.17.15 ] ~ [ linux-5.16.20 ] ~ [ linux-5.15.164 ] ~ [ linux-5.14.21 ] ~ [ linux-5.13.19 ] ~ [ linux-5.12.19 ] ~ [ linux-5.11.22 ] ~ [ linux-5.10.223 ] ~ [ linux-5.9.16 ] ~ [ linux-5.8.18 ] ~ [ linux-5.7.19 ] ~ [ linux-5.6.19 ] ~ [ linux-5.5.19 ] ~ [ linux-5.4.281 ] ~ [ linux-5.3.18 ] ~ [ linux-5.2.21 ] ~ [ linux-5.1.21 ] ~ [ linux-5.0.21 ] ~ [ linux-4.20.17 ] ~ [ linux-4.19.319 ] ~ [ linux-4.18.20 ] ~ [ linux-4.17.19 ] ~ [ linux-4.16.18 ] ~ [ linux-4.15.18 ] ~ [ linux-4.14.336 ] ~ [ linux-4.13.16 ] ~ [ linux-4.12.14 ] ~ [ linux-4.11.12 ] ~ [ linux-4.10.17 ] ~ [ linux-4.9.337 ] ~ [ linux-4.4.302 ] ~ [ linux-3.10.108 ] ~ [ linux-2.6.32.71 ] ~ [ linux-2.6.0 ] ~ [ linux-2.4.37.11 ] ~ [ unix-v6-master ] ~ [ ccs-tools-1.8.9 ] ~ [ policy-sample ] ~
Architecture: ~ [ i386 ] ~ [ alpha ] ~ [ m68k ] ~ [ mips ] ~ [ ppc ] ~ [ sparc ] ~ [ sparc64 ] ~

  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 <jiaxin.yu@mediatek.com>
  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 

~ [ source navigation ] ~ [ diff markup ] ~ [ identifier search ] ~

kernel.org | git.kernel.org | LWN.net | Project Home | SVN repository | Mail admin

Linux® is a registered trademark of Linus Torvalds in the United States and other countries.
TOMOYO® is a registered trademark of NTT DATA CORPORATION.

sflogo.php