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