ASoC: fsl-ssi: Support for SND_SOC_DAIFMT_CBM_CFS
Add SND_SOC_DAIFMT_CBM_CFS support for Freescale architecture. Successfully tested on i.MX 6Quad Wandboard and UDOO boards connected to the pcm1792a codec. In CBM_CFS mode, when using a sample size of 16 bits, we cannot use CCSR_SSI_SCR_I2S_MODE_MASTER since we get a frame sync every 16 bits. Signed-off-by: Michael Trimarchi <michael@amarulasolutions.com> Signed-off-by: Fabio Falzoi <fabio.falzoi84@gmail.com> Tested-by: Angelo Adamo <adamo.a60@gmail.com> Acked-by: Timur Tabi <timur@tabi.org> Signed-off-by: Mark Brown <broonie@linaro.org>
This commit is contained in:
@ -259,6 +259,11 @@ static bool fsl_ssi_is_i2s_master(struct fsl_ssi_private *ssi_private)
|
|||||||
SND_SOC_DAIFMT_CBS_CFS;
|
SND_SOC_DAIFMT_CBS_CFS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool fsl_ssi_is_i2s_cbm_cfs(struct fsl_ssi_private *ssi_private)
|
||||||
|
{
|
||||||
|
return (ssi_private->dai_fmt & SND_SOC_DAIFMT_MASTER_MASK) ==
|
||||||
|
SND_SOC_DAIFMT_CBM_CFS;
|
||||||
|
}
|
||||||
/**
|
/**
|
||||||
* fsl_ssi_isr: SSI interrupt handler
|
* fsl_ssi_isr: SSI interrupt handler
|
||||||
*
|
*
|
||||||
@ -705,6 +710,23 @@ static int fsl_ssi_hw_params(struct snd_pcm_substream *substream,
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!fsl_ssi_is_ac97(ssi_private)) {
|
||||||
|
u8 i2smode;
|
||||||
|
/*
|
||||||
|
* Switch to normal net mode in order to have a frame sync
|
||||||
|
* signal every 32 bits instead of 16 bits
|
||||||
|
*/
|
||||||
|
if (fsl_ssi_is_i2s_cbm_cfs(ssi_private) && sample_size == 16)
|
||||||
|
i2smode = CCSR_SSI_SCR_I2S_MODE_NORMAL |
|
||||||
|
CCSR_SSI_SCR_NET;
|
||||||
|
else
|
||||||
|
i2smode = ssi_private->i2s_mode;
|
||||||
|
|
||||||
|
regmap_update_bits(regs, CCSR_SSI_SCR,
|
||||||
|
CCSR_SSI_SCR_NET | CCSR_SSI_SCR_I2S_MODE_MASK,
|
||||||
|
channels == 1 ? 0 : i2smode);
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* FIXME: The documentation says that SxCCR[WL] should not be
|
* FIXME: The documentation says that SxCCR[WL] should not be
|
||||||
* modified while the SSI is enabled. The only time this can
|
* modified while the SSI is enabled. The only time this can
|
||||||
@ -724,11 +746,6 @@ static int fsl_ssi_hw_params(struct snd_pcm_substream *substream,
|
|||||||
regmap_update_bits(regs, CCSR_SSI_SRCCR, CCSR_SSI_SxCCR_WL_MASK,
|
regmap_update_bits(regs, CCSR_SSI_SRCCR, CCSR_SSI_SxCCR_WL_MASK,
|
||||||
wl);
|
wl);
|
||||||
|
|
||||||
if (!fsl_ssi_is_ac97(ssi_private))
|
|
||||||
regmap_update_bits(regs, CCSR_SSI_SCR,
|
|
||||||
CCSR_SSI_SCR_NET | CCSR_SSI_SCR_I2S_MODE_MASK,
|
|
||||||
channels == 1 ? 0 : ssi_private->i2s_mode);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -780,6 +797,7 @@ static int _fsl_ssi_set_dai_fmt(struct fsl_ssi_private *ssi_private,
|
|||||||
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
|
switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
|
||||||
case SND_SOC_DAIFMT_I2S:
|
case SND_SOC_DAIFMT_I2S:
|
||||||
switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
|
switch (fmt & SND_SOC_DAIFMT_MASTER_MASK) {
|
||||||
|
case SND_SOC_DAIFMT_CBM_CFS:
|
||||||
case SND_SOC_DAIFMT_CBS_CFS:
|
case SND_SOC_DAIFMT_CBS_CFS:
|
||||||
ssi_private->i2s_mode |= CCSR_SSI_SCR_I2S_MODE_MASTER;
|
ssi_private->i2s_mode |= CCSR_SSI_SCR_I2S_MODE_MASTER;
|
||||||
regmap_update_bits(regs, CCSR_SSI_STCCR,
|
regmap_update_bits(regs, CCSR_SSI_STCCR,
|
||||||
@ -853,6 +871,11 @@ static int _fsl_ssi_set_dai_fmt(struct fsl_ssi_private *ssi_private,
|
|||||||
case SND_SOC_DAIFMT_CBM_CFM:
|
case SND_SOC_DAIFMT_CBM_CFM:
|
||||||
scr &= ~CCSR_SSI_SCR_SYS_CLK_EN;
|
scr &= ~CCSR_SSI_SCR_SYS_CLK_EN;
|
||||||
break;
|
break;
|
||||||
|
case SND_SOC_DAIFMT_CBM_CFS:
|
||||||
|
strcr &= ~CCSR_SSI_STCR_TXDIR;
|
||||||
|
strcr |= CCSR_SSI_STCR_TFDIR;
|
||||||
|
scr &= ~CCSR_SSI_SCR_SYS_CLK_EN;
|
||||||
|
break;
|
||||||
default:
|
default:
|
||||||
return -EINVAL;
|
return -EINVAL;
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user