Skip to content

Commit 1f95c01

Browse files
wenliangwubroonie
authored andcommitted
ASoC: mediatek: mt8195: support pcm in platform driver
This patch adds mt8195 pcm dai driver. Signed-off-by: Trevor Wu <[email protected]> Link: https://lore.kernel.org/r/[email protected] Signed-off-by: Mark Brown <[email protected]>
1 parent 3de3eba commit 1f95c01

File tree

1 file changed

+389
-0
lines changed

1 file changed

+389
-0
lines changed
Lines changed: 389 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,389 @@
1+
// SPDX-License-Identifier: GPL-2.0
2+
/*
3+
* MediaTek ALSA SoC Audio DAI PCM I/F Control
4+
*
5+
* Copyright (c) 2020 MediaTek Inc.
6+
* Author: Bicycle Tsai <[email protected]>
7+
* Trevor Wu <[email protected]>
8+
*/
9+
10+
#include <linux/regmap.h>
11+
#include <sound/pcm_params.h>
12+
#include "mt8195-afe-clk.h"
13+
#include "mt8195-afe-common.h"
14+
#include "mt8195-reg.h"
15+
16+
enum {
17+
MTK_DAI_PCM_FMT_I2S,
18+
MTK_DAI_PCM_FMT_EIAJ,
19+
MTK_DAI_PCM_FMT_MODEA,
20+
MTK_DAI_PCM_FMT_MODEB,
21+
};
22+
23+
enum {
24+
MTK_DAI_PCM_CLK_A1SYS,
25+
MTK_DAI_PCM_CLK_A2SYS,
26+
MTK_DAI_PCM_CLK_26M_48K,
27+
MTK_DAI_PCM_CLK_26M_441K,
28+
};
29+
30+
struct mtk_dai_pcm_rate {
31+
unsigned int rate;
32+
unsigned int reg_value;
33+
};
34+
35+
struct mtk_dai_pcmif_priv {
36+
unsigned int slave_mode;
37+
unsigned int lrck_inv;
38+
unsigned int bck_inv;
39+
unsigned int format;
40+
};
41+
42+
static const struct mtk_dai_pcm_rate mtk_dai_pcm_rates[] = {
43+
{ .rate = 8000, .reg_value = 0, },
44+
{ .rate = 16000, .reg_value = 1, },
45+
{ .rate = 32000, .reg_value = 2, },
46+
{ .rate = 48000, .reg_value = 3, },
47+
{ .rate = 11025, .reg_value = 1, },
48+
{ .rate = 22050, .reg_value = 2, },
49+
{ .rate = 44100, .reg_value = 3, },
50+
};
51+
52+
static int mtk_dai_pcm_mode(unsigned int rate)
53+
{
54+
int i;
55+
56+
for (i = 0; i < ARRAY_SIZE(mtk_dai_pcm_rates); i++)
57+
if (mtk_dai_pcm_rates[i].rate == rate)
58+
return mtk_dai_pcm_rates[i].reg_value;
59+
60+
return -EINVAL;
61+
}
62+
63+
static const struct snd_kcontrol_new mtk_dai_pcm_o000_mix[] = {
64+
SOC_DAPM_SINGLE_AUTODISABLE("I000 Switch", AFE_CONN0, 0, 1, 0),
65+
SOC_DAPM_SINGLE_AUTODISABLE("I070 Switch", AFE_CONN0_2, 6, 1, 0),
66+
};
67+
68+
static const struct snd_kcontrol_new mtk_dai_pcm_o001_mix[] = {
69+
SOC_DAPM_SINGLE_AUTODISABLE("I001 Switch", AFE_CONN1, 1, 1, 0),
70+
SOC_DAPM_SINGLE_AUTODISABLE("I071 Switch", AFE_CONN1_2, 7, 1, 0),
71+
};
72+
73+
static const struct snd_soc_dapm_widget mtk_dai_pcm_widgets[] = {
74+
SND_SOC_DAPM_MIXER("I002", SND_SOC_NOPM, 0, 0, NULL, 0),
75+
SND_SOC_DAPM_MIXER("I003", SND_SOC_NOPM, 0, 0, NULL, 0),
76+
SND_SOC_DAPM_MIXER("O000", SND_SOC_NOPM, 0, 0,
77+
mtk_dai_pcm_o000_mix,
78+
ARRAY_SIZE(mtk_dai_pcm_o000_mix)),
79+
SND_SOC_DAPM_MIXER("O001", SND_SOC_NOPM, 0, 0,
80+
mtk_dai_pcm_o001_mix,
81+
ARRAY_SIZE(mtk_dai_pcm_o001_mix)),
82+
83+
SND_SOC_DAPM_INPUT("PCM1_INPUT"),
84+
SND_SOC_DAPM_OUTPUT("PCM1_OUTPUT"),
85+
};
86+
87+
static const struct snd_soc_dapm_route mtk_dai_pcm_routes[] = {
88+
{"I002", NULL, "PCM1 Capture"},
89+
{"I003", NULL, "PCM1 Capture"},
90+
91+
{"O000", "I000 Switch", "I000"},
92+
{"O001", "I001 Switch", "I001"},
93+
94+
{"O000", "I070 Switch", "I070"},
95+
{"O001", "I071 Switch", "I071"},
96+
97+
{"PCM1 Playback", NULL, "O000"},
98+
{"PCM1 Playback", NULL, "O001"},
99+
100+
{"PCM1_OUTPUT", NULL, "PCM1 Playback"},
101+
{"PCM1 Capture", NULL, "PCM1_INPUT"},
102+
};
103+
104+
static void mtk_dai_pcm_enable(struct mtk_base_afe *afe)
105+
{
106+
regmap_update_bits(afe->regmap, PCM_INTF_CON1,
107+
PCM_INTF_CON1_PCM_EN, PCM_INTF_CON1_PCM_EN);
108+
}
109+
110+
static void mtk_dai_pcm_disable(struct mtk_base_afe *afe)
111+
{
112+
regmap_update_bits(afe->regmap, PCM_INTF_CON1,
113+
PCM_INTF_CON1_PCM_EN, 0x0);
114+
}
115+
116+
static int mtk_dai_pcm_configure(struct snd_pcm_substream *substream,
117+
struct snd_soc_dai *dai)
118+
{
119+
struct snd_pcm_runtime * const runtime = substream->runtime;
120+
struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
121+
struct mt8195_afe_private *afe_priv = afe->platform_priv;
122+
struct mtk_dai_pcmif_priv *pcmif_priv = afe_priv->dai_priv[dai->id];
123+
unsigned int slave_mode = pcmif_priv->slave_mode;
124+
unsigned int lrck_inv = pcmif_priv->lrck_inv;
125+
unsigned int bck_inv = pcmif_priv->bck_inv;
126+
unsigned int fmt = pcmif_priv->format;
127+
unsigned int bit_width = dai->sample_bits;
128+
unsigned int val = 0;
129+
unsigned int mask = 0;
130+
int fs = 0;
131+
int mode = 0;
132+
133+
/* sync freq mode */
134+
fs = mt8195_afe_fs_timing(runtime->rate);
135+
if (fs < 0)
136+
return -EINVAL;
137+
val |= PCM_INTF_CON2_SYNC_FREQ_MODE(fs);
138+
mask |= PCM_INTF_CON2_SYNC_FREQ_MODE_MASK;
139+
140+
/* clk domain sel */
141+
if (runtime->rate % 8000)
142+
val |= PCM_INTF_CON2_CLK_DOMAIN_SEL(MTK_DAI_PCM_CLK_26M_441K);
143+
else
144+
val |= PCM_INTF_CON2_CLK_DOMAIN_SEL(MTK_DAI_PCM_CLK_26M_48K);
145+
mask |= PCM_INTF_CON2_CLK_DOMAIN_SEL_MASK;
146+
147+
regmap_update_bits(afe->regmap, PCM_INTF_CON2, mask, val);
148+
149+
val = 0;
150+
mask = 0;
151+
152+
/* pcm mode */
153+
mode = mtk_dai_pcm_mode(runtime->rate);
154+
if (mode < 0)
155+
return -EINVAL;
156+
val |= PCM_INTF_CON1_PCM_MODE(mode);
157+
mask |= PCM_INTF_CON1_PCM_MODE_MASK;
158+
159+
/* pcm format */
160+
val |= PCM_INTF_CON1_PCM_FMT(fmt);
161+
mask |= PCM_INTF_CON1_PCM_FMT_MASK;
162+
163+
/* pcm sync length */
164+
if (fmt == MTK_DAI_PCM_FMT_MODEA ||
165+
fmt == MTK_DAI_PCM_FMT_MODEB)
166+
val |= PCM_INTF_CON1_SYNC_LENGTH(1);
167+
else
168+
val |= PCM_INTF_CON1_SYNC_LENGTH(bit_width);
169+
mask |= PCM_INTF_CON1_SYNC_LENGTH_MASK;
170+
171+
/* pcm bits, word length */
172+
if (bit_width > 16) {
173+
val |= PCM_INTF_CON1_PCM_24BIT;
174+
val |= PCM_INTF_CON1_PCM_WLEN_64BCK;
175+
} else {
176+
val |= PCM_INTF_CON1_PCM_16BIT;
177+
val |= PCM_INTF_CON1_PCM_WLEN_32BCK;
178+
}
179+
mask |= PCM_INTF_CON1_PCM_BIT_MASK;
180+
mask |= PCM_INTF_CON1_PCM_WLEN_MASK;
181+
182+
/* master/slave */
183+
if (!slave_mode) {
184+
val |= PCM_INTF_CON1_PCM_MASTER;
185+
186+
if (lrck_inv)
187+
val |= PCM_INTF_CON1_SYNC_OUT_INV;
188+
if (bck_inv)
189+
val |= PCM_INTF_CON1_BCLK_OUT_INV;
190+
mask |= PCM_INTF_CON1_CLK_OUT_INV_MASK;
191+
} else {
192+
val |= PCM_INTF_CON1_PCM_SLAVE;
193+
194+
if (lrck_inv)
195+
val |= PCM_INTF_CON1_SYNC_IN_INV;
196+
if (bck_inv)
197+
val |= PCM_INTF_CON1_BCLK_IN_INV;
198+
mask |= PCM_INTF_CON1_CLK_IN_INV_MASK;
199+
200+
/* TODO: add asrc setting for slave mode */
201+
}
202+
mask |= PCM_INTF_CON1_PCM_M_S_MASK;
203+
204+
regmap_update_bits(afe->regmap, PCM_INTF_CON1, mask, val);
205+
206+
return 0;
207+
}
208+
209+
/* dai ops */
210+
static int mtk_dai_pcm_startup(struct snd_pcm_substream *substream,
211+
struct snd_soc_dai *dai)
212+
{
213+
struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
214+
struct mt8195_afe_private *afe_priv = afe->platform_priv;
215+
216+
if (dai->component->active)
217+
return 0;
218+
219+
mt8195_afe_enable_clk(afe, afe_priv->clk[MT8195_CLK_AUD_ASRC11]);
220+
mt8195_afe_enable_clk(afe, afe_priv->clk[MT8195_CLK_AUD_ASRC12]);
221+
mt8195_afe_enable_clk(afe, afe_priv->clk[MT8195_CLK_AUD_PCMIF]);
222+
223+
return 0;
224+
}
225+
226+
static void mtk_dai_pcm_shutdown(struct snd_pcm_substream *substream,
227+
struct snd_soc_dai *dai)
228+
{
229+
struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
230+
struct mt8195_afe_private *afe_priv = afe->platform_priv;
231+
232+
if (dai->component->active)
233+
return;
234+
235+
mtk_dai_pcm_disable(afe);
236+
237+
mt8195_afe_disable_clk(afe, afe_priv->clk[MT8195_CLK_AUD_PCMIF]);
238+
mt8195_afe_disable_clk(afe, afe_priv->clk[MT8195_CLK_AUD_ASRC12]);
239+
mt8195_afe_disable_clk(afe, afe_priv->clk[MT8195_CLK_AUD_ASRC11]);
240+
}
241+
242+
static int mtk_dai_pcm_prepare(struct snd_pcm_substream *substream,
243+
struct snd_soc_dai *dai)
244+
{
245+
struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
246+
int ret = 0;
247+
248+
if (snd_soc_dai_stream_active(dai, SNDRV_PCM_STREAM_PLAYBACK) &&
249+
snd_soc_dai_stream_active(dai, SNDRV_PCM_STREAM_CAPTURE))
250+
return 0;
251+
252+
ret = mtk_dai_pcm_configure(substream, dai);
253+
if (ret)
254+
return ret;
255+
256+
mtk_dai_pcm_enable(afe);
257+
258+
return 0;
259+
}
260+
261+
static int mtk_dai_pcm_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
262+
{
263+
struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
264+
struct mt8195_afe_private *afe_priv = afe->platform_priv;
265+
struct mtk_dai_pcmif_priv *pcmif_priv = afe_priv->dai_priv[dai->id];
266+
267+
dev_dbg(dai->dev, "%s fmt 0x%x\n", __func__, fmt);
268+
269+
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
270+
case SND_SOC_DAIFMT_I2S:
271+
pcmif_priv->format = MTK_DAI_PCM_FMT_I2S;
272+
break;
273+
case SND_SOC_DAIFMT_DSP_A:
274+
pcmif_priv->format = MTK_DAI_PCM_FMT_MODEA;
275+
break;
276+
case SND_SOC_DAIFMT_DSP_B:
277+
pcmif_priv->format = MTK_DAI_PCM_FMT_MODEB;
278+
break;
279+
default:
280+
return -EINVAL;
281+
}
282+
283+
switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
284+
case SND_SOC_DAIFMT_NB_NF:
285+
pcmif_priv->bck_inv = 0;
286+
pcmif_priv->lrck_inv = 0;
287+
break;
288+
case SND_SOC_DAIFMT_NB_IF:
289+
pcmif_priv->bck_inv = 0;
290+
pcmif_priv->lrck_inv = 1;
291+
break;
292+
case SND_SOC_DAIFMT_IB_NF:
293+
pcmif_priv->bck_inv = 1;
294+
pcmif_priv->lrck_inv = 0;
295+
break;
296+
case SND_SOC_DAIFMT_IB_IF:
297+
pcmif_priv->bck_inv = 1;
298+
pcmif_priv->lrck_inv = 1;
299+
break;
300+
default:
301+
return -EINVAL;
302+
}
303+
304+
switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
305+
case SND_SOC_DAIFMT_CBM_CFM:
306+
pcmif_priv->slave_mode = 1;
307+
break;
308+
case SND_SOC_DAIFMT_CBS_CFS:
309+
pcmif_priv->slave_mode = 0;
310+
break;
311+
default:
312+
return -EINVAL;
313+
}
314+
315+
return 0;
316+
}
317+
318+
static const struct snd_soc_dai_ops mtk_dai_pcm_ops = {
319+
.startup = mtk_dai_pcm_startup,
320+
.shutdown = mtk_dai_pcm_shutdown,
321+
.prepare = mtk_dai_pcm_prepare,
322+
.set_fmt = mtk_dai_pcm_set_fmt,
323+
};
324+
325+
/* dai driver */
326+
#define MTK_PCM_RATES (SNDRV_PCM_RATE_8000_48000)
327+
328+
#define MTK_PCM_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\
329+
SNDRV_PCM_FMTBIT_S24_LE |\
330+
SNDRV_PCM_FMTBIT_S32_LE)
331+
332+
static struct snd_soc_dai_driver mtk_dai_pcm_driver[] = {
333+
{
334+
.name = "PCM1",
335+
.id = MT8195_AFE_IO_PCM,
336+
.playback = {
337+
.stream_name = "PCM1 Playback",
338+
.channels_min = 1,
339+
.channels_max = 2,
340+
.rates = MTK_PCM_RATES,
341+
.formats = MTK_PCM_FORMATS,
342+
},
343+
.capture = {
344+
.stream_name = "PCM1 Capture",
345+
.channels_min = 1,
346+
.channels_max = 2,
347+
.rates = MTK_PCM_RATES,
348+
.formats = MTK_PCM_FORMATS,
349+
},
350+
.ops = &mtk_dai_pcm_ops,
351+
.symmetric_rate = 1,
352+
.symmetric_sample_bits = 1,
353+
},
354+
};
355+
356+
static int init_pcmif_priv_data(struct mtk_base_afe *afe)
357+
{
358+
struct mt8195_afe_private *afe_priv = afe->platform_priv;
359+
struct mtk_dai_pcmif_priv *pcmif_priv;
360+
361+
pcmif_priv = devm_kzalloc(afe->dev, sizeof(struct mtk_dai_pcmif_priv),
362+
GFP_KERNEL);
363+
if (!pcmif_priv)
364+
return -ENOMEM;
365+
366+
afe_priv->dai_priv[MT8195_AFE_IO_PCM] = pcmif_priv;
367+
return 0;
368+
}
369+
370+
int mt8195_dai_pcm_register(struct mtk_base_afe *afe)
371+
{
372+
struct mtk_base_afe_dai *dai;
373+
374+
dai = devm_kzalloc(afe->dev, sizeof(*dai), GFP_KERNEL);
375+
if (!dai)
376+
return -ENOMEM;
377+
378+
list_add(&dai->list, &afe->sub_dais);
379+
380+
dai->dai_drivers = mtk_dai_pcm_driver;
381+
dai->num_dai_drivers = ARRAY_SIZE(mtk_dai_pcm_driver);
382+
383+
dai->dapm_widgets = mtk_dai_pcm_widgets;
384+
dai->num_dapm_widgets = ARRAY_SIZE(mtk_dai_pcm_widgets);
385+
dai->dapm_routes = mtk_dai_pcm_routes;
386+
dai->num_dapm_routes = ARRAY_SIZE(mtk_dai_pcm_routes);
387+
388+
return init_pcmif_priv_data(afe);
389+
}

0 commit comments

Comments
 (0)