1*4882a593Smuzhiyun // SPDX-License-Identifier: GPL-2.0-only
2*4882a593Smuzhiyun /*
3*4882a593Smuzhiyun * Copyright (C) ST-Ericsson SA 2012
4*4882a593Smuzhiyun *
5*4882a593Smuzhiyun * Author: Ola Lilja (ola.o.lilja@stericsson.com)
6*4882a593Smuzhiyun * for ST-Ericsson.
7*4882a593Smuzhiyun *
8*4882a593Smuzhiyun * License terms:
9*4882a593Smuzhiyun */
10*4882a593Smuzhiyun
11*4882a593Smuzhiyun #include <asm/mach-types.h>
12*4882a593Smuzhiyun
13*4882a593Smuzhiyun #include <linux/module.h>
14*4882a593Smuzhiyun #include <linux/io.h>
15*4882a593Smuzhiyun #include <linux/spi/spi.h>
16*4882a593Smuzhiyun #include <linux/of.h>
17*4882a593Smuzhiyun
18*4882a593Smuzhiyun #include <sound/soc.h>
19*4882a593Smuzhiyun #include <sound/initval.h>
20*4882a593Smuzhiyun
21*4882a593Smuzhiyun #include "ux500_pcm.h"
22*4882a593Smuzhiyun #include "ux500_msp_dai.h"
23*4882a593Smuzhiyun
24*4882a593Smuzhiyun #include "mop500_ab8500.h"
25*4882a593Smuzhiyun
26*4882a593Smuzhiyun /* Define the whole MOP500 soundcard, linking platform to the codec-drivers */
27*4882a593Smuzhiyun SND_SOC_DAILINK_DEFS(link1,
28*4882a593Smuzhiyun DAILINK_COMP_ARRAY(COMP_CPU("ux500-msp-i2s.1")),
29*4882a593Smuzhiyun DAILINK_COMP_ARRAY(COMP_CODEC("ab8500-codec.0", "ab8500-codec-dai.0")),
30*4882a593Smuzhiyun DAILINK_COMP_ARRAY(COMP_PLATFORM("ux500-msp-i2s.1")));
31*4882a593Smuzhiyun
32*4882a593Smuzhiyun SND_SOC_DAILINK_DEFS(link2,
33*4882a593Smuzhiyun DAILINK_COMP_ARRAY(COMP_CPU("ux500-msp-i2s.3")),
34*4882a593Smuzhiyun DAILINK_COMP_ARRAY(COMP_CODEC("ab8500-codec.0", "ab8500-codec-dai.1")),
35*4882a593Smuzhiyun DAILINK_COMP_ARRAY(COMP_PLATFORM("ux500-msp-i2s.3")));
36*4882a593Smuzhiyun
37*4882a593Smuzhiyun static struct snd_soc_dai_link mop500_dai_links[] = {
38*4882a593Smuzhiyun {
39*4882a593Smuzhiyun .name = "ab8500_0",
40*4882a593Smuzhiyun .stream_name = "ab8500_0",
41*4882a593Smuzhiyun .init = mop500_ab8500_machine_init,
42*4882a593Smuzhiyun .ops = mop500_ab8500_ops,
43*4882a593Smuzhiyun SND_SOC_DAILINK_REG(link1),
44*4882a593Smuzhiyun },
45*4882a593Smuzhiyun {
46*4882a593Smuzhiyun .name = "ab8500_1",
47*4882a593Smuzhiyun .stream_name = "ab8500_1",
48*4882a593Smuzhiyun .init = NULL,
49*4882a593Smuzhiyun .ops = mop500_ab8500_ops,
50*4882a593Smuzhiyun SND_SOC_DAILINK_REG(link2),
51*4882a593Smuzhiyun },
52*4882a593Smuzhiyun };
53*4882a593Smuzhiyun
54*4882a593Smuzhiyun static struct snd_soc_card mop500_card = {
55*4882a593Smuzhiyun .name = "MOP500-card",
56*4882a593Smuzhiyun .owner = THIS_MODULE,
57*4882a593Smuzhiyun .probe = NULL,
58*4882a593Smuzhiyun .dai_link = mop500_dai_links,
59*4882a593Smuzhiyun .num_links = ARRAY_SIZE(mop500_dai_links),
60*4882a593Smuzhiyun };
61*4882a593Smuzhiyun
mop500_of_node_put(void)62*4882a593Smuzhiyun static void mop500_of_node_put(void)
63*4882a593Smuzhiyun {
64*4882a593Smuzhiyun int i;
65*4882a593Smuzhiyun
66*4882a593Smuzhiyun for (i = 0; i < 2; i++)
67*4882a593Smuzhiyun of_node_put(mop500_dai_links[i].cpus->of_node);
68*4882a593Smuzhiyun
69*4882a593Smuzhiyun /* Both links use the same codec, which is refcounted only once */
70*4882a593Smuzhiyun of_node_put(mop500_dai_links[0].codecs->of_node);
71*4882a593Smuzhiyun }
72*4882a593Smuzhiyun
mop500_of_probe(struct platform_device * pdev,struct device_node * np)73*4882a593Smuzhiyun static int mop500_of_probe(struct platform_device *pdev,
74*4882a593Smuzhiyun struct device_node *np)
75*4882a593Smuzhiyun {
76*4882a593Smuzhiyun struct device_node *codec_np, *msp_np[2];
77*4882a593Smuzhiyun int i;
78*4882a593Smuzhiyun
79*4882a593Smuzhiyun msp_np[0] = of_parse_phandle(np, "stericsson,cpu-dai", 0);
80*4882a593Smuzhiyun msp_np[1] = of_parse_phandle(np, "stericsson,cpu-dai", 1);
81*4882a593Smuzhiyun codec_np = of_parse_phandle(np, "stericsson,audio-codec", 0);
82*4882a593Smuzhiyun
83*4882a593Smuzhiyun if (!(msp_np[0] && msp_np[1] && codec_np)) {
84*4882a593Smuzhiyun dev_err(&pdev->dev, "Phandle missing or invalid\n");
85*4882a593Smuzhiyun for (i = 0; i < 2; i++)
86*4882a593Smuzhiyun of_node_put(msp_np[i]);
87*4882a593Smuzhiyun of_node_put(codec_np);
88*4882a593Smuzhiyun return -EINVAL;
89*4882a593Smuzhiyun }
90*4882a593Smuzhiyun
91*4882a593Smuzhiyun for (i = 0; i < 2; i++) {
92*4882a593Smuzhiyun mop500_dai_links[i].cpus->of_node = msp_np[i];
93*4882a593Smuzhiyun mop500_dai_links[i].cpus->dai_name = NULL;
94*4882a593Smuzhiyun mop500_dai_links[i].platforms->of_node = msp_np[i];
95*4882a593Smuzhiyun mop500_dai_links[i].platforms->name = NULL;
96*4882a593Smuzhiyun mop500_dai_links[i].codecs->of_node = codec_np;
97*4882a593Smuzhiyun mop500_dai_links[i].codecs->name = NULL;
98*4882a593Smuzhiyun }
99*4882a593Smuzhiyun
100*4882a593Smuzhiyun snd_soc_of_parse_card_name(&mop500_card, "stericsson,card-name");
101*4882a593Smuzhiyun
102*4882a593Smuzhiyun return 0;
103*4882a593Smuzhiyun }
104*4882a593Smuzhiyun
mop500_probe(struct platform_device * pdev)105*4882a593Smuzhiyun static int mop500_probe(struct platform_device *pdev)
106*4882a593Smuzhiyun {
107*4882a593Smuzhiyun struct device_node *np = pdev->dev.of_node;
108*4882a593Smuzhiyun int ret;
109*4882a593Smuzhiyun
110*4882a593Smuzhiyun dev_dbg(&pdev->dev, "%s: Enter.\n", __func__);
111*4882a593Smuzhiyun
112*4882a593Smuzhiyun mop500_card.dev = &pdev->dev;
113*4882a593Smuzhiyun
114*4882a593Smuzhiyun if (np) {
115*4882a593Smuzhiyun ret = mop500_of_probe(pdev, np);
116*4882a593Smuzhiyun if (ret)
117*4882a593Smuzhiyun return ret;
118*4882a593Smuzhiyun }
119*4882a593Smuzhiyun
120*4882a593Smuzhiyun dev_dbg(&pdev->dev, "%s: Card %s: Set platform drvdata.\n",
121*4882a593Smuzhiyun __func__, mop500_card.name);
122*4882a593Smuzhiyun
123*4882a593Smuzhiyun snd_soc_card_set_drvdata(&mop500_card, NULL);
124*4882a593Smuzhiyun
125*4882a593Smuzhiyun dev_dbg(&pdev->dev, "%s: Card %s: num_links = %d\n",
126*4882a593Smuzhiyun __func__, mop500_card.name, mop500_card.num_links);
127*4882a593Smuzhiyun dev_dbg(&pdev->dev, "%s: Card %s: DAI-link 0: name = %s\n",
128*4882a593Smuzhiyun __func__, mop500_card.name, mop500_card.dai_link[0].name);
129*4882a593Smuzhiyun dev_dbg(&pdev->dev, "%s: Card %s: DAI-link 0: stream_name = %s\n",
130*4882a593Smuzhiyun __func__, mop500_card.name,
131*4882a593Smuzhiyun mop500_card.dai_link[0].stream_name);
132*4882a593Smuzhiyun
133*4882a593Smuzhiyun ret = snd_soc_register_card(&mop500_card);
134*4882a593Smuzhiyun if (ret)
135*4882a593Smuzhiyun dev_err(&pdev->dev,
136*4882a593Smuzhiyun "Error: snd_soc_register_card failed (%d)!\n", ret);
137*4882a593Smuzhiyun
138*4882a593Smuzhiyun return ret;
139*4882a593Smuzhiyun }
140*4882a593Smuzhiyun
mop500_remove(struct platform_device * pdev)141*4882a593Smuzhiyun static int mop500_remove(struct platform_device *pdev)
142*4882a593Smuzhiyun {
143*4882a593Smuzhiyun struct snd_soc_card *mop500_card = platform_get_drvdata(pdev);
144*4882a593Smuzhiyun
145*4882a593Smuzhiyun pr_debug("%s: Enter.\n", __func__);
146*4882a593Smuzhiyun
147*4882a593Smuzhiyun snd_soc_unregister_card(mop500_card);
148*4882a593Smuzhiyun mop500_ab8500_remove(mop500_card);
149*4882a593Smuzhiyun mop500_of_node_put();
150*4882a593Smuzhiyun
151*4882a593Smuzhiyun return 0;
152*4882a593Smuzhiyun }
153*4882a593Smuzhiyun
154*4882a593Smuzhiyun static const struct of_device_id snd_soc_mop500_match[] = {
155*4882a593Smuzhiyun { .compatible = "stericsson,snd-soc-mop500", },
156*4882a593Smuzhiyun {},
157*4882a593Smuzhiyun };
158*4882a593Smuzhiyun MODULE_DEVICE_TABLE(of, snd_soc_mop500_match);
159*4882a593Smuzhiyun
160*4882a593Smuzhiyun static struct platform_driver snd_soc_mop500_driver = {
161*4882a593Smuzhiyun .driver = {
162*4882a593Smuzhiyun .name = "snd-soc-mop500",
163*4882a593Smuzhiyun .of_match_table = snd_soc_mop500_match,
164*4882a593Smuzhiyun },
165*4882a593Smuzhiyun .probe = mop500_probe,
166*4882a593Smuzhiyun .remove = mop500_remove,
167*4882a593Smuzhiyun };
168*4882a593Smuzhiyun
169*4882a593Smuzhiyun module_platform_driver(snd_soc_mop500_driver);
170*4882a593Smuzhiyun
171*4882a593Smuzhiyun MODULE_LICENSE("GPL v2");
172*4882a593Smuzhiyun MODULE_DESCRIPTION("ASoC MOP500 board driver");
173*4882a593Smuzhiyun MODULE_AUTHOR("Ola Lilja");
174