xref: /openbmc/linux/sound/soc/mediatek/mt8186/mt8186-mt6366-da7219-max98357.c (revision bbdd33769d319d1e7bb8fec09124a49b3573a2d3)
1 // SPDX-License-Identifier: GPL-2.0
2 //
3 // mt8186-mt6366-da7219-max98357.c
4 //	--  MT8186-MT6366-DA7219-MAX98357 ALSA SoC machine driver
5 //
6 // Copyright (c) 2022 MediaTek Inc.
7 // Author: Jiaxin Yu <jiaxin.yu@mediatek.com>
8 //
9 
10 #include <linux/input.h>
11 #include <linux/module.h>
12 #include <linux/of_device.h>
13 #include <linux/pm_runtime.h>
14 #include <sound/jack.h>
15 #include <sound/pcm_params.h>
16 #include <sound/soc.h>
17 
18 #include "../../codecs/da7219.h"
19 #include "../../codecs/mt6358.h"
20 #include "../common/mtk-afe-platform-driver.h"
21 #include "../common/mtk-dsp-sof-common.h"
22 #include "../common/mtk-soc-card.h"
23 #include "mt8186-afe-common.h"
24 #include "mt8186-afe-clk.h"
25 #include "mt8186-afe-gpio.h"
26 #include "mt8186-mt6366-common.h"
27 
28 #define DA7219_CODEC_DAI "da7219-hifi"
29 #define DA7219_DEV_NAME "da7219.5-001a"
30 
31 #define SOF_DMA_DL1 "SOF_DMA_DL1"
32 #define SOF_DMA_DL2 "SOF_DMA_DL2"
33 #define SOF_DMA_UL1 "SOF_DMA_UL1"
34 #define SOF_DMA_UL2 "SOF_DMA_UL2"
35 
36 struct mt8186_mt6366_da7219_max98357_priv {
37 	struct snd_soc_jack headset_jack, hdmi_jack;
38 };
39 
40 /* Headset jack detection DAPM pins */
41 static struct snd_soc_jack_pin mt8186_jack_pins[] = {
42 	{
43 		.pin = "Headphones",
44 		.mask = SND_JACK_HEADPHONE,
45 	},
46 	{
47 		.pin = "Headset Mic",
48 		.mask = SND_JACK_MICROPHONE,
49 	},
50 };
51 
52 static struct snd_soc_codec_conf mt8186_mt6366_da7219_max98357_codec_conf[] = {
53 	{
54 		.dlc = COMP_CODEC_CONF("mt6358-sound"),
55 		.name_prefix = "Mt6366",
56 	},
57 	{
58 		.dlc = COMP_CODEC_CONF("bt-sco"),
59 		.name_prefix = "Mt8186 bt",
60 	},
61 	{
62 		.dlc = COMP_CODEC_CONF("hdmi-audio-codec"),
63 		.name_prefix = "Mt8186 hdmi",
64 	},
65 };
66 
67 static int mt8186_da7219_init(struct snd_soc_pcm_runtime *rtd)
68 {
69 	struct snd_soc_component *cmpnt_afe =
70 		snd_soc_rtdcom_lookup(rtd, AFE_PCM_NAME);
71 	struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt_afe);
72 	struct mtk_soc_card_data *soc_card_data =
73 		snd_soc_card_get_drvdata(rtd->card);
74 	struct mt8186_mt6366_da7219_max98357_priv *priv = soc_card_data->mach_priv;
75 	struct snd_soc_jack *jack = &priv->headset_jack;
76 	struct snd_soc_component *cmpnt_codec =
77 		asoc_rtd_to_codec(rtd, 0)->component;
78 	int ret;
79 
80 	ret = mt8186_dai_i2s_set_share(afe, "I2S1", "I2S0");
81 	if (ret) {
82 		dev_err(rtd->dev, "Failed to set up shared clocks\n");
83 		return ret;
84 	}
85 
86 	/* Enable Headset and 4 Buttons Jack detection */
87 	ret = snd_soc_card_jack_new_pins(rtd->card, "Headset Jack",
88 				    SND_JACK_HEADSET | SND_JACK_BTN_0 |
89 				    SND_JACK_BTN_1 | SND_JACK_BTN_2 |
90 				    SND_JACK_BTN_3 | SND_JACK_LINEOUT,
91 				    jack, mt8186_jack_pins,
92 				    ARRAY_SIZE(mt8186_jack_pins));
93 	if (ret) {
94 		dev_err(rtd->dev, "Headset Jack creation failed: %d\n", ret);
95 		return ret;
96 	}
97 
98 	snd_jack_set_key(jack->jack, SND_JACK_BTN_0, KEY_PLAYPAUSE);
99 	snd_jack_set_key(jack->jack, SND_JACK_BTN_1, KEY_VOLUMEUP);
100 	snd_jack_set_key(jack->jack, SND_JACK_BTN_2, KEY_VOLUMEDOWN);
101 	snd_jack_set_key(jack->jack, SND_JACK_BTN_3, KEY_VOICECOMMAND);
102 
103 	snd_soc_component_set_jack(cmpnt_codec, &priv->headset_jack, NULL);
104 
105 	return 0;
106 }
107 
108 static int mt8186_da7219_i2s_hw_params(struct snd_pcm_substream *substream,
109 				       struct snd_pcm_hw_params *params)
110 {
111 	struct snd_soc_pcm_runtime *rtd = asoc_substream_to_rtd(substream);
112 	struct snd_soc_dai *codec_dai;
113 	unsigned int rate = params_rate(params);
114 	unsigned int mclk_fs_ratio = 256;
115 	unsigned int mclk_fs = rate * mclk_fs_ratio;
116 	unsigned int freq;
117 	int ret, j;
118 
119 	ret = snd_soc_dai_set_sysclk(asoc_rtd_to_cpu(rtd, 0), 0,
120 				     mclk_fs, SND_SOC_CLOCK_OUT);
121 	if (ret < 0) {
122 		dev_err(rtd->dev, "failed to set cpu dai sysclk: %d\n", ret);
123 		return ret;
124 	}
125 
126 	for_each_rtd_codec_dais(rtd, j, codec_dai) {
127 		if (!strcmp(codec_dai->component->name, DA7219_DEV_NAME)) {
128 			ret = snd_soc_dai_set_sysclk(codec_dai,
129 						     DA7219_CLKSRC_MCLK,
130 						     mclk_fs,
131 						     SND_SOC_CLOCK_IN);
132 			if (ret < 0) {
133 				dev_err(rtd->dev, "failed to set sysclk: %d\n",
134 					ret);
135 				return ret;
136 			}
137 
138 			if ((rate % 8000) == 0)
139 				freq = DA7219_PLL_FREQ_OUT_98304;
140 			else
141 				freq = DA7219_PLL_FREQ_OUT_90316;
142 
143 			ret = snd_soc_dai_set_pll(codec_dai, 0,
144 						  DA7219_SYSCLK_PLL_SRM,
145 						  0, freq);
146 			if (ret) {
147 				dev_err(rtd->dev, "failed to start PLL: %d\n",
148 					ret);
149 				return ret;
150 			}
151 		}
152 	}
153 
154 	return 0;
155 }
156 
157 static int mt8186_da7219_i2s_hw_free(struct snd_pcm_substream *substream)
158 {
159 	struct snd_soc_pcm_runtime *rtd = asoc_substream_to_rtd(substream);
160 	struct snd_soc_dai *codec_dai;
161 	int ret = 0, j;
162 
163 	for_each_rtd_codec_dais(rtd, j, codec_dai) {
164 		if (!strcmp(codec_dai->component->name, DA7219_DEV_NAME)) {
165 			ret = snd_soc_dai_set_pll(codec_dai,
166 						  0, DA7219_SYSCLK_MCLK, 0, 0);
167 			if (ret < 0) {
168 				dev_err(rtd->dev, "failed to stop PLL: %d\n",
169 					ret);
170 				return ret;
171 			}
172 		}
173 	}
174 
175 	return 0;
176 }
177 
178 static const struct snd_soc_ops mt8186_da7219_i2s_ops = {
179 	.hw_params = mt8186_da7219_i2s_hw_params,
180 	.hw_free = mt8186_da7219_i2s_hw_free,
181 };
182 
183 static int mt8186_mt6366_da7219_max98357_hdmi_init(struct snd_soc_pcm_runtime *rtd)
184 {
185 	struct snd_soc_component *cmpnt_afe =
186 		snd_soc_rtdcom_lookup(rtd, AFE_PCM_NAME);
187 	struct mtk_base_afe *afe = snd_soc_component_get_drvdata(cmpnt_afe);
188 	struct snd_soc_component *cmpnt_codec =
189 		asoc_rtd_to_codec(rtd, 0)->component;
190 	struct mtk_soc_card_data *soc_card_data =
191 		snd_soc_card_get_drvdata(rtd->card);
192 	struct mt8186_mt6366_da7219_max98357_priv *priv = soc_card_data->mach_priv;
193 	int ret;
194 
195 	ret = mt8186_dai_i2s_set_share(afe, "I2S2", "I2S3");
196 	if (ret) {
197 		dev_err(rtd->dev, "Failed to set up shared clocks\n");
198 		return ret;
199 	}
200 
201 	ret = snd_soc_card_jack_new(rtd->card, "HDMI Jack", SND_JACK_LINEOUT, &priv->hdmi_jack);
202 	if (ret) {
203 		dev_err(rtd->dev, "HDMI Jack creation failed: %d\n", ret);
204 		return ret;
205 	}
206 
207 	return snd_soc_component_set_jack(cmpnt_codec, &priv->hdmi_jack, NULL);
208 }
209 
210 static int mt8186_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
211 				  struct snd_pcm_hw_params *params,
212 				  snd_pcm_format_t fmt)
213 {
214 	struct snd_interval *channels = hw_param_interval(params,
215 		SNDRV_PCM_HW_PARAM_CHANNELS);
216 
217 	dev_dbg(rtd->dev, "%s(), fix format to %d\n", __func__, fmt);
218 
219 	/* fix BE i2s channel to 2 channel */
220 	channels->min = 2;
221 	channels->max = 2;
222 
223 	/* clean param mask first */
224 	snd_mask_reset_range(hw_param_mask(params, SNDRV_PCM_HW_PARAM_FORMAT),
225 			     0, (__force unsigned int)SNDRV_PCM_FORMAT_LAST);
226 
227 	params_set_format(params, fmt);
228 
229 	return 0;
230 }
231 
232 static int mt8186_i2s_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
233 				      struct snd_pcm_hw_params *params)
234 {
235 	return mt8186_hw_params_fixup(rtd, params, SNDRV_PCM_FORMAT_S32_LE);
236 }
237 
238 static int mt8186_anx7625_i2s_hw_params_fixup(struct snd_soc_pcm_runtime *rtd,
239 					      struct snd_pcm_hw_params *params)
240 {
241 	return mt8186_hw_params_fixup(rtd, params, SNDRV_PCM_FORMAT_S24_LE);
242 }
243 
244 /* fixup the BE DAI link to match any values from topology */
245 static int mt8186_sof_dai_link_fixup(struct snd_soc_pcm_runtime *rtd,
246 				     struct snd_pcm_hw_params *params)
247 {
248 	int ret;
249 
250 	ret = mtk_sof_dai_link_fixup(rtd, params);
251 
252 	if (!strcmp(rtd->dai_link->name, "I2S0") ||
253 	    !strcmp(rtd->dai_link->name, "I2S1") ||
254 	    !strcmp(rtd->dai_link->name, "I2S2"))
255 		mt8186_i2s_hw_params_fixup(rtd, params);
256 	else if (!strcmp(rtd->dai_link->name, "I2S3"))
257 		mt8186_anx7625_i2s_hw_params_fixup(rtd, params);
258 
259 	return ret;
260 }
261 
262 static int mt8186_mt6366_da7219_max98357_playback_startup(struct snd_pcm_substream *substream)
263 {
264 	static const unsigned int rates[] = {
265 		48000
266 	};
267 	static const unsigned int channels[] = {
268 		2
269 	};
270 	static const struct snd_pcm_hw_constraint_list constraints_rates = {
271 		.count = ARRAY_SIZE(rates),
272 		.list  = rates,
273 		.mask = 0,
274 	};
275 	static const struct snd_pcm_hw_constraint_list constraints_channels = {
276 		.count = ARRAY_SIZE(channels),
277 		.list  = channels,
278 		.mask = 0,
279 	};
280 
281 	struct snd_soc_pcm_runtime *rtd = asoc_substream_to_rtd(substream);
282 	struct snd_pcm_runtime *runtime = substream->runtime;
283 	int ret;
284 
285 	ret = snd_pcm_hw_constraint_list(runtime, 0,
286 					 SNDRV_PCM_HW_PARAM_RATE,
287 					 &constraints_rates);
288 	if (ret < 0) {
289 		dev_err(rtd->dev, "hw_constraint_list rate failed\n");
290 		return ret;
291 	}
292 
293 	ret = snd_pcm_hw_constraint_list(runtime, 0,
294 					 SNDRV_PCM_HW_PARAM_CHANNELS,
295 					 &constraints_channels);
296 	if (ret < 0) {
297 		dev_err(rtd->dev, "hw_constraint_list channel failed\n");
298 		return ret;
299 	}
300 
301 	return 0;
302 }
303 
304 static const struct snd_soc_ops mt8186_mt6366_da7219_max98357_playback_ops = {
305 	.startup = mt8186_mt6366_da7219_max98357_playback_startup,
306 };
307 
308 static int mt8186_mt6366_da7219_max98357_capture_startup(struct snd_pcm_substream *substream)
309 {
310 	static const unsigned int rates[] = {
311 		48000
312 	};
313 	static const unsigned int channels[] = {
314 		1, 2
315 	};
316 	static const struct snd_pcm_hw_constraint_list constraints_rates = {
317 		.count = ARRAY_SIZE(rates),
318 		.list  = rates,
319 		.mask = 0,
320 	};
321 	static const struct snd_pcm_hw_constraint_list constraints_channels = {
322 		.count = ARRAY_SIZE(channels),
323 		.list  = channels,
324 		.mask = 0,
325 	};
326 
327 	struct snd_soc_pcm_runtime *rtd = asoc_substream_to_rtd(substream);
328 	struct snd_pcm_runtime *runtime = substream->runtime;
329 	int ret;
330 
331 	ret = snd_pcm_hw_constraint_list(runtime, 0,
332 					 SNDRV_PCM_HW_PARAM_RATE,
333 					 &constraints_rates);
334 	if (ret < 0) {
335 		dev_err(rtd->dev, "hw_constraint_list rate failed\n");
336 		return ret;
337 	}
338 
339 	ret = snd_pcm_hw_constraint_list(runtime, 0,
340 					 SNDRV_PCM_HW_PARAM_CHANNELS,
341 					 &constraints_channels);
342 	if (ret < 0) {
343 		dev_err(rtd->dev, "hw_constraint_list channel failed\n");
344 		return ret;
345 	}
346 
347 	return 0;
348 }
349 
350 static const struct snd_soc_ops mt8186_mt6366_da7219_max98357_capture_ops = {
351 	.startup = mt8186_mt6366_da7219_max98357_capture_startup,
352 };
353 
354 /* FE */
355 SND_SOC_DAILINK_DEFS(playback1,
356 		     DAILINK_COMP_ARRAY(COMP_CPU("DL1")),
357 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
358 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
359 
360 SND_SOC_DAILINK_DEFS(playback12,
361 		     DAILINK_COMP_ARRAY(COMP_CPU("DL12")),
362 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
363 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
364 
365 SND_SOC_DAILINK_DEFS(playback2,
366 		     DAILINK_COMP_ARRAY(COMP_CPU("DL2")),
367 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
368 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
369 
370 SND_SOC_DAILINK_DEFS(playback3,
371 		     DAILINK_COMP_ARRAY(COMP_CPU("DL3")),
372 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
373 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
374 
375 SND_SOC_DAILINK_DEFS(playback4,
376 		     DAILINK_COMP_ARRAY(COMP_CPU("DL4")),
377 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
378 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
379 
380 SND_SOC_DAILINK_DEFS(playback5,
381 		     DAILINK_COMP_ARRAY(COMP_CPU("DL5")),
382 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
383 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
384 
385 SND_SOC_DAILINK_DEFS(playback6,
386 		     DAILINK_COMP_ARRAY(COMP_CPU("DL6")),
387 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
388 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
389 
390 SND_SOC_DAILINK_DEFS(playback7,
391 		     DAILINK_COMP_ARRAY(COMP_CPU("DL7")),
392 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
393 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
394 
395 SND_SOC_DAILINK_DEFS(playback8,
396 		     DAILINK_COMP_ARRAY(COMP_CPU("DL8")),
397 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
398 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
399 
400 SND_SOC_DAILINK_DEFS(capture1,
401 		     DAILINK_COMP_ARRAY(COMP_CPU("UL1")),
402 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
403 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
404 
405 SND_SOC_DAILINK_DEFS(capture2,
406 		     DAILINK_COMP_ARRAY(COMP_CPU("UL2")),
407 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
408 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
409 
410 SND_SOC_DAILINK_DEFS(capture3,
411 		     DAILINK_COMP_ARRAY(COMP_CPU("UL3")),
412 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
413 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
414 
415 SND_SOC_DAILINK_DEFS(capture4,
416 		     DAILINK_COMP_ARRAY(COMP_CPU("UL4")),
417 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
418 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
419 
420 SND_SOC_DAILINK_DEFS(capture5,
421 		     DAILINK_COMP_ARRAY(COMP_CPU("UL5")),
422 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
423 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
424 
425 SND_SOC_DAILINK_DEFS(capture6,
426 		     DAILINK_COMP_ARRAY(COMP_CPU("UL6")),
427 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
428 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
429 
430 SND_SOC_DAILINK_DEFS(capture7,
431 		     DAILINK_COMP_ARRAY(COMP_CPU("UL7")),
432 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
433 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
434 
435 /* hostless */
436 SND_SOC_DAILINK_DEFS(hostless_lpbk,
437 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless LPBK DAI")),
438 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
439 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
440 SND_SOC_DAILINK_DEFS(hostless_fm,
441 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless FM DAI")),
442 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
443 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
444 SND_SOC_DAILINK_DEFS(hostless_src1,
445 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless_SRC_1_DAI")),
446 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
447 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
448 SND_SOC_DAILINK_DEFS(hostless_src_bargein,
449 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless_SRC_Bargein_DAI")),
450 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
451 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
452 
453 /* BE */
454 SND_SOC_DAILINK_DEFS(adda,
455 		     DAILINK_COMP_ARRAY(COMP_CPU("ADDA")),
456 		     DAILINK_COMP_ARRAY(COMP_CODEC("mt6358-sound",
457 						   "mt6358-snd-codec-aif1"),
458 					COMP_CODEC("dmic-codec",
459 						   "dmic-hifi")),
460 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
461 SND_SOC_DAILINK_DEFS(i2s0,
462 		     DAILINK_COMP_ARRAY(COMP_CPU("I2S0")),
463 		     DAILINK_COMP_ARRAY(COMP_EMPTY()),
464 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
465 SND_SOC_DAILINK_DEFS(i2s1,
466 		     DAILINK_COMP_ARRAY(COMP_CPU("I2S1")),
467 		     DAILINK_COMP_ARRAY(COMP_EMPTY()),
468 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
469 SND_SOC_DAILINK_DEFS(i2s2,
470 		     DAILINK_COMP_ARRAY(COMP_CPU("I2S2")),
471 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
472 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
473 SND_SOC_DAILINK_DEFS(i2s3,
474 		     DAILINK_COMP_ARRAY(COMP_CPU("I2S3")),
475 		     DAILINK_COMP_ARRAY(COMP_EMPTY()),
476 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
477 SND_SOC_DAILINK_DEFS(hw_gain1,
478 		     DAILINK_COMP_ARRAY(COMP_CPU("HW Gain 1")),
479 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
480 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
481 SND_SOC_DAILINK_DEFS(hw_gain2,
482 		     DAILINK_COMP_ARRAY(COMP_CPU("HW Gain 2")),
483 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
484 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
485 SND_SOC_DAILINK_DEFS(hw_src1,
486 		     DAILINK_COMP_ARRAY(COMP_CPU("HW_SRC_1")),
487 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
488 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
489 SND_SOC_DAILINK_DEFS(hw_src2,
490 		     DAILINK_COMP_ARRAY(COMP_CPU("HW_SRC_2")),
491 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
492 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
493 SND_SOC_DAILINK_DEFS(connsys_i2s,
494 		     DAILINK_COMP_ARRAY(COMP_CPU("CONNSYS_I2S")),
495 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
496 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
497 SND_SOC_DAILINK_DEFS(pcm1,
498 		     DAILINK_COMP_ARRAY(COMP_CPU("PCM 1")),
499 		     DAILINK_COMP_ARRAY(COMP_CODEC("bt-sco", "bt-sco-pcm-wb")),
500 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
501 SND_SOC_DAILINK_DEFS(tdm_in,
502 		     DAILINK_COMP_ARRAY(COMP_CPU("TDM IN")),
503 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
504 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
505 
506 /* hostless */
507 SND_SOC_DAILINK_DEFS(hostless_ul1,
508 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless_UL1 DAI")),
509 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
510 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
511 SND_SOC_DAILINK_DEFS(hostless_ul2,
512 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless_UL2 DAI")),
513 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
514 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
515 SND_SOC_DAILINK_DEFS(hostless_ul3,
516 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless_UL3 DAI")),
517 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
518 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
519 SND_SOC_DAILINK_DEFS(hostless_ul5,
520 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless_UL5 DAI")),
521 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
522 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
523 SND_SOC_DAILINK_DEFS(hostless_ul6,
524 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless_UL6 DAI")),
525 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
526 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
527 SND_SOC_DAILINK_DEFS(hostless_hw_gain_aaudio,
528 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless HW Gain AAudio DAI")),
529 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
530 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
531 SND_SOC_DAILINK_DEFS(hostless_src_aaudio,
532 		     DAILINK_COMP_ARRAY(COMP_CPU("Hostless SRC AAudio DAI")),
533 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
534 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
535 SND_SOC_DAILINK_DEFS(AFE_SOF_DL1,
536 		     DAILINK_COMP_ARRAY(COMP_CPU("SOF_DL1")),
537 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
538 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
539 
540 SND_SOC_DAILINK_DEFS(AFE_SOF_DL2,
541 		     DAILINK_COMP_ARRAY(COMP_CPU("SOF_DL2")),
542 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
543 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
544 
545 SND_SOC_DAILINK_DEFS(AFE_SOF_UL1,
546 		     DAILINK_COMP_ARRAY(COMP_CPU("SOF_UL1")),
547 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
548 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
549 
550 SND_SOC_DAILINK_DEFS(AFE_SOF_UL2,
551 		     DAILINK_COMP_ARRAY(COMP_CPU("SOF_UL2")),
552 		     DAILINK_COMP_ARRAY(COMP_DUMMY()),
553 		     DAILINK_COMP_ARRAY(COMP_EMPTY()));
554 
555 static const struct sof_conn_stream g_sof_conn_streams[] = {
556 	{ "I2S1", "AFE_SOF_DL1", SOF_DMA_DL1, SNDRV_PCM_STREAM_PLAYBACK},
557 	{ "I2S3", "AFE_SOF_DL2", SOF_DMA_DL2, SNDRV_PCM_STREAM_PLAYBACK},
558 	{ "Primary Codec", "AFE_SOF_UL1", SOF_DMA_UL1, SNDRV_PCM_STREAM_CAPTURE},
559 	{ "I2S0", "AFE_SOF_UL2", SOF_DMA_UL2, SNDRV_PCM_STREAM_CAPTURE},
560 };
561 
562 static struct snd_soc_dai_link mt8186_mt6366_da7219_max98357_dai_links[] = {
563 	/* Front End DAI links */
564 	{
565 		.name = "Playback_1",
566 		.stream_name = "Playback_1",
567 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
568 			    SND_SOC_DPCM_TRIGGER_PRE},
569 		.dynamic = 1,
570 		.dpcm_playback = 1,
571 		.dpcm_merged_format = 1,
572 		.dpcm_merged_chan = 1,
573 		.dpcm_merged_rate = 1,
574 		.ops = &mt8186_mt6366_da7219_max98357_playback_ops,
575 		SND_SOC_DAILINK_REG(playback1),
576 	},
577 	{
578 		.name = "Playback_12",
579 		.stream_name = "Playback_12",
580 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
581 			    SND_SOC_DPCM_TRIGGER_PRE},
582 		.dynamic = 1,
583 		.dpcm_playback = 1,
584 		SND_SOC_DAILINK_REG(playback12),
585 	},
586 	{
587 		.name = "Playback_2",
588 		.stream_name = "Playback_2",
589 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
590 			    SND_SOC_DPCM_TRIGGER_PRE},
591 		.dynamic = 1,
592 		.dpcm_playback = 1,
593 		.dpcm_merged_format = 1,
594 		.dpcm_merged_chan = 1,
595 		.dpcm_merged_rate = 1,
596 		SND_SOC_DAILINK_REG(playback2),
597 	},
598 	{
599 		.name = "Playback_3",
600 		.stream_name = "Playback_3",
601 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
602 			    SND_SOC_DPCM_TRIGGER_PRE},
603 		.dynamic = 1,
604 		.dpcm_playback = 1,
605 		.dpcm_merged_format = 1,
606 		.dpcm_merged_chan = 1,
607 		.dpcm_merged_rate = 1,
608 		.ops = &mt8186_mt6366_da7219_max98357_playback_ops,
609 		SND_SOC_DAILINK_REG(playback3),
610 	},
611 	{
612 		.name = "Playback_4",
613 		.stream_name = "Playback_4",
614 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
615 			    SND_SOC_DPCM_TRIGGER_PRE},
616 		.dynamic = 1,
617 		.dpcm_playback = 1,
618 		SND_SOC_DAILINK_REG(playback4),
619 	},
620 	{
621 		.name = "Playback_5",
622 		.stream_name = "Playback_5",
623 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
624 			    SND_SOC_DPCM_TRIGGER_PRE},
625 		.dynamic = 1,
626 		.dpcm_playback = 1,
627 		SND_SOC_DAILINK_REG(playback5),
628 	},
629 	{
630 		.name = "Playback_6",
631 		.stream_name = "Playback_6",
632 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
633 			    SND_SOC_DPCM_TRIGGER_PRE},
634 		.dynamic = 1,
635 		.dpcm_playback = 1,
636 		SND_SOC_DAILINK_REG(playback6),
637 	},
638 	{
639 		.name = "Playback_7",
640 		.stream_name = "Playback_7",
641 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
642 			    SND_SOC_DPCM_TRIGGER_PRE},
643 		.dynamic = 1,
644 		.dpcm_playback = 1,
645 		SND_SOC_DAILINK_REG(playback7),
646 	},
647 	{
648 		.name = "Playback_8",
649 		.stream_name = "Playback_8",
650 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
651 			    SND_SOC_DPCM_TRIGGER_PRE},
652 		.dynamic = 1,
653 		.dpcm_playback = 1,
654 		SND_SOC_DAILINK_REG(playback8),
655 	},
656 	{
657 		.name = "Capture_1",
658 		.stream_name = "Capture_1",
659 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
660 			    SND_SOC_DPCM_TRIGGER_PRE},
661 		.dynamic = 1,
662 		.dpcm_capture = 1,
663 		SND_SOC_DAILINK_REG(capture1),
664 	},
665 	{
666 		.name = "Capture_2",
667 		.stream_name = "Capture_2",
668 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
669 			    SND_SOC_DPCM_TRIGGER_PRE},
670 		.dynamic = 1,
671 		.dpcm_capture = 1,
672 		.dpcm_merged_format = 1,
673 		.dpcm_merged_chan = 1,
674 		.dpcm_merged_rate = 1,
675 		.ops = &mt8186_mt6366_da7219_max98357_capture_ops,
676 		SND_SOC_DAILINK_REG(capture2),
677 	},
678 	{
679 		.name = "Capture_3",
680 		.stream_name = "Capture_3",
681 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
682 			    SND_SOC_DPCM_TRIGGER_PRE},
683 		.dynamic = 1,
684 		.dpcm_capture = 1,
685 		SND_SOC_DAILINK_REG(capture3),
686 	},
687 	{
688 		.name = "Capture_4",
689 		.stream_name = "Capture_4",
690 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
691 			    SND_SOC_DPCM_TRIGGER_PRE},
692 		.dynamic = 1,
693 		.dpcm_capture = 1,
694 		.dpcm_merged_format = 1,
695 		.dpcm_merged_chan = 1,
696 		.dpcm_merged_rate = 1,
697 		.ops = &mt8186_mt6366_da7219_max98357_capture_ops,
698 		SND_SOC_DAILINK_REG(capture4),
699 	},
700 	{
701 		.name = "Capture_5",
702 		.stream_name = "Capture_5",
703 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
704 			    SND_SOC_DPCM_TRIGGER_PRE},
705 		.dynamic = 1,
706 		.dpcm_capture = 1,
707 		SND_SOC_DAILINK_REG(capture5),
708 	},
709 	{
710 		.name = "Capture_6",
711 		.stream_name = "Capture_6",
712 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
713 			    SND_SOC_DPCM_TRIGGER_PRE},
714 		.dynamic = 1,
715 		.dpcm_capture = 1,
716 		.dpcm_merged_format = 1,
717 		.dpcm_merged_chan = 1,
718 		.dpcm_merged_rate = 1,
719 		SND_SOC_DAILINK_REG(capture6),
720 	},
721 	{
722 		.name = "Capture_7",
723 		.stream_name = "Capture_7",
724 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
725 			    SND_SOC_DPCM_TRIGGER_PRE},
726 		.dynamic = 1,
727 		.dpcm_capture = 1,
728 		SND_SOC_DAILINK_REG(capture7),
729 	},
730 	{
731 		.name = "Hostless_LPBK",
732 		.stream_name = "Hostless_LPBK",
733 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
734 			    SND_SOC_DPCM_TRIGGER_PRE},
735 		.dynamic = 1,
736 		.dpcm_playback = 1,
737 		.dpcm_capture = 1,
738 		.ignore_suspend = 1,
739 		SND_SOC_DAILINK_REG(hostless_lpbk),
740 	},
741 	{
742 		.name = "Hostless_FM",
743 		.stream_name = "Hostless_FM",
744 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
745 			    SND_SOC_DPCM_TRIGGER_PRE},
746 		.dynamic = 1,
747 		.dpcm_playback = 1,
748 		.dpcm_capture = 1,
749 		.ignore_suspend = 1,
750 		SND_SOC_DAILINK_REG(hostless_fm),
751 	},
752 	{
753 		.name = "Hostless_SRC_1",
754 		.stream_name = "Hostless_SRC_1",
755 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
756 			    SND_SOC_DPCM_TRIGGER_PRE},
757 		.dynamic = 1,
758 		.dpcm_playback = 1,
759 		.dpcm_capture = 1,
760 		.ignore_suspend = 1,
761 		SND_SOC_DAILINK_REG(hostless_src1),
762 	},
763 	{
764 		.name = "Hostless_SRC_Bargein",
765 		.stream_name = "Hostless_SRC_Bargein",
766 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
767 			    SND_SOC_DPCM_TRIGGER_PRE},
768 		.dynamic = 1,
769 		.dpcm_playback = 1,
770 		.dpcm_capture = 1,
771 		.ignore_suspend = 1,
772 		SND_SOC_DAILINK_REG(hostless_src_bargein),
773 	},
774 	{
775 		.name = "Hostless_HW_Gain_AAudio",
776 		.stream_name = "Hostless_HW_Gain_AAudio",
777 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
778 			    SND_SOC_DPCM_TRIGGER_PRE},
779 		.dynamic = 1,
780 		.dpcm_capture = 1,
781 		.ignore_suspend = 1,
782 		SND_SOC_DAILINK_REG(hostless_hw_gain_aaudio),
783 	},
784 	{
785 		.name = "Hostless_SRC_AAudio",
786 		.stream_name = "Hostless_SRC_AAudio",
787 		.trigger = {SND_SOC_DPCM_TRIGGER_PRE,
788 			    SND_SOC_DPCM_TRIGGER_PRE},
789 		.dynamic = 1,
790 		.dpcm_playback = 1,
791 		.dpcm_capture = 1,
792 		.ignore_suspend = 1,
793 		SND_SOC_DAILINK_REG(hostless_src_aaudio),
794 	},
795 	/* Back End DAI links */
796 	{
797 		.name = "Primary Codec",
798 		.no_pcm = 1,
799 		.dpcm_playback = 1,
800 		.dpcm_capture = 1,
801 		.ignore_suspend = 1,
802 		.init = mt8186_mt6366_init,
803 		SND_SOC_DAILINK_REG(adda),
804 	},
805 	{
806 		.name = "I2S3",
807 		.no_pcm = 1,
808 		.dai_fmt = SND_SOC_DAIFMT_I2S |
809 			   SND_SOC_DAIFMT_IB_IF |
810 			   SND_SOC_DAIFMT_CBM_CFM,
811 		.dpcm_playback = 1,
812 		.ignore_suspend = 1,
813 		.init = mt8186_mt6366_da7219_max98357_hdmi_init,
814 		.be_hw_params_fixup = mt8186_anx7625_i2s_hw_params_fixup,
815 		SND_SOC_DAILINK_REG(i2s3),
816 	},
817 	{
818 		.name = "I2S0",
819 		.no_pcm = 1,
820 		.dpcm_capture = 1,
821 		.ignore_suspend = 1,
822 		.be_hw_params_fixup = mt8186_i2s_hw_params_fixup,
823 		.ops = &mt8186_da7219_i2s_ops,
824 		SND_SOC_DAILINK_REG(i2s0),
825 	},
826 	{
827 		.name = "I2S1",
828 		.no_pcm = 1,
829 		.dpcm_playback = 1,
830 		.ignore_suspend = 1,
831 		.be_hw_params_fixup = mt8186_i2s_hw_params_fixup,
832 		.init = mt8186_da7219_init,
833 		.ops = &mt8186_da7219_i2s_ops,
834 		SND_SOC_DAILINK_REG(i2s1),
835 	},
836 	{
837 		.name = "I2S2",
838 		.no_pcm = 1,
839 		.dpcm_capture = 1,
840 		.ignore_suspend = 1,
841 		.be_hw_params_fixup = mt8186_i2s_hw_params_fixup,
842 		SND_SOC_DAILINK_REG(i2s2),
843 	},
844 	{
845 		.name = "HW Gain 1",
846 		.no_pcm = 1,
847 		.dpcm_playback = 1,
848 		.dpcm_capture = 1,
849 		.ignore_suspend = 1,
850 		SND_SOC_DAILINK_REG(hw_gain1),
851 	},
852 	{
853 		.name = "HW Gain 2",
854 		.no_pcm = 1,
855 		.dpcm_playback = 1,
856 		.dpcm_capture = 1,
857 		.ignore_suspend = 1,
858 		SND_SOC_DAILINK_REG(hw_gain2),
859 	},
860 	{
861 		.name = "HW_SRC_1",
862 		.no_pcm = 1,
863 		.dpcm_playback = 1,
864 		.dpcm_capture = 1,
865 		.ignore_suspend = 1,
866 		SND_SOC_DAILINK_REG(hw_src1),
867 	},
868 	{
869 		.name = "HW_SRC_2",
870 		.no_pcm = 1,
871 		.dpcm_playback = 1,
872 		.dpcm_capture = 1,
873 		.ignore_suspend = 1,
874 		SND_SOC_DAILINK_REG(hw_src2),
875 	},
876 	{
877 		.name = "CONNSYS_I2S",
878 		.no_pcm = 1,
879 		.dpcm_capture = 1,
880 		.ignore_suspend = 1,
881 		SND_SOC_DAILINK_REG(connsys_i2s),
882 	},
883 	{
884 		.name = "PCM 1",
885 		.dai_fmt = SND_SOC_DAIFMT_I2S |
886 			   SND_SOC_DAIFMT_NB_IF,
887 		.no_pcm = 1,
888 		.dpcm_playback = 1,
889 		.dpcm_capture = 1,
890 		.ignore_suspend = 1,
891 		SND_SOC_DAILINK_REG(pcm1),
892 	},
893 	{
894 		.name = "TDM IN",
895 		.no_pcm = 1,
896 		.dpcm_capture = 1,
897 		.ignore_suspend = 1,
898 		SND_SOC_DAILINK_REG(tdm_in),
899 	},
900 	/* dummy BE for ul memif to record from dl memif */
901 	{
902 		.name = "Hostless_UL1",
903 		.no_pcm = 1,
904 		.dpcm_capture = 1,
905 		.ignore_suspend = 1,
906 		SND_SOC_DAILINK_REG(hostless_ul1),
907 	},
908 	{
909 		.name = "Hostless_UL2",
910 		.no_pcm = 1,
911 		.dpcm_capture = 1,
912 		.ignore_suspend = 1,
913 		SND_SOC_DAILINK_REG(hostless_ul2),
914 	},
915 	{
916 		.name = "Hostless_UL3",
917 		.no_pcm = 1,
918 		.dpcm_capture = 1,
919 		.ignore_suspend = 1,
920 		SND_SOC_DAILINK_REG(hostless_ul3),
921 	},
922 	{
923 		.name = "Hostless_UL5",
924 		.no_pcm = 1,
925 		.dpcm_capture = 1,
926 		.ignore_suspend = 1,
927 		SND_SOC_DAILINK_REG(hostless_ul5),
928 	},
929 	{
930 		.name = "Hostless_UL6",
931 		.no_pcm = 1,
932 		.dpcm_capture = 1,
933 		.ignore_suspend = 1,
934 		SND_SOC_DAILINK_REG(hostless_ul6),
935 	},
936 	/* SOF BE */
937 	{
938 		.name = "AFE_SOF_DL1",
939 		.no_pcm = 1,
940 		.dpcm_playback = 1,
941 		SND_SOC_DAILINK_REG(AFE_SOF_DL1),
942 	},
943 	{
944 		.name = "AFE_SOF_DL2",
945 		.no_pcm = 1,
946 		.dpcm_playback = 1,
947 		SND_SOC_DAILINK_REG(AFE_SOF_DL2),
948 	},
949 	{
950 		.name = "AFE_SOF_UL1",
951 		.no_pcm = 1,
952 		.dpcm_capture = 1,
953 		SND_SOC_DAILINK_REG(AFE_SOF_UL1),
954 	},
955 	{
956 		.name = "AFE_SOF_UL2",
957 		.no_pcm = 1,
958 		.dpcm_capture = 1,
959 		SND_SOC_DAILINK_REG(AFE_SOF_UL2),
960 	},
961 };
962 
963 static const struct snd_soc_dapm_widget
964 mt8186_mt6366_da7219_max98357_widgets[] = {
965 	SND_SOC_DAPM_SPK("Speakers", NULL),
966 	SND_SOC_DAPM_HP("Headphones", NULL),
967 	SND_SOC_DAPM_MIC("Headset Mic", NULL),
968 	SND_SOC_DAPM_OUTPUT("HDMI1"),
969 	SND_SOC_DAPM_MIXER(SOF_DMA_DL1, SND_SOC_NOPM, 0, 0, NULL, 0),
970 	SND_SOC_DAPM_MIXER(SOF_DMA_DL2, SND_SOC_NOPM, 0, 0, NULL, 0),
971 	SND_SOC_DAPM_MIXER(SOF_DMA_UL1, SND_SOC_NOPM, 0, 0, NULL, 0),
972 	SND_SOC_DAPM_MIXER(SOF_DMA_UL2, SND_SOC_NOPM, 0, 0, NULL, 0),
973 };
974 
975 static const struct snd_soc_dapm_route
976 mt8186_mt6366_da7219_max98357_routes[] = {
977 	/* SPK */
978 	{ "Speakers", NULL, "Speaker"},
979 	/* Headset */
980 	{ "Headphones", NULL, "HPL" },
981 	{ "Headphones", NULL, "HPR" },
982 	{ "MIC", NULL, "Headset Mic" },
983 	/* HDMI */
984 	{ "HDMI1", NULL, "TX"},
985 	/* SOF Uplink */
986 	{SOF_DMA_UL1, NULL, "UL1_CH1"},
987 	{SOF_DMA_UL1, NULL, "UL1_CH2"},
988 	{SOF_DMA_UL2, NULL, "UL2_CH1"},
989 	{SOF_DMA_UL2, NULL, "UL2_CH2"},
990 	/* SOF Downlink */
991 	{"DSP_DL1_VIRT", NULL, SOF_DMA_DL1},
992 	{"DSP_DL2_VIRT", NULL, SOF_DMA_DL2},
993 };
994 
995 static const struct snd_kcontrol_new
996 mt8186_mt6366_da7219_max98357_controls[] = {
997 	SOC_DAPM_PIN_SWITCH("Speakers"),
998 	SOC_DAPM_PIN_SWITCH("Headphones"),
999 	SOC_DAPM_PIN_SWITCH("Headset Mic"),
1000 	SOC_DAPM_PIN_SWITCH("HDMI1"),
1001 };
1002 
1003 static struct snd_soc_card mt8186_mt6366_da7219_max98357_soc_card = {
1004 	.name = "mt8186_da7219_max98357",
1005 	.owner = THIS_MODULE,
1006 	.dai_link = mt8186_mt6366_da7219_max98357_dai_links,
1007 	.num_links = ARRAY_SIZE(mt8186_mt6366_da7219_max98357_dai_links),
1008 	.controls = mt8186_mt6366_da7219_max98357_controls,
1009 	.num_controls = ARRAY_SIZE(mt8186_mt6366_da7219_max98357_controls),
1010 	.dapm_widgets = mt8186_mt6366_da7219_max98357_widgets,
1011 	.num_dapm_widgets = ARRAY_SIZE(mt8186_mt6366_da7219_max98357_widgets),
1012 	.dapm_routes = mt8186_mt6366_da7219_max98357_routes,
1013 	.num_dapm_routes = ARRAY_SIZE(mt8186_mt6366_da7219_max98357_routes),
1014 	.codec_conf = mt8186_mt6366_da7219_max98357_codec_conf,
1015 	.num_configs = ARRAY_SIZE(mt8186_mt6366_da7219_max98357_codec_conf),
1016 };
1017 
1018 static int mt8186_mt6366_da7219_max98357_dev_probe(struct platform_device *pdev)
1019 {
1020 	struct snd_soc_card *card;
1021 	struct snd_soc_dai_link *dai_link;
1022 	struct mtk_soc_card_data *soc_card_data;
1023 	struct mt8186_mt6366_da7219_max98357_priv *mach_priv;
1024 	struct device_node *platform_node, *headset_codec, *playback_codec, *adsp_node;
1025 	int sof_on = 0;
1026 	int ret, i;
1027 
1028 	card = (struct snd_soc_card *)device_get_match_data(&pdev->dev);
1029 	if (!card)
1030 		return -EINVAL;
1031 	card->dev = &pdev->dev;
1032 
1033 	soc_card_data = devm_kzalloc(&pdev->dev, sizeof(*soc_card_data), GFP_KERNEL);
1034 	if (!soc_card_data)
1035 		return -ENOMEM;
1036 	mach_priv = devm_kzalloc(&pdev->dev, sizeof(*mach_priv), GFP_KERNEL);
1037 	if (!mach_priv)
1038 		return -ENOMEM;
1039 
1040 	soc_card_data->mach_priv = mach_priv;
1041 
1042 	adsp_node = of_parse_phandle(pdev->dev.of_node, "mediatek,adsp", 0);
1043 	if (adsp_node) {
1044 		struct mtk_sof_priv *sof_priv;
1045 
1046 		sof_priv = devm_kzalloc(&pdev->dev, sizeof(*sof_priv), GFP_KERNEL);
1047 		if (!sof_priv) {
1048 			ret = -ENOMEM;
1049 			goto err_adsp_node;
1050 		}
1051 		sof_priv->conn_streams = g_sof_conn_streams;
1052 		sof_priv->num_streams = ARRAY_SIZE(g_sof_conn_streams);
1053 		sof_priv->sof_dai_link_fixup = mt8186_sof_dai_link_fixup;
1054 		soc_card_data->sof_priv = sof_priv;
1055 		card->probe = mtk_sof_card_probe;
1056 		card->late_probe = mtk_sof_card_late_probe;
1057 		if (!card->topology_shortname_created) {
1058 			snprintf(card->topology_shortname, 32, "sof-%s", card->name);
1059 			card->topology_shortname_created = true;
1060 		}
1061 		card->name = card->topology_shortname;
1062 		sof_on = 1;
1063 	} else {
1064 		dev_info(&pdev->dev, "Probe without adsp\n");
1065 	}
1066 
1067 	if (of_property_read_bool(pdev->dev.of_node, "mediatek,dai-link")) {
1068 		ret = mtk_sof_dailink_parse_of(card, pdev->dev.of_node,
1069 					       "mediatek,dai-link",
1070 					       mt8186_mt6366_da7219_max98357_dai_links,
1071 					       ARRAY_SIZE(mt8186_mt6366_da7219_max98357_dai_links));
1072 		if (ret) {
1073 			dev_dbg(&pdev->dev, "Parse dai-link fail\n");
1074 			goto err_adsp_node;
1075 		}
1076 	} else {
1077 		if (!sof_on)
1078 			card->num_links = ARRAY_SIZE(mt8186_mt6366_da7219_max98357_dai_links)
1079 					- ARRAY_SIZE(g_sof_conn_streams);
1080 	}
1081 
1082 	platform_node = of_parse_phandle(pdev->dev.of_node, "mediatek,platform", 0);
1083 	if (!platform_node) {
1084 		ret = -EINVAL;
1085 		dev_err_probe(&pdev->dev, ret, "Property 'platform' missing or invalid\n");
1086 		goto err_platform_node;
1087 	}
1088 
1089 	playback_codec = of_get_child_by_name(pdev->dev.of_node, "playback-codecs");
1090 	if (!playback_codec) {
1091 		ret = -EINVAL;
1092 		dev_err_probe(&pdev->dev, ret, "Property 'speaker-codecs' missing or invalid\n");
1093 		goto err_playback_codec;
1094 	}
1095 
1096 	headset_codec = of_get_child_by_name(pdev->dev.of_node, "headset-codec");
1097 	if (!headset_codec) {
1098 		ret = -EINVAL;
1099 		dev_err_probe(&pdev->dev, ret, "Property 'headset-codec' missing or invalid\n");
1100 		goto err_headset_codec;
1101 	}
1102 
1103 	for_each_card_prelinks(card, i, dai_link) {
1104 		ret = mt8186_mt6366_card_set_be_link(card, dai_link, playback_codec, "I2S3");
1105 		if (ret) {
1106 			dev_err_probe(&pdev->dev, ret, "%s set speaker_codec fail\n",
1107 				      dai_link->name);
1108 			goto err_probe;
1109 		}
1110 
1111 		ret = mt8186_mt6366_card_set_be_link(card, dai_link, headset_codec, "I2S0");
1112 		if (ret) {
1113 			dev_err_probe(&pdev->dev, ret, "%s set headset_codec fail\n",
1114 				      dai_link->name);
1115 			goto err_probe;
1116 		}
1117 
1118 		ret = mt8186_mt6366_card_set_be_link(card, dai_link, headset_codec, "I2S1");
1119 		if (ret) {
1120 			dev_err_probe(&pdev->dev, ret, "%s set headset_codec fail\n",
1121 				      dai_link->name);
1122 			goto err_probe;
1123 		}
1124 
1125 		if (!strncmp(dai_link->name, "AFE_SOF", strlen("AFE_SOF")) && sof_on)
1126 			dai_link->platforms->of_node = adsp_node;
1127 
1128 		if (!dai_link->platforms->name && !dai_link->platforms->of_node)
1129 			dai_link->platforms->of_node = platform_node;
1130 	}
1131 
1132 	snd_soc_card_set_drvdata(card, soc_card_data);
1133 
1134 	ret = mt8186_afe_gpio_init(&pdev->dev);
1135 	if (ret) {
1136 		dev_err_probe(&pdev->dev, ret, "%s init gpio error\n", __func__);
1137 		goto err_probe;
1138 	}
1139 
1140 	ret = devm_snd_soc_register_card(&pdev->dev, card);
1141 	if (ret)
1142 		dev_err_probe(&pdev->dev, ret, "%s snd_soc_register_card fail\n", __func__);
1143 
1144 err_probe:
1145 	of_node_put(headset_codec);
1146 err_headset_codec:
1147 	of_node_put(playback_codec);
1148 err_playback_codec:
1149 	of_node_put(platform_node);
1150 err_platform_node:
1151 err_adsp_node:
1152 	of_node_put(adsp_node);
1153 
1154 	return ret;
1155 }
1156 
1157 #if IS_ENABLED(CONFIG_OF)
1158 static const struct of_device_id mt8186_mt6366_da7219_max98357_dt_match[] = {
1159 	{	.compatible = "mediatek,mt8186-mt6366-da7219-max98357-sound",
1160 		.data = &mt8186_mt6366_da7219_max98357_soc_card,
1161 	},
1162 	{}
1163 };
1164 MODULE_DEVICE_TABLE(of, mt8186_mt6366_da7219_max98357_dt_match);
1165 #endif
1166 
1167 static struct platform_driver mt8186_mt6366_da7219_max98357_driver = {
1168 	.driver = {
1169 		.name = "mt8186_mt6366_da7219_max98357",
1170 #if IS_ENABLED(CONFIG_OF)
1171 		.of_match_table = mt8186_mt6366_da7219_max98357_dt_match,
1172 #endif
1173 		.pm = &snd_soc_pm_ops,
1174 	},
1175 	.probe = mt8186_mt6366_da7219_max98357_dev_probe,
1176 };
1177 
1178 module_platform_driver(mt8186_mt6366_da7219_max98357_driver);
1179 
1180 /* Module information */
1181 MODULE_DESCRIPTION("MT8186-MT6366-DA7219-MAX98357 ALSA SoC machine driver");
1182 MODULE_AUTHOR("Jiaxin Yu <jiaxin.yu@mediatek.com>");
1183 MODULE_LICENSE("GPL v2");
1184 MODULE_ALIAS("mt8186_mt6366_da7219_max98357 soc card");
1185