]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
ASoC: wm8962: add .set_tdm_slot callback function
authorShengjiu Wang <shengjiu.wang@nxp.com>
Wed, 28 Jan 2026 02:59:55 +0000 (10:59 +0800)
committerMark Brown <broonie@kernel.org>
Wed, 28 Jan 2026 11:22:24 +0000 (11:22 +0000)
The slot_width can be different with the params_width(), for example,
DSP_A mode, slot_width = 32, but data format is S16_LE, if the word
length is configured to be 16, there is no sound on the right speaker.

So add .set_tdm_slot() callback function to configure the slot_width and
update the word length according to slot_width in hw_params().

Signed-off-by: Shengjiu Wang <shengjiu.wang@nxp.com>
Reviewed-by: Charles Keepax <ckeepax@opensource.cirrus.com>
Link: https://patch.msgid.link/20260128025955.2562331-1-shengjiu.wang@nxp.com
Signed-off-by: Mark Brown <broonie@kernel.org>
sound/soc/codecs/wm8962.c

index bff86446741631098103789df96726c3c1445ff6..8d2435bf44eab9d1a0b6511e2e668e98625933bc 100644 (file)
@@ -85,6 +85,8 @@ struct wm8962_priv {
 
        int irq;
        bool master_flag;
+       int tdm_width;
+       int tdm_slots;
 };
 
 /* We can't use the same notifier block for more than one supply and
@@ -2612,6 +2614,19 @@ static int wm8962_set_bias_level(struct snd_soc_component *component,
        return 0;
 }
 
+static int wm8962_set_tdm_slot(struct snd_soc_dai *dai, unsigned int tx_mask,
+                              unsigned int rx_mask, int slots, int slot_width)
+{
+       struct snd_soc_component *component = dai->component;
+       struct wm8962_priv *wm8962 = snd_soc_component_get_drvdata(component);
+
+       wm8962->tdm_width = slot_width;
+       /* External is one slot one channel, but internal is one slot two channels */
+       wm8962->tdm_slots = slots / 2;
+
+       return 0;
+}
+
 static const struct {
        int rate;
        int reg;
@@ -2639,10 +2654,21 @@ static int wm8962_hw_params(struct snd_pcm_substream *substream,
        int i;
        int aif0 = 0;
        int adctl3 = 0;
+       int width;
+
+       if (wm8962->tdm_width && wm8962->tdm_slots) {
+               wm8962->bclk = snd_soc_calc_bclk(params_rate(params),
+                                                wm8962->tdm_width,
+                                                params_channels(params),
+                                                wm8962->tdm_slots);
+               width = wm8962->tdm_width;
+       } else {
+               wm8962->bclk = snd_soc_params_to_bclk(params);
+               width = params_width(params);
 
-       wm8962->bclk = snd_soc_params_to_bclk(params);
-       if (params_channels(params) == 1)
-               wm8962->bclk *= 2;
+               if (params_channels(params) == 1)
+                       wm8962->bclk *= 2;
+       }
 
        wm8962->lrclk = params_rate(params);
 
@@ -2660,7 +2686,7 @@ static int wm8962_hw_params(struct snd_pcm_substream *substream,
        if (wm8962->lrclk % 8000 == 0)
                adctl3 |= WM8962_SAMPLE_RATE_INT_MODE;
 
-       switch (params_width(params)) {
+       switch (width) {
        case 16:
                break;
        case 20:
@@ -3039,6 +3065,7 @@ static const struct snd_soc_dai_ops wm8962_dai_ops = {
        .hw_params = wm8962_hw_params,
        .set_sysclk = wm8962_set_dai_sysclk,
        .set_fmt = wm8962_set_dai_fmt,
+       .set_tdm_slot = wm8962_set_tdm_slot,
        .mute_stream = wm8962_mute,
        .no_capture_mute = 1,
 };