1 // SPDX-License-Identifier: GPL-2.0+
3 // Copyright (c) 2014, Insignal Co., Ltd.
5 // Author: Claude <claude@insginal.co.kr>
7 #include <linux/module.h>
8 #include <linux/of_device.h>
9 #include <linux/platform_device.h>
10 #include <linux/clk.h>
12 #include <sound/soc.h>
13 #include <sound/soc-dapm.h>
14 #include <sound/pcm.h>
15 #include <sound/pcm_params.h>
19 static int arndale_hw_params(struct snd_pcm_substream
*substream
,
20 struct snd_pcm_hw_params
*params
)
22 struct snd_soc_pcm_runtime
*rtd
= substream
->private_data
;
23 struct snd_soc_dai
*cpu_dai
= rtd
->cpu_dai
;
24 struct snd_soc_dai
*codec_dai
= rtd
->codec_dai
;
30 rclk
= params_rate(params
) * rfs
;
32 ret
= snd_soc_dai_set_sysclk(cpu_dai
, SAMSUNG_I2S_CDCLK
,
33 0, SND_SOC_CLOCK_OUT
);
37 ret
= snd_soc_dai_set_sysclk(cpu_dai
, SAMSUNG_I2S_RCLKSRC_0
,
38 0, SND_SOC_CLOCK_OUT
);
43 ret
= snd_soc_dai_set_sysclk(codec_dai
, 0, rclk
, SND_SOC_CLOCK_OUT
);
50 static struct snd_soc_ops arndale_ops
= {
51 .hw_params
= arndale_hw_params
,
54 SND_SOC_DAILINK_DEFS(rt5631_hifi
,
55 DAILINK_COMP_ARRAY(COMP_EMPTY()),
56 DAILINK_COMP_ARRAY(COMP_CODEC(NULL
, "rt5631-hifi")),
57 DAILINK_COMP_ARRAY(COMP_EMPTY()));
59 static struct snd_soc_dai_link arndale_rt5631_dai
[] = {
61 .name
= "RT5631 HiFi",
62 .stream_name
= "Primary",
63 .dai_fmt
= SND_SOC_DAIFMT_I2S
64 | SND_SOC_DAIFMT_NB_NF
65 | SND_SOC_DAIFMT_CBS_CFS
,
67 SND_SOC_DAILINK_REG(rt5631_hifi
),
71 static struct snd_soc_card arndale_rt5631
= {
72 .name
= "Arndale RT5631",
74 .dai_link
= arndale_rt5631_dai
,
75 .num_links
= ARRAY_SIZE(arndale_rt5631_dai
),
78 static void arndale_put_of_nodes(struct snd_soc_card
*card
)
80 struct snd_soc_dai_link
*dai_link
;
83 for_each_card_prelinks(card
, i
, dai_link
) {
84 of_node_put(dai_link
->cpus
->of_node
);
85 of_node_put(dai_link
->codecs
->of_node
);
89 static int arndale_audio_probe(struct platform_device
*pdev
)
92 struct device_node
*np
= pdev
->dev
.of_node
;
93 struct snd_soc_card
*card
= &arndale_rt5631
;
95 card
->dev
= &pdev
->dev
;
97 for (n
= 0; np
&& n
< ARRAY_SIZE(arndale_rt5631_dai
); n
++) {
98 if (!arndale_rt5631_dai
[n
].cpus
->dai_name
) {
99 arndale_rt5631_dai
[n
].cpus
->of_node
= of_parse_phandle(np
,
100 "samsung,audio-cpu", n
);
102 if (!arndale_rt5631_dai
[n
].cpus
->of_node
) {
104 "Property 'samsung,audio-cpu' missing or invalid\n");
108 if (!arndale_rt5631_dai
[n
].platforms
->name
)
109 arndale_rt5631_dai
[n
].platforms
->of_node
=
110 arndale_rt5631_dai
[n
].cpus
->of_node
;
112 arndale_rt5631_dai
[n
].codecs
->name
= NULL
;
113 arndale_rt5631_dai
[n
].codecs
->of_node
= of_parse_phandle(np
,
114 "samsung,audio-codec", n
);
115 if (!arndale_rt5631_dai
[0].codecs
->of_node
) {
117 "Property 'samsung,audio-codec' missing or invalid\n");
119 goto err_put_of_nodes
;
123 ret
= devm_snd_soc_register_card(card
->dev
, card
);
125 dev_err(&pdev
->dev
, "snd_soc_register_card() failed: %d\n", ret
);
126 goto err_put_of_nodes
;
131 arndale_put_of_nodes(card
);
135 static int arndale_audio_remove(struct platform_device
*pdev
)
137 struct snd_soc_card
*card
= platform_get_drvdata(pdev
);
139 arndale_put_of_nodes(card
);
143 static const struct of_device_id samsung_arndale_rt5631_of_match
[] __maybe_unused
= {
144 { .compatible
= "samsung,arndale-rt5631", },
145 { .compatible
= "samsung,arndale-alc5631", },
148 MODULE_DEVICE_TABLE(of
, samsung_arndale_rt5631_of_match
);
150 static struct platform_driver arndale_audio_driver
= {
152 .name
= "arndale-audio",
153 .pm
= &snd_soc_pm_ops
,
154 .of_match_table
= of_match_ptr(samsung_arndale_rt5631_of_match
),
156 .probe
= arndale_audio_probe
,
157 .remove
= arndale_audio_remove
,
160 module_platform_driver(arndale_audio_driver
);
162 MODULE_AUTHOR("Claude <claude@insignal.co.kr>");
163 MODULE_DESCRIPTION("ALSA SoC Driver for Arndale Board");
164 MODULE_LICENSE("GPL");