xref: /openbmc/linux/sound/soc/codecs/mt6351.c (revision 39723d34)
1a74d51baSKai Chieh Chuang // SPDX-License-Identifier: GPL-2.0
23c76fbc3SKai Chieh Chuang //
33c76fbc3SKai Chieh Chuang // mt6351.c  --  mt6351 ALSA SoC audio codec driver
43c76fbc3SKai Chieh Chuang //
53c76fbc3SKai Chieh Chuang // Copyright (c) 2018 MediaTek Inc.
63c76fbc3SKai Chieh Chuang // Author: KaiChieh Chuang <kaichieh.chuang@mediatek.com>
7a74d51baSKai Chieh Chuang 
8a74d51baSKai Chieh Chuang #include <linux/dma-mapping.h>
9a74d51baSKai Chieh Chuang #include <linux/platform_device.h>
10a74d51baSKai Chieh Chuang #include <linux/slab.h>
11a74d51baSKai Chieh Chuang #include <linux/module.h>
12a74d51baSKai Chieh Chuang #include <linux/of_device.h>
13a74d51baSKai Chieh Chuang #include <linux/delay.h>
14a74d51baSKai Chieh Chuang 
15a74d51baSKai Chieh Chuang #include <sound/core.h>
16a74d51baSKai Chieh Chuang #include <sound/pcm.h>
17a74d51baSKai Chieh Chuang #include <sound/soc.h>
18a74d51baSKai Chieh Chuang #include <sound/tlv.h>
19a74d51baSKai Chieh Chuang 
20a74d51baSKai Chieh Chuang #include "mt6351.h"
21a74d51baSKai Chieh Chuang 
22a74d51baSKai Chieh Chuang /* MT6351_TOP_CLKSQ */
23a74d51baSKai Chieh Chuang #define RG_CLKSQ_EN_AUD_BIT (0)
24a74d51baSKai Chieh Chuang 
25a74d51baSKai Chieh Chuang /* MT6351_TOP_CKPDN_CON0 */
26a74d51baSKai Chieh Chuang #define RG_AUDNCP_CK_PDN_BIT (12)
27a74d51baSKai Chieh Chuang #define RG_AUDIF_CK_PDN_BIT (13)
28a74d51baSKai Chieh Chuang #define RG_AUD_CK_PDN_BIT (14)
29a74d51baSKai Chieh Chuang #define RG_ZCD13M_CK_PDN_BIT (15)
30a74d51baSKai Chieh Chuang 
31a74d51baSKai Chieh Chuang /* MT6351_AUDDEC_ANA_CON0 */
32a74d51baSKai Chieh Chuang #define RG_AUDDACLPWRUP_VAUDP32_BIT (0)
33a74d51baSKai Chieh Chuang #define RG_AUDDACRPWRUP_VAUDP32_BIT (1)
34a74d51baSKai Chieh Chuang #define RG_AUD_DAC_PWR_UP_VA32_BIT (2)
35a74d51baSKai Chieh Chuang #define RG_AUD_DAC_PWL_UP_VA32_BIT (3)
36a74d51baSKai Chieh Chuang 
37a74d51baSKai Chieh Chuang #define RG_AUDHSPWRUP_VAUDP32_BIT (4)
38a74d51baSKai Chieh Chuang 
39a74d51baSKai Chieh Chuang #define RG_AUDHPLPWRUP_VAUDP32_BIT (5)
40a74d51baSKai Chieh Chuang #define RG_AUDHPRPWRUP_VAUDP32_BIT (6)
41a74d51baSKai Chieh Chuang 
42a74d51baSKai Chieh Chuang #define RG_AUDHSMUXINPUTSEL_VAUDP32_SFT (7)
43a74d51baSKai Chieh Chuang #define RG_AUDHSMUXINPUTSEL_VAUDP32_MASK (0x3)
44a74d51baSKai Chieh Chuang 
45a74d51baSKai Chieh Chuang #define RG_AUDHPLMUXINPUTSEL_VAUDP32_SFT (9)
46a74d51baSKai Chieh Chuang #define RG_AUDHPLMUXINPUTSEL_VAUDP32_MASK (0x3)
47a74d51baSKai Chieh Chuang 
48a74d51baSKai Chieh Chuang #define RG_AUDHPRMUXINPUTSEL_VAUDP32_SFT (11)
49a74d51baSKai Chieh Chuang #define RG_AUDHPRMUXINPUTSEL_VAUDP32_MASK (0x3)
50a74d51baSKai Chieh Chuang 
51a74d51baSKai Chieh Chuang #define RG_AUDHSSCDISABLE_VAUDP32 (13)
52a74d51baSKai Chieh Chuang #define RG_AUDHPLSCDISABLE_VAUDP32_BIT (14)
53a74d51baSKai Chieh Chuang #define RG_AUDHPRSCDISABLE_VAUDP32_BIT (15)
54a74d51baSKai Chieh Chuang 
55a74d51baSKai Chieh Chuang /* MT6351_AUDDEC_ANA_CON1 */
56a74d51baSKai Chieh Chuang #define RG_HSOUTPUTSTBENH_VAUDP32_BIT (8)
57a74d51baSKai Chieh Chuang 
58a74d51baSKai Chieh Chuang /* MT6351_AUDDEC_ANA_CON3 */
59a74d51baSKai Chieh Chuang #define RG_AUDLOLPWRUP_VAUDP32_BIT (2)
60a74d51baSKai Chieh Chuang 
61a74d51baSKai Chieh Chuang #define RG_AUDLOLMUXINPUTSEL_VAUDP32_SFT (3)
62a74d51baSKai Chieh Chuang #define RG_AUDLOLMUXINPUTSEL_VAUDP32_MASK (0x3)
63a74d51baSKai Chieh Chuang 
64a74d51baSKai Chieh Chuang #define RG_AUDLOLSCDISABLE_VAUDP32_BIT (5)
65a74d51baSKai Chieh Chuang #define RG_LOOUTPUTSTBENH_VAUDP32_BIT (9)
66a74d51baSKai Chieh Chuang 
67a74d51baSKai Chieh Chuang /* MT6351_AUDDEC_ANA_CON6 */
68a74d51baSKai Chieh Chuang #define RG_ABIDEC_RSVD0_VAUDP32_HPL_BIT (8)
69a74d51baSKai Chieh Chuang #define RG_ABIDEC_RSVD0_VAUDP32_HPR_BIT (9)
70a74d51baSKai Chieh Chuang #define RG_ABIDEC_RSVD0_VAUDP32_HS_BIT (10)
71a74d51baSKai Chieh Chuang #define RG_ABIDEC_RSVD0_VAUDP32_LOL_BIT (11)
72a74d51baSKai Chieh Chuang 
73a74d51baSKai Chieh Chuang /* MT6351_AUDDEC_ANA_CON9 */
74a74d51baSKai Chieh Chuang #define RG_AUDIBIASPWRDN_VAUDP32_BIT (8)
75a74d51baSKai Chieh Chuang #define RG_RSTB_DECODER_VA32_BIT (9)
76a74d51baSKai Chieh Chuang #define RG_AUDGLB_PWRDN_VA32_BIT (12)
77a74d51baSKai Chieh Chuang 
78a74d51baSKai Chieh Chuang #define RG_LCLDO_DEC_EN_VA32_BIT (13)
79a74d51baSKai Chieh Chuang #define RG_LCLDO_DEC_REMOTE_SENSE_VA18_BIT (15)
80a74d51baSKai Chieh Chuang /* MT6351_AUDDEC_ANA_CON10 */
81a74d51baSKai Chieh Chuang #define RG_NVREG_EN_VAUDP32_BIT (8)
82a74d51baSKai Chieh Chuang 
83a74d51baSKai Chieh Chuang #define RG_AUDGLB_LP2_VOW_EN_VA32 10
84a74d51baSKai Chieh Chuang 
85a74d51baSKai Chieh Chuang /* MT6351_AFE_UL_DL_CON0 */
86a74d51baSKai Chieh Chuang #define RG_AFE_ON_BIT (0)
87a74d51baSKai Chieh Chuang 
88a74d51baSKai Chieh Chuang /* MT6351_AFE_DL_SRC2_CON0_L */
89a74d51baSKai Chieh Chuang #define RG_DL_2_SRC_ON_TMP_CTL_PRE_BIT (0)
90a74d51baSKai Chieh Chuang 
91a74d51baSKai Chieh Chuang /* MT6351_AFE_UL_SRC_CON0_L */
92a74d51baSKai Chieh Chuang #define UL_SRC_ON_TMP_CTL (0)
93a74d51baSKai Chieh Chuang 
94a74d51baSKai Chieh Chuang /* MT6351_AFE_TOP_CON0 */
95a74d51baSKai Chieh Chuang #define RG_DL_SINE_ON_SFT (0)
96a74d51baSKai Chieh Chuang #define RG_DL_SINE_ON_MASK (0x1)
97a74d51baSKai Chieh Chuang 
98a74d51baSKai Chieh Chuang #define RG_UL_SINE_ON_SFT (1)
99a74d51baSKai Chieh Chuang #define RG_UL_SINE_ON_MASK (0x1)
100a74d51baSKai Chieh Chuang 
101a74d51baSKai Chieh Chuang /* MT6351_AUDIO_TOP_CON0 */
102a74d51baSKai Chieh Chuang #define AUD_TOP_PDN_RESERVED_BIT 0
103a74d51baSKai Chieh Chuang #define AUD_TOP_PWR_CLK_DIS_CTL_BIT 2
104a74d51baSKai Chieh Chuang #define AUD_TOP_PDN_ADC_CTL_BIT 5
105a74d51baSKai Chieh Chuang #define AUD_TOP_PDN_DAC_CTL_BIT 6
106a74d51baSKai Chieh Chuang #define AUD_TOP_PDN_AFE_CTL_BIT 7
107a74d51baSKai Chieh Chuang 
108a74d51baSKai Chieh Chuang /* MT6351_AFE_SGEN_CFG0 */
109a74d51baSKai Chieh Chuang #define SGEN_C_MUTE_SW_CTL_BIT 6
110a74d51baSKai Chieh Chuang #define SGEN_C_DAC_EN_CTL_BIT 7
111a74d51baSKai Chieh Chuang 
112a74d51baSKai Chieh Chuang /* MT6351_AFE_NCP_CFG0 */
113a74d51baSKai Chieh Chuang #define RG_NCP_ON_BIT 0
114a74d51baSKai Chieh Chuang 
115a74d51baSKai Chieh Chuang /* MT6351_LDO_VUSB33_CON0 */
116a74d51baSKai Chieh Chuang #define RG_VUSB33_EN 1
117a74d51baSKai Chieh Chuang #define RG_VUSB33_ON_CTRL 3
118a74d51baSKai Chieh Chuang 
119a74d51baSKai Chieh Chuang /* MT6351_LDO_VA18_CON0 */
120a74d51baSKai Chieh Chuang #define RG_VA18_EN 1
121a74d51baSKai Chieh Chuang #define RG_VA18_ON_CTRL 3
122a74d51baSKai Chieh Chuang 
123a74d51baSKai Chieh Chuang /* MT6351_AUDENC_ANA_CON0 */
124a74d51baSKai Chieh Chuang #define RG_AUDPREAMPLON 0
125a74d51baSKai Chieh Chuang #define RG_AUDPREAMPLDCCEN 1
126a74d51baSKai Chieh Chuang #define RG_AUDPREAMPLDCPRECHARGE 2
127a74d51baSKai Chieh Chuang 
128a74d51baSKai Chieh Chuang #define RG_AUDPREAMPLINPUTSEL_SFT (4)
129a74d51baSKai Chieh Chuang #define RG_AUDPREAMPLINPUTSEL_MASK (0x3)
130a74d51baSKai Chieh Chuang 
131a74d51baSKai Chieh Chuang #define RG_AUDADCLPWRUP 12
132a74d51baSKai Chieh Chuang 
133a74d51baSKai Chieh Chuang #define RG_AUDADCLINPUTSEL_SFT (13)
134a74d51baSKai Chieh Chuang #define RG_AUDADCLINPUTSEL_MASK (0x3)
135a74d51baSKai Chieh Chuang 
136a74d51baSKai Chieh Chuang /* MT6351_AUDENC_ANA_CON1 */
137a74d51baSKai Chieh Chuang #define RG_AUDPREAMPRON 0
138a74d51baSKai Chieh Chuang #define RG_AUDPREAMPRDCCEN 1
139a74d51baSKai Chieh Chuang #define RG_AUDPREAMPRDCPRECHARGE 2
140a74d51baSKai Chieh Chuang 
141a74d51baSKai Chieh Chuang #define RG_AUDPREAMPRINPUTSEL_SFT (4)
142a74d51baSKai Chieh Chuang #define RG_AUDPREAMPRINPUTSEL_MASK (0x3)
143a74d51baSKai Chieh Chuang 
144a74d51baSKai Chieh Chuang #define RG_AUDADCRPWRUP 12
145a74d51baSKai Chieh Chuang 
146a74d51baSKai Chieh Chuang #define RG_AUDADCRINPUTSEL_SFT (13)
147a74d51baSKai Chieh Chuang #define RG_AUDADCRINPUTSEL_MASK (0x3)
148a74d51baSKai Chieh Chuang 
149a74d51baSKai Chieh Chuang /* MT6351_AUDENC_ANA_CON3 */
150a74d51baSKai Chieh Chuang #define RG_AUDADCCLKRSTB 6
151a74d51baSKai Chieh Chuang 
152a74d51baSKai Chieh Chuang /* MT6351_AUDENC_ANA_CON9 */
153a74d51baSKai Chieh Chuang #define RG_AUDPWDBMICBIAS0 0
154a74d51baSKai Chieh Chuang #define RG_AUDMICBIAS0VREF 4
155a74d51baSKai Chieh Chuang #define RG_AUDMICBIAS0LOWPEN 7
156a74d51baSKai Chieh Chuang 
157a74d51baSKai Chieh Chuang #define RG_AUDPWDBMICBIAS2 8
158a74d51baSKai Chieh Chuang #define RG_AUDMICBIAS2VREF 12
159a74d51baSKai Chieh Chuang #define RG_AUDMICBIAS2LOWPEN 15
160a74d51baSKai Chieh Chuang 
161a74d51baSKai Chieh Chuang /* MT6351_AUDENC_ANA_CON10 */
162a74d51baSKai Chieh Chuang #define RG_AUDPWDBMICBIAS1 0
163a74d51baSKai Chieh Chuang #define RG_AUDMICBIAS1DCSW1NEN 2
164a74d51baSKai Chieh Chuang #define RG_AUDMICBIAS1VREF 4
165a74d51baSKai Chieh Chuang #define RG_AUDMICBIAS1LOWPEN 7
166a74d51baSKai Chieh Chuang 
167a74d51baSKai Chieh Chuang enum {
168a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_HSOUTL,
169a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_HSOUTR,
170a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_HPOUTL,
171a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_HPOUTR,
172a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_LINEOUTL,
173a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_LINEOUTR,
174a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_MICAMP1,
175a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_MICAMP2,
176a74d51baSKai Chieh Chuang 	AUDIO_ANALOG_VOLUME_TYPE_MAX
177a74d51baSKai Chieh Chuang };
178a74d51baSKai Chieh Chuang 
179a74d51baSKai Chieh Chuang /* Supply subseq */
180a74d51baSKai Chieh Chuang enum {
181a74d51baSKai Chieh Chuang 	SUPPLY_SUBSEQ_SETTING,
182a74d51baSKai Chieh Chuang 	SUPPLY_SUBSEQ_ENABLE,
183a74d51baSKai Chieh Chuang 	SUPPLY_SUBSEQ_MICBIAS,
184a74d51baSKai Chieh Chuang };
185a74d51baSKai Chieh Chuang 
186a74d51baSKai Chieh Chuang #define REG_STRIDE 2
187a74d51baSKai Chieh Chuang 
188a74d51baSKai Chieh Chuang struct mt6351_priv {
189a74d51baSKai Chieh Chuang 	struct device *dev;
190a74d51baSKai Chieh Chuang 	struct regmap *regmap;
191a74d51baSKai Chieh Chuang 
192a74d51baSKai Chieh Chuang 	unsigned int dl_rate;
193a74d51baSKai Chieh Chuang 	unsigned int ul_rate;
194a74d51baSKai Chieh Chuang 
195a74d51baSKai Chieh Chuang 	int ana_gain[AUDIO_ANALOG_VOLUME_TYPE_MAX];
196a74d51baSKai Chieh Chuang 
197a74d51baSKai Chieh Chuang 	int hp_en_counter;
198a74d51baSKai Chieh Chuang };
199a74d51baSKai Chieh Chuang 
set_hp_gain_zero(struct snd_soc_component * cmpnt)200a74d51baSKai Chieh Chuang static void set_hp_gain_zero(struct snd_soc_component *cmpnt)
201a74d51baSKai Chieh Chuang {
202a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_ZCD_CON2,
203a74d51baSKai Chieh Chuang 			   0x1f << 7, 0x8 << 7);
204a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_ZCD_CON2,
205a74d51baSKai Chieh Chuang 			   0x1f << 0, 0x8 << 0);
206a74d51baSKai Chieh Chuang }
207a74d51baSKai Chieh Chuang 
get_cap_reg_val(struct snd_soc_component * cmpnt,unsigned int rate)208a74d51baSKai Chieh Chuang static unsigned int get_cap_reg_val(struct snd_soc_component *cmpnt,
209a74d51baSKai Chieh Chuang 				    unsigned int rate)
210a74d51baSKai Chieh Chuang {
211a74d51baSKai Chieh Chuang 	switch (rate) {
212a74d51baSKai Chieh Chuang 	case 8000:
213a74d51baSKai Chieh Chuang 		return 0;
214a74d51baSKai Chieh Chuang 	case 16000:
215a74d51baSKai Chieh Chuang 		return 1;
216a74d51baSKai Chieh Chuang 	case 32000:
217a74d51baSKai Chieh Chuang 		return 2;
218a74d51baSKai Chieh Chuang 	case 48000:
219a74d51baSKai Chieh Chuang 		return 3;
220a74d51baSKai Chieh Chuang 	case 96000:
221a74d51baSKai Chieh Chuang 		return 4;
222a74d51baSKai Chieh Chuang 	case 192000:
223a74d51baSKai Chieh Chuang 		return 5;
224a74d51baSKai Chieh Chuang 	default:
225a74d51baSKai Chieh Chuang 		dev_warn(cmpnt->dev, "%s(), error rate %d, return 3",
226a74d51baSKai Chieh Chuang 			 __func__, rate);
227a74d51baSKai Chieh Chuang 		return 3;
228a74d51baSKai Chieh Chuang 	}
229a74d51baSKai Chieh Chuang }
230a74d51baSKai Chieh Chuang 
get_play_reg_val(struct snd_soc_component * cmpnt,unsigned int rate)231a74d51baSKai Chieh Chuang static unsigned int get_play_reg_val(struct snd_soc_component *cmpnt,
232a74d51baSKai Chieh Chuang 				     unsigned int rate)
233a74d51baSKai Chieh Chuang {
234a74d51baSKai Chieh Chuang 	switch (rate) {
235a74d51baSKai Chieh Chuang 	case 8000:
236a74d51baSKai Chieh Chuang 		return 0;
237a74d51baSKai Chieh Chuang 	case 11025:
238a74d51baSKai Chieh Chuang 		return 1;
239a74d51baSKai Chieh Chuang 	case 12000:
240a74d51baSKai Chieh Chuang 		return 2;
241a74d51baSKai Chieh Chuang 	case 16000:
242a74d51baSKai Chieh Chuang 		return 3;
243a74d51baSKai Chieh Chuang 	case 22050:
244a74d51baSKai Chieh Chuang 		return 4;
245a74d51baSKai Chieh Chuang 	case 24000:
246a74d51baSKai Chieh Chuang 		return 5;
247a74d51baSKai Chieh Chuang 	case 32000:
248a74d51baSKai Chieh Chuang 		return 6;
249a74d51baSKai Chieh Chuang 	case 44100:
250a74d51baSKai Chieh Chuang 		return 7;
251a74d51baSKai Chieh Chuang 	case 48000:
252a74d51baSKai Chieh Chuang 	case 96000:
253a74d51baSKai Chieh Chuang 	case 192000:
254a74d51baSKai Chieh Chuang 		return 8;
255a74d51baSKai Chieh Chuang 	default:
256a74d51baSKai Chieh Chuang 		dev_warn(cmpnt->dev, "%s(), error rate %d, return 8",
257a74d51baSKai Chieh Chuang 			 __func__, rate);
258a74d51baSKai Chieh Chuang 		return 8;
259a74d51baSKai Chieh Chuang 	}
260a74d51baSKai Chieh Chuang }
261a74d51baSKai Chieh Chuang 
mt6351_codec_dai_hw_params(struct snd_pcm_substream * substream,struct snd_pcm_hw_params * params,struct snd_soc_dai * dai)262a74d51baSKai Chieh Chuang static int mt6351_codec_dai_hw_params(struct snd_pcm_substream *substream,
263a74d51baSKai Chieh Chuang 				      struct snd_pcm_hw_params *params,
264a74d51baSKai Chieh Chuang 				      struct snd_soc_dai *dai)
265a74d51baSKai Chieh Chuang {
266a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = dai->component;
267a74d51baSKai Chieh Chuang 	struct mt6351_priv *priv = snd_soc_component_get_drvdata(cmpnt);
268a74d51baSKai Chieh Chuang 	unsigned int rate = params_rate(params);
269a74d51baSKai Chieh Chuang 
270a74d51baSKai Chieh Chuang 	dev_dbg(priv->dev, "%s(), substream->stream %d, rate %d\n",
271a74d51baSKai Chieh Chuang 		__func__, substream->stream, rate);
272a74d51baSKai Chieh Chuang 
273a74d51baSKai Chieh Chuang 	if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK)
274a74d51baSKai Chieh Chuang 		priv->dl_rate = rate;
275a74d51baSKai Chieh Chuang 	else if (substream->stream == SNDRV_PCM_STREAM_CAPTURE)
276a74d51baSKai Chieh Chuang 		priv->ul_rate = rate;
277a74d51baSKai Chieh Chuang 
278a74d51baSKai Chieh Chuang 	return 0;
279a74d51baSKai Chieh Chuang }
280a74d51baSKai Chieh Chuang 
281a74d51baSKai Chieh Chuang static const struct snd_soc_dai_ops mt6351_codec_dai_ops = {
282a74d51baSKai Chieh Chuang 	.hw_params = mt6351_codec_dai_hw_params,
283a74d51baSKai Chieh Chuang };
284a74d51baSKai Chieh Chuang 
285*39723d34SCharles Keepax #define MT6351_FORMATS (SNDRV_PCM_FMTBIT_S16_LE | SNDRV_PCM_FMTBIT_U16_LE |\
286*39723d34SCharles Keepax 			SNDRV_PCM_FMTBIT_S24_LE | SNDRV_PCM_FMTBIT_U24_LE |\
287*39723d34SCharles Keepax 			SNDRV_PCM_FMTBIT_S32_LE | SNDRV_PCM_FMTBIT_U32_LE)
288a74d51baSKai Chieh Chuang 
289a74d51baSKai Chieh Chuang static struct snd_soc_dai_driver mt6351_dai_driver[] = {
290a74d51baSKai Chieh Chuang 	{
291a74d51baSKai Chieh Chuang 		.name = "mt6351-snd-codec-aif1",
292a74d51baSKai Chieh Chuang 		.playback = {
293a74d51baSKai Chieh Chuang 			.stream_name = "AIF1 Playback",
294a74d51baSKai Chieh Chuang 			.channels_min = 1,
295a74d51baSKai Chieh Chuang 			.channels_max = 2,
296a74d51baSKai Chieh Chuang 			.rates = SNDRV_PCM_RATE_8000_48000 |
297a74d51baSKai Chieh Chuang 				 SNDRV_PCM_RATE_96000 |
298a74d51baSKai Chieh Chuang 				 SNDRV_PCM_RATE_192000,
299a74d51baSKai Chieh Chuang 			.formats = MT6351_FORMATS,
300a74d51baSKai Chieh Chuang 		},
301a74d51baSKai Chieh Chuang 		.capture = {
302a74d51baSKai Chieh Chuang 			.stream_name = "AIF1 Capture",
303a74d51baSKai Chieh Chuang 			.channels_min = 1,
304a74d51baSKai Chieh Chuang 			.channels_max = 2,
305a74d51baSKai Chieh Chuang 			.rates = SNDRV_PCM_RATE_8000 |
306a74d51baSKai Chieh Chuang 				 SNDRV_PCM_RATE_16000 |
307a74d51baSKai Chieh Chuang 				 SNDRV_PCM_RATE_32000 |
308a74d51baSKai Chieh Chuang 				 SNDRV_PCM_RATE_48000 |
309a74d51baSKai Chieh Chuang 				 SNDRV_PCM_RATE_96000 |
310a74d51baSKai Chieh Chuang 				 SNDRV_PCM_RATE_192000,
311a74d51baSKai Chieh Chuang 			.formats = MT6351_FORMATS,
312a74d51baSKai Chieh Chuang 		},
313a74d51baSKai Chieh Chuang 		.ops = &mt6351_codec_dai_ops,
314a74d51baSKai Chieh Chuang 	},
315a74d51baSKai Chieh Chuang };
316a74d51baSKai Chieh Chuang 
317a74d51baSKai Chieh Chuang enum {
318a74d51baSKai Chieh Chuang 	HP_GAIN_SET_ZERO,
319a74d51baSKai Chieh Chuang 	HP_GAIN_RESTORE,
320a74d51baSKai Chieh Chuang };
321a74d51baSKai Chieh Chuang 
hp_gain_ramp_set(struct snd_soc_component * cmpnt,int hp_gain_ctl)322a74d51baSKai Chieh Chuang static void hp_gain_ramp_set(struct snd_soc_component *cmpnt, int hp_gain_ctl)
323a74d51baSKai Chieh Chuang {
324a74d51baSKai Chieh Chuang 	struct mt6351_priv *priv = snd_soc_component_get_drvdata(cmpnt);
325a74d51baSKai Chieh Chuang 	int idx, old_idx, offset, reg_idx;
326a74d51baSKai Chieh Chuang 
327a74d51baSKai Chieh Chuang 	if (hp_gain_ctl == HP_GAIN_SET_ZERO) {
328a74d51baSKai Chieh Chuang 		idx = 8;	/* 0dB */
329a74d51baSKai Chieh Chuang 		old_idx = priv->ana_gain[AUDIO_ANALOG_VOLUME_HPOUTL];
330a74d51baSKai Chieh Chuang 	} else {
331a74d51baSKai Chieh Chuang 		idx = priv->ana_gain[AUDIO_ANALOG_VOLUME_HPOUTL];
332a74d51baSKai Chieh Chuang 		old_idx = 8;	/* 0dB */
333a74d51baSKai Chieh Chuang 	}
334a74d51baSKai Chieh Chuang 	dev_dbg(priv->dev, "%s(), idx %d, old_idx %d\n",
335a74d51baSKai Chieh Chuang 		__func__, idx, old_idx);
336a74d51baSKai Chieh Chuang 
337a74d51baSKai Chieh Chuang 	if (idx > old_idx)
338a74d51baSKai Chieh Chuang 		offset = idx - old_idx;
339a74d51baSKai Chieh Chuang 	else
340a74d51baSKai Chieh Chuang 		offset = old_idx - idx;
341a74d51baSKai Chieh Chuang 
342a74d51baSKai Chieh Chuang 	reg_idx = old_idx;
343a74d51baSKai Chieh Chuang 
344a74d51baSKai Chieh Chuang 	while (offset > 0) {
345a74d51baSKai Chieh Chuang 		reg_idx = idx > old_idx ? reg_idx + 1 : reg_idx - 1;
346a74d51baSKai Chieh Chuang 
347a74d51baSKai Chieh Chuang 		/* check valid range, and set value */
348a74d51baSKai Chieh Chuang 		if ((reg_idx >= 0 && reg_idx <= 0x12) || reg_idx == 0x1f) {
349a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
350a74d51baSKai Chieh Chuang 					   MT6351_ZCD_CON2,
351a74d51baSKai Chieh Chuang 					   0xf9f,
352a74d51baSKai Chieh Chuang 					   (reg_idx << 7) | reg_idx);
353a74d51baSKai Chieh Chuang 			usleep_range(100, 120);
354a74d51baSKai Chieh Chuang 		}
355a74d51baSKai Chieh Chuang 		offset--;
356a74d51baSKai Chieh Chuang 	}
357a74d51baSKai Chieh Chuang }
358a74d51baSKai Chieh Chuang 
hp_zcd_enable(struct snd_soc_component * cmpnt)359a74d51baSKai Chieh Chuang static void hp_zcd_enable(struct snd_soc_component *cmpnt)
360a74d51baSKai Chieh Chuang {
361a74d51baSKai Chieh Chuang 	/* Enable ZCD, for minimize pop noise */
362a74d51baSKai Chieh Chuang 	/* when adjust gain during HP buffer on */
363a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_ZCD_CON0, 0x7 << 8, 0x1 << 8);
364a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_ZCD_CON0, 0x1 << 7, 0x0 << 7);
365a74d51baSKai Chieh Chuang 
366a74d51baSKai Chieh Chuang 	/* timeout, 1=5ms, 0=30ms */
367a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_ZCD_CON0, 0x1 << 6, 0x1 << 6);
368a74d51baSKai Chieh Chuang 
369a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_ZCD_CON0, 0x3 << 4, 0x0 << 4);
370a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_ZCD_CON0, 0x7 << 1, 0x5 << 1);
371a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_ZCD_CON0, 0x1 << 0, 0x1 << 0);
372a74d51baSKai Chieh Chuang }
373a74d51baSKai Chieh Chuang 
hp_zcd_disable(struct snd_soc_component * cmpnt)374a74d51baSKai Chieh Chuang static void hp_zcd_disable(struct snd_soc_component *cmpnt)
375a74d51baSKai Chieh Chuang {
376a74d51baSKai Chieh Chuang 	regmap_write(cmpnt->regmap, MT6351_ZCD_CON0, 0x0000);
377a74d51baSKai Chieh Chuang }
378a74d51baSKai Chieh Chuang 
379a74d51baSKai Chieh Chuang static const DECLARE_TLV_DB_SCALE(playback_tlv, -1000, 100, 0);
380a74d51baSKai Chieh Chuang static const DECLARE_TLV_DB_SCALE(pga_tlv, 0, 600, 0);
381a74d51baSKai Chieh Chuang 
382a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new mt6351_snd_controls[] = {
383a74d51baSKai Chieh Chuang 	/* dl pga gain */
384a74d51baSKai Chieh Chuang 	SOC_DOUBLE_TLV("Headphone Volume",
385a74d51baSKai Chieh Chuang 		       MT6351_ZCD_CON2, 0, 7, 0x12, 1,
386a74d51baSKai Chieh Chuang 		       playback_tlv),
387a74d51baSKai Chieh Chuang 	SOC_DOUBLE_TLV("Lineout Volume",
388a74d51baSKai Chieh Chuang 		       MT6351_ZCD_CON1, 0, 7, 0x12, 1,
389a74d51baSKai Chieh Chuang 		       playback_tlv),
390a74d51baSKai Chieh Chuang 	SOC_SINGLE_TLV("Handset Volume",
391a74d51baSKai Chieh Chuang 		       MT6351_ZCD_CON3, 0, 0x12, 1,
392a74d51baSKai Chieh Chuang 		       playback_tlv),
393a74d51baSKai Chieh Chuang        /* ul pga gain */
394a74d51baSKai Chieh Chuang 	SOC_DOUBLE_R_TLV("PGA Volume",
395a74d51baSKai Chieh Chuang 			 MT6351_AUDENC_ANA_CON0, MT6351_AUDENC_ANA_CON1,
396a74d51baSKai Chieh Chuang 			 8, 4, 0,
397a74d51baSKai Chieh Chuang 			 pga_tlv),
398a74d51baSKai Chieh Chuang };
399a74d51baSKai Chieh Chuang 
400a74d51baSKai Chieh Chuang /* MUX */
401a74d51baSKai Chieh Chuang 
402a74d51baSKai Chieh Chuang /* LOL MUX */
403a74d51baSKai Chieh Chuang static const char *const lo_in_mux_map[] = {
404a74d51baSKai Chieh Chuang 	"Open", "Mute", "Playback", "Test Mode",
405a74d51baSKai Chieh Chuang };
406a74d51baSKai Chieh Chuang 
407a74d51baSKai Chieh Chuang static int lo_in_mux_map_value[] = {
408a74d51baSKai Chieh Chuang 	0x0, 0x1, 0x2, 0x3,
409a74d51baSKai Chieh Chuang };
410a74d51baSKai Chieh Chuang 
411a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(lo_in_mux_map_enum,
412a74d51baSKai Chieh Chuang 				  MT6351_AUDDEC_ANA_CON3,
413a74d51baSKai Chieh Chuang 				  RG_AUDLOLMUXINPUTSEL_VAUDP32_SFT,
414a74d51baSKai Chieh Chuang 				  RG_AUDLOLMUXINPUTSEL_VAUDP32_MASK,
415a74d51baSKai Chieh Chuang 				  lo_in_mux_map,
416a74d51baSKai Chieh Chuang 				  lo_in_mux_map_value);
417a74d51baSKai Chieh Chuang 
418a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new lo_in_mux_control =
419a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("In Select", lo_in_mux_map_enum);
420a74d51baSKai Chieh Chuang 
421a74d51baSKai Chieh Chuang /*HP MUX */
422a74d51baSKai Chieh Chuang static const char *const hp_in_mux_map[] = {
423a74d51baSKai Chieh Chuang 	"Open", "LoudSPK Playback", "Audio Playback", "Test Mode",
424a74d51baSKai Chieh Chuang };
425a74d51baSKai Chieh Chuang 
426a74d51baSKai Chieh Chuang static int hp_in_mux_map_value[] = {
427a74d51baSKai Chieh Chuang 	0x0, 0x1, 0x2, 0x3,
428a74d51baSKai Chieh Chuang };
429a74d51baSKai Chieh Chuang 
430a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(hpl_in_mux_map_enum,
431a74d51baSKai Chieh Chuang 				  MT6351_AUDDEC_ANA_CON0,
432a74d51baSKai Chieh Chuang 				  RG_AUDHPLMUXINPUTSEL_VAUDP32_SFT,
433a74d51baSKai Chieh Chuang 				  RG_AUDHPLMUXINPUTSEL_VAUDP32_MASK,
434a74d51baSKai Chieh Chuang 				  hp_in_mux_map,
435a74d51baSKai Chieh Chuang 				  hp_in_mux_map_value);
436a74d51baSKai Chieh Chuang 
437a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new hpl_in_mux_control =
438a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("HPL Select", hpl_in_mux_map_enum);
439a74d51baSKai Chieh Chuang 
440a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(hpr_in_mux_map_enum,
441a74d51baSKai Chieh Chuang 				  MT6351_AUDDEC_ANA_CON0,
442a74d51baSKai Chieh Chuang 				  RG_AUDHPRMUXINPUTSEL_VAUDP32_SFT,
443a74d51baSKai Chieh Chuang 				  RG_AUDHPRMUXINPUTSEL_VAUDP32_MASK,
444a74d51baSKai Chieh Chuang 				  hp_in_mux_map,
445a74d51baSKai Chieh Chuang 				  hp_in_mux_map_value);
446a74d51baSKai Chieh Chuang 
447a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new hpr_in_mux_control =
448a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("HPR Select", hpr_in_mux_map_enum);
449a74d51baSKai Chieh Chuang 
450a74d51baSKai Chieh Chuang /* RCV MUX */
451a74d51baSKai Chieh Chuang static const char *const rcv_in_mux_map[] = {
452a74d51baSKai Chieh Chuang 	"Open", "Mute", "Voice Playback", "Test Mode",
453a74d51baSKai Chieh Chuang };
454a74d51baSKai Chieh Chuang 
455a74d51baSKai Chieh Chuang static int rcv_in_mux_map_value[] = {
456a74d51baSKai Chieh Chuang 	0x0, 0x1, 0x2, 0x3,
457a74d51baSKai Chieh Chuang };
458a74d51baSKai Chieh Chuang 
459a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(rcv_in_mux_map_enum,
460a74d51baSKai Chieh Chuang 				  MT6351_AUDDEC_ANA_CON0,
461a74d51baSKai Chieh Chuang 				  RG_AUDHSMUXINPUTSEL_VAUDP32_SFT,
462a74d51baSKai Chieh Chuang 				  RG_AUDHSMUXINPUTSEL_VAUDP32_MASK,
463a74d51baSKai Chieh Chuang 				  rcv_in_mux_map,
464a74d51baSKai Chieh Chuang 				  rcv_in_mux_map_value);
465a74d51baSKai Chieh Chuang 
466a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new rcv_in_mux_control =
467a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("RCV Select", rcv_in_mux_map_enum);
468a74d51baSKai Chieh Chuang 
469a74d51baSKai Chieh Chuang /* DAC In MUX */
470a74d51baSKai Chieh Chuang static const char *const dac_in_mux_map[] = {
471a74d51baSKai Chieh Chuang 	"Normal Path", "Sgen",
472a74d51baSKai Chieh Chuang };
473a74d51baSKai Chieh Chuang 
474a74d51baSKai Chieh Chuang static int dac_in_mux_map_value[] = {
475a74d51baSKai Chieh Chuang 	0x0, 0x1,
476a74d51baSKai Chieh Chuang };
477a74d51baSKai Chieh Chuang 
478a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(dac_in_mux_map_enum,
479a74d51baSKai Chieh Chuang 				  MT6351_AFE_TOP_CON0,
480a74d51baSKai Chieh Chuang 				  RG_DL_SINE_ON_SFT,
481a74d51baSKai Chieh Chuang 				  RG_DL_SINE_ON_MASK,
482a74d51baSKai Chieh Chuang 				  dac_in_mux_map,
483a74d51baSKai Chieh Chuang 				  dac_in_mux_map_value);
484a74d51baSKai Chieh Chuang 
485a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new dac_in_mux_control =
486a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("DAC Select", dac_in_mux_map_enum);
487a74d51baSKai Chieh Chuang 
488a74d51baSKai Chieh Chuang /* AIF Out MUX */
489a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(aif_out_mux_map_enum,
490a74d51baSKai Chieh Chuang 				  MT6351_AFE_TOP_CON0,
491a74d51baSKai Chieh Chuang 				  RG_UL_SINE_ON_SFT,
492a74d51baSKai Chieh Chuang 				  RG_UL_SINE_ON_MASK,
493a74d51baSKai Chieh Chuang 				  dac_in_mux_map,
494a74d51baSKai Chieh Chuang 				  dac_in_mux_map_value);
495a74d51baSKai Chieh Chuang 
496a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new aif_out_mux_control =
497a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("AIF Out Select", aif_out_mux_map_enum);
498a74d51baSKai Chieh Chuang 
499a74d51baSKai Chieh Chuang /* ADC L MUX */
500a74d51baSKai Chieh Chuang static const char *const adc_left_mux_map[] = {
501a74d51baSKai Chieh Chuang 	"Idle", "AIN0", "Left Preamplifier", "Idle_1",
502a74d51baSKai Chieh Chuang };
503a74d51baSKai Chieh Chuang 
504a74d51baSKai Chieh Chuang static int adc_left_mux_map_value[] = {
505a74d51baSKai Chieh Chuang 	0x0, 0x1, 0x2, 0x3,
506a74d51baSKai Chieh Chuang };
507a74d51baSKai Chieh Chuang 
508a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(adc_left_mux_map_enum,
509a74d51baSKai Chieh Chuang 				  MT6351_AUDENC_ANA_CON0,
510a74d51baSKai Chieh Chuang 				  RG_AUDADCLINPUTSEL_SFT,
511a74d51baSKai Chieh Chuang 				  RG_AUDADCLINPUTSEL_MASK,
512a74d51baSKai Chieh Chuang 				  adc_left_mux_map,
513a74d51baSKai Chieh Chuang 				  adc_left_mux_map_value);
514a74d51baSKai Chieh Chuang 
515a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new adc_left_mux_control =
516a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("ADC L Select", adc_left_mux_map_enum);
517a74d51baSKai Chieh Chuang 
518a74d51baSKai Chieh Chuang /* ADC R MUX */
519a74d51baSKai Chieh Chuang static const char *const adc_right_mux_map[] = {
520a74d51baSKai Chieh Chuang 	"Idle", "AIN0", "Right Preamplifier", "Idle_1",
521a74d51baSKai Chieh Chuang };
522a74d51baSKai Chieh Chuang 
523a74d51baSKai Chieh Chuang static int adc_right_mux_map_value[] = {
524a74d51baSKai Chieh Chuang 	0x0, 0x1, 0x2, 0x3,
525a74d51baSKai Chieh Chuang };
526a74d51baSKai Chieh Chuang 
527a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(adc_right_mux_map_enum,
528a74d51baSKai Chieh Chuang 				  MT6351_AUDENC_ANA_CON1,
529a74d51baSKai Chieh Chuang 				  RG_AUDADCRINPUTSEL_SFT,
530a74d51baSKai Chieh Chuang 				  RG_AUDADCRINPUTSEL_MASK,
531a74d51baSKai Chieh Chuang 				  adc_right_mux_map,
532a74d51baSKai Chieh Chuang 				  adc_right_mux_map_value);
533a74d51baSKai Chieh Chuang 
534a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new adc_right_mux_control =
535a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("ADC R Select", adc_right_mux_map_enum);
536a74d51baSKai Chieh Chuang 
537a74d51baSKai Chieh Chuang /* PGA L MUX */
538a74d51baSKai Chieh Chuang static const char *const pga_left_mux_map[] = {
539a74d51baSKai Chieh Chuang 	"None", "AIN0", "AIN1", "AIN2",
540a74d51baSKai Chieh Chuang };
541a74d51baSKai Chieh Chuang 
542a74d51baSKai Chieh Chuang static int pga_left_mux_map_value[] = {
543a74d51baSKai Chieh Chuang 	0x0, 0x1, 0x2, 0x3,
544a74d51baSKai Chieh Chuang };
545a74d51baSKai Chieh Chuang 
546a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(pga_left_mux_map_enum,
547a74d51baSKai Chieh Chuang 				  MT6351_AUDENC_ANA_CON0,
548a74d51baSKai Chieh Chuang 				  RG_AUDPREAMPLINPUTSEL_SFT,
549a74d51baSKai Chieh Chuang 				  RG_AUDPREAMPLINPUTSEL_MASK,
550a74d51baSKai Chieh Chuang 				  pga_left_mux_map,
551a74d51baSKai Chieh Chuang 				  pga_left_mux_map_value);
552a74d51baSKai Chieh Chuang 
553a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new pga_left_mux_control =
554a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("PGA L Select", pga_left_mux_map_enum);
555a74d51baSKai Chieh Chuang 
556a74d51baSKai Chieh Chuang /* PGA R MUX */
557a74d51baSKai Chieh Chuang static const char *const pga_right_mux_map[] = {
558a74d51baSKai Chieh Chuang 	"None", "AIN0", "AIN3", "AIN2",
559a74d51baSKai Chieh Chuang };
560a74d51baSKai Chieh Chuang 
561a74d51baSKai Chieh Chuang static int pga_right_mux_map_value[] = {
562a74d51baSKai Chieh Chuang 	0x0, 0x1, 0x2, 0x3,
563a74d51baSKai Chieh Chuang };
564a74d51baSKai Chieh Chuang 
565a74d51baSKai Chieh Chuang static SOC_VALUE_ENUM_SINGLE_DECL(pga_right_mux_map_enum,
566a74d51baSKai Chieh Chuang 				  MT6351_AUDENC_ANA_CON1,
567a74d51baSKai Chieh Chuang 				  RG_AUDPREAMPRINPUTSEL_SFT,
568a74d51baSKai Chieh Chuang 				  RG_AUDPREAMPRINPUTSEL_MASK,
569a74d51baSKai Chieh Chuang 				  pga_right_mux_map,
570a74d51baSKai Chieh Chuang 				  pga_right_mux_map_value);
571a74d51baSKai Chieh Chuang 
572a74d51baSKai Chieh Chuang static const struct snd_kcontrol_new pga_right_mux_control =
573a74d51baSKai Chieh Chuang 	SOC_DAPM_ENUM("PGA R Select", pga_right_mux_map_enum);
574a74d51baSKai Chieh Chuang 
mt_reg_set_clr_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)575a74d51baSKai Chieh Chuang static int mt_reg_set_clr_event(struct snd_soc_dapm_widget *w,
576a74d51baSKai Chieh Chuang 				struct snd_kcontrol *kcontrol,
577a74d51baSKai Chieh Chuang 				int event)
578a74d51baSKai Chieh Chuang {
579a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
580a74d51baSKai Chieh Chuang 
581a74d51baSKai Chieh Chuang 	switch (event) {
582a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMU:
583a74d51baSKai Chieh Chuang 		if (w->on_val) {
584a74d51baSKai Chieh Chuang 			/* SET REG */
585a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
586a74d51baSKai Chieh Chuang 					   w->reg + REG_STRIDE,
587a74d51baSKai Chieh Chuang 					   0x1 << w->shift,
588a74d51baSKai Chieh Chuang 					   0x1 << w->shift);
589a74d51baSKai Chieh Chuang 		} else {
590a74d51baSKai Chieh Chuang 			/* CLR REG */
591a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
592a74d51baSKai Chieh Chuang 					   w->reg + REG_STRIDE * 2,
593a74d51baSKai Chieh Chuang 					   0x1 << w->shift,
594a74d51baSKai Chieh Chuang 					   0x1 << w->shift);
595a74d51baSKai Chieh Chuang 		}
596a74d51baSKai Chieh Chuang 		break;
597a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMD:
598a74d51baSKai Chieh Chuang 		if (w->off_val) {
599a74d51baSKai Chieh Chuang 			/* SET REG */
600a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
601a74d51baSKai Chieh Chuang 					   w->reg + REG_STRIDE,
602a74d51baSKai Chieh Chuang 					   0x1 << w->shift,
603a74d51baSKai Chieh Chuang 					   0x1 << w->shift);
604a74d51baSKai Chieh Chuang 		} else {
605a74d51baSKai Chieh Chuang 			/* CLR REG */
606a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
607a74d51baSKai Chieh Chuang 					   w->reg + REG_STRIDE * 2,
608a74d51baSKai Chieh Chuang 					   0x1 << w->shift,
609a74d51baSKai Chieh Chuang 					   0x1 << w->shift);
610a74d51baSKai Chieh Chuang 		}
611a74d51baSKai Chieh Chuang 		break;
612a74d51baSKai Chieh Chuang 	default:
613a74d51baSKai Chieh Chuang 		break;
614a74d51baSKai Chieh Chuang 	}
615a74d51baSKai Chieh Chuang 
616a74d51baSKai Chieh Chuang 	return 0;
617a74d51baSKai Chieh Chuang }
618a74d51baSKai Chieh Chuang 
mt_ncp_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)619a74d51baSKai Chieh Chuang static int mt_ncp_event(struct snd_soc_dapm_widget *w,
620a74d51baSKai Chieh Chuang 			struct snd_kcontrol *kcontrol,
621a74d51baSKai Chieh Chuang 			int event)
622a74d51baSKai Chieh Chuang {
623a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
624a74d51baSKai Chieh Chuang 
625a74d51baSKai Chieh Chuang 	switch (event) {
626a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
627a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_NCP_CFG1,
628a74d51baSKai Chieh Chuang 				   0xffff, 0x1515);
629a74d51baSKai Chieh Chuang 		/* NCP: ck1 and ck2 clock frequecy adjust configure */
630a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_NCP_CFG0,
631a74d51baSKai Chieh Chuang 				   0xfffe, 0x8C00);
632a74d51baSKai Chieh Chuang 		break;
633a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMU:
634a74d51baSKai Chieh Chuang 		usleep_range(250, 270);
635a74d51baSKai Chieh Chuang 		break;
636a74d51baSKai Chieh Chuang 	default:
637a74d51baSKai Chieh Chuang 		break;
638a74d51baSKai Chieh Chuang 	}
639a74d51baSKai Chieh Chuang 
640a74d51baSKai Chieh Chuang 	return 0;
641a74d51baSKai Chieh Chuang }
642a74d51baSKai Chieh Chuang 
mt_sgen_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)643a74d51baSKai Chieh Chuang static int mt_sgen_event(struct snd_soc_dapm_widget *w,
644a74d51baSKai Chieh Chuang 			 struct snd_kcontrol *kcontrol,
645a74d51baSKai Chieh Chuang 			 int event)
646a74d51baSKai Chieh Chuang {
647a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
648a74d51baSKai Chieh Chuang 
649a74d51baSKai Chieh Chuang 	switch (event) {
650a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
651a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_SGEN_CFG0,
652a74d51baSKai Chieh Chuang 				   0xffef, 0x0008);
653a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_SGEN_CFG1,
654a74d51baSKai Chieh Chuang 				   0xffff, 0x0101);
655a74d51baSKai Chieh Chuang 		break;
656a74d51baSKai Chieh Chuang 	default:
657a74d51baSKai Chieh Chuang 		break;
658a74d51baSKai Chieh Chuang 	}
659a74d51baSKai Chieh Chuang 
660a74d51baSKai Chieh Chuang 	return 0;
661a74d51baSKai Chieh Chuang }
662a74d51baSKai Chieh Chuang 
mt_aif_in_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)663a74d51baSKai Chieh Chuang static int mt_aif_in_event(struct snd_soc_dapm_widget *w,
664a74d51baSKai Chieh Chuang 			   struct snd_kcontrol *kcontrol,
665a74d51baSKai Chieh Chuang 			   int event)
666a74d51baSKai Chieh Chuang {
667a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
668a74d51baSKai Chieh Chuang 	struct mt6351_priv *priv = snd_soc_component_get_drvdata(cmpnt);
669a74d51baSKai Chieh Chuang 
670a74d51baSKai Chieh Chuang 	dev_dbg(priv->dev, "%s(), event 0x%x, rate %d\n",
671a74d51baSKai Chieh Chuang 		__func__, event, priv->dl_rate);
672a74d51baSKai Chieh Chuang 
673a74d51baSKai Chieh Chuang 	switch (event) {
674a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
675a74d51baSKai Chieh Chuang 		/* sdm audio fifo clock power on */
676a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFUNC_AUD_CON2,
677a74d51baSKai Chieh Chuang 				   0xffff, 0x0006);
678a74d51baSKai Chieh Chuang 		/* scrambler clock on enable */
679a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFUNC_AUD_CON0,
680a74d51baSKai Chieh Chuang 				   0xffff, 0xC3A1);
681a74d51baSKai Chieh Chuang 		/* sdm power on */
682a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFUNC_AUD_CON2,
683a74d51baSKai Chieh Chuang 				   0xffff, 0x0003);
684a74d51baSKai Chieh Chuang 		/* sdm fifo enable */
685a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFUNC_AUD_CON2,
686a74d51baSKai Chieh Chuang 				   0xffff, 0x000B);
687a74d51baSKai Chieh Chuang 		/* set attenuation gain */
688a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_DL_SDM_CON1,
689a74d51baSKai Chieh Chuang 				   0xffff, 0x001E);
690a74d51baSKai Chieh Chuang 
691a74d51baSKai Chieh Chuang 		regmap_write(cmpnt->regmap, MT6351_AFE_PMIC_NEWIF_CFG0,
692a74d51baSKai Chieh Chuang 			     (get_play_reg_val(cmpnt, priv->dl_rate) << 12) |
693a74d51baSKai Chieh Chuang 			     0x330);
694a74d51baSKai Chieh Chuang 		regmap_write(cmpnt->regmap, MT6351_AFE_DL_SRC2_CON0_H,
695a74d51baSKai Chieh Chuang 			     (get_play_reg_val(cmpnt, priv->dl_rate) << 12) |
696a74d51baSKai Chieh Chuang 			     0x300);
697a74d51baSKai Chieh Chuang 
698a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_PMIC_NEWIF_CFG2,
699a74d51baSKai Chieh Chuang 				   0x8000, 0x8000);
700a74d51baSKai Chieh Chuang 		break;
701a74d51baSKai Chieh Chuang 	default:
702a74d51baSKai Chieh Chuang 		break;
703a74d51baSKai Chieh Chuang 	}
704a74d51baSKai Chieh Chuang 
705a74d51baSKai Chieh Chuang 	return 0;
706a74d51baSKai Chieh Chuang }
707a74d51baSKai Chieh Chuang 
mt_hp_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)708a74d51baSKai Chieh Chuang static int mt_hp_event(struct snd_soc_dapm_widget *w,
709a74d51baSKai Chieh Chuang 		       struct snd_kcontrol *kcontrol,
710a74d51baSKai Chieh Chuang 		       int event)
711a74d51baSKai Chieh Chuang {
712a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
713a74d51baSKai Chieh Chuang 	struct mt6351_priv *priv = snd_soc_component_get_drvdata(cmpnt);
714a74d51baSKai Chieh Chuang 	int reg;
715a74d51baSKai Chieh Chuang 
716a74d51baSKai Chieh Chuang 	dev_dbg(priv->dev, "%s(), event 0x%x, hp_en_counter %d\n",
717a74d51baSKai Chieh Chuang 		__func__, event, priv->hp_en_counter);
718a74d51baSKai Chieh Chuang 
719a74d51baSKai Chieh Chuang 	switch (event) {
720a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
721a74d51baSKai Chieh Chuang 		priv->hp_en_counter++;
722a74d51baSKai Chieh Chuang 		if (priv->hp_en_counter > 1)
723a74d51baSKai Chieh Chuang 			break;	/* already enabled, do nothing */
724a74d51baSKai Chieh Chuang 		else if (priv->hp_en_counter <= 0)
725a74d51baSKai Chieh Chuang 			dev_err(priv->dev, "%s(), hp_en_counter %d <= 0\n",
726a74d51baSKai Chieh Chuang 				__func__,
727a74d51baSKai Chieh Chuang 				priv->hp_en_counter);
728a74d51baSKai Chieh Chuang 
729a74d51baSKai Chieh Chuang 		hp_zcd_disable(cmpnt);
730a74d51baSKai Chieh Chuang 
731a74d51baSKai Chieh Chuang 		/* from yoyo HQA script */
732a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON6,
733a74d51baSKai Chieh Chuang 				   0x0700, 0x0700);
734a74d51baSKai Chieh Chuang 
735a74d51baSKai Chieh Chuang 		/* save target gain to restore after hardware open complete */
736a74d51baSKai Chieh Chuang 		regmap_read(cmpnt->regmap, MT6351_ZCD_CON2, &reg);
737a74d51baSKai Chieh Chuang 		priv->ana_gain[AUDIO_ANALOG_VOLUME_HPOUTL] = reg & 0x1f;
738a74d51baSKai Chieh Chuang 		priv->ana_gain[AUDIO_ANALOG_VOLUME_HPOUTR] = (reg >> 7) & 0x1f;
739a74d51baSKai Chieh Chuang 
740a74d51baSKai Chieh Chuang 		/* Set HPR/HPL gain as minimum (~ -40dB) */
741a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap,
742a74d51baSKai Chieh Chuang 				   MT6351_ZCD_CON2, 0xffff, 0x0F9F);
743a74d51baSKai Chieh Chuang 		/* Set HS gain as minimum (~ -40dB) */
744a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap,
745a74d51baSKai Chieh Chuang 				   MT6351_ZCD_CON3, 0xffff, 0x001F);
746a74d51baSKai Chieh Chuang 		/* De_OSC of HP */
747a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON2,
748a74d51baSKai Chieh Chuang 				   0x0001, 0x0001);
749a74d51baSKai Chieh Chuang 		/* enable output STBENH */
750a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON1,
751a74d51baSKai Chieh Chuang 				   0xffff, 0x2000);
752a74d51baSKai Chieh Chuang 		/* De_OSC of voice, enable output STBENH */
753a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON1,
754a74d51baSKai Chieh Chuang 				   0xffff, 0x2100);
755a74d51baSKai Chieh Chuang 		/* Enable voice driver */
756a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON0,
757a74d51baSKai Chieh Chuang 				   0x0010, 0xE090);
758a74d51baSKai Chieh Chuang 		/* Enable pre-charge buffer  */
759a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON1,
760a74d51baSKai Chieh Chuang 				   0xffff, 0x2140);
761a74d51baSKai Chieh Chuang 
762a74d51baSKai Chieh Chuang 		usleep_range(50, 60);
763a74d51baSKai Chieh Chuang 
764a74d51baSKai Chieh Chuang 		/* Apply digital DC compensation value to DAC */
765a74d51baSKai Chieh Chuang 		set_hp_gain_zero(cmpnt);
766a74d51baSKai Chieh Chuang 
767a74d51baSKai Chieh Chuang 		/* Enable HPR/HPL */
768a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON1,
769a74d51baSKai Chieh Chuang 				   0xffff, 0x2100);
770a74d51baSKai Chieh Chuang 		/* Disable pre-charge buffer */
771a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON1,
772a74d51baSKai Chieh Chuang 				   0xffff, 0x2000);
773a74d51baSKai Chieh Chuang 		/* Disable De_OSC of voice */
774a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON0,
775a74d51baSKai Chieh Chuang 				   0x0010, 0xF4EF);
776a74d51baSKai Chieh Chuang 		/* Disable voice buffer */
777a74d51baSKai Chieh Chuang 
778a74d51baSKai Chieh Chuang 		/* from yoyo HQ */
779a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON6,
780a74d51baSKai Chieh Chuang 				   0x0700, 0x0300);
781a74d51baSKai Chieh Chuang 
782a74d51baSKai Chieh Chuang 		/* Enable ZCD, for minimize pop noise */
783a74d51baSKai Chieh Chuang 		/* when adjust gain during HP buffer on */
784a74d51baSKai Chieh Chuang 		hp_zcd_enable(cmpnt);
785a74d51baSKai Chieh Chuang 
786a74d51baSKai Chieh Chuang 		/* apply volume setting */
787a74d51baSKai Chieh Chuang 		hp_gain_ramp_set(cmpnt, HP_GAIN_RESTORE);
788a74d51baSKai Chieh Chuang 
789a74d51baSKai Chieh Chuang 		break;
790a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMD:
791a74d51baSKai Chieh Chuang 		priv->hp_en_counter--;
792a74d51baSKai Chieh Chuang 		if (priv->hp_en_counter > 0)
793a74d51baSKai Chieh Chuang 			break;	/* still being used, don't close */
794a74d51baSKai Chieh Chuang 		else if (priv->hp_en_counter < 0)
795a74d51baSKai Chieh Chuang 			dev_err(priv->dev, "%s(), hp_en_counter %d <= 0\n",
796a74d51baSKai Chieh Chuang 				__func__,
797a74d51baSKai Chieh Chuang 				priv->hp_en_counter);
798a74d51baSKai Chieh Chuang 
799a74d51baSKai Chieh Chuang 		/* Disable AUD_ZCD */
800a74d51baSKai Chieh Chuang 		hp_zcd_disable(cmpnt);
801a74d51baSKai Chieh Chuang 
802a74d51baSKai Chieh Chuang 		/* Set HPR/HPL gain as -1dB, step by step */
803a74d51baSKai Chieh Chuang 		hp_gain_ramp_set(cmpnt, HP_GAIN_SET_ZERO);
804a74d51baSKai Chieh Chuang 
805a74d51baSKai Chieh Chuang 		set_hp_gain_zero(cmpnt);
806a74d51baSKai Chieh Chuang 		break;
807a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMD:
808a74d51baSKai Chieh Chuang 		if (priv->hp_en_counter > 0)
809a74d51baSKai Chieh Chuang 			break;	/* still being used, don't close */
810a74d51baSKai Chieh Chuang 		else if (priv->hp_en_counter < 0)
811a74d51baSKai Chieh Chuang 			dev_err(priv->dev, "%s(), hp_en_counter %d <= 0\n",
812a74d51baSKai Chieh Chuang 				__func__,
813a74d51baSKai Chieh Chuang 				priv->hp_en_counter);
814a74d51baSKai Chieh Chuang 
815a74d51baSKai Chieh Chuang 		/* reset*/
816a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap,
817a74d51baSKai Chieh Chuang 				   MT6351_AUDDEC_ANA_CON6,
818a74d51baSKai Chieh Chuang 				   0x0700,
819a74d51baSKai Chieh Chuang 				   0x0000);
820a74d51baSKai Chieh Chuang 		/* De_OSC of HP */
821a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap,
822a74d51baSKai Chieh Chuang 				   MT6351_AUDDEC_ANA_CON2,
823a74d51baSKai Chieh Chuang 				   0x0001,
824a74d51baSKai Chieh Chuang 				   0x0000);
825a74d51baSKai Chieh Chuang 
826a74d51baSKai Chieh Chuang 		/* apply volume setting */
827a74d51baSKai Chieh Chuang 		hp_gain_ramp_set(cmpnt, HP_GAIN_RESTORE);
828a74d51baSKai Chieh Chuang 		break;
829a74d51baSKai Chieh Chuang 	default:
830a74d51baSKai Chieh Chuang 		break;
831a74d51baSKai Chieh Chuang 	}
832a74d51baSKai Chieh Chuang 
833a74d51baSKai Chieh Chuang 	return 0;
834a74d51baSKai Chieh Chuang }
835a74d51baSKai Chieh Chuang 
mt_aif_out_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)836a74d51baSKai Chieh Chuang static int mt_aif_out_event(struct snd_soc_dapm_widget *w,
837a74d51baSKai Chieh Chuang 			    struct snd_kcontrol *kcontrol,
838a74d51baSKai Chieh Chuang 			    int event)
839a74d51baSKai Chieh Chuang {
840a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
841a74d51baSKai Chieh Chuang 	struct mt6351_priv *priv = snd_soc_component_get_drvdata(cmpnt);
842a74d51baSKai Chieh Chuang 
843a74d51baSKai Chieh Chuang 	dev_dbg(priv->dev, "%s(), event 0x%x, rate %d\n",
844a74d51baSKai Chieh Chuang 		__func__, event, priv->ul_rate);
845a74d51baSKai Chieh Chuang 
846a74d51baSKai Chieh Chuang 	switch (event) {
847a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
848a74d51baSKai Chieh Chuang 		/* dcclk_div=11'b00100000011, dcclk_ref_ck_sel=2'b00 */
849a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_DCCLK_CFG0,
850a74d51baSKai Chieh Chuang 				   0xffff, 0x2062);
851a74d51baSKai Chieh Chuang 		/* dcclk_pdn=1'b0 */
852a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_DCCLK_CFG0,
853a74d51baSKai Chieh Chuang 				   0xffff, 0x2060);
854a74d51baSKai Chieh Chuang 		/* dcclk_gen_on=1'b1 */
855a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_DCCLK_CFG0,
856a74d51baSKai Chieh Chuang 				   0xffff, 0x2061);
857a74d51baSKai Chieh Chuang 
858a74d51baSKai Chieh Chuang 		/* UL sample rate and mode configure */
859a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AFE_UL_SRC_CON0_H,
860a74d51baSKai Chieh Chuang 				   0x000E,
861a74d51baSKai Chieh Chuang 				   get_cap_reg_val(cmpnt, priv->ul_rate) << 1);
862a74d51baSKai Chieh Chuang 
863a74d51baSKai Chieh Chuang 		/* fixed 260k path for 8/16/32/48 */
864a74d51baSKai Chieh Chuang 		if (priv->ul_rate <= 48000) {
865a74d51baSKai Chieh Chuang 			/* anc ul path src on */
866a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
867a74d51baSKai Chieh Chuang 					   MT6351_AFE_HPANC_CFG0,
868a74d51baSKai Chieh Chuang 					   0x1 << 1,
869a74d51baSKai Chieh Chuang 					   0x1 << 1);
870a74d51baSKai Chieh Chuang 			/* ANC clk pdn release */
871a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
872a74d51baSKai Chieh Chuang 					   MT6351_AFE_HPANC_CFG0,
873a74d51baSKai Chieh Chuang 					   0x1 << 0,
874a74d51baSKai Chieh Chuang 					   0x0 << 0);
875a74d51baSKai Chieh Chuang 		}
876a74d51baSKai Chieh Chuang 		break;
877a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMD:
878a74d51baSKai Chieh Chuang 		/* fixed 260k path for 8/16/32/48 */
879a74d51baSKai Chieh Chuang 		if (priv->ul_rate <= 48000) {
880a74d51baSKai Chieh Chuang 			/* anc ul path src on */
881a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
882a74d51baSKai Chieh Chuang 					   MT6351_AFE_HPANC_CFG0,
883a74d51baSKai Chieh Chuang 					   0x1 << 1,
884a74d51baSKai Chieh Chuang 					   0x0 << 1);
885a74d51baSKai Chieh Chuang 			/* ANC clk pdn release */
886a74d51baSKai Chieh Chuang 			regmap_update_bits(cmpnt->regmap,
887a74d51baSKai Chieh Chuang 					   MT6351_AFE_HPANC_CFG0,
888a74d51baSKai Chieh Chuang 					   0x1 << 0,
889a74d51baSKai Chieh Chuang 					   0x1 << 0);
890a74d51baSKai Chieh Chuang 		}
891a74d51baSKai Chieh Chuang 		break;
892a74d51baSKai Chieh Chuang 	default:
893a74d51baSKai Chieh Chuang 		break;
894a74d51baSKai Chieh Chuang 	}
895a74d51baSKai Chieh Chuang 
896a74d51baSKai Chieh Chuang 	return 0;
897a74d51baSKai Chieh Chuang }
898a74d51baSKai Chieh Chuang 
mt_adc_clkgen_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)899a74d51baSKai Chieh Chuang static int mt_adc_clkgen_event(struct snd_soc_dapm_widget *w,
900a74d51baSKai Chieh Chuang 			       struct snd_kcontrol *kcontrol,
901a74d51baSKai Chieh Chuang 			       int event)
902a74d51baSKai Chieh Chuang {
903a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
904a74d51baSKai Chieh Chuang 
905a74d51baSKai Chieh Chuang 	switch (event) {
906a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
907a74d51baSKai Chieh Chuang 		/* Audio ADC clock gen. mode: 00_divided by 2 (Normal) */
908a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON3,
909a74d51baSKai Chieh Chuang 				   0x3 << 4, 0x0);
910a74d51baSKai Chieh Chuang 		break;
911a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMU:
912a74d51baSKai Chieh Chuang 		/* ADC CLK from: 00_13MHz from CLKSQ (Default) */
913a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON3,
914a74d51baSKai Chieh Chuang 				   0x3 << 2, 0x0);
915a74d51baSKai Chieh Chuang 		break;
916a74d51baSKai Chieh Chuang 	default:
917a74d51baSKai Chieh Chuang 		break;
918a74d51baSKai Chieh Chuang 	}
919a74d51baSKai Chieh Chuang 	return 0;
920a74d51baSKai Chieh Chuang }
921a74d51baSKai Chieh Chuang 
mt_pga_left_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)922a74d51baSKai Chieh Chuang static int mt_pga_left_event(struct snd_soc_dapm_widget *w,
923a74d51baSKai Chieh Chuang 			     struct snd_kcontrol *kcontrol,
924a74d51baSKai Chieh Chuang 			     int event)
925a74d51baSKai Chieh Chuang {
926a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
927a74d51baSKai Chieh Chuang 
928a74d51baSKai Chieh Chuang 	switch (event) {
929a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
930a74d51baSKai Chieh Chuang 		/* Audio L PGA precharge on */
931a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON0,
932a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDPREAMPLDCPRECHARGE,
933a74d51baSKai Chieh Chuang 				   0x1 << RG_AUDPREAMPLDCPRECHARGE);
934a74d51baSKai Chieh Chuang 		/* Audio L PGA mode: 1_DCC */
935a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON0,
936a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDPREAMPLDCCEN,
937a74d51baSKai Chieh Chuang 				   0x1 << RG_AUDPREAMPLDCCEN);
938a74d51baSKai Chieh Chuang 		break;
939a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMU:
940a74d51baSKai Chieh Chuang 		usleep_range(100, 120);
941a74d51baSKai Chieh Chuang 		/* Audio L PGA precharge off */
942a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON0,
943a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDPREAMPLDCPRECHARGE,
944a74d51baSKai Chieh Chuang 				   0x0 << RG_AUDPREAMPLDCPRECHARGE);
945a74d51baSKai Chieh Chuang 		break;
946a74d51baSKai Chieh Chuang 	default:
947a74d51baSKai Chieh Chuang 		break;
948a74d51baSKai Chieh Chuang 	}
949a74d51baSKai Chieh Chuang 	return 0;
950a74d51baSKai Chieh Chuang }
951a74d51baSKai Chieh Chuang 
mt_pga_right_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)952a74d51baSKai Chieh Chuang static int mt_pga_right_event(struct snd_soc_dapm_widget *w,
953a74d51baSKai Chieh Chuang 			      struct snd_kcontrol *kcontrol,
954a74d51baSKai Chieh Chuang 			      int event)
955a74d51baSKai Chieh Chuang {
956a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
957a74d51baSKai Chieh Chuang 
958a74d51baSKai Chieh Chuang 	switch (event) {
959a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
960a74d51baSKai Chieh Chuang 		/* Audio R PGA precharge on */
961a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON1,
962a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDPREAMPRDCPRECHARGE,
963a74d51baSKai Chieh Chuang 				   0x1 << RG_AUDPREAMPRDCPRECHARGE);
964a74d51baSKai Chieh Chuang 		/* Audio R PGA mode: 1_DCC */
965a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON1,
966a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDPREAMPRDCCEN,
967a74d51baSKai Chieh Chuang 				   0x1 << RG_AUDPREAMPRDCCEN);
968a74d51baSKai Chieh Chuang 		break;
969a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMU:
970a74d51baSKai Chieh Chuang 		usleep_range(100, 120);
971a74d51baSKai Chieh Chuang 		/* Audio R PGA precharge off */
972a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON1,
973a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDPREAMPRDCPRECHARGE,
974a74d51baSKai Chieh Chuang 				   0x0 << RG_AUDPREAMPRDCPRECHARGE);
975a74d51baSKai Chieh Chuang 		break;
976a74d51baSKai Chieh Chuang 	default:
977a74d51baSKai Chieh Chuang 		break;
978a74d51baSKai Chieh Chuang 	}
979a74d51baSKai Chieh Chuang 	return 0;
980a74d51baSKai Chieh Chuang }
981a74d51baSKai Chieh Chuang 
mt_mic_bias_0_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)982a74d51baSKai Chieh Chuang static int mt_mic_bias_0_event(struct snd_soc_dapm_widget *w,
983a74d51baSKai Chieh Chuang 			       struct snd_kcontrol *kcontrol,
984a74d51baSKai Chieh Chuang 			       int event)
985a74d51baSKai Chieh Chuang {
986a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
987a74d51baSKai Chieh Chuang 
988a74d51baSKai Chieh Chuang 	switch (event) {
989a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
990a74d51baSKai Chieh Chuang 		/* MIC Bias 0 LowPower: 0_Normal */
991a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON9,
992a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDMICBIAS0LOWPEN, 0x0);
993a74d51baSKai Chieh Chuang 		/* MISBIAS0 = 1P9V */
994a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON9,
995a74d51baSKai Chieh Chuang 				   0x7 << RG_AUDMICBIAS0VREF,
996a74d51baSKai Chieh Chuang 				   0x2 << RG_AUDMICBIAS0VREF);
997a74d51baSKai Chieh Chuang 		break;
998a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMD:
999a74d51baSKai Chieh Chuang 		/* MISBIAS0 = 1P97 */
1000a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON9,
1001a74d51baSKai Chieh Chuang 				   0x7 << RG_AUDMICBIAS0VREF,
1002a74d51baSKai Chieh Chuang 				   0x0 << RG_AUDMICBIAS0VREF);
1003a74d51baSKai Chieh Chuang 		break;
1004a74d51baSKai Chieh Chuang 	default:
1005a74d51baSKai Chieh Chuang 		break;
1006a74d51baSKai Chieh Chuang 	}
1007a74d51baSKai Chieh Chuang 	return 0;
1008a74d51baSKai Chieh Chuang }
1009a74d51baSKai Chieh Chuang 
mt_mic_bias_1_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)1010a74d51baSKai Chieh Chuang static int mt_mic_bias_1_event(struct snd_soc_dapm_widget *w,
1011a74d51baSKai Chieh Chuang 			       struct snd_kcontrol *kcontrol,
1012a74d51baSKai Chieh Chuang 			       int event)
1013a74d51baSKai Chieh Chuang {
1014a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
1015a74d51baSKai Chieh Chuang 
1016a74d51baSKai Chieh Chuang 	switch (event) {
1017a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
1018a74d51baSKai Chieh Chuang 		/* MIC Bias 1 LowPower: 0_Normal */
1019a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON10,
1020a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDMICBIAS1LOWPEN, 0x0);
1021a74d51baSKai Chieh Chuang 		/* MISBIAS1 = 2P7V */
1022a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON10,
1023a74d51baSKai Chieh Chuang 				   0x7 << RG_AUDMICBIAS1VREF,
1024a74d51baSKai Chieh Chuang 				   0x7 << RG_AUDMICBIAS1VREF);
1025a74d51baSKai Chieh Chuang 		break;
1026a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMD:
1027a74d51baSKai Chieh Chuang 		/* MISBIAS1 = 1P7V */
1028a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON10,
1029a74d51baSKai Chieh Chuang 				   0x7 << RG_AUDMICBIAS1VREF,
1030a74d51baSKai Chieh Chuang 				   0x0 << RG_AUDMICBIAS1VREF);
1031a74d51baSKai Chieh Chuang 		break;
1032a74d51baSKai Chieh Chuang 	default:
1033a74d51baSKai Chieh Chuang 		break;
1034a74d51baSKai Chieh Chuang 	}
1035a74d51baSKai Chieh Chuang 	return 0;
1036a74d51baSKai Chieh Chuang }
1037a74d51baSKai Chieh Chuang 
mt_mic_bias_2_event(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)1038a74d51baSKai Chieh Chuang static int mt_mic_bias_2_event(struct snd_soc_dapm_widget *w,
1039a74d51baSKai Chieh Chuang 			       struct snd_kcontrol *kcontrol,
1040a74d51baSKai Chieh Chuang 			       int event)
1041a74d51baSKai Chieh Chuang {
1042a74d51baSKai Chieh Chuang 	struct snd_soc_component *cmpnt = snd_soc_dapm_to_component(w->dapm);
1043a74d51baSKai Chieh Chuang 
1044a74d51baSKai Chieh Chuang 	switch (event) {
1045a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_PRE_PMU:
1046a74d51baSKai Chieh Chuang 		/* MIC Bias 2 LowPower: 0_Normal */
1047a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON9,
1048a74d51baSKai Chieh Chuang 				   0x3 << RG_AUDMICBIAS2LOWPEN, 0x0);
1049a74d51baSKai Chieh Chuang 		/* MISBIAS2 = 1P9V */
1050a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON9,
1051a74d51baSKai Chieh Chuang 				   0x7 << RG_AUDMICBIAS2VREF,
1052a74d51baSKai Chieh Chuang 				   0x2 << RG_AUDMICBIAS2VREF);
1053a74d51baSKai Chieh Chuang 		break;
1054a74d51baSKai Chieh Chuang 	case SND_SOC_DAPM_POST_PMD:
1055a74d51baSKai Chieh Chuang 		/* MISBIAS2 = 1P97 */
1056a74d51baSKai Chieh Chuang 		regmap_update_bits(cmpnt->regmap, MT6351_AUDENC_ANA_CON9,
1057a74d51baSKai Chieh Chuang 				   0x7 << RG_AUDMICBIAS2VREF,
1058a74d51baSKai Chieh Chuang 				   0x0 << RG_AUDMICBIAS2VREF);
1059a74d51baSKai Chieh Chuang 		break;
1060a74d51baSKai Chieh Chuang 	default:
1061a74d51baSKai Chieh Chuang 		break;
1062a74d51baSKai Chieh Chuang 	}
1063a74d51baSKai Chieh Chuang 	return 0;
1064a74d51baSKai Chieh Chuang }
1065a74d51baSKai Chieh Chuang 
1066a74d51baSKai Chieh Chuang /* DAPM Widgets */
1067a74d51baSKai Chieh Chuang static const struct snd_soc_dapm_widget mt6351_dapm_widgets[] = {
1068a74d51baSKai Chieh Chuang 	/* Digital Clock */
1069a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUDIO_TOP_AFE_CTL", MT6351_AUDIO_TOP_CON0,
1070a74d51baSKai Chieh Chuang 			    AUD_TOP_PDN_AFE_CTL_BIT, 1, NULL, 0),
1071a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUDIO_TOP_DAC_CTL", MT6351_AUDIO_TOP_CON0,
1072a74d51baSKai Chieh Chuang 			    AUD_TOP_PDN_DAC_CTL_BIT, 1, NULL, 0),
1073a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUDIO_TOP_ADC_CTL", MT6351_AUDIO_TOP_CON0,
1074a74d51baSKai Chieh Chuang 			    AUD_TOP_PDN_ADC_CTL_BIT, 1, NULL, 0),
1075a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUDIO_TOP_PWR_CLK", MT6351_AUDIO_TOP_CON0,
1076a74d51baSKai Chieh Chuang 			    AUD_TOP_PWR_CLK_DIS_CTL_BIT, 1, NULL, 0),
1077a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUDIO_TOP_PDN_RESERVED", MT6351_AUDIO_TOP_CON0,
1078a74d51baSKai Chieh Chuang 			    AUD_TOP_PDN_RESERVED_BIT, 1, NULL, 0),
1079a74d51baSKai Chieh Chuang 
1080a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("NCP", MT6351_AFE_NCP_CFG0,
1081a74d51baSKai Chieh Chuang 			    RG_NCP_ON_BIT, 0,
1082a74d51baSKai Chieh Chuang 			    mt_ncp_event,
1083a74d51baSKai Chieh Chuang 			    SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU),
1084a74d51baSKai Chieh Chuang 
1085a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("DL Digital Clock", SND_SOC_NOPM,
1086a74d51baSKai Chieh Chuang 			    0, 0, NULL, 0),
1087a74d51baSKai Chieh Chuang 
1088a74d51baSKai Chieh Chuang 	/* Global Supply*/
1089a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUDGLB", MT6351_AUDDEC_ANA_CON9,
1090a74d51baSKai Chieh Chuang 			    RG_AUDGLB_PWRDN_VA32_BIT, 1, NULL, 0),
1091a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("CLKSQ Audio", MT6351_TOP_CLKSQ,
1092a74d51baSKai Chieh Chuang 			    RG_CLKSQ_EN_AUD_BIT, 0,
1093a74d51baSKai Chieh Chuang 			    mt_reg_set_clr_event,
1094a74d51baSKai Chieh Chuang 			    SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
1095a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("ZCD13M_CK", MT6351_TOP_CKPDN_CON0,
1096a74d51baSKai Chieh Chuang 			    RG_ZCD13M_CK_PDN_BIT, 1,
1097a74d51baSKai Chieh Chuang 			    mt_reg_set_clr_event,
1098a74d51baSKai Chieh Chuang 			    SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
1099a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUD_CK", MT6351_TOP_CKPDN_CON0,
1100a74d51baSKai Chieh Chuang 			    RG_AUD_CK_PDN_BIT, 1,
1101a74d51baSKai Chieh Chuang 			    mt_reg_set_clr_event,
1102a74d51baSKai Chieh Chuang 			    SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
1103a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUDIF_CK", MT6351_TOP_CKPDN_CON0,
1104a74d51baSKai Chieh Chuang 			    RG_AUDIF_CK_PDN_BIT, 1,
1105a74d51baSKai Chieh Chuang 			    mt_reg_set_clr_event,
1106a74d51baSKai Chieh Chuang 			    SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
1107a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUDNCP_CK", MT6351_TOP_CKPDN_CON0,
1108a74d51baSKai Chieh Chuang 			    RG_AUDNCP_CK_PDN_BIT, 1,
1109a74d51baSKai Chieh Chuang 			    mt_reg_set_clr_event,
1110a74d51baSKai Chieh Chuang 			    SND_SOC_DAPM_POST_PMU | SND_SOC_DAPM_PRE_PMD),
1111a74d51baSKai Chieh Chuang 
1112a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AFE_ON", MT6351_AFE_UL_DL_CON0, RG_AFE_ON_BIT, 0,
1113a74d51baSKai Chieh Chuang 			    NULL, 0),
1114a74d51baSKai Chieh Chuang 
1115a74d51baSKai Chieh Chuang 	/* AIF Rx*/
1116a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_AIF_IN_E("AIF_RX", "AIF1 Playback", 0,
1117a74d51baSKai Chieh Chuang 			      MT6351_AFE_DL_SRC2_CON0_L,
1118a74d51baSKai Chieh Chuang 			      RG_DL_2_SRC_ON_TMP_CTL_PRE_BIT, 0,
1119a74d51baSKai Chieh Chuang 			      mt_aif_in_event, SND_SOC_DAPM_PRE_PMU),
1120a74d51baSKai Chieh Chuang 
1121a74d51baSKai Chieh Chuang 	/* DL Supply */
1122a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("DL Power Supply", SND_SOC_NOPM,
1123a74d51baSKai Chieh Chuang 			    0, 0, NULL, 0),
1124a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("NV Regulator", MT6351_AUDDEC_ANA_CON10,
1125a74d51baSKai Chieh Chuang 			    RG_NVREG_EN_VAUDP32_BIT, 0, NULL, 0),
1126a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("AUD_CLK", MT6351_AUDDEC_ANA_CON9,
1127a74d51baSKai Chieh Chuang 			    RG_RSTB_DECODER_VA32_BIT, 0, NULL, 0),
1128a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("IBIST", MT6351_AUDDEC_ANA_CON9,
1129a74d51baSKai Chieh Chuang 			    RG_AUDIBIASPWRDN_VAUDP32_BIT, 1, NULL, 0),
1130a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("LDO", MT6351_AUDDEC_ANA_CON9,
1131a74d51baSKai Chieh Chuang 			    RG_LCLDO_DEC_EN_VA32_BIT, 0, NULL, 0),
1132a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("LDO_REMOTE_SENSE", MT6351_AUDDEC_ANA_CON9,
1133a74d51baSKai Chieh Chuang 			    RG_LCLDO_DEC_REMOTE_SENSE_VA18_BIT, 0, NULL, 0),
1134a74d51baSKai Chieh Chuang 
1135a74d51baSKai Chieh Chuang 	/* DAC */
1136a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("DAC In Mux", SND_SOC_NOPM, 0, 0, &dac_in_mux_control),
1137a74d51baSKai Chieh Chuang 
1138a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_DAC("DACL", NULL, MT6351_AUDDEC_ANA_CON0,
1139a74d51baSKai Chieh Chuang 			 RG_AUDDACLPWRUP_VAUDP32_BIT, 0),
1140a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("DACL_BIASGEN", MT6351_AUDDEC_ANA_CON0,
1141a74d51baSKai Chieh Chuang 			    RG_AUD_DAC_PWL_UP_VA32_BIT, 0, NULL, 0),
1142a74d51baSKai Chieh Chuang 
1143a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_DAC("DACR", NULL, MT6351_AUDDEC_ANA_CON0,
1144a74d51baSKai Chieh Chuang 			 RG_AUDDACRPWRUP_VAUDP32_BIT, 0),
1145a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("DACR_BIASGEN", MT6351_AUDDEC_ANA_CON0,
1146a74d51baSKai Chieh Chuang 			    RG_AUD_DAC_PWR_UP_VA32_BIT, 0, NULL, 0),
1147a74d51baSKai Chieh Chuang 	/* LOL */
1148a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("LOL Mux", SND_SOC_NOPM, 0, 0, &lo_in_mux_control),
1149a74d51baSKai Chieh Chuang 
1150a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("LO Stability Enh", MT6351_AUDDEC_ANA_CON3,
1151a74d51baSKai Chieh Chuang 			    RG_LOOUTPUTSTBENH_VAUDP32_BIT, 0, NULL, 0),
1152a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("LOL Bias Gen", MT6351_AUDDEC_ANA_CON6,
1153a74d51baSKai Chieh Chuang 			    RG_ABIDEC_RSVD0_VAUDP32_LOL_BIT, 0, NULL, 0),
1154a74d51baSKai Chieh Chuang 
1155a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_OUT_DRV("LOL Buffer", MT6351_AUDDEC_ANA_CON3,
1156a74d51baSKai Chieh Chuang 			     RG_AUDLOLPWRUP_VAUDP32_BIT, 0, NULL, 0),
1157a74d51baSKai Chieh Chuang 
1158a74d51baSKai Chieh Chuang 	/* Headphone */
1159a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("HPL Mux", SND_SOC_NOPM, 0, 0, &hpl_in_mux_control),
1160a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("HPR Mux", SND_SOC_NOPM, 0, 0, &hpr_in_mux_control),
1161a74d51baSKai Chieh Chuang 
1162a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_OUT_DRV_E("HPL Power", MT6351_AUDDEC_ANA_CON0,
1163a74d51baSKai Chieh Chuang 			       RG_AUDHPLPWRUP_VAUDP32_BIT, 0, NULL, 0,
1164a74d51baSKai Chieh Chuang 			       mt_hp_event,
1165a74d51baSKai Chieh Chuang 			       SND_SOC_DAPM_PRE_PMU |
1166a74d51baSKai Chieh Chuang 			       SND_SOC_DAPM_PRE_PMD |
1167a74d51baSKai Chieh Chuang 			       SND_SOC_DAPM_POST_PMD),
1168a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_OUT_DRV_E("HPR Power", MT6351_AUDDEC_ANA_CON0,
1169a74d51baSKai Chieh Chuang 			       RG_AUDHPRPWRUP_VAUDP32_BIT, 0, NULL, 0,
1170a74d51baSKai Chieh Chuang 			       mt_hp_event,
1171a74d51baSKai Chieh Chuang 			       SND_SOC_DAPM_PRE_PMU |
1172a74d51baSKai Chieh Chuang 			       SND_SOC_DAPM_PRE_PMD |
1173a74d51baSKai Chieh Chuang 			       SND_SOC_DAPM_POST_PMD),
1174a74d51baSKai Chieh Chuang 
1175a74d51baSKai Chieh Chuang 	/* Receiver */
1176a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("RCV Mux", SND_SOC_NOPM, 0, 0, &rcv_in_mux_control),
1177a74d51baSKai Chieh Chuang 
1178a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("RCV Stability Enh", MT6351_AUDDEC_ANA_CON1,
1179a74d51baSKai Chieh Chuang 			    RG_HSOUTPUTSTBENH_VAUDP32_BIT, 0, NULL, 0),
1180a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("RCV Bias Gen", MT6351_AUDDEC_ANA_CON6,
1181a74d51baSKai Chieh Chuang 			    RG_ABIDEC_RSVD0_VAUDP32_HS_BIT, 0, NULL, 0),
1182a74d51baSKai Chieh Chuang 
1183a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_OUT_DRV("RCV Buffer", MT6351_AUDDEC_ANA_CON0,
1184a74d51baSKai Chieh Chuang 			     RG_AUDHSPWRUP_VAUDP32_BIT, 0, NULL, 0),
1185a74d51baSKai Chieh Chuang 
1186a74d51baSKai Chieh Chuang 	/* Outputs */
1187a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_OUTPUT("Receiver"),
1188a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_OUTPUT("Headphone L"),
1189a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_OUTPUT("Headphone R"),
1190a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_OUTPUT("LINEOUT L"),
1191a74d51baSKai Chieh Chuang 
1192a74d51baSKai Chieh Chuang 	/* SGEN */
1193a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("SGEN DL Enable", MT6351_AFE_SGEN_CFG0,
1194a74d51baSKai Chieh Chuang 			    SGEN_C_DAC_EN_CTL_BIT, 0, NULL, 0),
1195a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("SGEN MUTE", MT6351_AFE_SGEN_CFG0,
1196a74d51baSKai Chieh Chuang 			    SGEN_C_MUTE_SW_CTL_BIT, 1,
1197a74d51baSKai Chieh Chuang 			    mt_sgen_event, SND_SOC_DAPM_PRE_PMU),
1198a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY("SGEN DL SRC", MT6351_AFE_DL_SRC2_CON0_L,
1199a74d51baSKai Chieh Chuang 			    RG_DL_2_SRC_ON_TMP_CTL_PRE_BIT, 0, NULL, 0),
1200a74d51baSKai Chieh Chuang 
1201a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_INPUT("SGEN DL"),
1202a74d51baSKai Chieh Chuang 
1203a74d51baSKai Chieh Chuang 	/* Uplinks */
1204a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_AIF_OUT_E("AIF1TX", "AIF1 Capture", 0,
1205a74d51baSKai Chieh Chuang 			       MT6351_AFE_UL_SRC_CON0_L,
1206a74d51baSKai Chieh Chuang 			       UL_SRC_ON_TMP_CTL, 0,
1207a74d51baSKai Chieh Chuang 			       mt_aif_out_event,
1208a74d51baSKai Chieh Chuang 			       SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_PRE_PMD),
1209a74d51baSKai Chieh Chuang 
1210a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("VUSB33_LDO", SUPPLY_SUBSEQ_ENABLE,
1211a74d51baSKai Chieh Chuang 			      MT6351_LDO_VUSB33_CON0, RG_VUSB33_EN, 0,
1212a74d51baSKai Chieh Chuang 			      NULL, 0),
1213a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("VUSB33_LDO_CTRL", SUPPLY_SUBSEQ_SETTING,
1214a74d51baSKai Chieh Chuang 			      MT6351_LDO_VUSB33_CON0, RG_VUSB33_ON_CTRL, 1,
1215a74d51baSKai Chieh Chuang 			      NULL, 0),
1216a74d51baSKai Chieh Chuang 
1217a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("VA18_LDO", SUPPLY_SUBSEQ_ENABLE,
1218a74d51baSKai Chieh Chuang 			      MT6351_LDO_VA18_CON0, RG_VA18_EN, 0, NULL, 0),
1219a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("VA18_LDO_CTRL", SUPPLY_SUBSEQ_SETTING,
1220a74d51baSKai Chieh Chuang 			      MT6351_LDO_VA18_CON0, RG_VA18_ON_CTRL, 1,
1221a74d51baSKai Chieh Chuang 			      NULL, 0),
1222a74d51baSKai Chieh Chuang 
1223a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("ADC CLKGEN", SUPPLY_SUBSEQ_ENABLE,
1224a74d51baSKai Chieh Chuang 			      MT6351_AUDENC_ANA_CON3, RG_AUDADCCLKRSTB, 0,
1225a74d51baSKai Chieh Chuang 			      mt_adc_clkgen_event,
1226a74d51baSKai Chieh Chuang 			      SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU),
1227a74d51baSKai Chieh Chuang 
1228a74d51baSKai Chieh Chuang 	/* Uplinks MUX */
1229a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("AIF Out Mux", SND_SOC_NOPM, 0, 0,
1230a74d51baSKai Chieh Chuang 			 &aif_out_mux_control),
1231a74d51baSKai Chieh Chuang 
1232a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("ADC L Mux", SND_SOC_NOPM, 0, 0,
1233a74d51baSKai Chieh Chuang 			 &adc_left_mux_control),
1234a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("ADC R Mux", SND_SOC_NOPM, 0, 0,
1235a74d51baSKai Chieh Chuang 			 &adc_right_mux_control),
1236a74d51baSKai Chieh Chuang 
1237a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_ADC("ADC L", NULL,
1238a74d51baSKai Chieh Chuang 			 MT6351_AUDENC_ANA_CON0, RG_AUDADCLPWRUP, 0),
1239a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_ADC("ADC R", NULL,
1240a74d51baSKai Chieh Chuang 			 MT6351_AUDENC_ANA_CON1, RG_AUDADCRPWRUP, 0),
1241a74d51baSKai Chieh Chuang 
1242a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("PGA L Mux", SND_SOC_NOPM, 0, 0,
1243a74d51baSKai Chieh Chuang 			 &pga_left_mux_control),
1244a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_MUX("PGA R Mux", SND_SOC_NOPM, 0, 0,
1245a74d51baSKai Chieh Chuang 			 &pga_right_mux_control),
1246a74d51baSKai Chieh Chuang 
1247a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_PGA_E("PGA L", MT6351_AUDENC_ANA_CON0, RG_AUDPREAMPLON, 0,
1248a74d51baSKai Chieh Chuang 			   NULL, 0,
1249a74d51baSKai Chieh Chuang 			   mt_pga_left_event,
1250a74d51baSKai Chieh Chuang 			   SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU),
1251a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_PGA_E("PGA R", MT6351_AUDENC_ANA_CON1, RG_AUDPREAMPRON, 0,
1252a74d51baSKai Chieh Chuang 			   NULL, 0,
1253a74d51baSKai Chieh Chuang 			   mt_pga_right_event,
1254a74d51baSKai Chieh Chuang 			   SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU),
1255a74d51baSKai Chieh Chuang 
1256a74d51baSKai Chieh Chuang 	/* main mic mic bias */
1257a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("Mic Bias 0", SUPPLY_SUBSEQ_MICBIAS,
1258a74d51baSKai Chieh Chuang 			      MT6351_AUDENC_ANA_CON9, RG_AUDPWDBMICBIAS0, 0,
1259a74d51baSKai Chieh Chuang 			      mt_mic_bias_0_event,
1260a74d51baSKai Chieh Chuang 			      SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
1261a74d51baSKai Chieh Chuang 	/* ref mic mic bias */
1262a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("Mic Bias 2", SUPPLY_SUBSEQ_MICBIAS,
1263a74d51baSKai Chieh Chuang 			      MT6351_AUDENC_ANA_CON9, RG_AUDPWDBMICBIAS2, 0,
1264a74d51baSKai Chieh Chuang 			      mt_mic_bias_2_event,
1265a74d51baSKai Chieh Chuang 			      SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
1266a74d51baSKai Chieh Chuang 	/* headset mic1/2 mic bias */
1267a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("Mic Bias 1", SUPPLY_SUBSEQ_MICBIAS,
1268a74d51baSKai Chieh Chuang 			      MT6351_AUDENC_ANA_CON10, RG_AUDPWDBMICBIAS1, 0,
1269a74d51baSKai Chieh Chuang 			      mt_mic_bias_1_event,
1270a74d51baSKai Chieh Chuang 			      SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMD),
1271a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_SUPPLY_S("Mic Bias 1 DCC pull high", SUPPLY_SUBSEQ_MICBIAS,
1272a74d51baSKai Chieh Chuang 			      MT6351_AUDENC_ANA_CON10,
1273a74d51baSKai Chieh Chuang 			      RG_AUDMICBIAS1DCSW1NEN, 0,
1274a74d51baSKai Chieh Chuang 			      NULL, 0),
1275a74d51baSKai Chieh Chuang 
1276a74d51baSKai Chieh Chuang 	/* UL input */
1277a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_INPUT("AIN0"),
1278a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_INPUT("AIN1"),
1279a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_INPUT("AIN2"),
1280a74d51baSKai Chieh Chuang 	SND_SOC_DAPM_INPUT("AIN3"),
1281a74d51baSKai Chieh Chuang };
1282a74d51baSKai Chieh Chuang 
1283a74d51baSKai Chieh Chuang static const struct snd_soc_dapm_route mt6351_dapm_routes[] = {
1284a74d51baSKai Chieh Chuang 	/* Capture */
1285a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "AIF Out Mux"},
1286a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "VUSB33_LDO"},
1287a74d51baSKai Chieh Chuang 	{"VUSB33_LDO", NULL, "VUSB33_LDO_CTRL"},
1288a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "VA18_LDO"},
1289a74d51baSKai Chieh Chuang 	{"VA18_LDO", NULL, "VA18_LDO_CTRL"},
1290a74d51baSKai Chieh Chuang 
1291a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "AUDGLB"},
1292a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "CLKSQ Audio"},
1293a74d51baSKai Chieh Chuang 
1294a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "AFE_ON"},
1295a74d51baSKai Chieh Chuang 
1296a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "AUDIO_TOP_AFE_CTL"},
1297a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "AUDIO_TOP_ADC_CTL"},
1298a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "AUDIO_TOP_PWR_CLK"},
1299a74d51baSKai Chieh Chuang 	{"AIF1TX", NULL, "AUDIO_TOP_PDN_RESERVED"},
1300a74d51baSKai Chieh Chuang 
1301a74d51baSKai Chieh Chuang 	{"AIF Out Mux", "Normal Path", "ADC L"},
1302a74d51baSKai Chieh Chuang 	{"AIF Out Mux", "Normal Path", "ADC R"},
1303a74d51baSKai Chieh Chuang 
1304a74d51baSKai Chieh Chuang 	{"ADC L", NULL, "ADC L Mux"},
1305a74d51baSKai Chieh Chuang 	{"ADC L", NULL, "AUD_CK"},
1306a74d51baSKai Chieh Chuang 	{"ADC L", NULL, "AUDIF_CK"},
1307a74d51baSKai Chieh Chuang 	{"ADC L", NULL, "ADC CLKGEN"},
1308a74d51baSKai Chieh Chuang 	{"ADC R", NULL, "ADC R Mux"},
1309a74d51baSKai Chieh Chuang 	{"ADC R", NULL, "AUD_CK"},
1310a74d51baSKai Chieh Chuang 	{"ADC R", NULL, "AUDIF_CK"},
1311a74d51baSKai Chieh Chuang 	{"ADC R", NULL, "ADC CLKGEN"},
1312a74d51baSKai Chieh Chuang 
1313a74d51baSKai Chieh Chuang 	{"ADC L Mux", "AIN0", "AIN0"},
1314a74d51baSKai Chieh Chuang 	{"ADC L Mux", "Left Preamplifier", "PGA L"},
1315a74d51baSKai Chieh Chuang 
1316a74d51baSKai Chieh Chuang 	{"ADC R Mux", "AIN0", "AIN0"},
1317a74d51baSKai Chieh Chuang 	{"ADC R Mux", "Right Preamplifier", "PGA R"},
1318a74d51baSKai Chieh Chuang 
1319a74d51baSKai Chieh Chuang 	{"PGA L", NULL, "PGA L Mux"},
1320a74d51baSKai Chieh Chuang 	{"PGA R", NULL, "PGA R Mux"},
1321a74d51baSKai Chieh Chuang 
1322a74d51baSKai Chieh Chuang 	{"PGA L Mux", "AIN0", "AIN0"},
1323a74d51baSKai Chieh Chuang 	{"PGA L Mux", "AIN1", "AIN1"},
1324a74d51baSKai Chieh Chuang 	{"PGA L Mux", "AIN2", "AIN2"},
1325a74d51baSKai Chieh Chuang 
1326a74d51baSKai Chieh Chuang 	{"PGA R Mux", "AIN0", "AIN0"},
1327a74d51baSKai Chieh Chuang 	{"PGA R Mux", "AIN3", "AIN3"},
1328a74d51baSKai Chieh Chuang 	{"PGA R Mux", "AIN2", "AIN2"},
1329a74d51baSKai Chieh Chuang 
1330a74d51baSKai Chieh Chuang 	{"AIN0", NULL, "Mic Bias 0"},
1331a74d51baSKai Chieh Chuang 	{"AIN2", NULL, "Mic Bias 2"},
1332a74d51baSKai Chieh Chuang 
1333a74d51baSKai Chieh Chuang 	{"AIN1", NULL, "Mic Bias 1"},
1334a74d51baSKai Chieh Chuang 	{"AIN1", NULL, "Mic Bias 1 DCC pull high"},
1335a74d51baSKai Chieh Chuang 
1336a74d51baSKai Chieh Chuang 	/* DL Supply */
1337a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "AUDGLB"},
1338a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "CLKSQ Audio"},
1339a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "ZCD13M_CK"},
1340a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "AUD_CK"},
1341a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "AUDIF_CK"},
1342a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "AUDNCP_CK"},
1343a74d51baSKai Chieh Chuang 
1344a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "NV Regulator"},
1345a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "AUD_CLK"},
1346a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "IBIST"},
1347a74d51baSKai Chieh Chuang 	{"DL Power Supply", NULL, "LDO"},
1348a74d51baSKai Chieh Chuang 	{"LDO", NULL, "LDO_REMOTE_SENSE"},
1349a74d51baSKai Chieh Chuang 
1350a74d51baSKai Chieh Chuang 	/* DL Digital Supply */
1351a74d51baSKai Chieh Chuang 	{"DL Digital Clock", NULL, "AUDIO_TOP_AFE_CTL"},
1352a74d51baSKai Chieh Chuang 	{"DL Digital Clock", NULL, "AUDIO_TOP_DAC_CTL"},
1353a74d51baSKai Chieh Chuang 	{"DL Digital Clock", NULL, "AUDIO_TOP_PWR_CLK"},
1354a74d51baSKai Chieh Chuang 	{"DL Digital Clock", NULL, "AUDIO_TOP_PDN_RESERVED"},
1355a74d51baSKai Chieh Chuang 	{"DL Digital Clock", NULL, "NCP"},
1356a74d51baSKai Chieh Chuang 	{"DL Digital Clock", NULL, "AFE_ON"},
1357a74d51baSKai Chieh Chuang 
1358a74d51baSKai Chieh Chuang 	{"AIF_RX", NULL, "DL Digital Clock"},
1359a74d51baSKai Chieh Chuang 
1360a74d51baSKai Chieh Chuang 	/* DL Path */
1361a74d51baSKai Chieh Chuang 	{"DAC In Mux", "Normal Path", "AIF_RX"},
1362a74d51baSKai Chieh Chuang 
1363a74d51baSKai Chieh Chuang 	{"DAC In Mux", "Sgen", "SGEN DL"},
1364a74d51baSKai Chieh Chuang 	{"SGEN DL", NULL, "SGEN DL SRC"},
1365a74d51baSKai Chieh Chuang 	{"SGEN DL", NULL, "SGEN MUTE"},
1366a74d51baSKai Chieh Chuang 	{"SGEN DL", NULL, "SGEN DL Enable"},
1367a74d51baSKai Chieh Chuang 	{"SGEN DL", NULL, "DL Digital Clock"},
1368a74d51baSKai Chieh Chuang 
1369a74d51baSKai Chieh Chuang 	{"DACL", NULL, "DAC In Mux"},
1370a74d51baSKai Chieh Chuang 	{"DACL", NULL, "DL Power Supply"},
1371a74d51baSKai Chieh Chuang 	{"DACL", NULL, "DACL_BIASGEN"},
1372a74d51baSKai Chieh Chuang 
1373a74d51baSKai Chieh Chuang 	{"DACR", NULL, "DAC In Mux"},
1374a74d51baSKai Chieh Chuang 	{"DACR", NULL, "DL Power Supply"},
1375a74d51baSKai Chieh Chuang 	{"DACR", NULL, "DACR_BIASGEN"},
1376a74d51baSKai Chieh Chuang 
1377a74d51baSKai Chieh Chuang 	{"LOL Mux", "Playback", "DACL"},
1378a74d51baSKai Chieh Chuang 
1379a74d51baSKai Chieh Chuang 	{"LOL Buffer", NULL, "LOL Mux"},
1380a74d51baSKai Chieh Chuang 	{"LOL Buffer", NULL, "LO Stability Enh"},
1381a74d51baSKai Chieh Chuang 	{"LOL Buffer", NULL, "LOL Bias Gen"},
1382a74d51baSKai Chieh Chuang 
1383a74d51baSKai Chieh Chuang 	{"LINEOUT L", NULL, "LOL Buffer"},
1384a74d51baSKai Chieh Chuang 
1385a74d51baSKai Chieh Chuang 	/* Headphone Path */
1386a74d51baSKai Chieh Chuang 	{"HPL Mux", "Audio Playback", "DACL"},
1387a74d51baSKai Chieh Chuang 	{"HPR Mux", "Audio Playback", "DACR"},
1388a74d51baSKai Chieh Chuang 
1389a74d51baSKai Chieh Chuang 	{"HPL Mux", "LoudSPK Playback", "DACL"},
1390a74d51baSKai Chieh Chuang 	{"HPR Mux", "LoudSPK Playback", "DACR"},
1391a74d51baSKai Chieh Chuang 
1392a74d51baSKai Chieh Chuang 	{"HPL Power", NULL, "HPL Mux"},
1393a74d51baSKai Chieh Chuang 	{"HPR Power", NULL, "HPR Mux"},
1394a74d51baSKai Chieh Chuang 
1395a74d51baSKai Chieh Chuang 	{"Headphone L", NULL, "HPL Power"},
1396a74d51baSKai Chieh Chuang 	{"Headphone R", NULL, "HPR Power"},
1397a74d51baSKai Chieh Chuang 
1398a74d51baSKai Chieh Chuang 	/* Receiver Path */
1399a74d51baSKai Chieh Chuang 	{"RCV Mux", "Voice Playback", "DACL"},
1400a74d51baSKai Chieh Chuang 
1401a74d51baSKai Chieh Chuang 	{"RCV Buffer", NULL, "RCV Mux"},
1402a74d51baSKai Chieh Chuang 	{"RCV Buffer", NULL, "RCV Stability Enh"},
1403a74d51baSKai Chieh Chuang 	{"RCV Buffer", NULL, "RCV Bias Gen"},
1404a74d51baSKai Chieh Chuang 
1405a74d51baSKai Chieh Chuang 	{"Receiver", NULL, "RCV Buffer"},
1406a74d51baSKai Chieh Chuang };
1407a74d51baSKai Chieh Chuang 
mt6351_codec_init_reg(struct snd_soc_component * cmpnt)1408a74d51baSKai Chieh Chuang static int mt6351_codec_init_reg(struct snd_soc_component *cmpnt)
1409a74d51baSKai Chieh Chuang {
1410a74d51baSKai Chieh Chuang 	/* Disable CLKSQ 26MHz */
1411a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_TOP_CLKSQ, 0x0001, 0x0);
1412a74d51baSKai Chieh Chuang 	/* disable AUDGLB */
1413a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON9,
1414a74d51baSKai Chieh Chuang 			   0x1000, 0x1000);
1415a74d51baSKai Chieh Chuang 	/* Turn off AUDNCP_CLKDIV engine clock,Turn off AUD 26M */
1416a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_TOP_CKPDN_CON0_SET,
1417a74d51baSKai Chieh Chuang 			   0x3800, 0x3800);
1418a74d51baSKai Chieh Chuang 	/* Disable HeadphoneL/HeadphoneR/voice short circuit protection */
1419a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON0,
1420a74d51baSKai Chieh Chuang 			   0xe000, 0xe000);
1421a74d51baSKai Chieh Chuang 	/* [5] = 1, disable LO buffer left short circuit protection */
1422a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_AUDDEC_ANA_CON3,
1423a74d51baSKai Chieh Chuang 			   0x20, 0x20);
1424a74d51baSKai Chieh Chuang 	/* Reverse the PMIC clock*/
1425a74d51baSKai Chieh Chuang 	regmap_update_bits(cmpnt->regmap, MT6351_AFE_PMIC_NEWIF_CFG2,
1426a74d51baSKai Chieh Chuang 			   0x8000, 0x8000);
1427d61780c1SPierre-Louis Bossart 	return 0;
1428a74d51baSKai Chieh Chuang }
1429a74d51baSKai Chieh Chuang 
mt6351_codec_probe(struct snd_soc_component * cmpnt)1430a74d51baSKai Chieh Chuang static int mt6351_codec_probe(struct snd_soc_component *cmpnt)
1431a74d51baSKai Chieh Chuang {
1432a74d51baSKai Chieh Chuang 	struct mt6351_priv *priv = snd_soc_component_get_drvdata(cmpnt);
1433a74d51baSKai Chieh Chuang 
1434a74d51baSKai Chieh Chuang 	snd_soc_component_init_regmap(cmpnt, priv->regmap);
1435a74d51baSKai Chieh Chuang 
1436a74d51baSKai Chieh Chuang 	mt6351_codec_init_reg(cmpnt);
1437a74d51baSKai Chieh Chuang 	return 0;
1438a74d51baSKai Chieh Chuang }
1439a74d51baSKai Chieh Chuang 
1440a74d51baSKai Chieh Chuang static const struct snd_soc_component_driver mt6351_soc_component_driver = {
1441a74d51baSKai Chieh Chuang 	.probe = mt6351_codec_probe,
1442a74d51baSKai Chieh Chuang 	.controls = mt6351_snd_controls,
1443a74d51baSKai Chieh Chuang 	.num_controls = ARRAY_SIZE(mt6351_snd_controls),
1444a74d51baSKai Chieh Chuang 	.dapm_widgets = mt6351_dapm_widgets,
1445a74d51baSKai Chieh Chuang 	.num_dapm_widgets = ARRAY_SIZE(mt6351_dapm_widgets),
1446a74d51baSKai Chieh Chuang 	.dapm_routes = mt6351_dapm_routes,
1447a74d51baSKai Chieh Chuang 	.num_dapm_routes = ARRAY_SIZE(mt6351_dapm_routes),
1448*39723d34SCharles Keepax 	.endianness = 1,
1449a74d51baSKai Chieh Chuang };
1450a74d51baSKai Chieh Chuang 
mt6351_codec_driver_probe(struct platform_device * pdev)1451a74d51baSKai Chieh Chuang static int mt6351_codec_driver_probe(struct platform_device *pdev)
1452a74d51baSKai Chieh Chuang {
1453a74d51baSKai Chieh Chuang 	struct mt6351_priv *priv;
1454a74d51baSKai Chieh Chuang 
1455a74d51baSKai Chieh Chuang 	priv = devm_kzalloc(&pdev->dev,
1456a74d51baSKai Chieh Chuang 			    sizeof(struct mt6351_priv),
1457a74d51baSKai Chieh Chuang 			    GFP_KERNEL);
1458a74d51baSKai Chieh Chuang 	if (!priv)
1459a74d51baSKai Chieh Chuang 		return -ENOMEM;
1460a74d51baSKai Chieh Chuang 
1461a74d51baSKai Chieh Chuang 	dev_set_drvdata(&pdev->dev, priv);
1462a74d51baSKai Chieh Chuang 
1463a74d51baSKai Chieh Chuang 	priv->dev = &pdev->dev;
1464a74d51baSKai Chieh Chuang 
1465a74d51baSKai Chieh Chuang 	priv->regmap = dev_get_regmap(pdev->dev.parent, NULL);
1466aaa730caSWei Yongjun 	if (!priv->regmap)
1467aaa730caSWei Yongjun 		return -ENODEV;
1468a74d51baSKai Chieh Chuang 
1469a74d51baSKai Chieh Chuang 	dev_dbg(priv->dev, "%s(), dev name %s\n",
1470a74d51baSKai Chieh Chuang 		__func__, dev_name(&pdev->dev));
1471a74d51baSKai Chieh Chuang 
1472a74d51baSKai Chieh Chuang 	return devm_snd_soc_register_component(&pdev->dev,
1473a74d51baSKai Chieh Chuang 					       &mt6351_soc_component_driver,
1474a74d51baSKai Chieh Chuang 					       mt6351_dai_driver,
1475a74d51baSKai Chieh Chuang 					       ARRAY_SIZE(mt6351_dai_driver));
1476a74d51baSKai Chieh Chuang }
1477a74d51baSKai Chieh Chuang 
1478a74d51baSKai Chieh Chuang static const struct of_device_id mt6351_of_match[] = {
1479a74d51baSKai Chieh Chuang 	{.compatible = "mediatek,mt6351-sound",},
1480a74d51baSKai Chieh Chuang 	{}
1481a74d51baSKai Chieh Chuang };
1482a74d51baSKai Chieh Chuang 
1483a74d51baSKai Chieh Chuang static struct platform_driver mt6351_codec_driver = {
1484a74d51baSKai Chieh Chuang 	.driver = {
1485a74d51baSKai Chieh Chuang 		.name = "mt6351-sound",
1486a74d51baSKai Chieh Chuang 		.of_match_table = mt6351_of_match,
1487a74d51baSKai Chieh Chuang 	},
1488a74d51baSKai Chieh Chuang 	.probe = mt6351_codec_driver_probe,
1489a74d51baSKai Chieh Chuang };
1490a74d51baSKai Chieh Chuang 
1491a74d51baSKai Chieh Chuang module_platform_driver(mt6351_codec_driver)
1492a74d51baSKai Chieh Chuang 
1493a74d51baSKai Chieh Chuang /* Module information */
1494a74d51baSKai Chieh Chuang MODULE_DESCRIPTION("MT6351 ALSA SoC codec driver");
1495a74d51baSKai Chieh Chuang MODULE_AUTHOR("KaiChieh Chuang <kaichieh.chuang@mediatek.com>");
1496a74d51baSKai Chieh Chuang MODULE_LICENSE("GPL v2");
1497