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