ASoC: audio_graph_card2: Add support for variable slot widths
authorRichard Fitzgerald <rf@opensource.cirrus.com>
Mon, 28 Feb 2022 17:27:54 +0000 (17:27 +0000)
committerMark Brown <broonie@kernel.org>
Mon, 7 Mar 2022 13:12:58 +0000 (13:12 +0000)
Some audio hardware cannot support the same slot width for all sample
widths, or a slot width equal to the sample width for all sample widths.
This is usually due either to limitations of the audio serial port or
system clocking restrictions.
A typical example would be:

- 16-bit samples in 16-bit slots
- 24-bit samples in 32-bit slots

The new dai-tdm-slot-width-map property allows setting a mapping of
sample widths and the corresponding tdm slot widths and slot counts.
Although the slot count is usually the same for all cases this does
allow for adding padding slots to maintain the same bitclk frequency.

The property is added to each endpoint node that needs the component
DAI to be told the TDM slot width and count.

Signed-off-by: Richard Fitzgerald <rf@opensource.cirrus.com>
Link: https://lore.kernel.org/r/20220228172754.453783-3-rf@opensource.cirrus.com
Signed-off-by: Mark Brown <broonie@kernel.org>
include/sound/simple_card_utils.h
sound/soc/generic/audio-graph-card2.c
sound/soc/generic/simple-card-utils.c

index 5ee269c59aac67294dff48b5ed5d7020157e1ea5..8faa649f712b80644c6d9ed0de9d24f92bcf6043 100644 (file)
 #define asoc_simple_init_mic(card, sjack, prefix) \
        asoc_simple_init_jack(card, sjack, 0, prefix, NULL)
 
+struct asoc_simple_tdm_width_map {
+       u8 sample_bits;
+       u8 slot_count;
+       u16 slot_width;
+};
+
 struct asoc_simple_dai {
        const char *name;
        unsigned int sysclk;
@@ -26,6 +32,8 @@ struct asoc_simple_dai {
        unsigned int rx_slot_mask;
        struct clk *clk;
        bool clk_fixed;
+       struct asoc_simple_tdm_width_map *tdm_width_map;
+       int n_tdm_widths;
 };
 
 struct asoc_simple_data {
@@ -132,6 +140,9 @@ int asoc_simple_parse_daifmt(struct device *dev,
                             struct device_node *codec,
                             char *prefix,
                             unsigned int *retfmt);
+int asoc_simple_parse_tdm_width_map(struct device *dev, struct device_node *np,
+                                   struct asoc_simple_dai *dai);
+
 __printf(3, 4)
 int asoc_simple_set_dailink_name(struct device *dev,
                                 struct snd_soc_dai_link *dai_link,
index c3947347dda3573c8363b4b4ee4ce24baafc1bfb..c0f3907a01fd416f687981885ac37efa55232d58 100644 (file)
@@ -503,6 +503,10 @@ static int __graph_parse_node(struct asoc_simple_priv *priv,
        if (ret < 0)
                return ret;
 
+       ret = asoc_simple_parse_tdm_width_map(dev, ep, dai);
+       if (ret < 0)
+               return ret;
+
        ret = asoc_simple_parse_clk(dev, ep, dai, dlc);
        if (ret < 0)
                return ret;
index a4babfb63175e93bec0394c99336bb3abd5f820d..14c8b3a74c2d31a1cd31563d0e68b8cb7c4ccc09 100644 (file)
@@ -12,6 +12,7 @@
 #include <linux/of_gpio.h>
 #include <linux/of_graph.h>
 #include <sound/jack.h>
+#include <sound/pcm_params.h>
 #include <sound/simple_card_utils.h>
 
 void asoc_simple_convert_fixup(struct asoc_simple_data *data,
@@ -87,6 +88,51 @@ int asoc_simple_parse_daifmt(struct device *dev,
 }
 EXPORT_SYMBOL_GPL(asoc_simple_parse_daifmt);
 
+int asoc_simple_parse_tdm_width_map(struct device *dev, struct device_node *np,
+                                   struct asoc_simple_dai *dai)
+{
+       u32 *array_values, *p;
+       int n, i, ret;
+
+       if (!of_property_read_bool(np, "dai-tdm-slot-width-map"))
+               return 0;
+
+       n = of_property_count_elems_of_size(np, "dai-tdm-slot-width-map", sizeof(u32));
+       if (n % 3) {
+               dev_err(dev, "Invalid number of cells for dai-tdm-slot-width-map\n");
+               return -EINVAL;
+       }
+
+       dai->tdm_width_map = devm_kcalloc(dev, n, sizeof(*dai->tdm_width_map), GFP_KERNEL);
+       if (!dai->tdm_width_map)
+               return -ENOMEM;
+
+       array_values = kcalloc(n, sizeof(*array_values), GFP_KERNEL);
+       if (!array_values)
+               return -ENOMEM;
+
+       ret = of_property_read_u32_array(np, "dai-tdm-slot-width-map", array_values, n);
+       if (ret < 0) {
+               dev_err(dev, "Could not read dai-tdm-slot-width-map: %d\n", ret);
+               goto out;
+       }
+
+       p = array_values;
+       for (i = 0; i < n / 3; ++i) {
+               dai->tdm_width_map[i].sample_bits = *p++;
+               dai->tdm_width_map[i].slot_width = *p++;
+               dai->tdm_width_map[i].slot_count = *p++;
+       }
+
+       dai->n_tdm_widths = i;
+       ret = 0;
+out:
+       kfree(array_values);
+
+       return ret;
+}
+EXPORT_SYMBOL_GPL(asoc_simple_parse_tdm_width_map);
+
 int asoc_simple_set_dailink_name(struct device *dev,
                                 struct snd_soc_dai_link *dai_link,
                                 const char *fmt, ...)
@@ -309,6 +355,42 @@ static int asoc_simple_set_clk_rate(struct device *dev,
        return clk_set_rate(simple_dai->clk, rate);
 }
 
+static int asoc_simple_set_tdm(struct snd_soc_dai *dai,
+                               struct asoc_simple_dai *simple_dai,
+                               struct snd_pcm_hw_params *params)
+{
+       int sample_bits = params_width(params);
+       int slot_width = simple_dai->slot_width;
+       int slot_count = simple_dai->slots;
+       int i, ret;
+
+       if (!simple_dai || !simple_dai->tdm_width_map)
+               return 0;
+
+       if (slot_width == 0)
+               slot_width = sample_bits;
+
+       for (i = 0; i < simple_dai->n_tdm_widths; ++i) {
+               if (simple_dai->tdm_width_map[i].sample_bits == sample_bits) {
+                       slot_width = simple_dai->tdm_width_map[i].slot_width;
+                       slot_count = simple_dai->tdm_width_map[i].slot_count;
+                       break;
+               }
+       }
+
+       ret = snd_soc_dai_set_tdm_slot(dai,
+                                      simple_dai->tx_slot_mask,
+                                      simple_dai->rx_slot_mask,
+                                      slot_count,
+                                      slot_width);
+       if (ret && ret != -ENOTSUPP) {
+               dev_err(dai->dev, "simple-card: set_tdm_slot error: %d\n", ret);
+               return ret;
+       }
+
+       return 0;
+}
+
 int asoc_simple_hw_params(struct snd_pcm_substream *substream,
                          struct snd_pcm_hw_params *params)
 {
@@ -362,6 +444,21 @@ int asoc_simple_hw_params(struct snd_pcm_substream *substream,
                                return ret;
                }
        }
+
+       for_each_prop_dai_codec(props, i, pdai) {
+               sdai = asoc_rtd_to_codec(rtd, i);
+               ret = asoc_simple_set_tdm(sdai, pdai, params);
+               if (ret < 0)
+                       return ret;
+       }
+
+       for_each_prop_dai_cpu(props, i, pdai) {
+               sdai = asoc_rtd_to_cpu(rtd, i);
+               ret = asoc_simple_set_tdm(sdai, pdai, params);
+               if (ret < 0)
+                       return ret;
+       }
+
        return 0;
 }
 EXPORT_SYMBOL_GPL(asoc_simple_hw_params);