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