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 <bicycle.tsai@mediatek.com>
7 *         Trevor Wu <trevor.wu@mediatek.com>
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
16enum {
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
23enum {
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
30struct mtk_dai_pcm_rate {
31	unsigned int rate;
32	unsigned int reg_value;
33};
34
35struct 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
42static 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
52static 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
63static 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
68static 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
73static 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_SUPPLY("PCM_EN", PCM_INTF_CON1,
84			    PCM_INTF_CON1_PCM_EN_SHIFT, 0, NULL, 0),
85
86	SND_SOC_DAPM_INPUT("PCM1_INPUT"),
87	SND_SOC_DAPM_OUTPUT("PCM1_OUTPUT"),
88
89	SND_SOC_DAPM_CLOCK_SUPPLY("aud_asrc11"),
90	SND_SOC_DAPM_CLOCK_SUPPLY("aud_asrc12"),
91	SND_SOC_DAPM_CLOCK_SUPPLY("aud_pcmif"),
92};
93
94static const struct snd_soc_dapm_route mtk_dai_pcm_routes[] = {
95	{"I002", NULL, "PCM1 Capture"},
96	{"I003", NULL, "PCM1 Capture"},
97
98	{"O000", "I000 Switch", "I000"},
99	{"O001", "I001 Switch", "I001"},
100
101	{"O000", "I070 Switch", "I070"},
102	{"O001", "I071 Switch", "I071"},
103
104	{"PCM1 Playback", NULL, "O000"},
105	{"PCM1 Playback", NULL, "O001"},
106
107	{"PCM1 Playback", NULL, "PCM_EN"},
108	{"PCM1 Playback", NULL, "aud_asrc12"},
109	{"PCM1 Playback", NULL, "aud_pcmif"},
110
111	{"PCM1 Capture", NULL, "PCM_EN"},
112	{"PCM1 Capture", NULL, "aud_asrc11"},
113	{"PCM1 Capture", NULL, "aud_pcmif"},
114
115	{"PCM1_OUTPUT", NULL, "PCM1 Playback"},
116	{"PCM1 Capture", NULL, "PCM1_INPUT"},
117};
118
119static int mtk_dai_pcm_configure(struct snd_pcm_substream *substream,
120				 struct snd_soc_dai *dai)
121{
122	struct snd_pcm_runtime * const runtime = substream->runtime;
123	struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
124	struct mt8195_afe_private *afe_priv = afe->platform_priv;
125	struct mtk_dai_pcmif_priv *pcmif_priv;
126	unsigned int slave_mode;
127	unsigned int lrck_inv;
128	unsigned int bck_inv;
129	unsigned int fmt;
130	unsigned int bit_width = dai->sample_bits;
131	unsigned int val = 0;
132	unsigned int mask = 0;
133	int fs = 0;
134	int mode = 0;
135
136	if (dai->id != MT8195_AFE_IO_PCM)
137		return -EINVAL;
138
139	pcmif_priv = afe_priv->dai_priv[dai->id];
140	slave_mode = pcmif_priv->slave_mode;
141	lrck_inv = pcmif_priv->lrck_inv;
142	bck_inv = pcmif_priv->bck_inv;
143	fmt = pcmif_priv->format;
144
145	/* sync freq mode */
146	fs = mt8195_afe_fs_timing(runtime->rate);
147	if (fs < 0)
148		return -EINVAL;
149	val |= PCM_INTF_CON2_SYNC_FREQ_MODE(fs);
150	mask |= PCM_INTF_CON2_SYNC_FREQ_MODE_MASK;
151
152	/* clk domain sel */
153	if (runtime->rate % 8000)
154		val |= PCM_INTF_CON2_CLK_DOMAIN_SEL(MTK_DAI_PCM_CLK_26M_441K);
155	else
156		val |= PCM_INTF_CON2_CLK_DOMAIN_SEL(MTK_DAI_PCM_CLK_26M_48K);
157	mask |= PCM_INTF_CON2_CLK_DOMAIN_SEL_MASK;
158
159	regmap_update_bits(afe->regmap, PCM_INTF_CON2, mask, val);
160
161	val = 0;
162	mask = 0;
163
164	/* pcm mode */
165	mode = mtk_dai_pcm_mode(runtime->rate);
166	if (mode < 0)
167		return -EINVAL;
168	val |= PCM_INTF_CON1_PCM_MODE(mode);
169	mask |= PCM_INTF_CON1_PCM_MODE_MASK;
170
171	/* pcm format */
172	val |= PCM_INTF_CON1_PCM_FMT(fmt);
173	mask |= PCM_INTF_CON1_PCM_FMT_MASK;
174
175	/* pcm sync length */
176	if (fmt == MTK_DAI_PCM_FMT_MODEA ||
177	    fmt == MTK_DAI_PCM_FMT_MODEB)
178		val |= PCM_INTF_CON1_SYNC_LENGTH(1);
179	else
180		val |= PCM_INTF_CON1_SYNC_LENGTH(bit_width);
181	mask |= PCM_INTF_CON1_SYNC_LENGTH_MASK;
182
183	/* pcm bits, word length */
184	if (bit_width > 16) {
185		val |= PCM_INTF_CON1_PCM_24BIT;
186		val |= PCM_INTF_CON1_PCM_WLEN_64BCK;
187	} else {
188		val |= PCM_INTF_CON1_PCM_16BIT;
189		val |= PCM_INTF_CON1_PCM_WLEN_32BCK;
190	}
191	mask |= PCM_INTF_CON1_PCM_BIT_MASK;
192	mask |= PCM_INTF_CON1_PCM_WLEN_MASK;
193
194	/* master/slave */
195	if (!slave_mode) {
196		val |= PCM_INTF_CON1_PCM_MASTER;
197
198		if (lrck_inv)
199			val |= PCM_INTF_CON1_SYNC_OUT_INV;
200		if (bck_inv)
201			val |= PCM_INTF_CON1_BCLK_OUT_INV;
202		mask |= PCM_INTF_CON1_CLK_OUT_INV_MASK;
203	} else {
204		val |= PCM_INTF_CON1_PCM_SLAVE;
205
206		if (lrck_inv)
207			val |= PCM_INTF_CON1_SYNC_IN_INV;
208		if (bck_inv)
209			val |= PCM_INTF_CON1_BCLK_IN_INV;
210		mask |= PCM_INTF_CON1_CLK_IN_INV_MASK;
211
212		/* TODO: add asrc setting for slave mode */
213	}
214	mask |= PCM_INTF_CON1_PCM_M_S_MASK;
215
216	regmap_update_bits(afe->regmap, PCM_INTF_CON1, mask, val);
217
218	return 0;
219}
220
221/* dai ops */
222static int mtk_dai_pcm_prepare(struct snd_pcm_substream *substream,
223			       struct snd_soc_dai *dai)
224{
225	struct snd_soc_dapm_widget *p = snd_soc_dai_get_widget_playback(dai);
226	struct snd_soc_dapm_widget *c = snd_soc_dai_get_widget_capture(dai);
227
228	dev_dbg(dai->dev, "%s(), id %d, stream %d, widget active p %d, c %d\n",
229		__func__, dai->id, substream->stream,
230		p->active, c->active);
231
232	if (p->active || c->active)
233		return 0;
234
235	return mtk_dai_pcm_configure(substream, dai);
236}
237
238static int mtk_dai_pcm_set_fmt(struct snd_soc_dai *dai, unsigned int fmt)
239{
240	struct mtk_base_afe *afe = snd_soc_dai_get_drvdata(dai);
241	struct mt8195_afe_private *afe_priv = afe->platform_priv;
242	struct mtk_dai_pcmif_priv *pcmif_priv;
243
244	dev_dbg(dai->dev, "%s fmt 0x%x\n", __func__, fmt);
245
246	if (dai->id != MT8195_AFE_IO_PCM)
247		return -EINVAL;
248
249	pcmif_priv = afe_priv->dai_priv[dai->id];
250
251	switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
252	case SND_SOC_DAIFMT_I2S:
253		pcmif_priv->format = MTK_DAI_PCM_FMT_I2S;
254		break;
255	case SND_SOC_DAIFMT_DSP_A:
256		pcmif_priv->format = MTK_DAI_PCM_FMT_MODEA;
257		break;
258	case SND_SOC_DAIFMT_DSP_B:
259		pcmif_priv->format = MTK_DAI_PCM_FMT_MODEB;
260		break;
261	default:
262		return -EINVAL;
263	}
264
265	switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
266	case SND_SOC_DAIFMT_NB_NF:
267		pcmif_priv->bck_inv = 0;
268		pcmif_priv->lrck_inv = 0;
269		break;
270	case SND_SOC_DAIFMT_NB_IF:
271		pcmif_priv->bck_inv = 0;
272		pcmif_priv->lrck_inv = 1;
273		break;
274	case SND_SOC_DAIFMT_IB_NF:
275		pcmif_priv->bck_inv = 1;
276		pcmif_priv->lrck_inv = 0;
277		break;
278	case SND_SOC_DAIFMT_IB_IF:
279		pcmif_priv->bck_inv = 1;
280		pcmif_priv->lrck_inv = 1;
281		break;
282	default:
283		return -EINVAL;
284	}
285
286	switch (fmt & SND_SOC_DAIFMT_CLOCK_PROVIDER_MASK) {
287	case SND_SOC_DAIFMT_BC_FC:
288		pcmif_priv->slave_mode = 1;
289		break;
290	case SND_SOC_DAIFMT_BP_FP:
291		pcmif_priv->slave_mode = 0;
292		break;
293	default:
294		return -EINVAL;
295	}
296
297	return 0;
298}
299
300static const struct snd_soc_dai_ops mtk_dai_pcm_ops = {
301	.prepare	= mtk_dai_pcm_prepare,
302	.set_fmt	= mtk_dai_pcm_set_fmt,
303};
304
305/* dai driver */
306#define MTK_PCM_RATES (SNDRV_PCM_RATE_8000_48000)
307
308#define MTK_PCM_FORMATS (SNDRV_PCM_FMTBIT_S16_LE |\
309			 SNDRV_PCM_FMTBIT_S24_LE |\
310			 SNDRV_PCM_FMTBIT_S32_LE)
311
312static struct snd_soc_dai_driver mtk_dai_pcm_driver[] = {
313	{
314		.name = "PCM1",
315		.id = MT8195_AFE_IO_PCM,
316		.playback = {
317			.stream_name = "PCM1 Playback",
318			.channels_min = 1,
319			.channels_max = 2,
320			.rates = MTK_PCM_RATES,
321			.formats = MTK_PCM_FORMATS,
322		},
323		.capture = {
324			.stream_name = "PCM1 Capture",
325			.channels_min = 1,
326			.channels_max = 2,
327			.rates = MTK_PCM_RATES,
328			.formats = MTK_PCM_FORMATS,
329		},
330		.ops = &mtk_dai_pcm_ops,
331		.symmetric_rate = 1,
332		.symmetric_sample_bits = 1,
333	},
334};
335
336static int init_pcmif_priv_data(struct mtk_base_afe *afe)
337{
338	struct mt8195_afe_private *afe_priv = afe->platform_priv;
339	struct mtk_dai_pcmif_priv *pcmif_priv;
340
341	pcmif_priv = devm_kzalloc(afe->dev, sizeof(struct mtk_dai_pcmif_priv),
342				  GFP_KERNEL);
343	if (!pcmif_priv)
344		return -ENOMEM;
345
346	afe_priv->dai_priv[MT8195_AFE_IO_PCM] = pcmif_priv;
347	return 0;
348}
349
350int mt8195_dai_pcm_register(struct mtk_base_afe *afe)
351{
352	struct mtk_base_afe_dai *dai;
353
354	dai = devm_kzalloc(afe->dev, sizeof(*dai), GFP_KERNEL);
355	if (!dai)
356		return -ENOMEM;
357
358	list_add(&dai->list, &afe->sub_dais);
359
360	dai->dai_drivers = mtk_dai_pcm_driver;
361	dai->num_dai_drivers = ARRAY_SIZE(mtk_dai_pcm_driver);
362
363	dai->dapm_widgets = mtk_dai_pcm_widgets;
364	dai->num_dapm_widgets = ARRAY_SIZE(mtk_dai_pcm_widgets);
365	dai->dapm_routes = mtk_dai_pcm_routes;
366	dai->num_dapm_routes = ARRAY_SIZE(mtk_dai_pcm_routes);
367
368	return init_pcmif_priv_data(afe);
369}
370