1 // SPDX-License-Identifier: GPL-2.0
3 // Phytec pcm030 driver for the PSC of the Freescale MPC52xx
4 // configured as AC97 interface
6 // Copyright 2008 Jon Smirl, Digispeaker
7 // Author: Jon Smirl <jonsmirl@gmail.com>
9 #include <linux/init.h>
10 #include <linux/module.h>
11 #include <linux/device.h>
14 #include <sound/soc.h>
16 #include "mpc5200_dma.h"
18 #define DRV_NAME "pcm030-audio-fabric"
20 struct pcm030_audio_data
{
21 struct snd_soc_card
*card
;
22 struct platform_device
*codec_device
;
25 SND_SOC_DAILINK_DEFS(analog
,
26 DAILINK_COMP_ARRAY(COMP_CPU("mpc5200-psc-ac97.0")),
27 DAILINK_COMP_ARRAY(COMP_CODEC("wm9712-codec", "wm9712-hifi")),
28 DAILINK_COMP_ARRAY(COMP_EMPTY()));
30 SND_SOC_DAILINK_DEFS(iec958
,
31 DAILINK_COMP_ARRAY(COMP_CPU("mpc5200-psc-ac97.1")),
32 DAILINK_COMP_ARRAY(COMP_CODEC("wm9712-codec", "wm9712-aux")),
33 DAILINK_COMP_ARRAY(COMP_EMPTY()));
35 static struct snd_soc_dai_link pcm030_fabric_dai
[] = {
38 .stream_name
= "AC97 Analog",
39 SND_SOC_DAILINK_REG(analog
),
43 .stream_name
= "AC97 IEC958",
44 SND_SOC_DAILINK_REG(iec958
),
48 static struct snd_soc_card pcm030_card
= {
51 .dai_link
= pcm030_fabric_dai
,
52 .num_links
= ARRAY_SIZE(pcm030_fabric_dai
),
55 static int pcm030_fabric_probe(struct platform_device
*op
)
57 struct device_node
*np
= op
->dev
.of_node
;
58 struct device_node
*platform_np
;
59 struct snd_soc_card
*card
= &pcm030_card
;
60 struct pcm030_audio_data
*pdata
;
61 struct snd_soc_dai_link
*dai_link
;
65 if (!of_machine_is_compatible("phytec,pcm030"))
68 pdata
= devm_kzalloc(&op
->dev
, sizeof(struct pcm030_audio_data
),
77 platform_np
= of_parse_phandle(np
, "asoc-platform", 0);
79 dev_err(&op
->dev
, "ac97 not registered\n");
83 for_each_card_prelinks(card
, i
, dai_link
)
84 dai_link
->platforms
->of_node
= platform_np
;
86 ret
= request_module("snd-soc-wm9712");
88 dev_err(&op
->dev
, "request_module returned: %d\n", ret
);
90 pdata
->codec_device
= platform_device_alloc("wm9712-codec", -1);
91 if (!pdata
->codec_device
)
92 dev_err(&op
->dev
, "platform_device_alloc() failed\n");
94 ret
= platform_device_add(pdata
->codec_device
);
96 dev_err(&op
->dev
, "platform_device_add() failed: %d\n", ret
);
97 platform_device_put(pdata
->codec_device
);
100 ret
= snd_soc_register_card(card
);
102 dev_err(&op
->dev
, "snd_soc_register_card() failed: %d\n", ret
);
103 platform_device_unregister(pdata
->codec_device
);
106 platform_set_drvdata(op
, pdata
);
111 static void pcm030_fabric_remove(struct platform_device
*op
)
113 struct pcm030_audio_data
*pdata
= platform_get_drvdata(op
);
115 snd_soc_unregister_card(pdata
->card
);
116 platform_device_unregister(pdata
->codec_device
);
119 static const struct of_device_id pcm030_audio_match
[] = {
120 { .compatible
= "phytec,pcm030-audio-fabric", },
123 MODULE_DEVICE_TABLE(of
, pcm030_audio_match
);
125 static struct platform_driver pcm030_fabric_driver
= {
126 .probe
= pcm030_fabric_probe
,
127 .remove
= pcm030_fabric_remove
,
130 .of_match_table
= pcm030_audio_match
,
134 module_platform_driver(pcm030_fabric_driver
);
137 MODULE_AUTHOR("Jon Smirl <jonsmirl@gmail.com>");
138 MODULE_DESCRIPTION(DRV_NAME
": mpc5200 pcm030 fabric driver");
139 MODULE_LICENSE("GPL");