xref: /OK3568_Linux_fs/kernel/sound/soc/codecs/jz4725b.c (revision 4882a59341e53eb6f0b4789bf948001014eff981)
1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0
2*4882a593Smuzhiyun //
3*4882a593Smuzhiyun // JZ4725B CODEC driver
4*4882a593Smuzhiyun //
5*4882a593Smuzhiyun // Copyright (C) 2019, Paul Cercueil <paul@crapouillou.net>
6*4882a593Smuzhiyun 
7*4882a593Smuzhiyun #include <linux/kernel.h>
8*4882a593Smuzhiyun #include <linux/module.h>
9*4882a593Smuzhiyun #include <linux/platform_device.h>
10*4882a593Smuzhiyun #include <linux/slab.h>
11*4882a593Smuzhiyun #include <linux/io.h>
12*4882a593Smuzhiyun #include <linux/iopoll.h>
13*4882a593Smuzhiyun #include <linux/regmap.h>
14*4882a593Smuzhiyun #include <linux/clk.h>
15*4882a593Smuzhiyun 
16*4882a593Smuzhiyun #include <linux/delay.h>
17*4882a593Smuzhiyun 
18*4882a593Smuzhiyun #include <sound/core.h>
19*4882a593Smuzhiyun #include <sound/pcm.h>
20*4882a593Smuzhiyun #include <sound/pcm_params.h>
21*4882a593Smuzhiyun #include <sound/initval.h>
22*4882a593Smuzhiyun #include <sound/soc.h>
23*4882a593Smuzhiyun #include <sound/tlv.h>
24*4882a593Smuzhiyun 
25*4882a593Smuzhiyun #define ICDC_RGADW_OFFSET		0x00
26*4882a593Smuzhiyun #define ICDC_RGDATA_OFFSET		0x04
27*4882a593Smuzhiyun 
28*4882a593Smuzhiyun /* ICDC internal register access control register(RGADW) */
29*4882a593Smuzhiyun #define ICDC_RGADW_RGWR			BIT(16)
30*4882a593Smuzhiyun 
31*4882a593Smuzhiyun #define ICDC_RGADW_RGADDR_OFFSET	8
32*4882a593Smuzhiyun #define	ICDC_RGADW_RGADDR_MASK		GENMASK(14, ICDC_RGADW_RGADDR_OFFSET)
33*4882a593Smuzhiyun 
34*4882a593Smuzhiyun #define ICDC_RGADW_RGDIN_OFFSET		0
35*4882a593Smuzhiyun #define	ICDC_RGADW_RGDIN_MASK		GENMASK(7, ICDC_RGADW_RGDIN_OFFSET)
36*4882a593Smuzhiyun 
37*4882a593Smuzhiyun /* ICDC internal register data output register (RGDATA)*/
38*4882a593Smuzhiyun #define ICDC_RGDATA_IRQ			BIT(8)
39*4882a593Smuzhiyun 
40*4882a593Smuzhiyun #define ICDC_RGDATA_RGDOUT_OFFSET	0
41*4882a593Smuzhiyun #define ICDC_RGDATA_RGDOUT_MASK		GENMASK(7, ICDC_RGDATA_RGDOUT_OFFSET)
42*4882a593Smuzhiyun 
43*4882a593Smuzhiyun /* JZ internal register space */
44*4882a593Smuzhiyun enum {
45*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_AICR,
46*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CR1,
47*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CR2,
48*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CCR1,
49*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CCR2,
50*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_PMR1,
51*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_PMR2,
52*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CRR,
53*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_ICR,
54*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_IFR,
55*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR1,
56*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR2,
57*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR3,
58*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR4,
59*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR5,
60*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR6,
61*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR7,
62*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR8,
63*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR9,
64*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CGR10,
65*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_TR1,
66*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_TR2,
67*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_CR3,
68*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_AGC1,
69*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_AGC2,
70*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_AGC3,
71*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_AGC4,
72*4882a593Smuzhiyun 	JZ4725B_CODEC_REG_AGC5,
73*4882a593Smuzhiyun };
74*4882a593Smuzhiyun 
75*4882a593Smuzhiyun #define REG_AICR_CONFIG1_OFFSET		0
76*4882a593Smuzhiyun #define REG_AICR_CONFIG1_MASK		(0xf << REG_AICR_CONFIG1_OFFSET)
77*4882a593Smuzhiyun 
78*4882a593Smuzhiyun #define REG_CR1_SB_MICBIAS_OFFSET	7
79*4882a593Smuzhiyun #define REG_CR1_MONO_OFFSET		6
80*4882a593Smuzhiyun #define REG_CR1_DAC_MUTE_OFFSET		5
81*4882a593Smuzhiyun #define REG_CR1_HP_DIS_OFFSET		4
82*4882a593Smuzhiyun #define REG_CR1_DACSEL_OFFSET		3
83*4882a593Smuzhiyun #define REG_CR1_BYPASS_OFFSET		2
84*4882a593Smuzhiyun 
85*4882a593Smuzhiyun #define REG_CR2_DAC_DEEMP_OFFSET	7
86*4882a593Smuzhiyun #define REG_CR2_DAC_ADWL_OFFSET		5
87*4882a593Smuzhiyun #define REG_CR2_DAC_ADWL_MASK		(0x3 << REG_CR2_DAC_ADWL_OFFSET)
88*4882a593Smuzhiyun #define REG_CR2_ADC_ADWL_OFFSET		3
89*4882a593Smuzhiyun #define REG_CR2_ADC_ADWL_MASK		(0x3 << REG_CR2_ADC_ADWL_OFFSET)
90*4882a593Smuzhiyun #define REG_CR2_ADC_HPF_OFFSET		2
91*4882a593Smuzhiyun 
92*4882a593Smuzhiyun #define REG_CR3_SB_MIC1_OFFSET		7
93*4882a593Smuzhiyun #define REG_CR3_SB_MIC2_OFFSET		6
94*4882a593Smuzhiyun #define REG_CR3_SIDETONE1_OFFSET	5
95*4882a593Smuzhiyun #define REG_CR3_SIDETONE2_OFFSET	4
96*4882a593Smuzhiyun #define REG_CR3_MICDIFF_OFFSET		3
97*4882a593Smuzhiyun #define REG_CR3_MICSTEREO_OFFSET	2
98*4882a593Smuzhiyun #define REG_CR3_INSEL_OFFSET		0
99*4882a593Smuzhiyun #define REG_CR3_INSEL_MASK		(0x3 << REG_CR3_INSEL_OFFSET)
100*4882a593Smuzhiyun 
101*4882a593Smuzhiyun #define REG_CCR1_CONFIG4_OFFSET		0
102*4882a593Smuzhiyun #define REG_CCR1_CONFIG4_MASK		(0xf << REG_CCR1_CONFIG4_OFFSET)
103*4882a593Smuzhiyun 
104*4882a593Smuzhiyun #define REG_CCR2_DFREQ_OFFSET		4
105*4882a593Smuzhiyun #define REG_CCR2_DFREQ_MASK		(0xf << REG_CCR2_DFREQ_OFFSET)
106*4882a593Smuzhiyun #define REG_CCR2_AFREQ_OFFSET		0
107*4882a593Smuzhiyun #define REG_CCR2_AFREQ_MASK		(0xf << REG_CCR2_AFREQ_OFFSET)
108*4882a593Smuzhiyun 
109*4882a593Smuzhiyun #define REG_PMR1_SB_DAC_OFFSET		7
110*4882a593Smuzhiyun #define REG_PMR1_SB_OUT_OFFSET		6
111*4882a593Smuzhiyun #define REG_PMR1_SB_MIX_OFFSET		5
112*4882a593Smuzhiyun #define REG_PMR1_SB_ADC_OFFSET		4
113*4882a593Smuzhiyun #define REG_PMR1_SB_LIN_OFFSET		3
114*4882a593Smuzhiyun #define REG_PMR1_SB_IND_OFFSET		0
115*4882a593Smuzhiyun 
116*4882a593Smuzhiyun #define REG_PMR2_LRGI_OFFSET		7
117*4882a593Smuzhiyun #define REG_PMR2_RLGI_OFFSET		6
118*4882a593Smuzhiyun #define REG_PMR2_LRGOD_OFFSET		5
119*4882a593Smuzhiyun #define REG_PMR2_RLGOD_OFFSET		4
120*4882a593Smuzhiyun #define REG_PMR2_GIM_OFFSET		3
121*4882a593Smuzhiyun #define REG_PMR2_SB_MC_OFFSET		2
122*4882a593Smuzhiyun #define REG_PMR2_SB_OFFSET		1
123*4882a593Smuzhiyun #define REG_PMR2_SB_SLEEP_OFFSET	0
124*4882a593Smuzhiyun 
125*4882a593Smuzhiyun #define REG_IFR_RAMP_UP_DONE_OFFSET	3
126*4882a593Smuzhiyun #define REG_IFR_RAMP_DOWN_DONE_OFFSET	2
127*4882a593Smuzhiyun 
128*4882a593Smuzhiyun #define REG_CGR1_GODL_OFFSET		4
129*4882a593Smuzhiyun #define REG_CGR1_GODL_MASK		(0xf << REG_CGR1_GODL_OFFSET)
130*4882a593Smuzhiyun #define REG_CGR1_GODR_OFFSET		0
131*4882a593Smuzhiyun #define REG_CGR1_GODR_MASK		(0xf << REG_CGR1_GODR_OFFSET)
132*4882a593Smuzhiyun 
133*4882a593Smuzhiyun #define REG_CGR2_GO1R_OFFSET		0
134*4882a593Smuzhiyun #define REG_CGR2_GO1R_MASK		(0x1f << REG_CGR2_GO1R_OFFSET)
135*4882a593Smuzhiyun 
136*4882a593Smuzhiyun #define REG_CGR3_GO1L_OFFSET		0
137*4882a593Smuzhiyun #define REG_CGR3_GO1L_MASK		(0x1f << REG_CGR3_GO1L_OFFSET)
138*4882a593Smuzhiyun 
139*4882a593Smuzhiyun #define REG_CGR10_GIL_OFFSET		0
140*4882a593Smuzhiyun #define REG_CGR10_GIR_OFFSET		4
141*4882a593Smuzhiyun 
142*4882a593Smuzhiyun struct jz_icdc {
143*4882a593Smuzhiyun 	struct regmap *regmap;
144*4882a593Smuzhiyun 	void __iomem *base;
145*4882a593Smuzhiyun 	struct clk *clk;
146*4882a593Smuzhiyun };
147*4882a593Smuzhiyun 
148*4882a593Smuzhiyun static const SNDRV_CTL_TLVD_DECLARE_DB_SCALE(jz4725b_adc_tlv,     0, 150, 0);
149*4882a593Smuzhiyun static const SNDRV_CTL_TLVD_DECLARE_DB_SCALE(jz4725b_dac_tlv, -2250, 150, 0);
150*4882a593Smuzhiyun 
151*4882a593Smuzhiyun static const struct snd_kcontrol_new jz4725b_codec_controls[] = {
152*4882a593Smuzhiyun 	SOC_DOUBLE_TLV("Master Playback Volume",
153*4882a593Smuzhiyun 		       JZ4725B_CODEC_REG_CGR1,
154*4882a593Smuzhiyun 		       REG_CGR1_GODL_OFFSET,
155*4882a593Smuzhiyun 		       REG_CGR1_GODR_OFFSET,
156*4882a593Smuzhiyun 		       0xf, 1, jz4725b_dac_tlv),
157*4882a593Smuzhiyun 	SOC_DOUBLE_TLV("Master Capture Volume",
158*4882a593Smuzhiyun 		       JZ4725B_CODEC_REG_CGR10,
159*4882a593Smuzhiyun 		       REG_CGR10_GIL_OFFSET,
160*4882a593Smuzhiyun 		       REG_CGR10_GIR_OFFSET,
161*4882a593Smuzhiyun 		       0xf, 0, jz4725b_adc_tlv),
162*4882a593Smuzhiyun 
163*4882a593Smuzhiyun 	SOC_SINGLE("Master Playback Switch", JZ4725B_CODEC_REG_CR1,
164*4882a593Smuzhiyun 		   REG_CR1_DAC_MUTE_OFFSET, 1, 1),
165*4882a593Smuzhiyun 
166*4882a593Smuzhiyun 	SOC_SINGLE("Deemphasize Filter Playback Switch",
167*4882a593Smuzhiyun 		   JZ4725B_CODEC_REG_CR2,
168*4882a593Smuzhiyun 		   REG_CR2_DAC_DEEMP_OFFSET, 1, 0),
169*4882a593Smuzhiyun 
170*4882a593Smuzhiyun 	SOC_SINGLE("High-Pass Filter Capture Switch",
171*4882a593Smuzhiyun 		   JZ4725B_CODEC_REG_CR2,
172*4882a593Smuzhiyun 		   REG_CR2_ADC_HPF_OFFSET, 1, 0),
173*4882a593Smuzhiyun };
174*4882a593Smuzhiyun 
175*4882a593Smuzhiyun static const char * const jz4725b_codec_adc_src_texts[] = {
176*4882a593Smuzhiyun 	"Mic 1", "Mic 2", "Line In", "Mixer",
177*4882a593Smuzhiyun };
178*4882a593Smuzhiyun static const unsigned int jz4725b_codec_adc_src_values[] = { 0, 1, 2, 3, };
179*4882a593Smuzhiyun static SOC_VALUE_ENUM_SINGLE_DECL(jz4725b_codec_adc_src_enum,
180*4882a593Smuzhiyun 				  JZ4725B_CODEC_REG_CR3,
181*4882a593Smuzhiyun 				  REG_CR3_INSEL_OFFSET,
182*4882a593Smuzhiyun 				  REG_CR3_INSEL_MASK,
183*4882a593Smuzhiyun 				  jz4725b_codec_adc_src_texts,
184*4882a593Smuzhiyun 				  jz4725b_codec_adc_src_values);
185*4882a593Smuzhiyun static const struct snd_kcontrol_new jz4725b_codec_adc_src_ctrl =
186*4882a593Smuzhiyun 	SOC_DAPM_ENUM("ADC Source Capture Route", jz4725b_codec_adc_src_enum);
187*4882a593Smuzhiyun 
188*4882a593Smuzhiyun static const struct snd_kcontrol_new jz4725b_codec_mixer_controls[] = {
189*4882a593Smuzhiyun 	SOC_DAPM_SINGLE("Line In Bypass", JZ4725B_CODEC_REG_CR1,
190*4882a593Smuzhiyun 			REG_CR1_BYPASS_OFFSET, 1, 0),
191*4882a593Smuzhiyun };
192*4882a593Smuzhiyun 
jz4725b_out_stage_enable(struct snd_soc_dapm_widget * w,struct snd_kcontrol * kcontrol,int event)193*4882a593Smuzhiyun static int jz4725b_out_stage_enable(struct snd_soc_dapm_widget *w,
194*4882a593Smuzhiyun 				    struct snd_kcontrol *kcontrol,
195*4882a593Smuzhiyun 				    int event)
196*4882a593Smuzhiyun {
197*4882a593Smuzhiyun 	struct snd_soc_component *codec = snd_soc_dapm_to_component(w->dapm);
198*4882a593Smuzhiyun 	struct jz_icdc *icdc = snd_soc_component_get_drvdata(codec);
199*4882a593Smuzhiyun 	struct regmap *map = icdc->regmap;
200*4882a593Smuzhiyun 	unsigned int val;
201*4882a593Smuzhiyun 
202*4882a593Smuzhiyun 	switch (event) {
203*4882a593Smuzhiyun 	case SND_SOC_DAPM_PRE_PMU:
204*4882a593Smuzhiyun 		return regmap_update_bits(map, JZ4725B_CODEC_REG_IFR,
205*4882a593Smuzhiyun 					  BIT(REG_IFR_RAMP_UP_DONE_OFFSET), 0);
206*4882a593Smuzhiyun 	case SND_SOC_DAPM_POST_PMU:
207*4882a593Smuzhiyun 		return regmap_read_poll_timeout(map, JZ4725B_CODEC_REG_IFR,
208*4882a593Smuzhiyun 			       val, val & BIT(REG_IFR_RAMP_UP_DONE_OFFSET),
209*4882a593Smuzhiyun 			       100000, 500000);
210*4882a593Smuzhiyun 	case SND_SOC_DAPM_PRE_PMD:
211*4882a593Smuzhiyun 		return regmap_update_bits(map, JZ4725B_CODEC_REG_IFR,
212*4882a593Smuzhiyun 				BIT(REG_IFR_RAMP_DOWN_DONE_OFFSET), 0);
213*4882a593Smuzhiyun 	case SND_SOC_DAPM_POST_PMD:
214*4882a593Smuzhiyun 		return regmap_read_poll_timeout(map, JZ4725B_CODEC_REG_IFR,
215*4882a593Smuzhiyun 			       val, val & BIT(REG_IFR_RAMP_DOWN_DONE_OFFSET),
216*4882a593Smuzhiyun 			       100000, 500000);
217*4882a593Smuzhiyun 	default:
218*4882a593Smuzhiyun 		return -EINVAL;
219*4882a593Smuzhiyun 	}
220*4882a593Smuzhiyun }
221*4882a593Smuzhiyun 
222*4882a593Smuzhiyun static const struct snd_soc_dapm_widget jz4725b_codec_dapm_widgets[] = {
223*4882a593Smuzhiyun 	/* DAC */
224*4882a593Smuzhiyun 	SND_SOC_DAPM_DAC("DAC", "Playback",
225*4882a593Smuzhiyun 			 JZ4725B_CODEC_REG_PMR1, REG_PMR1_SB_DAC_OFFSET, 1),
226*4882a593Smuzhiyun 
227*4882a593Smuzhiyun 	/* ADC */
228*4882a593Smuzhiyun 	SND_SOC_DAPM_ADC("ADC", "Capture",
229*4882a593Smuzhiyun 			 JZ4725B_CODEC_REG_PMR1, REG_PMR1_SB_ADC_OFFSET, 1),
230*4882a593Smuzhiyun 
231*4882a593Smuzhiyun 	SND_SOC_DAPM_MUX("ADC Source Capture Route", SND_SOC_NOPM, 0, 0,
232*4882a593Smuzhiyun 			 &jz4725b_codec_adc_src_ctrl),
233*4882a593Smuzhiyun 
234*4882a593Smuzhiyun 	/* Mixer */
235*4882a593Smuzhiyun 	SND_SOC_DAPM_MIXER("Mixer", JZ4725B_CODEC_REG_PMR1,
236*4882a593Smuzhiyun 			   REG_PMR1_SB_MIX_OFFSET, 1,
237*4882a593Smuzhiyun 			   jz4725b_codec_mixer_controls,
238*4882a593Smuzhiyun 			   ARRAY_SIZE(jz4725b_codec_mixer_controls)),
239*4882a593Smuzhiyun 	SND_SOC_DAPM_MIXER("DAC to Mixer", JZ4725B_CODEC_REG_CR1,
240*4882a593Smuzhiyun 			   REG_CR1_DACSEL_OFFSET, 0, NULL, 0),
241*4882a593Smuzhiyun 
242*4882a593Smuzhiyun 	SND_SOC_DAPM_MIXER("Line In", JZ4725B_CODEC_REG_PMR1,
243*4882a593Smuzhiyun 			   REG_PMR1_SB_LIN_OFFSET, 1, NULL, 0),
244*4882a593Smuzhiyun 	SND_SOC_DAPM_MIXER("HP Out", JZ4725B_CODEC_REG_CR1,
245*4882a593Smuzhiyun 			   REG_CR1_HP_DIS_OFFSET, 1, NULL, 0),
246*4882a593Smuzhiyun 
247*4882a593Smuzhiyun 	SND_SOC_DAPM_MIXER("Mic 1", JZ4725B_CODEC_REG_CR3,
248*4882a593Smuzhiyun 			   REG_CR3_SB_MIC1_OFFSET, 1, NULL, 0),
249*4882a593Smuzhiyun 	SND_SOC_DAPM_MIXER("Mic 2", JZ4725B_CODEC_REG_CR3,
250*4882a593Smuzhiyun 			   REG_CR3_SB_MIC2_OFFSET, 1, NULL, 0),
251*4882a593Smuzhiyun 
252*4882a593Smuzhiyun 	SND_SOC_DAPM_MIXER_E("Out Stage", JZ4725B_CODEC_REG_PMR1,
253*4882a593Smuzhiyun 			     REG_PMR1_SB_OUT_OFFSET, 1, NULL, 0,
254*4882a593Smuzhiyun 			     jz4725b_out_stage_enable,
255*4882a593Smuzhiyun 			     SND_SOC_DAPM_PRE_PMU | SND_SOC_DAPM_POST_PMU |
256*4882a593Smuzhiyun 			     SND_SOC_DAPM_PRE_PMD | SND_SOC_DAPM_POST_PMD),
257*4882a593Smuzhiyun 	SND_SOC_DAPM_MIXER("Mixer to ADC", JZ4725B_CODEC_REG_PMR1,
258*4882a593Smuzhiyun 			   REG_PMR1_SB_IND_OFFSET, 1, NULL, 0),
259*4882a593Smuzhiyun 
260*4882a593Smuzhiyun 	SND_SOC_DAPM_SUPPLY("Mic Bias", JZ4725B_CODEC_REG_CR1,
261*4882a593Smuzhiyun 			    REG_CR1_SB_MICBIAS_OFFSET, 1, NULL, 0),
262*4882a593Smuzhiyun 
263*4882a593Smuzhiyun 	/* Pins */
264*4882a593Smuzhiyun 	SND_SOC_DAPM_INPUT("MIC1P"),
265*4882a593Smuzhiyun 	SND_SOC_DAPM_INPUT("MIC1N"),
266*4882a593Smuzhiyun 	SND_SOC_DAPM_INPUT("MIC2P"),
267*4882a593Smuzhiyun 	SND_SOC_DAPM_INPUT("MIC2N"),
268*4882a593Smuzhiyun 
269*4882a593Smuzhiyun 	SND_SOC_DAPM_INPUT("LLINEIN"),
270*4882a593Smuzhiyun 	SND_SOC_DAPM_INPUT("RLINEIN"),
271*4882a593Smuzhiyun 
272*4882a593Smuzhiyun 	SND_SOC_DAPM_OUTPUT("LHPOUT"),
273*4882a593Smuzhiyun 	SND_SOC_DAPM_OUTPUT("RHPOUT"),
274*4882a593Smuzhiyun };
275*4882a593Smuzhiyun 
276*4882a593Smuzhiyun static const struct snd_soc_dapm_route jz4725b_codec_dapm_routes[] = {
277*4882a593Smuzhiyun 	{"Mic 1", NULL, "MIC1P"},
278*4882a593Smuzhiyun 	{"Mic 1", NULL, "MIC1N"},
279*4882a593Smuzhiyun 	{"Mic 2", NULL, "MIC2P"},
280*4882a593Smuzhiyun 	{"Mic 2", NULL, "MIC2N"},
281*4882a593Smuzhiyun 
282*4882a593Smuzhiyun 	{"Line In", NULL, "LLINEIN"},
283*4882a593Smuzhiyun 	{"Line In", NULL, "RLINEIN"},
284*4882a593Smuzhiyun 
285*4882a593Smuzhiyun 	{"Mixer", "Line In Bypass", "Line In"},
286*4882a593Smuzhiyun 	{"DAC to Mixer", NULL, "DAC"},
287*4882a593Smuzhiyun 	{"Mixer", NULL, "DAC to Mixer"},
288*4882a593Smuzhiyun 
289*4882a593Smuzhiyun 	{"Mixer to ADC", NULL, "Mixer"},
290*4882a593Smuzhiyun 	{"ADC Source Capture Route", "Mixer", "Mixer to ADC"},
291*4882a593Smuzhiyun 	{"ADC Source Capture Route", "Line In", "Line In"},
292*4882a593Smuzhiyun 	{"ADC Source Capture Route", "Mic 1", "Mic 1"},
293*4882a593Smuzhiyun 	{"ADC Source Capture Route", "Mic 2", "Mic 2"},
294*4882a593Smuzhiyun 	{"ADC", NULL, "ADC Source Capture Route"},
295*4882a593Smuzhiyun 
296*4882a593Smuzhiyun 	{"Out Stage", NULL, "Mixer"},
297*4882a593Smuzhiyun 	{"HP Out", NULL, "Out Stage"},
298*4882a593Smuzhiyun 	{"LHPOUT", NULL, "HP Out"},
299*4882a593Smuzhiyun 	{"RHPOUT", NULL, "HP Out"},
300*4882a593Smuzhiyun };
301*4882a593Smuzhiyun 
jz4725b_codec_set_bias_level(struct snd_soc_component * component,enum snd_soc_bias_level level)302*4882a593Smuzhiyun static int jz4725b_codec_set_bias_level(struct snd_soc_component *component,
303*4882a593Smuzhiyun 					enum snd_soc_bias_level level)
304*4882a593Smuzhiyun {
305*4882a593Smuzhiyun 	struct jz_icdc *icdc = snd_soc_component_get_drvdata(component);
306*4882a593Smuzhiyun 	struct regmap *map = icdc->regmap;
307*4882a593Smuzhiyun 
308*4882a593Smuzhiyun 	switch (level) {
309*4882a593Smuzhiyun 	case SND_SOC_BIAS_ON:
310*4882a593Smuzhiyun 		regmap_update_bits(map, JZ4725B_CODEC_REG_PMR2,
311*4882a593Smuzhiyun 				   BIT(REG_PMR2_SB_SLEEP_OFFSET), 0);
312*4882a593Smuzhiyun 		break;
313*4882a593Smuzhiyun 	case SND_SOC_BIAS_PREPARE:
314*4882a593Smuzhiyun 		/* Enable sound hardware */
315*4882a593Smuzhiyun 		regmap_update_bits(map, JZ4725B_CODEC_REG_PMR2,
316*4882a593Smuzhiyun 				   BIT(REG_PMR2_SB_OFFSET), 0);
317*4882a593Smuzhiyun 		msleep(224);
318*4882a593Smuzhiyun 		break;
319*4882a593Smuzhiyun 	case SND_SOC_BIAS_STANDBY:
320*4882a593Smuzhiyun 		regmap_update_bits(map, JZ4725B_CODEC_REG_PMR2,
321*4882a593Smuzhiyun 				   BIT(REG_PMR2_SB_SLEEP_OFFSET),
322*4882a593Smuzhiyun 				   BIT(REG_PMR2_SB_SLEEP_OFFSET));
323*4882a593Smuzhiyun 		break;
324*4882a593Smuzhiyun 	case SND_SOC_BIAS_OFF:
325*4882a593Smuzhiyun 		regmap_update_bits(map, JZ4725B_CODEC_REG_PMR2,
326*4882a593Smuzhiyun 				   BIT(REG_PMR2_SB_OFFSET),
327*4882a593Smuzhiyun 				   BIT(REG_PMR2_SB_OFFSET));
328*4882a593Smuzhiyun 		break;
329*4882a593Smuzhiyun 	}
330*4882a593Smuzhiyun 
331*4882a593Smuzhiyun 	return 0;
332*4882a593Smuzhiyun }
333*4882a593Smuzhiyun 
jz4725b_codec_dev_probe(struct snd_soc_component * component)334*4882a593Smuzhiyun static int jz4725b_codec_dev_probe(struct snd_soc_component *component)
335*4882a593Smuzhiyun {
336*4882a593Smuzhiyun 	struct jz_icdc *icdc = snd_soc_component_get_drvdata(component);
337*4882a593Smuzhiyun 	struct regmap *map = icdc->regmap;
338*4882a593Smuzhiyun 
339*4882a593Smuzhiyun 	clk_prepare_enable(icdc->clk);
340*4882a593Smuzhiyun 
341*4882a593Smuzhiyun 	/* Write CONFIGn (n=1 to 8) bits.
342*4882a593Smuzhiyun 	 * The value 0x0f is specified in the datasheet as a requirement.
343*4882a593Smuzhiyun 	 */
344*4882a593Smuzhiyun 	regmap_write(map, JZ4725B_CODEC_REG_AICR,
345*4882a593Smuzhiyun 		     0xf << REG_AICR_CONFIG1_OFFSET);
346*4882a593Smuzhiyun 	regmap_write(map, JZ4725B_CODEC_REG_CCR1,
347*4882a593Smuzhiyun 		     0x0 << REG_CCR1_CONFIG4_OFFSET);
348*4882a593Smuzhiyun 
349*4882a593Smuzhiyun 	return 0;
350*4882a593Smuzhiyun }
351*4882a593Smuzhiyun 
jz4725b_codec_dev_remove(struct snd_soc_component * component)352*4882a593Smuzhiyun static void jz4725b_codec_dev_remove(struct snd_soc_component *component)
353*4882a593Smuzhiyun {
354*4882a593Smuzhiyun 	struct jz_icdc *icdc = snd_soc_component_get_drvdata(component);
355*4882a593Smuzhiyun 
356*4882a593Smuzhiyun 	clk_disable_unprepare(icdc->clk);
357*4882a593Smuzhiyun }
358*4882a593Smuzhiyun 
359*4882a593Smuzhiyun static const struct snd_soc_component_driver jz4725b_codec = {
360*4882a593Smuzhiyun 	.probe			= jz4725b_codec_dev_probe,
361*4882a593Smuzhiyun 	.remove			= jz4725b_codec_dev_remove,
362*4882a593Smuzhiyun 	.set_bias_level		= jz4725b_codec_set_bias_level,
363*4882a593Smuzhiyun 	.controls		= jz4725b_codec_controls,
364*4882a593Smuzhiyun 	.num_controls		= ARRAY_SIZE(jz4725b_codec_controls),
365*4882a593Smuzhiyun 	.dapm_widgets		= jz4725b_codec_dapm_widgets,
366*4882a593Smuzhiyun 	.num_dapm_widgets	= ARRAY_SIZE(jz4725b_codec_dapm_widgets),
367*4882a593Smuzhiyun 	.dapm_routes		= jz4725b_codec_dapm_routes,
368*4882a593Smuzhiyun 	.num_dapm_routes	= ARRAY_SIZE(jz4725b_codec_dapm_routes),
369*4882a593Smuzhiyun 	.suspend_bias_off	= 1,
370*4882a593Smuzhiyun 	.use_pmdown_time	= 1,
371*4882a593Smuzhiyun };
372*4882a593Smuzhiyun 
373*4882a593Smuzhiyun static const unsigned int jz4725b_codec_sample_rates[] = {
374*4882a593Smuzhiyun 	96000, 48000, 44100, 32000,
375*4882a593Smuzhiyun 	24000, 22050, 16000, 12000,
376*4882a593Smuzhiyun 	11025, 9600, 8000,
377*4882a593Smuzhiyun };
378*4882a593Smuzhiyun 
jz4725b_codec_hw_params(struct snd_pcm_substream * substream,struct snd_pcm_hw_params * params,struct snd_soc_dai * dai)379*4882a593Smuzhiyun static int jz4725b_codec_hw_params(struct snd_pcm_substream *substream,
380*4882a593Smuzhiyun 	struct snd_pcm_hw_params *params, struct snd_soc_dai *dai)
381*4882a593Smuzhiyun {
382*4882a593Smuzhiyun 	struct jz_icdc *icdc = snd_soc_component_get_drvdata(dai->component);
383*4882a593Smuzhiyun 	unsigned int rate, bit_width;
384*4882a593Smuzhiyun 
385*4882a593Smuzhiyun 	switch (params_format(params)) {
386*4882a593Smuzhiyun 	case SNDRV_PCM_FORMAT_S16_LE:
387*4882a593Smuzhiyun 		bit_width = 0;
388*4882a593Smuzhiyun 		break;
389*4882a593Smuzhiyun 	case SNDRV_PCM_FORMAT_S18_3LE:
390*4882a593Smuzhiyun 		bit_width = 1;
391*4882a593Smuzhiyun 		break;
392*4882a593Smuzhiyun 	case SNDRV_PCM_FORMAT_S20_3LE:
393*4882a593Smuzhiyun 		bit_width = 2;
394*4882a593Smuzhiyun 		break;
395*4882a593Smuzhiyun 	case SNDRV_PCM_FORMAT_S24_3LE:
396*4882a593Smuzhiyun 		bit_width = 3;
397*4882a593Smuzhiyun 		break;
398*4882a593Smuzhiyun 	default:
399*4882a593Smuzhiyun 		return -EINVAL;
400*4882a593Smuzhiyun 	}
401*4882a593Smuzhiyun 
402*4882a593Smuzhiyun 	for (rate = 0; rate < ARRAY_SIZE(jz4725b_codec_sample_rates); rate++) {
403*4882a593Smuzhiyun 		if (jz4725b_codec_sample_rates[rate] == params_rate(params))
404*4882a593Smuzhiyun 			break;
405*4882a593Smuzhiyun 	}
406*4882a593Smuzhiyun 
407*4882a593Smuzhiyun 	if (rate == ARRAY_SIZE(jz4725b_codec_sample_rates))
408*4882a593Smuzhiyun 		return -EINVAL;
409*4882a593Smuzhiyun 
410*4882a593Smuzhiyun 	if (substream->stream == SNDRV_PCM_STREAM_PLAYBACK) {
411*4882a593Smuzhiyun 		regmap_update_bits(icdc->regmap,
412*4882a593Smuzhiyun 				   JZ4725B_CODEC_REG_CR2,
413*4882a593Smuzhiyun 				   REG_CR2_DAC_ADWL_MASK,
414*4882a593Smuzhiyun 				   bit_width << REG_CR2_DAC_ADWL_OFFSET);
415*4882a593Smuzhiyun 
416*4882a593Smuzhiyun 		regmap_update_bits(icdc->regmap,
417*4882a593Smuzhiyun 				   JZ4725B_CODEC_REG_CCR2,
418*4882a593Smuzhiyun 				   REG_CCR2_DFREQ_MASK,
419*4882a593Smuzhiyun 				   rate << REG_CCR2_DFREQ_OFFSET);
420*4882a593Smuzhiyun 	} else {
421*4882a593Smuzhiyun 		regmap_update_bits(icdc->regmap,
422*4882a593Smuzhiyun 				   JZ4725B_CODEC_REG_CR2,
423*4882a593Smuzhiyun 				   REG_CR2_ADC_ADWL_MASK,
424*4882a593Smuzhiyun 				   bit_width << REG_CR2_ADC_ADWL_OFFSET);
425*4882a593Smuzhiyun 
426*4882a593Smuzhiyun 		regmap_update_bits(icdc->regmap,
427*4882a593Smuzhiyun 				   JZ4725B_CODEC_REG_CCR2,
428*4882a593Smuzhiyun 				   REG_CCR2_AFREQ_MASK,
429*4882a593Smuzhiyun 				   rate << REG_CCR2_AFREQ_OFFSET);
430*4882a593Smuzhiyun 	}
431*4882a593Smuzhiyun 
432*4882a593Smuzhiyun 	return 0;
433*4882a593Smuzhiyun }
434*4882a593Smuzhiyun 
435*4882a593Smuzhiyun static const struct snd_soc_dai_ops jz4725b_codec_dai_ops = {
436*4882a593Smuzhiyun 	.hw_params = jz4725b_codec_hw_params,
437*4882a593Smuzhiyun };
438*4882a593Smuzhiyun 
439*4882a593Smuzhiyun #define JZ_ICDC_FORMATS (SNDRV_PCM_FMTBIT_S16_LE  | SNDRV_PCM_FMTBIT_S18_3LE | \
440*4882a593Smuzhiyun 			 SNDRV_PCM_FMTBIT_S20_3LE | SNDRV_PCM_FMTBIT_S24_3LE)
441*4882a593Smuzhiyun 
442*4882a593Smuzhiyun static struct snd_soc_dai_driver jz4725b_codec_dai = {
443*4882a593Smuzhiyun 	.name = "jz4725b-hifi",
444*4882a593Smuzhiyun 	.playback = {
445*4882a593Smuzhiyun 		.stream_name = "Playback",
446*4882a593Smuzhiyun 		.channels_min = 2,
447*4882a593Smuzhiyun 		.channels_max = 2,
448*4882a593Smuzhiyun 		.rates = SNDRV_PCM_RATE_8000_96000,
449*4882a593Smuzhiyun 		.formats = JZ_ICDC_FORMATS,
450*4882a593Smuzhiyun 	},
451*4882a593Smuzhiyun 	.capture = {
452*4882a593Smuzhiyun 		.stream_name = "Capture",
453*4882a593Smuzhiyun 		.channels_min = 2,
454*4882a593Smuzhiyun 		.channels_max = 2,
455*4882a593Smuzhiyun 		.rates = SNDRV_PCM_RATE_8000_96000,
456*4882a593Smuzhiyun 		.formats = JZ_ICDC_FORMATS,
457*4882a593Smuzhiyun 	},
458*4882a593Smuzhiyun 	.ops = &jz4725b_codec_dai_ops,
459*4882a593Smuzhiyun };
460*4882a593Smuzhiyun 
jz4725b_codec_volatile(struct device * dev,unsigned int reg)461*4882a593Smuzhiyun static bool jz4725b_codec_volatile(struct device *dev, unsigned int reg)
462*4882a593Smuzhiyun {
463*4882a593Smuzhiyun 	return reg == JZ4725B_CODEC_REG_IFR;
464*4882a593Smuzhiyun }
465*4882a593Smuzhiyun 
jz4725b_codec_can_access_reg(struct device * dev,unsigned int reg)466*4882a593Smuzhiyun static bool jz4725b_codec_can_access_reg(struct device *dev, unsigned int reg)
467*4882a593Smuzhiyun {
468*4882a593Smuzhiyun 	return (reg != JZ4725B_CODEC_REG_TR1) && (reg != JZ4725B_CODEC_REG_TR2);
469*4882a593Smuzhiyun }
470*4882a593Smuzhiyun 
jz4725b_codec_io_wait(struct jz_icdc * icdc)471*4882a593Smuzhiyun static int jz4725b_codec_io_wait(struct jz_icdc *icdc)
472*4882a593Smuzhiyun {
473*4882a593Smuzhiyun 	u32 reg;
474*4882a593Smuzhiyun 
475*4882a593Smuzhiyun 	return readl_poll_timeout(icdc->base + ICDC_RGADW_OFFSET, reg,
476*4882a593Smuzhiyun 				  !(reg & ICDC_RGADW_RGWR), 1000, 10000);
477*4882a593Smuzhiyun }
478*4882a593Smuzhiyun 
jz4725b_codec_reg_read(void * context,unsigned int reg,unsigned int * val)479*4882a593Smuzhiyun static int jz4725b_codec_reg_read(void *context, unsigned int reg,
480*4882a593Smuzhiyun 				  unsigned int *val)
481*4882a593Smuzhiyun {
482*4882a593Smuzhiyun 	struct jz_icdc *icdc = context;
483*4882a593Smuzhiyun 	unsigned int i;
484*4882a593Smuzhiyun 	u32 tmp;
485*4882a593Smuzhiyun 	int ret;
486*4882a593Smuzhiyun 
487*4882a593Smuzhiyun 	ret = jz4725b_codec_io_wait(icdc);
488*4882a593Smuzhiyun 	if (ret)
489*4882a593Smuzhiyun 		return ret;
490*4882a593Smuzhiyun 
491*4882a593Smuzhiyun 	tmp = readl(icdc->base + ICDC_RGADW_OFFSET);
492*4882a593Smuzhiyun 	tmp = (tmp & ~ICDC_RGADW_RGADDR_MASK)
493*4882a593Smuzhiyun 	    | (reg << ICDC_RGADW_RGADDR_OFFSET);
494*4882a593Smuzhiyun 	writel(tmp, icdc->base + ICDC_RGADW_OFFSET);
495*4882a593Smuzhiyun 
496*4882a593Smuzhiyun 	/* wait 6+ cycles */
497*4882a593Smuzhiyun 	for (i = 0; i < 6; i++)
498*4882a593Smuzhiyun 		*val = readl(icdc->base + ICDC_RGDATA_OFFSET) &
499*4882a593Smuzhiyun 			ICDC_RGDATA_RGDOUT_MASK;
500*4882a593Smuzhiyun 
501*4882a593Smuzhiyun 	return 0;
502*4882a593Smuzhiyun }
503*4882a593Smuzhiyun 
jz4725b_codec_reg_write(void * context,unsigned int reg,unsigned int val)504*4882a593Smuzhiyun static int jz4725b_codec_reg_write(void *context, unsigned int reg,
505*4882a593Smuzhiyun 				   unsigned int val)
506*4882a593Smuzhiyun {
507*4882a593Smuzhiyun 	struct jz_icdc *icdc = context;
508*4882a593Smuzhiyun 	int ret;
509*4882a593Smuzhiyun 
510*4882a593Smuzhiyun 	ret = jz4725b_codec_io_wait(icdc);
511*4882a593Smuzhiyun 	if (ret)
512*4882a593Smuzhiyun 		return ret;
513*4882a593Smuzhiyun 
514*4882a593Smuzhiyun 	writel(ICDC_RGADW_RGWR | (reg << ICDC_RGADW_RGADDR_OFFSET) | val,
515*4882a593Smuzhiyun 			icdc->base + ICDC_RGADW_OFFSET);
516*4882a593Smuzhiyun 
517*4882a593Smuzhiyun 	ret = jz4725b_codec_io_wait(icdc);
518*4882a593Smuzhiyun 	if (ret)
519*4882a593Smuzhiyun 		return ret;
520*4882a593Smuzhiyun 
521*4882a593Smuzhiyun 	return 0;
522*4882a593Smuzhiyun }
523*4882a593Smuzhiyun 
524*4882a593Smuzhiyun static const u8 jz4725b_codec_reg_defaults[] = {
525*4882a593Smuzhiyun 	0x0c, 0xaa, 0x78, 0x00, 0x00, 0xff, 0x03, 0x51,
526*4882a593Smuzhiyun 	0x3f, 0x00, 0x00, 0x04, 0x04, 0x04, 0x04, 0x04,
527*4882a593Smuzhiyun 	0x04, 0x0a, 0x0a, 0x00, 0x00, 0x00, 0xc0, 0x34,
528*4882a593Smuzhiyun 	0x07, 0x44, 0x1f, 0x00,
529*4882a593Smuzhiyun };
530*4882a593Smuzhiyun 
531*4882a593Smuzhiyun static const struct regmap_config jz4725b_codec_regmap_config = {
532*4882a593Smuzhiyun 	.reg_bits = 7,
533*4882a593Smuzhiyun 	.val_bits = 8,
534*4882a593Smuzhiyun 
535*4882a593Smuzhiyun 	.max_register = JZ4725B_CODEC_REG_AGC5,
536*4882a593Smuzhiyun 	.volatile_reg = jz4725b_codec_volatile,
537*4882a593Smuzhiyun 	.readable_reg = jz4725b_codec_can_access_reg,
538*4882a593Smuzhiyun 	.writeable_reg = jz4725b_codec_can_access_reg,
539*4882a593Smuzhiyun 
540*4882a593Smuzhiyun 	.reg_read = jz4725b_codec_reg_read,
541*4882a593Smuzhiyun 	.reg_write = jz4725b_codec_reg_write,
542*4882a593Smuzhiyun 
543*4882a593Smuzhiyun 	.reg_defaults_raw = jz4725b_codec_reg_defaults,
544*4882a593Smuzhiyun 	.num_reg_defaults_raw = ARRAY_SIZE(jz4725b_codec_reg_defaults),
545*4882a593Smuzhiyun 	.cache_type = REGCACHE_FLAT,
546*4882a593Smuzhiyun };
547*4882a593Smuzhiyun 
jz4725b_codec_probe(struct platform_device * pdev)548*4882a593Smuzhiyun static int jz4725b_codec_probe(struct platform_device *pdev)
549*4882a593Smuzhiyun {
550*4882a593Smuzhiyun 	struct device *dev = &pdev->dev;
551*4882a593Smuzhiyun 	struct jz_icdc *icdc;
552*4882a593Smuzhiyun 	int ret;
553*4882a593Smuzhiyun 
554*4882a593Smuzhiyun 	icdc = devm_kzalloc(dev, sizeof(*icdc), GFP_KERNEL);
555*4882a593Smuzhiyun 	if (!icdc)
556*4882a593Smuzhiyun 		return -ENOMEM;
557*4882a593Smuzhiyun 
558*4882a593Smuzhiyun 	icdc->base = devm_platform_ioremap_resource(pdev, 0);
559*4882a593Smuzhiyun 	if (IS_ERR(icdc->base))
560*4882a593Smuzhiyun 		return PTR_ERR(icdc->base);
561*4882a593Smuzhiyun 
562*4882a593Smuzhiyun 	icdc->regmap = devm_regmap_init(dev, NULL, icdc,
563*4882a593Smuzhiyun 					&jz4725b_codec_regmap_config);
564*4882a593Smuzhiyun 	if (IS_ERR(icdc->regmap))
565*4882a593Smuzhiyun 		return PTR_ERR(icdc->regmap);
566*4882a593Smuzhiyun 
567*4882a593Smuzhiyun 	icdc->clk = devm_clk_get(&pdev->dev, "aic");
568*4882a593Smuzhiyun 	if (IS_ERR(icdc->clk))
569*4882a593Smuzhiyun 		return PTR_ERR(icdc->clk);
570*4882a593Smuzhiyun 
571*4882a593Smuzhiyun 	platform_set_drvdata(pdev, icdc);
572*4882a593Smuzhiyun 
573*4882a593Smuzhiyun 	ret = devm_snd_soc_register_component(dev, &jz4725b_codec,
574*4882a593Smuzhiyun 					      &jz4725b_codec_dai, 1);
575*4882a593Smuzhiyun 	if (ret)
576*4882a593Smuzhiyun 		dev_err(dev, "Failed to register codec\n");
577*4882a593Smuzhiyun 
578*4882a593Smuzhiyun 	return ret;
579*4882a593Smuzhiyun }
580*4882a593Smuzhiyun 
581*4882a593Smuzhiyun static const struct of_device_id jz4725b_codec_of_matches[] = {
582*4882a593Smuzhiyun 	{ .compatible = "ingenic,jz4725b-codec", },
583*4882a593Smuzhiyun 	{ }
584*4882a593Smuzhiyun };
585*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, jz4725b_codec_of_matches);
586*4882a593Smuzhiyun 
587*4882a593Smuzhiyun static struct platform_driver jz4725b_codec_driver = {
588*4882a593Smuzhiyun 	.probe = jz4725b_codec_probe,
589*4882a593Smuzhiyun 	.driver = {
590*4882a593Smuzhiyun 		.name = "jz4725b-codec",
591*4882a593Smuzhiyun 		.of_match_table = jz4725b_codec_of_matches,
592*4882a593Smuzhiyun 	},
593*4882a593Smuzhiyun };
594*4882a593Smuzhiyun module_platform_driver(jz4725b_codec_driver);
595*4882a593Smuzhiyun 
596*4882a593Smuzhiyun MODULE_DESCRIPTION("JZ4725B SoC internal codec driver");
597*4882a593Smuzhiyun MODULE_AUTHOR("Paul Cercueil <paul@crapouillou.net>");
598*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
599