ASoC: wm8962: add .set_tdm_slot callback function

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>
This commit is contained in:
Shengjiu Wang 2026-01-28 10:59:55 +08:00 committed by Mark Brown
parent 75ca86026b
commit ab3f4f0c7f
No known key found for this signature in database
GPG key ID: 24D68B725D5487D0

View 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;
wm8962->bclk = snd_soc_params_to_bclk(params);
if (params_channels(params) == 1)
wm8962->bclk *= 2;
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);
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,
};