1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0+
2*4882a593Smuzhiyun //
3*4882a593Smuzhiyun // Copyright 2012 Freescale Semiconductor, Inc.
4*4882a593Smuzhiyun // Copyright 2012 Linaro Ltd.
5*4882a593Smuzhiyun
6*4882a593Smuzhiyun #include <linux/gpio.h>
7*4882a593Smuzhiyun #include <linux/module.h>
8*4882a593Smuzhiyun #include <linux/of.h>
9*4882a593Smuzhiyun #include <linux/of_platform.h>
10*4882a593Smuzhiyun #include <linux/i2c.h>
11*4882a593Smuzhiyun #include <linux/of_gpio.h>
12*4882a593Smuzhiyun #include <sound/soc.h>
13*4882a593Smuzhiyun #include <sound/jack.h>
14*4882a593Smuzhiyun
15*4882a593Smuzhiyun #include "imx-audmux.h"
16*4882a593Smuzhiyun
17*4882a593Smuzhiyun #define DAI_NAME_SIZE 32
18*4882a593Smuzhiyun #define MUX_PORT_MAX 7
19*4882a593Smuzhiyun
20*4882a593Smuzhiyun struct imx_es8328_data {
21*4882a593Smuzhiyun struct device *dev;
22*4882a593Smuzhiyun struct snd_soc_dai_link dai;
23*4882a593Smuzhiyun struct snd_soc_card card;
24*4882a593Smuzhiyun char codec_dai_name[DAI_NAME_SIZE];
25*4882a593Smuzhiyun char platform_name[DAI_NAME_SIZE];
26*4882a593Smuzhiyun int jack_gpio;
27*4882a593Smuzhiyun };
28*4882a593Smuzhiyun
29*4882a593Smuzhiyun static struct snd_soc_jack_gpio headset_jack_gpios[] = {
30*4882a593Smuzhiyun {
31*4882a593Smuzhiyun .gpio = -1,
32*4882a593Smuzhiyun .name = "headset-gpio",
33*4882a593Smuzhiyun .report = SND_JACK_HEADSET,
34*4882a593Smuzhiyun .invert = 0,
35*4882a593Smuzhiyun .debounce_time = 200,
36*4882a593Smuzhiyun },
37*4882a593Smuzhiyun };
38*4882a593Smuzhiyun
39*4882a593Smuzhiyun static struct snd_soc_jack headset_jack;
40*4882a593Smuzhiyun
imx_es8328_dai_init(struct snd_soc_pcm_runtime * rtd)41*4882a593Smuzhiyun static int imx_es8328_dai_init(struct snd_soc_pcm_runtime *rtd)
42*4882a593Smuzhiyun {
43*4882a593Smuzhiyun struct imx_es8328_data *data = container_of(rtd->card,
44*4882a593Smuzhiyun struct imx_es8328_data, card);
45*4882a593Smuzhiyun int ret = 0;
46*4882a593Smuzhiyun
47*4882a593Smuzhiyun /* Headphone jack detection */
48*4882a593Smuzhiyun if (gpio_is_valid(data->jack_gpio)) {
49*4882a593Smuzhiyun ret = snd_soc_card_jack_new(rtd->card, "Headphone",
50*4882a593Smuzhiyun SND_JACK_HEADPHONE | SND_JACK_BTN_0,
51*4882a593Smuzhiyun &headset_jack, NULL, 0);
52*4882a593Smuzhiyun if (ret)
53*4882a593Smuzhiyun return ret;
54*4882a593Smuzhiyun
55*4882a593Smuzhiyun headset_jack_gpios[0].gpio = data->jack_gpio;
56*4882a593Smuzhiyun ret = snd_soc_jack_add_gpios(&headset_jack,
57*4882a593Smuzhiyun ARRAY_SIZE(headset_jack_gpios),
58*4882a593Smuzhiyun headset_jack_gpios);
59*4882a593Smuzhiyun }
60*4882a593Smuzhiyun
61*4882a593Smuzhiyun return ret;
62*4882a593Smuzhiyun }
63*4882a593Smuzhiyun
64*4882a593Smuzhiyun static const struct snd_soc_dapm_widget imx_es8328_dapm_widgets[] = {
65*4882a593Smuzhiyun SND_SOC_DAPM_MIC("Mic Jack", NULL),
66*4882a593Smuzhiyun SND_SOC_DAPM_HP("Headphone", NULL),
67*4882a593Smuzhiyun SND_SOC_DAPM_SPK("Speaker", NULL),
68*4882a593Smuzhiyun SND_SOC_DAPM_REGULATOR_SUPPLY("audio-amp", 1, 0),
69*4882a593Smuzhiyun };
70*4882a593Smuzhiyun
imx_es8328_probe(struct platform_device * pdev)71*4882a593Smuzhiyun static int imx_es8328_probe(struct platform_device *pdev)
72*4882a593Smuzhiyun {
73*4882a593Smuzhiyun struct device_node *np = pdev->dev.of_node;
74*4882a593Smuzhiyun struct device_node *ssi_np = NULL, *codec_np = NULL;
75*4882a593Smuzhiyun struct platform_device *ssi_pdev;
76*4882a593Smuzhiyun struct imx_es8328_data *data;
77*4882a593Smuzhiyun struct snd_soc_dai_link_component *comp;
78*4882a593Smuzhiyun u32 int_port, ext_port;
79*4882a593Smuzhiyun int ret;
80*4882a593Smuzhiyun struct device *dev = &pdev->dev;
81*4882a593Smuzhiyun
82*4882a593Smuzhiyun ret = of_property_read_u32(np, "mux-int-port", &int_port);
83*4882a593Smuzhiyun if (ret) {
84*4882a593Smuzhiyun dev_err(dev, "mux-int-port missing or invalid\n");
85*4882a593Smuzhiyun goto fail;
86*4882a593Smuzhiyun }
87*4882a593Smuzhiyun if (int_port > MUX_PORT_MAX || int_port == 0) {
88*4882a593Smuzhiyun dev_err(dev, "mux-int-port: hardware only has %d mux ports\n",
89*4882a593Smuzhiyun MUX_PORT_MAX);
90*4882a593Smuzhiyun ret = -EINVAL;
91*4882a593Smuzhiyun goto fail;
92*4882a593Smuzhiyun }
93*4882a593Smuzhiyun
94*4882a593Smuzhiyun ret = of_property_read_u32(np, "mux-ext-port", &ext_port);
95*4882a593Smuzhiyun if (ret) {
96*4882a593Smuzhiyun dev_err(dev, "mux-ext-port missing or invalid\n");
97*4882a593Smuzhiyun goto fail;
98*4882a593Smuzhiyun }
99*4882a593Smuzhiyun if (ext_port > MUX_PORT_MAX || ext_port == 0) {
100*4882a593Smuzhiyun dev_err(dev, "mux-ext-port: hardware only has %d mux ports\n",
101*4882a593Smuzhiyun MUX_PORT_MAX);
102*4882a593Smuzhiyun ret = -EINVAL;
103*4882a593Smuzhiyun goto fail;
104*4882a593Smuzhiyun }
105*4882a593Smuzhiyun
106*4882a593Smuzhiyun /*
107*4882a593Smuzhiyun * The port numbering in the hardware manual starts at 1, while
108*4882a593Smuzhiyun * the audmux API expects it starts at 0.
109*4882a593Smuzhiyun */
110*4882a593Smuzhiyun int_port--;
111*4882a593Smuzhiyun ext_port--;
112*4882a593Smuzhiyun ret = imx_audmux_v2_configure_port(int_port,
113*4882a593Smuzhiyun IMX_AUDMUX_V2_PTCR_SYN |
114*4882a593Smuzhiyun IMX_AUDMUX_V2_PTCR_TFSEL(ext_port) |
115*4882a593Smuzhiyun IMX_AUDMUX_V2_PTCR_TCSEL(ext_port) |
116*4882a593Smuzhiyun IMX_AUDMUX_V2_PTCR_TFSDIR |
117*4882a593Smuzhiyun IMX_AUDMUX_V2_PTCR_TCLKDIR,
118*4882a593Smuzhiyun IMX_AUDMUX_V2_PDCR_RXDSEL(ext_port));
119*4882a593Smuzhiyun if (ret) {
120*4882a593Smuzhiyun dev_err(dev, "audmux internal port setup failed\n");
121*4882a593Smuzhiyun return ret;
122*4882a593Smuzhiyun }
123*4882a593Smuzhiyun ret = imx_audmux_v2_configure_port(ext_port,
124*4882a593Smuzhiyun IMX_AUDMUX_V2_PTCR_SYN,
125*4882a593Smuzhiyun IMX_AUDMUX_V2_PDCR_RXDSEL(int_port));
126*4882a593Smuzhiyun if (ret) {
127*4882a593Smuzhiyun dev_err(dev, "audmux external port setup failed\n");
128*4882a593Smuzhiyun return ret;
129*4882a593Smuzhiyun }
130*4882a593Smuzhiyun
131*4882a593Smuzhiyun ssi_np = of_parse_phandle(pdev->dev.of_node, "ssi-controller", 0);
132*4882a593Smuzhiyun codec_np = of_parse_phandle(pdev->dev.of_node, "audio-codec", 0);
133*4882a593Smuzhiyun if (!ssi_np || !codec_np) {
134*4882a593Smuzhiyun dev_err(dev, "phandle missing or invalid\n");
135*4882a593Smuzhiyun ret = -EINVAL;
136*4882a593Smuzhiyun goto fail;
137*4882a593Smuzhiyun }
138*4882a593Smuzhiyun
139*4882a593Smuzhiyun ssi_pdev = of_find_device_by_node(ssi_np);
140*4882a593Smuzhiyun if (!ssi_pdev) {
141*4882a593Smuzhiyun dev_err(dev, "failed to find SSI platform device\n");
142*4882a593Smuzhiyun ret = -EINVAL;
143*4882a593Smuzhiyun goto fail;
144*4882a593Smuzhiyun }
145*4882a593Smuzhiyun
146*4882a593Smuzhiyun data = devm_kzalloc(dev, sizeof(*data), GFP_KERNEL);
147*4882a593Smuzhiyun if (!data) {
148*4882a593Smuzhiyun ret = -ENOMEM;
149*4882a593Smuzhiyun goto put_device;
150*4882a593Smuzhiyun }
151*4882a593Smuzhiyun
152*4882a593Smuzhiyun comp = devm_kzalloc(dev, 3 * sizeof(*comp), GFP_KERNEL);
153*4882a593Smuzhiyun if (!comp) {
154*4882a593Smuzhiyun ret = -ENOMEM;
155*4882a593Smuzhiyun goto put_device;
156*4882a593Smuzhiyun }
157*4882a593Smuzhiyun
158*4882a593Smuzhiyun data->dev = dev;
159*4882a593Smuzhiyun
160*4882a593Smuzhiyun data->jack_gpio = of_get_named_gpio(pdev->dev.of_node, "jack-gpio", 0);
161*4882a593Smuzhiyun
162*4882a593Smuzhiyun data->dai.cpus = &comp[0];
163*4882a593Smuzhiyun data->dai.codecs = &comp[1];
164*4882a593Smuzhiyun data->dai.platforms = &comp[2];
165*4882a593Smuzhiyun
166*4882a593Smuzhiyun data->dai.num_cpus = 1;
167*4882a593Smuzhiyun data->dai.num_codecs = 1;
168*4882a593Smuzhiyun data->dai.num_platforms = 1;
169*4882a593Smuzhiyun
170*4882a593Smuzhiyun data->dai.name = "hifi";
171*4882a593Smuzhiyun data->dai.stream_name = "hifi";
172*4882a593Smuzhiyun data->dai.codecs->dai_name = "es8328-hifi-analog";
173*4882a593Smuzhiyun data->dai.codecs->of_node = codec_np;
174*4882a593Smuzhiyun data->dai.cpus->of_node = ssi_np;
175*4882a593Smuzhiyun data->dai.platforms->of_node = ssi_np;
176*4882a593Smuzhiyun data->dai.init = &imx_es8328_dai_init;
177*4882a593Smuzhiyun data->dai.dai_fmt = SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF |
178*4882a593Smuzhiyun SND_SOC_DAIFMT_CBM_CFM;
179*4882a593Smuzhiyun
180*4882a593Smuzhiyun data->card.dev = dev;
181*4882a593Smuzhiyun data->card.dapm_widgets = imx_es8328_dapm_widgets;
182*4882a593Smuzhiyun data->card.num_dapm_widgets = ARRAY_SIZE(imx_es8328_dapm_widgets);
183*4882a593Smuzhiyun ret = snd_soc_of_parse_card_name(&data->card, "model");
184*4882a593Smuzhiyun if (ret) {
185*4882a593Smuzhiyun dev_err(dev, "Unable to parse card name\n");
186*4882a593Smuzhiyun goto put_device;
187*4882a593Smuzhiyun }
188*4882a593Smuzhiyun ret = snd_soc_of_parse_audio_routing(&data->card, "audio-routing");
189*4882a593Smuzhiyun if (ret) {
190*4882a593Smuzhiyun dev_err(dev, "Unable to parse routing: %d\n", ret);
191*4882a593Smuzhiyun goto put_device;
192*4882a593Smuzhiyun }
193*4882a593Smuzhiyun data->card.num_links = 1;
194*4882a593Smuzhiyun data->card.owner = THIS_MODULE;
195*4882a593Smuzhiyun data->card.dai_link = &data->dai;
196*4882a593Smuzhiyun
197*4882a593Smuzhiyun ret = snd_soc_register_card(&data->card);
198*4882a593Smuzhiyun if (ret) {
199*4882a593Smuzhiyun dev_err(dev, "Unable to register: %d\n", ret);
200*4882a593Smuzhiyun goto put_device;
201*4882a593Smuzhiyun }
202*4882a593Smuzhiyun
203*4882a593Smuzhiyun platform_set_drvdata(pdev, data);
204*4882a593Smuzhiyun put_device:
205*4882a593Smuzhiyun put_device(&ssi_pdev->dev);
206*4882a593Smuzhiyun fail:
207*4882a593Smuzhiyun of_node_put(ssi_np);
208*4882a593Smuzhiyun of_node_put(codec_np);
209*4882a593Smuzhiyun
210*4882a593Smuzhiyun return ret;
211*4882a593Smuzhiyun }
212*4882a593Smuzhiyun
imx_es8328_remove(struct platform_device * pdev)213*4882a593Smuzhiyun static int imx_es8328_remove(struct platform_device *pdev)
214*4882a593Smuzhiyun {
215*4882a593Smuzhiyun struct imx_es8328_data *data = platform_get_drvdata(pdev);
216*4882a593Smuzhiyun
217*4882a593Smuzhiyun snd_soc_unregister_card(&data->card);
218*4882a593Smuzhiyun
219*4882a593Smuzhiyun return 0;
220*4882a593Smuzhiyun }
221*4882a593Smuzhiyun
222*4882a593Smuzhiyun static const struct of_device_id imx_es8328_dt_ids[] = {
223*4882a593Smuzhiyun { .compatible = "fsl,imx-audio-es8328", },
224*4882a593Smuzhiyun { /* sentinel */ }
225*4882a593Smuzhiyun };
226*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, imx_es8328_dt_ids);
227*4882a593Smuzhiyun
228*4882a593Smuzhiyun static struct platform_driver imx_es8328_driver = {
229*4882a593Smuzhiyun .driver = {
230*4882a593Smuzhiyun .name = "imx-es8328",
231*4882a593Smuzhiyun .of_match_table = imx_es8328_dt_ids,
232*4882a593Smuzhiyun },
233*4882a593Smuzhiyun .probe = imx_es8328_probe,
234*4882a593Smuzhiyun .remove = imx_es8328_remove,
235*4882a593Smuzhiyun };
236*4882a593Smuzhiyun module_platform_driver(imx_es8328_driver);
237*4882a593Smuzhiyun
238*4882a593Smuzhiyun MODULE_AUTHOR("Sean Cross <xobs@kosagi.com>");
239*4882a593Smuzhiyun MODULE_DESCRIPTION("Kosagi i.MX6 ES8328 ASoC machine driver");
240*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
241*4882a593Smuzhiyun MODULE_ALIAS("platform:imx-audio-es8328");
242