Commit 5a504963 authored by Stephen Warren's avatar Stephen Warren Committed by Mark Brown

ASoC: Allow DAI links to be specified using device tree nodes

DAI link endpoints and platform (DMA) devices are currently specified
by name. When instantiating sound cards from device tree, it may be more
convenient to refer to these devices by phandle in the device tree, and
for code to describe DAI links using the "struct device_node *"
("of_node") those phandles map to.

This change adds new fields to snd_soc_dai_link which can "name" devices
using of_node, enhances soc_bind_dai_link() to allow binding based on
of_node, and enhances snd_soc_register_card() to ensure that illegal
combinations of name and of_node are not used.
Signed-off-by: default avatarStephen Warren <swarren@nvidia.com>
Acked-by: default avatarLiam Girdwood <lrg@ti.com>
Signed-off-by: default avatarMark Brown <broonie@opensource.wolfsonmicro.com>
parent 82150101
...@@ -231,6 +231,7 @@ enum snd_soc_bias_level { ...@@ -231,6 +231,7 @@ enum snd_soc_bias_level {
SND_SOC_BIAS_ON = 3, SND_SOC_BIAS_ON = 3,
}; };
struct device_node;
struct snd_jack; struct snd_jack;
struct snd_soc_card; struct snd_soc_card;
struct snd_soc_pcm_stream; struct snd_soc_pcm_stream;
...@@ -703,8 +704,11 @@ struct snd_soc_dai_link { ...@@ -703,8 +704,11 @@ struct snd_soc_dai_link {
const char *name; /* Codec name */ const char *name; /* Codec name */
const char *stream_name; /* Stream name */ const char *stream_name; /* Stream name */
const char *codec_name; /* for multi-codec */ const char *codec_name; /* for multi-codec */
const struct device_node *codec_of_node;
const char *platform_name; /* for multi-platform */ const char *platform_name; /* for multi-platform */
const struct device_node *platform_of_node;
const char *cpu_dai_name; const char *cpu_dai_name;
const struct device_node *cpu_dai_of_node;
const char *codec_dai_name; const char *codec_dai_name;
unsigned int dai_fmt; /* format to set on init */ unsigned int dai_fmt; /* format to set on init */
......
...@@ -764,8 +764,13 @@ static int soc_bind_dai_link(struct snd_soc_card *card, int num) ...@@ -764,8 +764,13 @@ static int soc_bind_dai_link(struct snd_soc_card *card, int num)
} }
/* no, then find CPU DAI from registered DAIs*/ /* no, then find CPU DAI from registered DAIs*/
list_for_each_entry(cpu_dai, &dai_list, list) { list_for_each_entry(cpu_dai, &dai_list, list) {
if (strcmp(cpu_dai->name, dai_link->cpu_dai_name)) if (dai_link->cpu_dai_of_node) {
continue; if (cpu_dai->dev->of_node != dai_link->cpu_dai_of_node)
continue;
} else {
if (strcmp(cpu_dai->name, dai_link->cpu_dai_name))
continue;
}
rtd->cpu_dai = cpu_dai; rtd->cpu_dai = cpu_dai;
goto find_codec; goto find_codec;
...@@ -781,8 +786,13 @@ static int soc_bind_dai_link(struct snd_soc_card *card, int num) ...@@ -781,8 +786,13 @@ static int soc_bind_dai_link(struct snd_soc_card *card, int num)
/* no, then find CODEC from registered CODECs*/ /* no, then find CODEC from registered CODECs*/
list_for_each_entry(codec, &codec_list, list) { list_for_each_entry(codec, &codec_list, list) {
if (strcmp(codec->name, dai_link->codec_name)) if (dai_link->codec_of_node) {
continue; if (codec->dev->of_node != dai_link->codec_of_node)
continue;
} else {
if (strcmp(codec->name, dai_link->codec_name))
continue;
}
rtd->codec = codec; rtd->codec = codec;
...@@ -814,13 +824,19 @@ static int soc_bind_dai_link(struct snd_soc_card *card, int num) ...@@ -814,13 +824,19 @@ static int soc_bind_dai_link(struct snd_soc_card *card, int num)
/* if there's no platform we match on the empty platform */ /* if there's no platform we match on the empty platform */
platform_name = dai_link->platform_name; platform_name = dai_link->platform_name;
if (!platform_name) if (!platform_name && !dai_link->platform_of_node)
platform_name = "snd-soc-dummy"; platform_name = "snd-soc-dummy";
/* no, then find one from the set of registered platforms */ /* no, then find one from the set of registered platforms */
list_for_each_entry(platform, &platform_list, list) { list_for_each_entry(platform, &platform_list, list) {
if (strcmp(platform->name, platform_name)) if (dai_link->platform_of_node) {
continue; if (platform->dev->of_node !=
dai_link->platform_of_node)
continue;
} else {
if (strcmp(platform->name, platform_name))
continue;
}
rtd->platform = platform; rtd->platform = platform;
goto out; goto out;
...@@ -2831,6 +2847,40 @@ int snd_soc_register_card(struct snd_soc_card *card) ...@@ -2831,6 +2847,40 @@ int snd_soc_register_card(struct snd_soc_card *card)
if (!card->name || !card->dev) if (!card->name || !card->dev)
return -EINVAL; return -EINVAL;
for (i = 0; i < card->num_links; i++) {
struct snd_soc_dai_link *link = &card->dai_link[i];
/*
* Codec must be specified by 1 of name or OF node,
* not both or neither.
*/
if (!!link->codec_name == !!link->codec_of_node) {
dev_err(card->dev,
"Neither/both codec name/of_node are set\n");
return -EINVAL;
}
/*
* Platform may be specified by either name or OF node, but
* can be left unspecified, and a dummy platform will be used.
*/
if (link->platform_name && link->platform_of_node) {
dev_err(card->dev,
"Both platform name/of_node are set\n");
return -EINVAL;
}
/*
* CPU DAI must be specified by 1 of name or OF node,
* not both or neither.
*/
if (!!link->cpu_dai_name == !!link->cpu_dai_of_node) {
dev_err(card->dev,
"Neither/both cpu_dai name/of_node are set\n");
return -EINVAL;
}
}
dev_set_drvdata(card->dev, card); dev_set_drvdata(card->dev, card);
snd_soc_initialize_card_lists(card); snd_soc_initialize_card_lists(card);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment