[alsa-devel] [PATCH] ASoC:simple-card: Add multi-CODEC support

Benoit Cousson bcousson at baylibre.com
Wed Sep 10 13:43:06 CEST 2014


Hi Jean-Francois,

On 10/09/2014 13:28, Jean-Francois Moine wrote:
> This patch adds multi-CODEC support to the simple-card.
>
> Signed-off-by: Jean-Francois Moine <moinejf at free.fr>
> ---
>   .../devicetree/bindings/sound/simple-card.txt      | 28 +++++++--------
>   sound/soc/generic/simple-card.c                    | 41 +++++++++++++++++++++-
>   2 files changed, 52 insertions(+), 17 deletions(-)
>
> diff --git a/Documentation/devicetree/bindings/sound/simple-card.txt b/Documentation/devicetree/bindings/sound/simple-card.txt
> index c2e9841..c217687 100644
> --- a/Documentation/devicetree/bindings/sound/simple-card.txt
> +++ b/Documentation/devicetree/bindings/sound/simple-card.txt
> @@ -37,6 +37,8 @@ Required dai-link subnodes:
>
>   - cpu					: CPU   sub-node
>   - codec					: CODEC sub-node
> +					  In case of multi-CODECs, there may
> +					  be many of such sub-nodes.
>
>   Optional dai-link subnode properties:
>
> @@ -115,37 +117,31 @@ sh_fsi2: sh_fsi2 at ec230000 {
>   	interrupts = <0 146 0x4>;
>   };
>
> -Example 2 - many DAI links:
> +Example 2 - many DAI links and multi-CODECs:
>
>   sound {
>   	compatible = "simple-audio-card";
>   	simple-audio-card,name = "Cubox Audio";
>
> -	simple-audio-card,dai-link at 0 {		/* I2S - HDMI */
> +	simple-audio-card,dai-link at 0 {		/* S/PDIF - HDMI & S/PDIF */
>   		format = "i2s";
>   		cpu {
> -			sound-dai = <&audio1 0>;
> -		};
> -		codec {
> -			sound-dai = <&tda998x 0>;
> -		};
> -	};
> -
> -	simple-audio-card,dai-link at 1 {		/* S/PDIF - HDMI */
> -		cpu {
>   			sound-dai = <&audio1 1>;
>   		};
> -		codec {
> -			sound-dai = <&tda998x 1>;
> +		codec at 0 {
> +			sound-dai = <&hdmi 0>;
> +		};
> +		codec at 1 {
> +			sound-dai = <&spdif_codec>;
>   		};

I don't have strong opinion on that, but in my case, I was considering 
using a simple list instead of several nodes.
I don't like having to add fake address just to ensure uniqueness.

Something like that:

  sound-dais = <&spdif_codec 1>, <&hdmi 0>;

That being said, it will require changing the name with a plural form, 
and ensuring we have the same number of parameters for each codec.

That was just my .2 cents.

Regards,
Benoit



>   	};
>
> -	simple-audio-card,dai-link at 2 {		/* S/PDIF - S/PDIF */
> +	simple-audio-card,dai-link at 1 {		/* I2S - HDMI */
>   		cpu {
> -			sound-dai = <&audio1 1>;
> +			sound-dai = <&audio1 0>;
>   		};
>   		codec {
> -			sound-dai = <&spdif_codec>;
> +			sound-dai = <&hdmi 1>;
>   		};
>   	};
>   };
> diff --git a/sound/soc/generic/simple-card.c b/sound/soc/generic/simple-card.c
> index 4053152..bf0ce08 100644
> --- a/sound/soc/generic/simple-card.c
> +++ b/sound/soc/generic/simple-card.c
> @@ -179,11 +179,12 @@ static int asoc_simple_card_dai_link_of(struct device_node *node,
>   	struct device_node *np = NULL;
>   	struct device_node *bitclkmaster = NULL;
>   	struct device_node *framemaster = NULL;
> +	struct snd_soc_dai_link_component *component;
>   	unsigned int daifmt;
>   	char *name;
>   	char prop[128];
>   	char *prefix = "";
> -	int ret, cpu_args;
> +	int ret, cpu_args, num_codec_dais;
>
>   	/* For single DAI link & old style of DT node */
>   	if (is_top_level_node)
> @@ -225,7 +226,16 @@ static int asoc_simple_card_dai_link_of(struct device_node *node,
>   	}
>
>   	of_node_put(np);
> +
> +	/* count the number of codec DAIs */
>   	snprintf(prop, sizeof(prop), "%scodec", prefix);
> +	num_codec_dais = 0;
> +	for_each_child_of_node(node, np) {
> +		if (strcmp(np->name, prop) == 0)
> +			num_codec_dais++;
> +	}
> +
> +	/* treat the first DAI */
>   	np = of_get_child_by_name(node, prop);
>   	if (!np) {
>   		ret = -EINVAL;
> @@ -307,6 +317,35 @@ static int asoc_simple_card_dai_link_of(struct device_node *node,
>   	if (!cpu_args)
>   		dai_link->cpu_dai_name = NULL;
>
> +	/* handle multi-codec DAIs */
> +	if (num_codec_dais == 1)
> +		goto out;
> +	dai_link->codecs = component =
> +			   devm_kzalloc(dev,
> +					sizeof *component * num_codec_dais,
> +					GFP_KERNEL);
> +	dai_link->num_codecs = num_codec_dais;
> +	component->of_node = dai_link->codec_of_node;
> +	dai_link->codec_of_node = NULL;
> +	component->dai_name = dai_link->codec_dai_name;
> +	dai_link->codec_dai_name = NULL;
> +	for (;;) {
> +		np = of_get_next_child(node, np);
> +		if (!np)
> +			break;
> +		component++;
> +		component->of_node = of_parse_phandle(np, "sound-dai", 0);
> +		if (!component->of_node) {
> +			ret = -ENODEV;
> +			dev_err(dev, "Bad sound-dai\n");
> +			goto dai_link_of_err;
> +		}
> +		ret = snd_soc_of_get_dai_name(np, &component->dai_name);
> +		if (ret < 0)
> +			goto dai_link_of_err;
> +	}
> +
> +out:
>   dai_link_of_err:
>   	if (np)
>   		of_node_put(np);
>


-- 
Benoît Cousson
BayLibre
Embedded Linux Technology Lab
www.baylibre.com


More information about the Alsa-devel mailing list