ASoC: tas2552: Add TDM support
TDM support is achieved using DSP transfer mode and setting a programmable offset which specifies where data begins with respect to the frame sync. Signed-off-by: Peter Ujfalusi <peter.ujfalusi@ti.com> Signed-off-by: Mark Brown <broonie@kernel.org>
This commit is contained in:
parent
1b68c7dca2
commit
3f747a810e
2 changed files with 58 additions and 4 deletions
|
@ -78,6 +78,9 @@ struct tas2552_data {
|
||||||
unsigned char regs[TAS2552_VBAT_DATA];
|
unsigned char regs[TAS2552_VBAT_DATA];
|
||||||
unsigned int pll_clkin;
|
unsigned int pll_clkin;
|
||||||
unsigned int pdm_clk;
|
unsigned int pdm_clk;
|
||||||
|
|
||||||
|
unsigned int dai_fmt;
|
||||||
|
unsigned int tdm_delay;
|
||||||
};
|
};
|
||||||
|
|
||||||
/* Input mux controls */
|
/* Input mux controls */
|
||||||
|
@ -191,10 +194,29 @@ static int tas2552_hw_params(struct snd_pcm_substream *substream,
|
||||||
#define TAS2552_DAI_FMT_MASK (TAS2552_BCLKDIR | \
|
#define TAS2552_DAI_FMT_MASK (TAS2552_BCLKDIR | \
|
||||||
TAS2552_WCLKDIR | \
|
TAS2552_WCLKDIR | \
|
||||||
TAS2552_DATAFORMAT_MASK)
|
TAS2552_DATAFORMAT_MASK)
|
||||||
|
static int tas2552_prepare(struct snd_pcm_substream *substream,
|
||||||
|
struct snd_soc_dai *dai)
|
||||||
|
{
|
||||||
|
struct snd_soc_codec *codec = dai->codec;
|
||||||
|
struct tas2552_data *tas2552 = snd_soc_codec_get_drvdata(codec);
|
||||||
|
int delay = 0;
|
||||||
|
|
||||||
|
/* TDM slot selection only valid in DSP_A/_B mode */
|
||||||
|
if (tas2552->dai_fmt == SND_SOC_DAIFMT_DSP_A)
|
||||||
|
delay += (tas2552->tdm_delay + 1);
|
||||||
|
else if (tas2552->dai_fmt == SND_SOC_DAIFMT_DSP_B)
|
||||||
|
delay += tas2552->tdm_delay;
|
||||||
|
|
||||||
|
/* Configure data delay */
|
||||||
|
snd_soc_write(codec, TAS2552_SER_CTRL_2, delay);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int tas2552_set_dai_fmt(struct snd_soc_dai *dai, unsigned int fmt)
|
static int tas2552_set_dai_fmt(struct snd_soc_dai *dai, unsigned int fmt)
|
||||||
{
|
{
|
||||||
struct snd_soc_codec *codec = dai->codec;
|
struct snd_soc_codec *codec = dai->codec;
|
||||||
u8 delay = 0;
|
struct tas2552_data *tas2552 = dev_get_drvdata(codec->dev);
|
||||||
u8 serial_format;
|
u8 serial_format;
|
||||||
|
|
||||||
switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
|
switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
|
||||||
|
@ -220,7 +242,6 @@ static int tas2552_set_dai_fmt(struct snd_soc_dai *dai, unsigned int fmt)
|
||||||
case (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF):
|
case (SND_SOC_DAIFMT_I2S | SND_SOC_DAIFMT_NB_NF):
|
||||||
break;
|
break;
|
||||||
case (SND_SOC_DAIFMT_DSP_A | SND_SOC_DAIFMT_IB_NF):
|
case (SND_SOC_DAIFMT_DSP_A | SND_SOC_DAIFMT_IB_NF):
|
||||||
delay = 1;
|
|
||||||
case (SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_IB_NF):
|
case (SND_SOC_DAIFMT_DSP_B | SND_SOC_DAIFMT_IB_NF):
|
||||||
serial_format |= TAS2552_DATAFORMAT_DSP;
|
serial_format |= TAS2552_DATAFORMAT_DSP;
|
||||||
break;
|
break;
|
||||||
|
@ -234,11 +255,10 @@ static int tas2552_set_dai_fmt(struct snd_soc_dai *dai, unsigned int fmt)
|
||||||
dev_vdbg(codec->dev, "DAI Format is not found\n");
|
dev_vdbg(codec->dev, "DAI Format is not found\n");
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
tas2552->dai_fmt = fmt & SND_SOC_DAIFMT_FORMAT_MASK;
|
||||||
|
|
||||||
snd_soc_update_bits(codec, TAS2552_SER_CTRL_1, TAS2552_DAI_FMT_MASK,
|
snd_soc_update_bits(codec, TAS2552_SER_CTRL_1, TAS2552_DAI_FMT_MASK,
|
||||||
serial_format);
|
serial_format);
|
||||||
snd_soc_write(codec, TAS2552_SER_CTRL_2, delay);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -278,6 +298,35 @@ static int tas2552_set_dai_sysclk(struct snd_soc_dai *dai, int clk_id,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int tas2552_set_dai_tdm_slot(struct snd_soc_dai *dai,
|
||||||
|
unsigned int tx_mask, unsigned int rx_mask,
|
||||||
|
int slots, int slot_width)
|
||||||
|
{
|
||||||
|
struct snd_soc_codec *codec = dai->codec;
|
||||||
|
struct tas2552_data *tas2552 = snd_soc_codec_get_drvdata(codec);
|
||||||
|
unsigned int lsb;
|
||||||
|
|
||||||
|
if (unlikely(!tx_mask)) {
|
||||||
|
dev_err(codec->dev, "tx masks need to be non 0\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* TDM based on DSP mode requires slots to be adjacent */
|
||||||
|
lsb = __ffs(tx_mask);
|
||||||
|
if ((lsb + 1) != __fls(tx_mask)) {
|
||||||
|
dev_err(codec->dev, "Invalid mask, slots must be adjacent\n");
|
||||||
|
return -EINVAL;
|
||||||
|
}
|
||||||
|
|
||||||
|
tas2552->tdm_delay = lsb * slot_width;
|
||||||
|
|
||||||
|
/* DOUT in high-impedance on inactive bit clocks */
|
||||||
|
snd_soc_update_bits(codec, TAS2552_DOUT,
|
||||||
|
TAS2552_SDOUT_TRISTATE, TAS2552_SDOUT_TRISTATE);
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
static int tas2552_mute(struct snd_soc_dai *dai, int mute)
|
static int tas2552_mute(struct snd_soc_dai *dai, int mute)
|
||||||
{
|
{
|
||||||
u8 cfg1_reg = 0;
|
u8 cfg1_reg = 0;
|
||||||
|
@ -330,8 +379,10 @@ static const struct dev_pm_ops tas2552_pm = {
|
||||||
|
|
||||||
static struct snd_soc_dai_ops tas2552_speaker_dai_ops = {
|
static struct snd_soc_dai_ops tas2552_speaker_dai_ops = {
|
||||||
.hw_params = tas2552_hw_params,
|
.hw_params = tas2552_hw_params,
|
||||||
|
.prepare = tas2552_prepare,
|
||||||
.set_sysclk = tas2552_set_dai_sysclk,
|
.set_sysclk = tas2552_set_dai_sysclk,
|
||||||
.set_fmt = tas2552_set_dai_fmt,
|
.set_fmt = tas2552_set_dai_fmt,
|
||||||
|
.set_tdm_slot = tas2552_set_dai_tdm_slot,
|
||||||
.digital_mute = tas2552_mute,
|
.digital_mute = tas2552_mute,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -62,6 +62,9 @@
|
||||||
#define TAS2552_LIM_EN (1 << 2)
|
#define TAS2552_LIM_EN (1 << 2)
|
||||||
#define TAS2552_IVSENSE_EN (1 << 1)
|
#define TAS2552_IVSENSE_EN (1 << 1)
|
||||||
|
|
||||||
|
/* DOUT Register Masks */
|
||||||
|
#define TAS2552_SDOUT_TRISTATE (1 << 2)
|
||||||
|
|
||||||
/* Serial Interface Control Register Masks */
|
/* Serial Interface Control Register Masks */
|
||||||
#define TAS2552_DATAFORMAT_I2S (0x0 << 2)
|
#define TAS2552_DATAFORMAT_I2S (0x0 << 2)
|
||||||
#define TAS2552_DATAFORMAT_DSP (0x1 << 2)
|
#define TAS2552_DATAFORMAT_DSP (0x1 << 2)
|
||||||
|
|
Loading…
Reference in a new issue