Re: [PATCH v2 1/3] ASoC: rockchip: Parse dai links from dts

2017-08-16 Thread jeffy

Hi Matthias,

On 08/17/2017 07:50 AM, Matthias Kaehlcke wrote:

El Thu, Aug 17, 2017 at 06:55:20AM +0800 jeffy ha dit:


hi matthias,

thanks for your suggestion.

On 08/17/2017 05:59 AM, Matthias Kaehlcke wrote:

El Thu, Aug 10, 2017 at 12:54:56PM +0800 Jeffy Chen ha dit:


Refactor rockchip_sound_probe, parse dai links from dts instead of
hard coding them.

Mark doesn't seem to be overly convinced that 'rockchip,codec-names'
is a good idea (https://lkml.org/lkml/2017/8/10/511).

How about using something like this instead:

static const char *dailink_compat[] = {
[DAILINK_MAX98357A] = "maxim,max98357a",
[DAILINK_RT5514] = "realtek,rt5514",
[DAILINK_DA7219] = "dlg,da7219",
};

i've thought about this too, but i'm working on converting rt5514
dsp(spi) from codec name matching to of_node, and it would have the
same compatible with rt5514(i2c)


Bummer!

I wonder if a relatively inoffensive DT hack would be an appropriate
solution in this case, since the conflicting compatible string is a
somewhat special case and this change only affects the DT and the
driver/glue of a specific device (family).

The hack would consist in adding an additional 'compatible' entry to
the DT entry of the codec, which is ignored by the rt5514 driver, and
only used by the sound glue to identify it:

--- a/arch/arm64/boot/dts/rockchip/rk3399-gru.dtsi
+++ b/arch/arm64/boot/dts/rockchip/rk3399-gru.dtsi
@@ -671,7 +671,7 @@ ap_i2c_mic: &i2c1 {
 i2c-scl-rising-time-ns = <300>;

 headsetcodec: rt5514@57 {
-   compatible = "realtek,rt5514";
+   compatible = "realtek,rt5514", "realtek,rt5514-i2c";


And then use "realtek,rt5514-i2c" in dailink_compat.

this should work, i'll do that in new version, thanks.



Mark, would you prefer a hack like this over the list of codec names
or do you have any other suggestions?

Matthias








Re: [PATCH v2 1/3] ASoC: rockchip: Parse dai links from dts

2017-08-16 Thread Matthias Kaehlcke
El Thu, Aug 17, 2017 at 06:55:20AM +0800 jeffy ha dit:

> hi matthias,
> 
> thanks for your suggestion.
> 
> On 08/17/2017 05:59 AM, Matthias Kaehlcke wrote:
> >El Thu, Aug 10, 2017 at 12:54:56PM +0800 Jeffy Chen ha dit:
> >
> >>>Refactor rockchip_sound_probe, parse dai links from dts instead of
> >>>hard coding them.
> >Mark doesn't seem to be overly convinced that 'rockchip,codec-names'
> >is a good idea (https://lkml.org/lkml/2017/8/10/511).
> >
> >How about using something like this instead:
> >
> >static const char *dailink_compat[] = {
> > [DAILINK_MAX98357A] = "maxim,max98357a",
> > [DAILINK_RT5514] = "realtek,rt5514",
> > [DAILINK_DA7219] = "dlg,da7219",
> >};
> i've thought about this too, but i'm working on converting rt5514
> dsp(spi) from codec name matching to of_node, and it would have the
> same compatible with rt5514(i2c)

Bummer!

I wonder if a relatively inoffensive DT hack would be an appropriate
solution in this case, since the conflicting compatible string is a
somewhat special case and this change only affects the DT and the
driver/glue of a specific device (family).

The hack would consist in adding an additional 'compatible' entry to
the DT entry of the codec, which is ignored by the rt5514 driver, and
only used by the sound glue to identify it:

--- a/arch/arm64/boot/dts/rockchip/rk3399-gru.dtsi
+++ b/arch/arm64/boot/dts/rockchip/rk3399-gru.dtsi
@@ -671,7 +671,7 @@ ap_i2c_mic: &i2c1 {
i2c-scl-rising-time-ns = <300>;
 
headsetcodec: rt5514@57 {
-   compatible = "realtek,rt5514";
+   compatible = "realtek,rt5514", "realtek,rt5514-i2c";


And then use "realtek,rt5514-i2c" in dailink_compat.

Mark, would you prefer a hack like this over the list of codec names
or do you have any other suggestions?

Matthias


Re: [PATCH v2 1/3] ASoC: rockchip: Parse dai links from dts

2017-08-16 Thread jeffy

hi matthias,

thanks for your suggestion.

On 08/17/2017 05:59 AM, Matthias Kaehlcke wrote:

El Thu, Aug 10, 2017 at 12:54:56PM +0800 Jeffy Chen ha dit:


>Refactor rockchip_sound_probe, parse dai links from dts instead of
>hard coding them.

Mark doesn't seem to be overly convinced that 'rockchip,codec-names'
is a good idea (https://lkml.org/lkml/2017/8/10/511).

How about using something like this instead:

static const char *dailink_compat[] = {
[DAILINK_MAX98357A] = "maxim,max98357a",
[DAILINK_RT5514] = "realtek,rt5514",
[DAILINK_DA7219] = "dlg,da7219",
};
i've thought about this too, but i'm working on converting rt5514 
dsp(spi) from codec name matching to of_node, and it would have the same 
compatible with rt5514(i2c)




Re: [PATCH v2 1/3] ASoC: rockchip: Parse dai links from dts

2017-08-16 Thread Matthias Kaehlcke
El Thu, Aug 10, 2017 at 12:54:56PM +0800 Jeffy Chen ha dit:

> Refactor rockchip_sound_probe, parse dai links from dts instead of
> hard coding them.

Mark doesn't seem to be overly convinced that 'rockchip,codec-names'
is a good idea (https://lkml.org/lkml/2017/8/10/511).

How about using something like this instead:

static const char *dailink_compat[] = {
[DAILINK_MAX98357A] = "maxim,max98357a",
[DAILINK_RT5514] = "realtek,rt5514",
[DAILINK_DA7219] = "dlg,da7219",
};

...

static int rockchip_sound_probe(struct platform_device *pdev)
{
...

for (i = 0; ; i++) {
struct device_node *codec_node;
int j;

codec_node = of_parse_phandle(pdev->dev.of_node, 
"rockchip,codec", i);
if (!codec_node)
break;

for (j = 0; j < DAILINK_ENTITIES; j++) {
if (of_device_is_compatible(codec_node, 
dailink_compat[j]))
break;
}

if (j == DAILINK_ENTITIES) {
dev_err(...);
return -EINVAL;
}

rockchip_dailinks[j].codec_of_node = codec_node;
rockchip_dailinks[j].platform_of_node = cpu_node;
rockchip_dailinks[j].cpu_of_node = cpu_node;

/* not strictly needed, could also check for
   rockchip_dailinks[DAILINK_RT5514].cpu_of_node or so
   */
if (j == DAILINK_RT5514)
has_rt5514 = true;
}
...
}

Matthias

> Signed-off-by: Jeffy Chen 
> ---
> 
> Changes in v2:
> Let rockchip,codec-names be a required property, because we plan to
> add more supported codecs to the fixed dai link list in the driver.
> 
>  sound/soc/rockchip/rk3399_gru_sound.c | 125 
> ++
>  1 file changed, 81 insertions(+), 44 deletions(-)
> 
> diff --git a/sound/soc/rockchip/rk3399_gru_sound.c 
> b/sound/soc/rockchip/rk3399_gru_sound.c
> index 3475c61..03b7fae 100644
> --- a/sound/soc/rockchip/rk3399_gru_sound.c
> +++ b/sound/soc/rockchip/rk3399_gru_sound.c
> @@ -247,9 +247,7 @@ enum {
>   DAILINK_RT5514_DSP,
>  };
>  
> -#define DAILINK_ENTITIES (DAILINK_DA7219 + 1)
> -
> -static struct snd_soc_dai_link rockchip_dailinks[] = {
> +static const struct snd_soc_dai_link rockchip_dais[] = {
>   [DAILINK_MAX98357A] = {
>   .name = "MAX98357A",
>   .stream_name = "MAX98357A PCM",
> @@ -290,8 +288,6 @@ static struct snd_soc_dai_link rockchip_dailinks[] = {
>  static struct snd_soc_card rockchip_sound_card = {
>   .name = "rk3399-gru-sound",
>   .owner = THIS_MODULE,
> - .dai_link = rockchip_dailinks,
> - .num_links =  ARRAY_SIZE(rockchip_dailinks),
>   .dapm_widgets = rockchip_dapm_widgets,
>   .num_dapm_widgets = ARRAY_SIZE(rockchip_dapm_widgets),
>   .dapm_routes = rockchip_dapm_routes,
> @@ -305,71 +301,112 @@ static int rockchip_sound_match_stub(struct device 
> *dev, void *data)
>   return 1;
>  }
>  
> -static int rockchip_sound_probe(struct platform_device *pdev)
> +static int rockchip_sound_of_parse_dais(struct device *dev,
> + struct snd_soc_card *card)
>  {
> - struct snd_soc_card *card = &rockchip_sound_card;
> + struct device *rt5514_dev;
> + struct device_driver *rt5514_drv;
>   struct device_node *cpu_node;
> - struct device *dev;
> - struct device_driver *drv;
> - int i, ret;
> -
> - cpu_node = of_parse_phandle(pdev->dev.of_node, "rockchip,cpu", 0);
> - if (!cpu_node) {
> - dev_err(&pdev->dev, "Property 'rockchip,cpu' missing or 
> invalid\n");
> - return -EINVAL;
> - }
> -
> - for (i = 0; i < DAILINK_ENTITIES; i++) {
> - rockchip_dailinks[i].platform_of_node = cpu_node;
> - rockchip_dailinks[i].cpu_of_node = cpu_node;
> + struct device_node *np_codec;
> + struct snd_soc_dai_link *dai;
> + bool has_rt5514 = false;
> + int i, index, ret;
> +
> + card->dai_link = devm_kzalloc(dev, sizeof(rockchip_dais),
> +   GFP_KERNEL);
> + if (!card->dai_link)
> + return -ENOMEM;
> +
> + cpu_node = of_parse_phandle(dev->of_node, "rockchip,cpu", 0);
> +
> + card->num_links = 0;
> + for (i = 0; i < DAILINK_RT5514_DSP; i++) {
> + index = of_property_match_string(dev->of_node,
> + "rockchip,codec-names",
> + rockchip_dais[i].name);
> + if (index < 0)
> + continue;
> +
> + np_codec = of_parse_phandle(dev->of_node,
> + "rockchip,codec", index);
> + if (!np_codec) {
> + dev_err(dev, "Missing 'rockchip,codec' for %s\n",
> + rockchip_dai

[PATCH v2 1/3] ASoC: rockchip: Parse dai links from dts

2017-08-09 Thread Jeffy Chen
Refactor rockchip_sound_probe, parse dai links from dts instead of
hard coding them.

Signed-off-by: Jeffy Chen 
---

Changes in v2:
Let rockchip,codec-names be a required property, because we plan to
add more supported codecs to the fixed dai link list in the driver.

 sound/soc/rockchip/rk3399_gru_sound.c | 125 ++
 1 file changed, 81 insertions(+), 44 deletions(-)

diff --git a/sound/soc/rockchip/rk3399_gru_sound.c 
b/sound/soc/rockchip/rk3399_gru_sound.c
index 3475c61..03b7fae 100644
--- a/sound/soc/rockchip/rk3399_gru_sound.c
+++ b/sound/soc/rockchip/rk3399_gru_sound.c
@@ -247,9 +247,7 @@ enum {
DAILINK_RT5514_DSP,
 };
 
-#define DAILINK_ENTITIES   (DAILINK_DA7219 + 1)
-
-static struct snd_soc_dai_link rockchip_dailinks[] = {
+static const struct snd_soc_dai_link rockchip_dais[] = {
[DAILINK_MAX98357A] = {
.name = "MAX98357A",
.stream_name = "MAX98357A PCM",
@@ -290,8 +288,6 @@ static struct snd_soc_dai_link rockchip_dailinks[] = {
 static struct snd_soc_card rockchip_sound_card = {
.name = "rk3399-gru-sound",
.owner = THIS_MODULE,
-   .dai_link = rockchip_dailinks,
-   .num_links =  ARRAY_SIZE(rockchip_dailinks),
.dapm_widgets = rockchip_dapm_widgets,
.num_dapm_widgets = ARRAY_SIZE(rockchip_dapm_widgets),
.dapm_routes = rockchip_dapm_routes,
@@ -305,71 +301,112 @@ static int rockchip_sound_match_stub(struct device *dev, 
void *data)
return 1;
 }
 
-static int rockchip_sound_probe(struct platform_device *pdev)
+static int rockchip_sound_of_parse_dais(struct device *dev,
+   struct snd_soc_card *card)
 {
-   struct snd_soc_card *card = &rockchip_sound_card;
+   struct device *rt5514_dev;
+   struct device_driver *rt5514_drv;
struct device_node *cpu_node;
-   struct device *dev;
-   struct device_driver *drv;
-   int i, ret;
-
-   cpu_node = of_parse_phandle(pdev->dev.of_node, "rockchip,cpu", 0);
-   if (!cpu_node) {
-   dev_err(&pdev->dev, "Property 'rockchip,cpu' missing or 
invalid\n");
-   return -EINVAL;
-   }
-
-   for (i = 0; i < DAILINK_ENTITIES; i++) {
-   rockchip_dailinks[i].platform_of_node = cpu_node;
-   rockchip_dailinks[i].cpu_of_node = cpu_node;
+   struct device_node *np_codec;
+   struct snd_soc_dai_link *dai;
+   bool has_rt5514 = false;
+   int i, index, ret;
+
+   card->dai_link = devm_kzalloc(dev, sizeof(rockchip_dais),
+ GFP_KERNEL);
+   if (!card->dai_link)
+   return -ENOMEM;
+
+   cpu_node = of_parse_phandle(dev->of_node, "rockchip,cpu", 0);
+
+   card->num_links = 0;
+   for (i = 0; i < DAILINK_RT5514_DSP; i++) {
+   index = of_property_match_string(dev->of_node,
+   "rockchip,codec-names",
+   rockchip_dais[i].name);
+   if (index < 0)
+   continue;
+
+   np_codec = of_parse_phandle(dev->of_node,
+   "rockchip,codec", index);
+   if (!np_codec) {
+   dev_err(dev, "Missing 'rockchip,codec' for %s\n",
+   rockchip_dais[i].name);
+   return -EINVAL;
+   }
+   if (!of_device_is_available(np_codec))
+   continue;
 
-   rockchip_dailinks[i].codec_of_node =
-   of_parse_phandle(pdev->dev.of_node, "rockchip,codec", 
i);
-   if (!rockchip_dailinks[i].codec_of_node) {
-   dev_err(&pdev->dev,
-   "Property[%d] 'rockchip,codec' missing or 
invalid\n", i);
+   if (!cpu_node) {
+   dev_err(dev, "Missing 'rockchip,cpu' for %s\n",
+   rockchip_dais[i].name);
return -EINVAL;
}
+
+   dai = &card->dai_link[card->num_links++];
+   *dai = rockchip_dais[i];
+
+   dai->codec_of_node = np_codec;
+   dai->platform_of_node = cpu_node;
+   dai->cpu_of_node = cpu_node;
+
+   if (i == DAILINK_RT5514)
+   has_rt5514 = true;
}
 
+   if (!has_rt5514)
+   return 0;
+
/**
 * To acquire the spi driver of the rt5514 and set the dai-links names
 * for soc_bind_dai_link
 */
-   drv = driver_find("rt5514", &spi_bus_type);
-   if (!drv) {
-   dev_err(&pdev->dev, "Can not find the rt5514 driver at the spi 
bus\n");
+   rt5514_drv = driver_find("rt5514", &spi_bus_type);
+   if (!rt5514_drv) {
+   dev_err(dev, "Can not find the rt5514 driver at the spi bus\n");
return -EINVAL;
}
 
-   dev = driver_