1 // SPDX-License-Identifier: GPL-2.0
3 * mt8173-max98090.c -- MT8173 MAX98090 ALSA SoC machine driver
5 * Copyright (c) 2015 MediaTek Inc.
6 * Author: Koro Chen <koro.chen@mediatek.com>
9 #include <linux/module.h>
10 #include <sound/soc.h>
11 #include <sound/jack.h>
12 #include <linux/gpio.h>
13 #include "../../codecs/max98090.h"
15 static struct snd_soc_jack mt8173_max98090_jack
;
17 static struct snd_soc_jack_pin mt8173_max98090_jack_pins
[] = {
20 .mask
= SND_JACK_HEADPHONE
,
24 .mask
= SND_JACK_MICROPHONE
,
28 static const struct snd_soc_dapm_widget mt8173_max98090_widgets
[] = {
29 SND_SOC_DAPM_SPK("Speaker", NULL
),
30 SND_SOC_DAPM_MIC("Int Mic", NULL
),
31 SND_SOC_DAPM_HP("Headphone", NULL
),
32 SND_SOC_DAPM_MIC("Headset Mic", NULL
),
35 static const struct snd_soc_dapm_route mt8173_max98090_routes
[] = {
36 {"Speaker", NULL
, "SPKL"},
37 {"Speaker", NULL
, "SPKR"},
38 {"DMICL", NULL
, "Int Mic"},
39 {"Headphone", NULL
, "HPL"},
40 {"Headphone", NULL
, "HPR"},
41 {"Headset Mic", NULL
, "MICBIAS"},
42 {"IN34", NULL
, "Headset Mic"},
45 static const struct snd_kcontrol_new mt8173_max98090_controls
[] = {
46 SOC_DAPM_PIN_SWITCH("Speaker"),
47 SOC_DAPM_PIN_SWITCH("Int Mic"),
48 SOC_DAPM_PIN_SWITCH("Headphone"),
49 SOC_DAPM_PIN_SWITCH("Headset Mic"),
52 static int mt8173_max98090_hw_params(struct snd_pcm_substream
*substream
,
53 struct snd_pcm_hw_params
*params
)
55 struct snd_soc_pcm_runtime
*rtd
= substream
->private_data
;
56 struct snd_soc_dai
*codec_dai
= rtd
->codec_dai
;
58 return snd_soc_dai_set_sysclk(codec_dai
, 0, params_rate(params
) * 256,
62 static const struct snd_soc_ops mt8173_max98090_ops
= {
63 .hw_params
= mt8173_max98090_hw_params
,
66 static int mt8173_max98090_init(struct snd_soc_pcm_runtime
*runtime
)
69 struct snd_soc_card
*card
= runtime
->card
;
70 struct snd_soc_component
*component
= runtime
->codec_dai
->component
;
72 /* enable jack detection */
73 ret
= snd_soc_card_jack_new(card
, "Headphone", SND_JACK_HEADPHONE
,
74 &mt8173_max98090_jack
,
75 mt8173_max98090_jack_pins
,
76 ARRAY_SIZE(mt8173_max98090_jack_pins
));
78 dev_err(card
->dev
, "Can't create a new Jack %d\n", ret
);
82 return max98090_mic_detect(component
, &mt8173_max98090_jack
);
85 SND_SOC_DAILINK_DEFS(playback
,
86 DAILINK_COMP_ARRAY(COMP_CPU("DL1")),
87 DAILINK_COMP_ARRAY(COMP_DUMMY()),
88 DAILINK_COMP_ARRAY(COMP_EMPTY()));
90 SND_SOC_DAILINK_DEFS(capture
,
91 DAILINK_COMP_ARRAY(COMP_CPU("VUL")),
92 DAILINK_COMP_ARRAY(COMP_DUMMY()),
93 DAILINK_COMP_ARRAY(COMP_EMPTY()));
95 SND_SOC_DAILINK_DEFS(hifi
,
96 DAILINK_COMP_ARRAY(COMP_CPU("I2S")),
97 DAILINK_COMP_ARRAY(COMP_CODEC(NULL
, "HiFi")),
98 DAILINK_COMP_ARRAY(COMP_EMPTY()));
100 /* Digital audio interface glue - connects codec <---> CPU */
101 static struct snd_soc_dai_link mt8173_max98090_dais
[] = {
102 /* Front End DAI links */
104 .name
= "MAX98090 Playback",
105 .stream_name
= "MAX98090 Playback",
106 .trigger
= {SND_SOC_DPCM_TRIGGER_POST
, SND_SOC_DPCM_TRIGGER_POST
},
109 SND_SOC_DAILINK_REG(playback
),
112 .name
= "MAX98090 Capture",
113 .stream_name
= "MAX98090 Capture",
114 .trigger
= {SND_SOC_DPCM_TRIGGER_POST
, SND_SOC_DPCM_TRIGGER_POST
},
117 SND_SOC_DAILINK_REG(capture
),
119 /* Back End DAI links */
123 .init
= mt8173_max98090_init
,
124 .ops
= &mt8173_max98090_ops
,
125 .dai_fmt
= SND_SOC_DAIFMT_I2S
| SND_SOC_DAIFMT_NB_NF
|
126 SND_SOC_DAIFMT_CBS_CFS
,
129 SND_SOC_DAILINK_REG(hifi
),
133 static struct snd_soc_card mt8173_max98090_card
= {
134 .name
= "mt8173-max98090",
135 .owner
= THIS_MODULE
,
136 .dai_link
= mt8173_max98090_dais
,
137 .num_links
= ARRAY_SIZE(mt8173_max98090_dais
),
138 .controls
= mt8173_max98090_controls
,
139 .num_controls
= ARRAY_SIZE(mt8173_max98090_controls
),
140 .dapm_widgets
= mt8173_max98090_widgets
,
141 .num_dapm_widgets
= ARRAY_SIZE(mt8173_max98090_widgets
),
142 .dapm_routes
= mt8173_max98090_routes
,
143 .num_dapm_routes
= ARRAY_SIZE(mt8173_max98090_routes
),
146 static int mt8173_max98090_dev_probe(struct platform_device
*pdev
)
148 struct snd_soc_card
*card
= &mt8173_max98090_card
;
149 struct device_node
*codec_node
, *platform_node
;
150 struct snd_soc_dai_link
*dai_link
;
153 platform_node
= of_parse_phandle(pdev
->dev
.of_node
,
154 "mediatek,platform", 0);
155 if (!platform_node
) {
156 dev_err(&pdev
->dev
, "Property 'platform' missing or invalid\n");
159 for_each_card_prelinks(card
, i
, dai_link
) {
160 if (dai_link
->platforms
->name
)
162 dai_link
->platforms
->of_node
= platform_node
;
165 codec_node
= of_parse_phandle(pdev
->dev
.of_node
,
166 "mediatek,audio-codec", 0);
169 "Property 'audio-codec' missing or invalid\n");
172 for_each_card_prelinks(card
, i
, dai_link
) {
173 if (dai_link
->codecs
->name
)
175 dai_link
->codecs
->of_node
= codec_node
;
177 card
->dev
= &pdev
->dev
;
179 ret
= devm_snd_soc_register_card(&pdev
->dev
, card
);
181 dev_err(&pdev
->dev
, "%s snd_soc_register_card fail %d\n",
186 static const struct of_device_id mt8173_max98090_dt_match
[] = {
187 { .compatible
= "mediatek,mt8173-max98090", },
190 MODULE_DEVICE_TABLE(of
, mt8173_max98090_dt_match
);
192 static struct platform_driver mt8173_max98090_driver
= {
194 .name
= "mt8173-max98090",
195 .of_match_table
= mt8173_max98090_dt_match
,
197 .pm
= &snd_soc_pm_ops
,
200 .probe
= mt8173_max98090_dev_probe
,
203 module_platform_driver(mt8173_max98090_driver
);
205 /* Module information */
206 MODULE_DESCRIPTION("MT8173 MAX98090 ALSA SoC machine driver");
207 MODULE_AUTHOR("Koro Chen <koro.chen@mediatek.com>");
208 MODULE_LICENSE("GPL v2");
209 MODULE_ALIAS("platform:mt8173-max98090");