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

TOMOYO Linux Cross Reference
Linux/sound/soc/codecs/rk3308_codec.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-only
  2 /*
  3  * Rockchip RK3308 internal audio codec driver
  4  *
  5  * Copyright (c) 2018, Fuzhou Rockchip Electronics Co., Ltd All rights reserved.
  6  * Copyright (c) 2024, Vivax-Metrotech Ltd
  7  */
  8 
  9 #include <linux/clk.h>
 10 #include <linux/device.h>
 11 #include <linux/delay.h>
 12 #include <linux/init.h>
 13 #include <linux/io.h>
 14 #include <linux/mfd/syscon.h>
 15 #include <linux/module.h>
 16 #include <linux/of.h>
 17 #include <linux/platform_device.h>
 18 #include <linux/regmap.h>
 19 #include <linux/reset.h>
 20 #include <linux/util_macros.h>
 21 #include <sound/core.h>
 22 #include <sound/pcm.h>
 23 #include <sound/pcm_params.h>
 24 #include <sound/soc.h>
 25 #include <sound/tlv.h>
 26 
 27 #include "rk3308_codec.h"
 28 
 29 #define ADC_LR_GROUP_MAX                4
 30 
 31 #define GRF_CHIP_ID                     0x800
 32 
 33 enum {
 34         ACODEC_VERSION_A = 'A',
 35         ACODEC_VERSION_B,
 36         ACODEC_VERSION_C,
 37 };
 38 
 39 struct rk3308_codec_priv {
 40         const struct device *dev;
 41         struct regmap *regmap;
 42         struct regmap *grf;
 43         struct reset_control *reset;
 44         struct clk *hclk;
 45         struct clk *mclk_rx;
 46         struct clk *mclk_tx;
 47         struct snd_soc_component *component;
 48         unsigned char codec_ver;
 49 };
 50 
 51 static struct clk_bulk_data rk3308_codec_clocks[] = {
 52         { .id = "hclk" },
 53         { .id = "mclk_rx" },
 54         { .id = "mclk_tx" },
 55 };
 56 
 57 static const DECLARE_TLV_DB_SCALE(rk3308_codec_adc_alc_gain_tlv,   -1800, 150, 0);
 58 static const DECLARE_TLV_DB_SCALE(rk3308_codec_dac_hpout_gain_tlv, -3900, 150, 0);
 59 static const DECLARE_TLV_DB_SCALE(rk3308_codec_dac_hpmix_gain_tlv,  -600, 600, 0);
 60 
 61 static const DECLARE_TLV_DB_RANGE(rk3308_codec_dac_lineout_gain_tlv,
 62         0, 0, TLV_DB_SCALE_ITEM(-600, 0, 0),
 63         1, 1, TLV_DB_SCALE_ITEM(-300, 0, 0),
 64         2, 2, TLV_DB_SCALE_ITEM(-150, 0, 0),
 65         3, 3, TLV_DB_SCALE_ITEM(0,    0, 0),
 66 );
 67 
 68 static const char * const rk3308_codec_hpf_cutoff_text[] = {
 69         "20 Hz", "245 Hz", "612 Hz"
 70 };
 71 
 72 static SOC_ENUM_SINGLE_DECL(rk3308_codec_hpf_cutoff_enum12, RK3308_ADC_DIG_CON04(0), 0,
 73                             rk3308_codec_hpf_cutoff_text);
 74 static SOC_ENUM_SINGLE_DECL(rk3308_codec_hpf_cutoff_enum34, RK3308_ADC_DIG_CON04(1), 0,
 75                             rk3308_codec_hpf_cutoff_text);
 76 static SOC_ENUM_SINGLE_DECL(rk3308_codec_hpf_cutoff_enum56, RK3308_ADC_DIG_CON04(2), 0,
 77                             rk3308_codec_hpf_cutoff_text);
 78 static SOC_ENUM_SINGLE_DECL(rk3308_codec_hpf_cutoff_enum78, RK3308_ADC_DIG_CON04(3), 0,
 79                             rk3308_codec_hpf_cutoff_text);
 80 
 81 static const struct snd_kcontrol_new rk3308_codec_controls[] = {
 82         /* Despite the register names, these set the gain when AGC is OFF */
 83         SOC_SINGLE_RANGE_TLV("MIC1 Capture Volume",
 84                              RK3308_ADC_ANA_CON03(0),
 85                              RK3308_ADC_CH1_ALC_GAIN_SFT,
 86                              RK3308_ADC_CH1_ALC_GAIN_MIN,
 87                              RK3308_ADC_CH1_ALC_GAIN_MAX,
 88                              0, rk3308_codec_adc_alc_gain_tlv),
 89         SOC_SINGLE_RANGE_TLV("MIC2 Capture Volume",
 90                              RK3308_ADC_ANA_CON04(0),
 91                              RK3308_ADC_CH2_ALC_GAIN_SFT,
 92                              RK3308_ADC_CH2_ALC_GAIN_MIN,
 93                              RK3308_ADC_CH2_ALC_GAIN_MAX,
 94                              0, rk3308_codec_adc_alc_gain_tlv),
 95         SOC_SINGLE_RANGE_TLV("MIC3 Capture Volume",
 96                              RK3308_ADC_ANA_CON03(1),
 97                              RK3308_ADC_CH1_ALC_GAIN_SFT,
 98                              RK3308_ADC_CH1_ALC_GAIN_MIN,
 99                              RK3308_ADC_CH1_ALC_GAIN_MAX,
100                              0, rk3308_codec_adc_alc_gain_tlv),
101         SOC_SINGLE_RANGE_TLV("MIC4 Capture Volume",
102                              RK3308_ADC_ANA_CON04(1),
103                              RK3308_ADC_CH2_ALC_GAIN_SFT,
104                              RK3308_ADC_CH2_ALC_GAIN_MIN,
105                              RK3308_ADC_CH2_ALC_GAIN_MAX,
106                              0, rk3308_codec_adc_alc_gain_tlv),
107         SOC_SINGLE_RANGE_TLV("MIC5 Capture Volume",
108                              RK3308_ADC_ANA_CON03(2),
109                              RK3308_ADC_CH1_ALC_GAIN_SFT,
110                              RK3308_ADC_CH1_ALC_GAIN_MIN,
111                              RK3308_ADC_CH1_ALC_GAIN_MAX,
112                              0, rk3308_codec_adc_alc_gain_tlv),
113         SOC_SINGLE_RANGE_TLV("MIC6 Capture Volume",
114                              RK3308_ADC_ANA_CON04(2),
115                              RK3308_ADC_CH2_ALC_GAIN_SFT,
116                              RK3308_ADC_CH2_ALC_GAIN_MIN,
117                              RK3308_ADC_CH2_ALC_GAIN_MAX,
118                              0, rk3308_codec_adc_alc_gain_tlv),
119         SOC_SINGLE_RANGE_TLV("MIC7 Capture Volume",
120                              RK3308_ADC_ANA_CON03(3),
121                              RK3308_ADC_CH1_ALC_GAIN_SFT,
122                              RK3308_ADC_CH1_ALC_GAIN_MIN,
123                              RK3308_ADC_CH1_ALC_GAIN_MAX,
124                              0, rk3308_codec_adc_alc_gain_tlv),
125         SOC_SINGLE_RANGE_TLV("MIC8 Capture Volume",
126                              RK3308_ADC_ANA_CON04(3),
127                              RK3308_ADC_CH2_ALC_GAIN_SFT,
128                              RK3308_ADC_CH2_ALC_GAIN_MIN,
129                              RK3308_ADC_CH2_ALC_GAIN_MAX,
130                              0, rk3308_codec_adc_alc_gain_tlv),
131 
132         SOC_SINGLE("MIC1 Capture Switch", RK3308_ADC_ANA_CON00(0), 3, 1, 0),
133         SOC_SINGLE("MIC2 Capture Switch", RK3308_ADC_ANA_CON00(0), 7, 1, 0),
134         SOC_SINGLE("MIC3 Capture Switch", RK3308_ADC_ANA_CON00(1), 3, 1, 0),
135         SOC_SINGLE("MIC4 Capture Switch", RK3308_ADC_ANA_CON00(1), 7, 1, 0),
136         SOC_SINGLE("MIC5 Capture Switch", RK3308_ADC_ANA_CON00(2), 3, 1, 0),
137         SOC_SINGLE("MIC6 Capture Switch", RK3308_ADC_ANA_CON00(2), 7, 1, 0),
138         SOC_SINGLE("MIC7 Capture Switch", RK3308_ADC_ANA_CON00(3), 3, 1, 0),
139         SOC_SINGLE("MIC8 Capture Switch", RK3308_ADC_ANA_CON00(3), 7, 1, 0),
140 
141         SOC_SINGLE("MIC12 HPF Capture Switch", RK3308_ADC_DIG_CON04(0), 2, 1, 1),
142         SOC_SINGLE("MIC34 HPF Capture Switch", RK3308_ADC_DIG_CON04(1), 2, 1, 1),
143         SOC_SINGLE("MIC56 HPF Capture Switch", RK3308_ADC_DIG_CON04(2), 2, 1, 1),
144         SOC_SINGLE("MIC78 HPF Capture Switch", RK3308_ADC_DIG_CON04(3), 2, 1, 1),
145 
146         SOC_ENUM("MIC12 HPF Cutoff", rk3308_codec_hpf_cutoff_enum12),
147         SOC_ENUM("MIC34 HPF Cutoff", rk3308_codec_hpf_cutoff_enum34),
148         SOC_ENUM("MIC56 HPF Cutoff", rk3308_codec_hpf_cutoff_enum56),
149         SOC_ENUM("MIC78 HPF Cutoff", rk3308_codec_hpf_cutoff_enum78),
150 
151         SOC_DOUBLE_TLV("Line Out Playback Volume",
152                        RK3308_DAC_ANA_CON04,
153                        RK3308_DAC_L_LINEOUT_GAIN_SFT,
154                        RK3308_DAC_R_LINEOUT_GAIN_SFT,
155                        RK3308_DAC_x_LINEOUT_GAIN_MAX,
156                        0, rk3308_codec_dac_lineout_gain_tlv),
157         SOC_DOUBLE("Line Out Playback Switch",
158                    RK3308_DAC_ANA_CON04,
159                    RK3308_DAC_L_LINEOUT_MUTE_SFT,
160                    RK3308_DAC_R_LINEOUT_MUTE_SFT, 1, 0),
161         SOC_DOUBLE_R_TLV("Headphone Playback Volume",
162                          RK3308_DAC_ANA_CON05,
163                          RK3308_DAC_ANA_CON06,
164                          RK3308_DAC_x_HPOUT_GAIN_SFT,
165                          RK3308_DAC_x_HPOUT_GAIN_MAX,
166                          0, rk3308_codec_dac_hpout_gain_tlv),
167         SOC_DOUBLE("Headphone Playback Switch",
168                    RK3308_DAC_ANA_CON03,
169                    RK3308_DAC_L_HPOUT_MUTE_SFT,
170                    RK3308_DAC_R_HPOUT_MUTE_SFT, 1, 0),
171         SOC_DOUBLE_RANGE_TLV("DAC HPMIX Playback Volume",
172                              RK3308_DAC_ANA_CON12,
173                              RK3308_DAC_L_HPMIX_GAIN_SFT,
174                              RK3308_DAC_R_HPMIX_GAIN_SFT,
175                              1, 2, 0, rk3308_codec_dac_hpmix_gain_tlv),
176 };
177 
178 static int rk3308_codec_pop_sound_set(struct snd_soc_dapm_widget *w,
179                                       struct snd_kcontrol *kcontrol,
180                                       int event)
181 {
182         struct snd_soc_component *component = snd_soc_dapm_to_component(w->dapm);
183         struct rk3308_codec_priv *rk3308 = snd_soc_component_get_drvdata(component);
184         unsigned int val = (event == SND_SOC_DAPM_POST_PMU) ?
185                 RK3308_DAC_HPOUT_POP_SOUND_x_WORK :
186                 RK3308_DAC_HPOUT_POP_SOUND_x_INIT;
187         unsigned int mask = RK3308_DAC_HPOUT_POP_SOUND_x_MSK;
188 
189         regmap_update_bits(rk3308->regmap, RK3308_DAC_ANA_CON01,
190                            mask << w->shift, val << w->shift);
191 
192         return 0;
193 }
194 
195 static const struct snd_soc_dapm_widget rk3308_codec_dapm_widgets[] = {
196         SND_SOC_DAPM_INPUT("MIC1"),
197         SND_SOC_DAPM_INPUT("MIC2"),
198         SND_SOC_DAPM_INPUT("MIC3"),
199         SND_SOC_DAPM_INPUT("MIC4"),
200         SND_SOC_DAPM_INPUT("MIC5"),
201         SND_SOC_DAPM_INPUT("MIC6"),
202         SND_SOC_DAPM_INPUT("MIC7"),
203         SND_SOC_DAPM_INPUT("MIC8"),
204 
205         SND_SOC_DAPM_SUPPLY("ADC_CURRENT_EN12", RK3308_ADC_ANA_CON06(0), 0, 0, NULL, 0),
206         SND_SOC_DAPM_SUPPLY("ADC_CURRENT_EN34", RK3308_ADC_ANA_CON06(1), 0, 0, NULL, 0),
207         SND_SOC_DAPM_SUPPLY("ADC_CURRENT_EN56", RK3308_ADC_ANA_CON06(2), 0, 0, NULL, 0),
208         SND_SOC_DAPM_SUPPLY("ADC_CURRENT_EN78", RK3308_ADC_ANA_CON06(3), 0, 0, NULL, 0),
209 
210         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC1_EN", RK3308_ADC_ANA_CON00(0), 1, 1, 1, 0),
211         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC2_EN", RK3308_ADC_ANA_CON00(0), 5, 1, 1, 0),
212         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC3_EN", RK3308_ADC_ANA_CON00(1), 1, 1, 1, 0),
213         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC4_EN", RK3308_ADC_ANA_CON00(1), 5, 1, 1, 0),
214         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC5_EN", RK3308_ADC_ANA_CON00(2), 1, 1, 1, 0),
215         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC6_EN", RK3308_ADC_ANA_CON00(2), 5, 1, 1, 0),
216         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC7_EN", RK3308_ADC_ANA_CON00(3), 1, 1, 1, 0),
217         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC8_EN", RK3308_ADC_ANA_CON00(3), 5, 1, 1, 0),
218 
219         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC1_WORK", RK3308_ADC_ANA_CON00(0), 2, 1, 1, 0),
220         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC2_WORK", RK3308_ADC_ANA_CON00(0), 6, 1, 1, 0),
221         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC3_WORK", RK3308_ADC_ANA_CON00(1), 2, 1, 1, 0),
222         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC4_WORK", RK3308_ADC_ANA_CON00(1), 6, 1, 1, 0),
223         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC5_WORK", RK3308_ADC_ANA_CON00(2), 2, 1, 1, 0),
224         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC6_WORK", RK3308_ADC_ANA_CON00(2), 6, 1, 1, 0),
225         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC7_WORK", RK3308_ADC_ANA_CON00(3), 2, 1, 1, 0),
226         SND_SOC_DAPM_REG(snd_soc_dapm_mic, "MIC8_WORK", RK3308_ADC_ANA_CON00(3), 6, 1, 1, 0),
227 
228         /*
229          * In theory MIC1 and MIC2 can switch to LINE IN, but this is not
230          * supported so all we can do is enabling the MIC input.
231          */
232         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "CH1_IN_SEL", RK3308_ADC_ANA_CON07(0), 4, 1, 1, 0),
233         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "CH2_IN_SEL", RK3308_ADC_ANA_CON07(0), 6, 1, 1, 0),
234 
235         SND_SOC_DAPM_SUPPLY("ADC1_BUF_REF_EN", RK3308_ADC_ANA_CON00(0), 0, 0, NULL, 0),
236         SND_SOC_DAPM_SUPPLY("ADC2_BUF_REF_EN", RK3308_ADC_ANA_CON00(0), 4, 0, NULL, 0),
237         SND_SOC_DAPM_SUPPLY("ADC3_BUF_REF_EN", RK3308_ADC_ANA_CON00(1), 0, 0, NULL, 0),
238         SND_SOC_DAPM_SUPPLY("ADC4_BUF_REF_EN", RK3308_ADC_ANA_CON00(1), 4, 0, NULL, 0),
239         SND_SOC_DAPM_SUPPLY("ADC5_BUF_REF_EN", RK3308_ADC_ANA_CON00(2), 0, 0, NULL, 0),
240         SND_SOC_DAPM_SUPPLY("ADC6_BUF_REF_EN", RK3308_ADC_ANA_CON00(2), 4, 0, NULL, 0),
241         SND_SOC_DAPM_SUPPLY("ADC7_BUF_REF_EN", RK3308_ADC_ANA_CON00(3), 0, 0, NULL, 0),
242         SND_SOC_DAPM_SUPPLY("ADC8_BUF_REF_EN", RK3308_ADC_ANA_CON00(3), 4, 0, NULL, 0),
243 
244         SND_SOC_DAPM_SUPPLY("ADC_MCLK_GATE", RK3308_GLB_CON, 5, 1, NULL, 0),
245 
246         SND_SOC_DAPM_SUPPLY("ADC1_CLK_EN", RK3308_ADC_ANA_CON05(0), 0, 0, NULL, 0),
247         SND_SOC_DAPM_SUPPLY("ADC2_CLK_EN", RK3308_ADC_ANA_CON05(0), 4, 0, NULL, 0),
248         SND_SOC_DAPM_SUPPLY("ADC3_CLK_EN", RK3308_ADC_ANA_CON05(1), 0, 0, NULL, 0),
249         SND_SOC_DAPM_SUPPLY("ADC4_CLK_EN", RK3308_ADC_ANA_CON05(1), 4, 0, NULL, 0),
250         SND_SOC_DAPM_SUPPLY("ADC5_CLK_EN", RK3308_ADC_ANA_CON05(2), 0, 0, NULL, 0),
251         SND_SOC_DAPM_SUPPLY("ADC6_CLK_EN", RK3308_ADC_ANA_CON05(2), 4, 0, NULL, 0),
252         SND_SOC_DAPM_SUPPLY("ADC7_CLK_EN", RK3308_ADC_ANA_CON05(3), 0, 0, NULL, 0),
253         SND_SOC_DAPM_SUPPLY("ADC8_CLK_EN", RK3308_ADC_ANA_CON05(3), 4, 0, NULL, 0),
254 
255         /* The "ALC" name from the TRM is misleading, these are needed even without ALC/AGC */
256         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC1_EN", RK3308_ADC_ANA_CON02(0), 0, 1, 1, 0),
257         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC2_EN", RK3308_ADC_ANA_CON02(0), 4, 1, 1, 0),
258         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC3_EN", RK3308_ADC_ANA_CON02(1), 0, 1, 1, 0),
259         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC4_EN", RK3308_ADC_ANA_CON02(1), 4, 1, 1, 0),
260         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC5_EN", RK3308_ADC_ANA_CON02(2), 0, 1, 1, 0),
261         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC6_EN", RK3308_ADC_ANA_CON02(2), 4, 1, 1, 0),
262         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC7_EN", RK3308_ADC_ANA_CON02(3), 0, 1, 1, 0),
263         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC8_EN", RK3308_ADC_ANA_CON02(3), 4, 1, 1, 0),
264 
265         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ADC1_EN", RK3308_ADC_ANA_CON05(0), 1, 1, 1, 0),
266         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ADC2_EN", RK3308_ADC_ANA_CON05(0), 5, 1, 1, 0),
267         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ADC3_EN", RK3308_ADC_ANA_CON05(1), 1, 1, 1, 0),
268         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ADC4_EN", RK3308_ADC_ANA_CON05(1), 5, 1, 1, 0),
269         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ADC5_EN", RK3308_ADC_ANA_CON05(2), 1, 1, 1, 0),
270         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ADC6_EN", RK3308_ADC_ANA_CON05(2), 5, 1, 1, 0),
271         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ADC7_EN", RK3308_ADC_ANA_CON05(3), 1, 1, 1, 0),
272         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ADC8_EN", RK3308_ADC_ANA_CON05(3), 5, 1, 1, 0),
273 
274         SND_SOC_DAPM_ADC("ADC1_WORK", "Capture", RK3308_ADC_ANA_CON05(0), 2, 0),
275         SND_SOC_DAPM_ADC("ADC2_WORK", "Capture", RK3308_ADC_ANA_CON05(0), 6, 0),
276         SND_SOC_DAPM_ADC("ADC3_WORK", "Capture", RK3308_ADC_ANA_CON05(1), 2, 0),
277         SND_SOC_DAPM_ADC("ADC4_WORK", "Capture", RK3308_ADC_ANA_CON05(1), 6, 0),
278         SND_SOC_DAPM_ADC("ADC5_WORK", "Capture", RK3308_ADC_ANA_CON05(2), 2, 0),
279         SND_SOC_DAPM_ADC("ADC6_WORK", "Capture", RK3308_ADC_ANA_CON05(2), 6, 0),
280         SND_SOC_DAPM_ADC("ADC7_WORK", "Capture", RK3308_ADC_ANA_CON05(3), 2, 0),
281         SND_SOC_DAPM_ADC("ADC8_WORK", "Capture", RK3308_ADC_ANA_CON05(3), 6, 0),
282 
283         /* The "ALC" name from the TRM is misleading, these are needed even without ALC/AGC */
284         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC1_WORK", RK3308_ADC_ANA_CON02(0), 1, 1, 1, 0),
285         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC2_WORK", RK3308_ADC_ANA_CON02(0), 5, 1, 1, 0),
286         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC3_WORK", RK3308_ADC_ANA_CON02(1), 1, 1, 1, 0),
287         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC4_WORK", RK3308_ADC_ANA_CON02(1), 5, 1, 1, 0),
288         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC5_WORK", RK3308_ADC_ANA_CON02(2), 1, 1, 1, 0),
289         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC6_WORK", RK3308_ADC_ANA_CON02(2), 5, 1, 1, 0),
290         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC7_WORK", RK3308_ADC_ANA_CON02(3), 1, 1, 1, 0),
291         SND_SOC_DAPM_REG(snd_soc_dapm_adc, "ALC8_WORK", RK3308_ADC_ANA_CON02(3), 5, 1, 1, 0),
292 
293         SND_SOC_DAPM_SUPPLY("MICBIAS Current", RK3308_ADC_ANA_CON08(0), 4, 0, NULL, 0),
294         SND_SOC_DAPM_SUPPLY("MICBIAS1", RK3308_ADC_ANA_CON07(1), 3, 0, NULL, 0),
295         SND_SOC_DAPM_SUPPLY("MICBIAS2", RK3308_ADC_ANA_CON07(2), 3, 0, NULL, 0),
296 
297         SND_SOC_DAPM_OUT_DRV("DAC_L_HPMIX_EN",   RK3308_DAC_ANA_CON13, 0, 0, NULL, 0),
298         SND_SOC_DAPM_OUT_DRV("DAC_R_HPMIX_EN",   RK3308_DAC_ANA_CON13, 4, 0, NULL, 0),
299         SND_SOC_DAPM_OUT_DRV("DAC_L_HPMIX_WORK", RK3308_DAC_ANA_CON13, 1, 0, NULL, 0),
300         SND_SOC_DAPM_OUT_DRV("DAC_R_HPMIX_WORK", RK3308_DAC_ANA_CON13, 5, 0, NULL, 0),
301         /* HPMIX is not actually acting as a mixer as the only supported input is I2S */
302         SND_SOC_DAPM_OUT_DRV("DAC_L_HPMIX_SEL",  RK3308_DAC_ANA_CON12, 2, 0, NULL, 0),
303         SND_SOC_DAPM_OUT_DRV("DAC_R_HPMIX_SEL",  RK3308_DAC_ANA_CON12, 6, 0, NULL, 0),
304         SND_SOC_DAPM_MIXER("DAC HPMIX Left",     RK3308_DAC_ANA_CON13, 2, 0, NULL, 0),
305         SND_SOC_DAPM_MIXER("DAC HPMIX Right",    RK3308_DAC_ANA_CON13, 6, 0, NULL, 0),
306 
307         SND_SOC_DAPM_SUPPLY("DAC_MCLK_GATE", RK3308_GLB_CON, 4, 1, NULL, 0),
308 
309         SND_SOC_DAPM_SUPPLY("DAC_CURRENT_EN", RK3308_DAC_ANA_CON00, 0, 0, NULL, 0),
310         SND_SOC_DAPM_SUPPLY("DAC_L_REF_EN",   RK3308_DAC_ANA_CON02, 0, 0, NULL, 0),
311         SND_SOC_DAPM_SUPPLY("DAC_R_REF_EN",   RK3308_DAC_ANA_CON02, 4, 0, NULL, 0),
312         SND_SOC_DAPM_SUPPLY("DAC_L_CLK_EN",   RK3308_DAC_ANA_CON02, 1, 0, NULL, 0),
313         SND_SOC_DAPM_SUPPLY("DAC_R_CLK_EN",   RK3308_DAC_ANA_CON02, 5, 0, NULL, 0),
314         SND_SOC_DAPM_DAC("DAC_L_DAC_WORK", NULL, RK3308_DAC_ANA_CON02, 3, 0),
315         SND_SOC_DAPM_DAC("DAC_R_DAC_WORK", NULL, RK3308_DAC_ANA_CON02, 7, 0),
316 
317         SND_SOC_DAPM_SUPPLY("DAC_BUF_REF_L", RK3308_DAC_ANA_CON01, 2, 0, NULL, 0),
318         SND_SOC_DAPM_SUPPLY("DAC_BUF_REF_R", RK3308_DAC_ANA_CON01, 6, 0, NULL, 0),
319         SND_SOC_DAPM_OUT_DRV_E("HPOUT_POP_SOUND_L", SND_SOC_NOPM, 0, 0, NULL, 0,
320                                rk3308_codec_pop_sound_set,
321                                SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
322         SND_SOC_DAPM_OUT_DRV_E("HPOUT_POP_SOUND_R", SND_SOC_NOPM, 4, 0, NULL, 0,
323                                rk3308_codec_pop_sound_set,
324                                SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
325         SND_SOC_DAPM_OUT_DRV("L_HPOUT_EN", RK3308_DAC_ANA_CON03, 1, 0, NULL, 0),
326         SND_SOC_DAPM_OUT_DRV("R_HPOUT_EN", RK3308_DAC_ANA_CON03, 5, 0, NULL, 0),
327         SND_SOC_DAPM_OUT_DRV("L_HPOUT_WORK", RK3308_DAC_ANA_CON03, 2, 0, NULL, 0),
328         SND_SOC_DAPM_OUT_DRV("R_HPOUT_WORK", RK3308_DAC_ANA_CON03, 6, 0, NULL, 0),
329         SND_SOC_DAPM_OUTPUT("HPOUT_L"),
330         SND_SOC_DAPM_OUTPUT("HPOUT_R"),
331 
332         SND_SOC_DAPM_OUT_DRV("L_LINEOUT_EN", RK3308_DAC_ANA_CON04, 0, 0, NULL, 0),
333         SND_SOC_DAPM_OUT_DRV("R_LINEOUT_EN", RK3308_DAC_ANA_CON04, 4, 0, NULL, 0),
334         SND_SOC_DAPM_OUTPUT("LINEOUT_L"),
335         SND_SOC_DAPM_OUTPUT("LINEOUT_R"),
336 };
337 
338 static const struct snd_soc_dapm_route rk3308_codec_dapm_routes[] = {
339         { "MICBIAS1", NULL, "MICBIAS Current" },
340         { "MICBIAS2", NULL, "MICBIAS Current" },
341 
342         { "MIC1_EN", NULL, "MIC1" },
343         { "MIC2_EN", NULL, "MIC2" },
344         { "MIC3_EN", NULL, "MIC3" },
345         { "MIC4_EN", NULL, "MIC4" },
346         { "MIC5_EN", NULL, "MIC5" },
347         { "MIC6_EN", NULL, "MIC6" },
348         { "MIC7_EN", NULL, "MIC7" },
349         { "MIC8_EN", NULL, "MIC8" },
350 
351         { "MIC1_WORK", NULL, "MIC1_EN" },
352         { "MIC2_WORK", NULL, "MIC2_EN" },
353         { "MIC3_WORK", NULL, "MIC3_EN" },
354         { "MIC4_WORK", NULL, "MIC4_EN" },
355         { "MIC5_WORK", NULL, "MIC5_EN" },
356         { "MIC6_WORK", NULL, "MIC6_EN" },
357         { "MIC7_WORK", NULL, "MIC7_EN" },
358         { "MIC8_WORK", NULL, "MIC8_EN" },
359 
360         { "CH1_IN_SEL", NULL, "MIC1_WORK" },
361         { "CH2_IN_SEL", NULL, "MIC2_WORK" },
362 
363         { "ALC1_EN", NULL, "CH1_IN_SEL" },
364         { "ALC2_EN", NULL, "CH2_IN_SEL" },
365         { "ALC3_EN", NULL, "MIC3_WORK" },
366         { "ALC4_EN", NULL, "MIC4_WORK" },
367         { "ALC5_EN", NULL, "MIC5_WORK" },
368         { "ALC6_EN", NULL, "MIC6_WORK" },
369         { "ALC7_EN", NULL, "MIC7_WORK" },
370         { "ALC8_EN", NULL, "MIC8_WORK" },
371 
372         { "ADC1_EN", NULL, "ALC1_EN" },
373         { "ADC2_EN", NULL, "ALC2_EN" },
374         { "ADC3_EN", NULL, "ALC3_EN" },
375         { "ADC4_EN", NULL, "ALC4_EN" },
376         { "ADC5_EN", NULL, "ALC5_EN" },
377         { "ADC6_EN", NULL, "ALC6_EN" },
378         { "ADC7_EN", NULL, "ALC7_EN" },
379         { "ADC8_EN", NULL, "ALC8_EN" },
380 
381         { "ADC1_WORK", NULL, "ADC1_EN" },
382         { "ADC2_WORK", NULL, "ADC2_EN" },
383         { "ADC3_WORK", NULL, "ADC3_EN" },
384         { "ADC4_WORK", NULL, "ADC4_EN" },
385         { "ADC5_WORK", NULL, "ADC5_EN" },
386         { "ADC6_WORK", NULL, "ADC6_EN" },
387         { "ADC7_WORK", NULL, "ADC7_EN" },
388         { "ADC8_WORK", NULL, "ADC8_EN" },
389 
390         { "ADC1_BUF_REF_EN", NULL, "ADC_CURRENT_EN12" },
391         { "ADC2_BUF_REF_EN", NULL, "ADC_CURRENT_EN12" },
392         { "ADC3_BUF_REF_EN", NULL, "ADC_CURRENT_EN34" },
393         { "ADC4_BUF_REF_EN", NULL, "ADC_CURRENT_EN34" },
394         { "ADC5_BUF_REF_EN", NULL, "ADC_CURRENT_EN56" },
395         { "ADC6_BUF_REF_EN", NULL, "ADC_CURRENT_EN56" },
396         { "ADC7_BUF_REF_EN", NULL, "ADC_CURRENT_EN78" },
397         { "ADC8_BUF_REF_EN", NULL, "ADC_CURRENT_EN78" },
398 
399         { "ADC1_WORK", NULL, "ADC1_BUF_REF_EN" },
400         { "ADC2_WORK", NULL, "ADC2_BUF_REF_EN" },
401         { "ADC3_WORK", NULL, "ADC3_BUF_REF_EN" },
402         { "ADC4_WORK", NULL, "ADC4_BUF_REF_EN" },
403         { "ADC5_WORK", NULL, "ADC5_BUF_REF_EN" },
404         { "ADC6_WORK", NULL, "ADC6_BUF_REF_EN" },
405         { "ADC7_WORK", NULL, "ADC7_BUF_REF_EN" },
406         { "ADC8_WORK", NULL, "ADC8_BUF_REF_EN" },
407 
408         { "ADC1_CLK_EN", NULL, "ADC_MCLK_GATE" },
409         { "ADC2_CLK_EN", NULL, "ADC_MCLK_GATE" },
410         { "ADC3_CLK_EN", NULL, "ADC_MCLK_GATE" },
411         { "ADC4_CLK_EN", NULL, "ADC_MCLK_GATE" },
412         { "ADC5_CLK_EN", NULL, "ADC_MCLK_GATE" },
413         { "ADC6_CLK_EN", NULL, "ADC_MCLK_GATE" },
414         { "ADC7_CLK_EN", NULL, "ADC_MCLK_GATE" },
415         { "ADC8_CLK_EN", NULL, "ADC_MCLK_GATE" },
416 
417         { "ADC1_WORK", NULL, "ADC1_CLK_EN" },
418         { "ADC2_WORK", NULL, "ADC2_CLK_EN" },
419         { "ADC3_WORK", NULL, "ADC3_CLK_EN" },
420         { "ADC4_WORK", NULL, "ADC4_CLK_EN" },
421         { "ADC5_WORK", NULL, "ADC5_CLK_EN" },
422         { "ADC6_WORK", NULL, "ADC6_CLK_EN" },
423         { "ADC7_WORK", NULL, "ADC7_CLK_EN" },
424         { "ADC8_WORK", NULL, "ADC8_CLK_EN" },
425 
426         { "ALC1_WORK", NULL, "ADC1_WORK" },
427         { "ALC2_WORK", NULL, "ADC2_WORK" },
428         { "ALC3_WORK", NULL, "ADC3_WORK" },
429         { "ALC4_WORK", NULL, "ADC4_WORK" },
430         { "ALC5_WORK", NULL, "ADC5_WORK" },
431         { "ALC6_WORK", NULL, "ADC6_WORK" },
432         { "ALC7_WORK", NULL, "ADC7_WORK" },
433         { "ALC8_WORK", NULL, "ADC8_WORK" },
434 
435         { "HiFi Capture", NULL, "ALC1_WORK" },
436         { "HiFi Capture", NULL, "ALC2_WORK" },
437         { "HiFi Capture", NULL, "ALC3_WORK" },
438         { "HiFi Capture", NULL, "ALC4_WORK" },
439         { "HiFi Capture", NULL, "ALC5_WORK" },
440         { "HiFi Capture", NULL, "ALC6_WORK" },
441         { "HiFi Capture", NULL, "ALC7_WORK" },
442         { "HiFi Capture", NULL, "ALC8_WORK" },
443 
444         { "DAC_L_HPMIX_EN", NULL, "HiFi Playback" },
445         { "DAC_R_HPMIX_EN", NULL, "HiFi Playback" },
446         { "DAC_L_HPMIX_WORK", NULL, "DAC_L_HPMIX_EN" },
447         { "DAC_R_HPMIX_WORK", NULL, "DAC_R_HPMIX_EN" },
448         { "DAC HPMIX Left",  NULL, "DAC_L_HPMIX_WORK" },
449         { "DAC HPMIX Right", NULL, "DAC_R_HPMIX_WORK" },
450 
451         { "DAC_L_DAC_WORK", NULL, "DAC HPMIX Left"  },
452         { "DAC_R_DAC_WORK", NULL, "DAC HPMIX Right" },
453 
454         { "DAC_L_REF_EN", NULL, "DAC_CURRENT_EN" },
455         { "DAC_R_REF_EN", NULL, "DAC_CURRENT_EN" },
456         { "DAC_L_CLK_EN", NULL, "DAC_L_REF_EN" },
457         { "DAC_R_CLK_EN", NULL, "DAC_R_REF_EN" },
458         { "DAC_L_CLK_EN", NULL, "DAC_MCLK_GATE" },
459         { "DAC_R_CLK_EN", NULL, "DAC_MCLK_GATE" },
460         { "DAC_L_DAC_WORK", NULL, "DAC_L_CLK_EN" },
461         { "DAC_R_DAC_WORK", NULL, "DAC_R_CLK_EN" },
462         { "DAC_L_HPMIX_SEL", NULL, "DAC_L_DAC_WORK" },
463         { "DAC_R_HPMIX_SEL", NULL, "DAC_R_DAC_WORK" },
464 
465         { "HPOUT_L", NULL, "DAC_BUF_REF_L" },
466         { "HPOUT_R", NULL, "DAC_BUF_REF_R" },
467         { "L_HPOUT_EN", NULL, "DAC_L_HPMIX_SEL" },
468         { "R_HPOUT_EN", NULL, "DAC_R_HPMIX_SEL" },
469         { "L_HPOUT_WORK", NULL, "L_HPOUT_EN" },
470         { "R_HPOUT_WORK", NULL, "R_HPOUT_EN" },
471         { "HPOUT_POP_SOUND_L", NULL, "L_HPOUT_WORK" },
472         { "HPOUT_POP_SOUND_R", NULL, "R_HPOUT_WORK" },
473         { "HPOUT_L", NULL, "HPOUT_POP_SOUND_L" },
474         { "HPOUT_R", NULL, "HPOUT_POP_SOUND_R" },
475 
476         { "L_LINEOUT_EN", NULL, "DAC_L_HPMIX_SEL" },
477         { "R_LINEOUT_EN", NULL, "DAC_R_HPMIX_SEL" },
478         { "LINEOUT_L", NULL, "L_LINEOUT_EN" },
479         { "LINEOUT_R", NULL, "R_LINEOUT_EN" },
480 };
481 
482 static int rk3308_codec_set_dai_fmt(struct snd_soc_dai *codec_dai,
483                                     unsigned int fmt)
484 {
485         struct snd_soc_component *component = codec_dai->component;
486         struct rk3308_codec_priv *rk3308 = snd_soc_component_get_drvdata(component);
487         const unsigned int inv_bits = fmt & SND_SOC_DAIFMT_INV_MASK;
488         const bool inv_bitclk =
489                 (inv_bits & SND_SOC_DAIFMT_IB_IF) ||
490                 (inv_bits & SND_SOC_DAIFMT_IB_NF);
491         const bool inv_frmclk =
492                 (inv_bits & SND_SOC_DAIFMT_IB_IF) ||
493                 (inv_bits & SND_SOC_DAIFMT_NB_IF);
494         const unsigned int dac_master_bits = rk3308->codec_ver < ACODEC_VERSION_C ?
495                 RK3308_DAC_IO_MODE_MASTER   | RK3308_DAC_MODE_MASTER  :
496                 RK3308BS_DAC_IO_MODE_MASTER | RK3308BS_DAC_MODE_MASTER;
497         unsigned int adc_aif1 = 0, adc_aif2 = 0, dac_aif1 = 0, dac_aif2 = 0;
498         bool is_master = false;
499         int grp;
500 
501         switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
502         case SND_SOC_DAIFMT_CBC_CFC:
503                 break;
504         case SND_SOC_DAIFMT_CBP_CFP:
505                 adc_aif2 |= RK3308_ADC_IO_MODE_MASTER;
506                 adc_aif2 |= RK3308_ADC_MODE_MASTER;
507                 dac_aif2 |= dac_master_bits;
508                 is_master = true;
509                 break;
510         default:
511                 return -EINVAL;
512         }
513 
514         switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
515         case SND_SOC_DAIFMT_DSP_A:
516                 adc_aif1 |= RK3308_ADC_I2S_MODE_PCM;
517                 dac_aif1 |= RK3308_DAC_I2S_MODE_PCM;
518                 break;
519         case SND_SOC_DAIFMT_I2S:
520                 adc_aif1 |= RK3308_ADC_I2S_MODE_I2S;
521                 dac_aif1 |= RK3308_DAC_I2S_MODE_I2S;
522                 break;
523         case SND_SOC_DAIFMT_RIGHT_J:
524                 adc_aif1 |= RK3308_ADC_I2S_MODE_RJ;
525                 dac_aif1 |= RK3308_DAC_I2S_MODE_RJ;
526                 break;
527         case SND_SOC_DAIFMT_LEFT_J:
528                 adc_aif1 |= RK3308_ADC_I2S_MODE_LJ;
529                 dac_aif1 |= RK3308_DAC_I2S_MODE_LJ;
530                 break;
531         default:
532                 return -EINVAL;
533         }
534 
535         if (inv_bitclk) {
536                 adc_aif2 |= RK3308_ADC_I2S_BIT_CLK_POL_REVERSAL;
537                 dac_aif2 |= RK3308_DAC_I2S_BIT_CLK_POL_REVERSAL;
538         }
539 
540         if (inv_frmclk) {
541                 adc_aif1 |= RK3308_ADC_I2S_LRC_POL_REVERSAL;
542                 dac_aif1 |= RK3308_DAC_I2S_LRC_POL_REVERSAL;
543         }
544 
545         /*
546          * Hold ADC Digital registers start at master mode
547          *
548          * There are 8 ADCs which use the same internal SCLK and LRCK for
549          * master mode. We need to make sure that they are in effect at the
550          * same time, otherwise they will cause abnormal clocks.
551          */
552         if (is_master)
553                 regmap_clear_bits(rk3308->regmap, RK3308_GLB_CON, RK3308_ADC_DIG_WORK);
554 
555         for (grp = 0; grp < ADC_LR_GROUP_MAX; grp++) {
556                 regmap_update_bits(rk3308->regmap, RK3308_ADC_DIG_CON01(grp),
557                                    RK3308_ADC_I2S_LRC_POL_REVERSAL |
558                                    RK3308_ADC_I2S_MODE_MSK,
559                                    adc_aif1);
560                 regmap_update_bits(rk3308->regmap, RK3308_ADC_DIG_CON02(grp),
561                                    RK3308_ADC_IO_MODE_MASTER |
562                                    RK3308_ADC_MODE_MASTER |
563                                    RK3308_ADC_I2S_BIT_CLK_POL_REVERSAL,
564                                    adc_aif2);
565         }
566 
567         /* Hold ADC Digital registers end at master mode */
568         if (is_master)
569                 regmap_set_bits(rk3308->regmap, RK3308_GLB_CON, RK3308_ADC_DIG_WORK);
570 
571         regmap_update_bits(rk3308->regmap, RK3308_DAC_DIG_CON01,
572                            RK3308_DAC_I2S_LRC_POL_REVERSAL |
573                            RK3308_DAC_I2S_MODE_MSK,
574                            dac_aif1);
575         regmap_update_bits(rk3308->regmap, RK3308_DAC_DIG_CON02,
576                            dac_master_bits | RK3308_DAC_I2S_BIT_CLK_POL_REVERSAL,
577                            dac_aif2);
578 
579         return 0;
580 }
581 
582 static int rk3308_codec_dac_dig_config(struct rk3308_codec_priv *rk3308,
583                                        struct snd_pcm_hw_params *params)
584 {
585         unsigned int dac_aif1 = 0;
586 
587         switch (params_format(params)) {
588         case SNDRV_PCM_FORMAT_S16_LE:
589                 dac_aif1 |= RK3308_DAC_I2S_VALID_LEN_16BITS;
590                 break;
591         case SNDRV_PCM_FORMAT_S20_3LE:
592                 dac_aif1 |= RK3308_DAC_I2S_VALID_LEN_20BITS;
593                 break;
594         case SNDRV_PCM_FORMAT_S24_LE:
595                 dac_aif1 |= RK3308_DAC_I2S_VALID_LEN_24BITS;
596                 break;
597         case SNDRV_PCM_FORMAT_S32_LE:
598                 dac_aif1 |= RK3308_DAC_I2S_VALID_LEN_32BITS;
599                 break;
600         default:
601                 return -EINVAL;
602         }
603 
604         regmap_update_bits(rk3308->regmap, RK3308_DAC_DIG_CON01,
605                            RK3308_DAC_I2S_VALID_LEN_MSK, dac_aif1);
606         regmap_set_bits(rk3308->regmap, RK3308_DAC_DIG_CON02, RK3308_DAC_I2S_WORK);
607 
608         return 0;
609 }
610 
611 static int rk3308_codec_adc_dig_config(struct rk3308_codec_priv *rk3308,
612                                        struct snd_pcm_hw_params *params)
613 {
614         unsigned int adc_aif1 = 0;
615         /*
616          * grp 0 = ADC1 and ADC2
617          * grp 1 = ADC3 and ADC4
618          * grp 2 = ADC5 and ADC6
619          * grp 3 = ADC7 and ADC8
620          */
621         u32 used_adc_grps;
622         int grp;
623 
624         switch (params_channels(params)) {
625         case 1:
626                 adc_aif1 |= RK3308_ADC_I2S_MONO;
627                 used_adc_grps = 1;
628                 break;
629         case 2:
630         case 4:
631         case 6:
632         case 8:
633                 used_adc_grps = params_channels(params) / 2;
634                 break;
635         default:
636                 dev_err(rk3308->dev, "Invalid channel number %d\n", params_channels(params));
637                 return -EINVAL;
638         }
639 
640         switch (params_format(params)) {
641         case SNDRV_PCM_FORMAT_S16_LE:
642                 adc_aif1 |= RK3308_ADC_I2S_VALID_LEN_16BITS;
643                 break;
644         case SNDRV_PCM_FORMAT_S20_3LE:
645                 adc_aif1 |= RK3308_ADC_I2S_VALID_LEN_20BITS;
646                 break;
647         case SNDRV_PCM_FORMAT_S24_LE:
648                 adc_aif1 |= RK3308_ADC_I2S_VALID_LEN_24BITS;
649                 break;
650         case SNDRV_PCM_FORMAT_S32_LE:
651                 adc_aif1 |= RK3308_ADC_I2S_VALID_LEN_32BITS;
652                 break;
653         default:
654                 return -EINVAL;
655         }
656 
657         for (grp = 0; grp < used_adc_grps; grp++) {
658                 regmap_update_bits(rk3308->regmap,
659                                    RK3308_ADC_DIG_CON03(grp),
660                                    RK3308_ADC_L_CH_BIST_MSK | RK3308_ADC_R_CH_BIST_MSK,
661                                    RK3308_ADC_L_CH_NORMAL_LEFT | RK3308_ADC_R_CH_NORMAL_RIGHT);
662                 regmap_update_bits(rk3308->regmap, RK3308_ADC_DIG_CON01(grp),
663                                    RK3308_ADC_I2S_VALID_LEN_MSK | RK3308_ADC_I2S_MONO, adc_aif1);
664                 regmap_set_bits(rk3308->regmap, RK3308_ADC_DIG_CON02(grp), RK3308_ADC_I2S_WORK);
665         }
666 
667         return 0;
668 }
669 
670 static int rk3308_codec_hw_params(struct snd_pcm_substream *substream,
671                                   struct snd_pcm_hw_params *params,
672                                   struct snd_soc_dai *dai)
673 {
674         struct snd_soc_component *component = dai->component;
675         struct rk3308_codec_priv *rk3308 = snd_soc_component_get_drvdata(component);
676 
677         return (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) ?
678                 rk3308_codec_dac_dig_config(rk3308, params) :
679                 rk3308_codec_adc_dig_config(rk3308, params);
680 }
681 
682 static const struct snd_soc_dai_ops rk3308_codec_dai_ops = {
683         .hw_params = rk3308_codec_hw_params,
684         .set_fmt = rk3308_codec_set_dai_fmt,
685 };
686 
687 static struct snd_soc_dai_driver rk3308_codec_dai_driver = {
688         .name = "rk3308-hifi",
689         .playback = {
690                 .stream_name = "HiFi Playback",
691                 .channels_min = 2,
692                 .channels_max = 2,
693                 .rates = SNDRV_PCM_RATE_8000_192000,
694                 .formats = (SNDRV_PCM_FMTBIT_S16_LE |
695                             SNDRV_PCM_FMTBIT_S20_3LE |
696                             SNDRV_PCM_FMTBIT_S24_LE |
697                             SNDRV_PCM_FMTBIT_S32_LE),
698         },
699         .capture = {
700                 .stream_name = "HiFi Capture",
701                 .channels_min = 1,
702                 .channels_max = 8,
703                 .rates = SNDRV_PCM_RATE_8000_192000,
704                 .formats = (SNDRV_PCM_FMTBIT_S16_LE |
705                             SNDRV_PCM_FMTBIT_S20_3LE |
706                             SNDRV_PCM_FMTBIT_S24_LE |
707                             SNDRV_PCM_FMTBIT_S32_LE),
708         },
709         .ops = &rk3308_codec_dai_ops,
710 };
711 
712 static void rk3308_codec_reset(struct snd_soc_component *component)
713 {
714         struct rk3308_codec_priv *rk3308 = snd_soc_component_get_drvdata(component);
715 
716         reset_control_assert(rk3308->reset);
717         usleep_range(10000, 11000);     /* estimated value */
718         reset_control_deassert(rk3308->reset);
719 
720         regmap_write(rk3308->regmap, RK3308_GLB_CON, 0x00);
721         usleep_range(10000, 11000);     /* estimated value */
722         regmap_write(rk3308->regmap, RK3308_GLB_CON,
723                      RK3308_SYS_WORK |
724                      RK3308_DAC_DIG_WORK |
725                      RK3308_ADC_DIG_WORK);
726 }
727 
728 /*
729  * Initialize register whose default after HW reset is problematic or which
730  * are never modified.
731  */
732 static int rk3308_codec_initialize(struct rk3308_codec_priv *rk3308)
733 {
734         int grp;
735 
736         /*
737          * Init ADC digital vol to 0 dB (reset value is 0xff, undocumented).
738          * Range: -97dB ~ +32dB.
739          */
740         if (rk3308->codec_ver == ACODEC_VERSION_C) {
741                 for (grp = 0; grp < ADC_LR_GROUP_MAX; grp++) {
742                         regmap_write(rk3308->regmap, RK3308_ADC_DIG_CON05(grp),
743                                            RK3308_ADC_DIG_VOL_CON_x_0DB);
744                         regmap_write(rk3308->regmap, RK3308_ADC_DIG_CON06(grp),
745                                            RK3308_ADC_DIG_VOL_CON_x_0DB);
746                 }
747         }
748 
749         /* set HPMIX default gains (reset value is 0, which is illegal) */
750         regmap_update_bits(rk3308->regmap, RK3308_DAC_ANA_CON12,
751                            RK3308_DAC_L_HPMIX_GAIN_MSK |
752                            RK3308_DAC_R_HPMIX_GAIN_MSK,
753                            RK3308_DAC_L_HPMIX_GAIN_NDB_6 |
754                            RK3308_DAC_R_HPMIX_GAIN_NDB_6);
755 
756         /* recover DAC digital gain to 0 dB (reset value is 0xff, undocumented) */
757         if (rk3308->codec_ver == ACODEC_VERSION_C)
758                 regmap_write(rk3308->regmap, RK3308_DAC_DIG_CON04,
759                              RK3308BS_DAC_DIG_GAIN_0DB);
760 
761         /*
762          * Unconditionally enable zero-cross detection (needed for AGC,
763          * harmless without AGC)
764          */
765         for (grp = 0; grp < ADC_LR_GROUP_MAX; grp++)
766                 regmap_set_bits(rk3308->regmap, RK3308_ADC_ANA_CON02(grp),
767                                 RK3308_ADC_CH1_ZEROCROSS_DET_EN |
768                                 RK3308_ADC_CH2_ZEROCROSS_DET_EN);
769 
770         return 0;
771 }
772 
773 static int rk3308_codec_probe(struct snd_soc_component *component)
774 {
775         struct rk3308_codec_priv *rk3308 = snd_soc_component_get_drvdata(component);
776 
777         rk3308->component = component;
778 
779         rk3308_codec_reset(component);
780         rk3308_codec_initialize(rk3308);
781 
782         return 0;
783 }
784 
785 static int rk3308_codec_set_bias_level(struct snd_soc_component *component,
786                                        enum snd_soc_bias_level level)
787 {
788         struct rk3308_codec_priv *rk3308 = snd_soc_component_get_drvdata(component);
789 
790         switch (level) {
791         case SND_SOC_BIAS_ON:
792                 break;
793         case SND_SOC_BIAS_PREPARE:
794                 break;
795         case SND_SOC_BIAS_STANDBY:
796                 if (snd_soc_component_get_bias_level(component) != SND_SOC_BIAS_OFF)
797                         break;
798 
799                 /* Sequence from TRM Section 8.6.3 "Power Up" */
800                 regmap_set_bits(rk3308->regmap, RK3308_DAC_ANA_CON02,
801                                 RK3308_DAC_L_DAC_EN | RK3308_DAC_R_DAC_EN);
802                 regmap_update_bits(rk3308->regmap, RK3308_ADC_ANA_CON10(0),
803                                    RK3308_ADC_CURRENT_CHARGE_MSK, 1);
804                 regmap_set_bits(rk3308->regmap, RK3308_ADC_ANA_CON10(0),
805                                 RK3308_ADC_REF_EN);
806                 regmap_update_bits(rk3308->regmap, RK3308_ADC_ANA_CON10(0),
807                                    RK3308_ADC_CURRENT_CHARGE_MSK, 0x7f);
808                 msleep(20);     /* estimated value */
809                 break;
810         case SND_SOC_BIAS_OFF:
811                 /* Sequence from TRM Section 8.6.4 "Power Down" */
812                 regmap_update_bits(rk3308->regmap, RK3308_ADC_ANA_CON10(0),
813                                    RK3308_ADC_CURRENT_CHARGE_MSK, 1);
814                 regmap_clear_bits(rk3308->regmap, RK3308_ADC_ANA_CON10(0),
815                                   RK3308_ADC_REF_EN);
816                 regmap_clear_bits(rk3308->regmap, RK3308_DAC_ANA_CON02,
817                                   RK3308_DAC_L_DAC_EN | RK3308_DAC_R_DAC_EN);
818                 msleep(20);     /* estimated value */
819                 break;
820         }
821         return 0;
822 }
823 
824 static const struct snd_soc_component_driver rk3308_codec_component_driver = {
825         .probe = rk3308_codec_probe,
826         .set_bias_level = rk3308_codec_set_bias_level,
827         .controls = rk3308_codec_controls,
828         .num_controls = ARRAY_SIZE(rk3308_codec_controls),
829         .dapm_widgets = rk3308_codec_dapm_widgets,
830         .num_dapm_widgets = ARRAY_SIZE(rk3308_codec_dapm_widgets),
831         .dapm_routes = rk3308_codec_dapm_routes,
832         .num_dapm_routes = ARRAY_SIZE(rk3308_codec_dapm_routes),
833 };
834 
835 static const struct regmap_config rk3308_codec_regmap_config = {
836         .reg_bits = 32,
837         .reg_stride = 4,
838         .val_bits = 32,
839         .max_register = RK3308_DAC_ANA_CON15,
840 };
841 
842 static int rk3308_codec_get_version(struct rk3308_codec_priv *rk3308)
843 {
844         unsigned int chip_id;
845         int err;
846 
847         err = regmap_read(rk3308->grf, GRF_CHIP_ID, &chip_id);
848         if (err)
849                 return err;
850 
851         switch (chip_id) {
852         case 3306:
853                 rk3308->codec_ver = ACODEC_VERSION_A;
854                 break;
855         case 0x3308:
856                 rk3308->codec_ver = ACODEC_VERSION_B;
857                 return dev_err_probe(rk3308->dev, -EINVAL, "Chip version B not supported\n");
858         case 0x3308c:
859                 rk3308->codec_ver = ACODEC_VERSION_C;
860                 break;
861         default:
862                 return dev_err_probe(rk3308->dev, -EINVAL, "Unknown chip_id: 0x%x\n", chip_id);
863         }
864 
865         dev_info(rk3308->dev, "Found codec version %c\n", rk3308->codec_ver);
866         return 0;
867 }
868 
869 static int rk3308_codec_set_micbias_level(struct rk3308_codec_priv *rk3308)
870 {
871         struct device_node *np = rk3308->dev->of_node;
872         u32 percent;
873         u32 mult;
874         int err;
875 
876         err = of_property_read_u32(np, "rockchip,micbias-avdd-percent", &percent);
877         if (err == -EINVAL)
878                 return 0;
879         if (err)
880                 return dev_err_probe(rk3308->dev, err,
881                                      "Error reading 'rockchip,micbias-avdd-percent'\n");
882 
883         /* Convert percent to register value, linerarly (50% -> 0, 5% step = +1) */
884         mult = (percent - 50) / 5;
885 
886         /* Check range and that the percent was an exact value allowed */
887         if (mult > RK3308_ADC_LEVEL_RANGE_MICBIAS_MAX || mult * 5 + 50 != percent)
888                 return dev_err_probe(rk3308->dev, -EINVAL,
889                                      "Invalid value %u for 'rockchip,micbias-avdd-percent'\n",
890                                      percent);
891 
892         regmap_update_bits(rk3308->regmap, RK3308_ADC_ANA_CON07(0),
893                            RK3308_ADC_LEVEL_RANGE_MICBIAS_MSK,
894                            mult << RK3308_ADC_LEVEL_RANGE_MICBIAS_SFT);
895 
896         return 0;
897 }
898 
899 static int rk3308_codec_platform_probe(struct platform_device *pdev)
900 {
901         struct device_node *np = pdev->dev.of_node;
902         struct device *dev = &pdev->dev;
903         struct rk3308_codec_priv *rk3308;
904         void __iomem *base;
905         int err;
906 
907         rk3308 = devm_kzalloc(&pdev->dev, sizeof(*rk3308), GFP_KERNEL);
908         if (!rk3308)
909                 return -ENOMEM;
910 
911         rk3308->dev = dev;
912 
913         rk3308->grf = syscon_regmap_lookup_by_phandle(np, "rockchip,grf");
914         if (IS_ERR(rk3308->grf))
915                 return dev_err_probe(dev, PTR_ERR(rk3308->grf), "Error getting GRF\n");
916 
917         rk3308->reset = devm_reset_control_get_optional_exclusive(dev, "codec");
918         if (IS_ERR(rk3308->reset))
919                 return dev_err_probe(dev, PTR_ERR(rk3308->reset), "Failed to get reset control\n");
920 
921         err = devm_clk_bulk_get(dev, ARRAY_SIZE(rk3308_codec_clocks), rk3308_codec_clocks);
922         if (err)
923                 return dev_err_probe(dev, err, "Failed to get clocks\n");
924 
925         err = clk_bulk_prepare_enable(ARRAY_SIZE(rk3308_codec_clocks), rk3308_codec_clocks);
926         if (err)
927                 return dev_err_probe(dev, err, "Failed to enable clocks\n");
928 
929         err = rk3308_codec_get_version(rk3308);
930         if (err)
931                 return err;
932 
933         base = devm_platform_ioremap_resource(pdev, 0);
934         if (IS_ERR(base))
935                 return PTR_ERR(base);
936 
937         rk3308->regmap = devm_regmap_init_mmio(dev, base, &rk3308_codec_regmap_config);
938         if (IS_ERR(rk3308->regmap))
939                 return dev_err_probe(dev, PTR_ERR(rk3308->regmap),
940                                      "Failed to init regmap\n");
941 
942         platform_set_drvdata(pdev, rk3308);
943 
944         err = rk3308_codec_set_micbias_level(rk3308);
945         if (err)
946                 return err;
947 
948         err = devm_snd_soc_register_component(dev, &rk3308_codec_component_driver,
949                                               &rk3308_codec_dai_driver, 1);
950         if (err)
951                 return dev_err_probe(dev, err, "Failed to register codec\n");
952 
953         return 0;
954 }
955 
956 static const struct of_device_id __maybe_unused rk3308_codec_of_match[] = {
957         { .compatible = "rockchip,rk3308-codec", },
958         {},
959 };
960 MODULE_DEVICE_TABLE(of, rk3308_codec_of_match);
961 
962 static struct platform_driver rk3308_codec_driver = {
963         .driver = {
964                 .name = "rk3308-acodec",
965                 .of_match_table = rk3308_codec_of_match,
966         },
967         .probe = rk3308_codec_platform_probe,
968 };
969 module_platform_driver(rk3308_codec_driver);
970 
971 MODULE_AUTHOR("Xing Zheng <zhengxing@rock-chips.com>");
972 MODULE_AUTHOR("Luca Ceresoli <luca.ceresoli@bootlin.com>");
973 MODULE_DESCRIPTION("ASoC RK3308 Codec Driver");
974 MODULE_LICENSE("GPL");
975 

~ [ 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