[alsa-devel] [PATCH 4/4] pxa-ssp: switch from network mode to psp

pHilipp Zabel philipp.zabel at gmail.com
Wed Mar 4 21:56:01 CET 2009


On Wed, Mar 4, 2009 at 9:17 PM, Daniel Mack <daniel at caiaq.de> wrote:
> This patch uses the SSP's PSP functionality to provide I2S timings. The
> particular problem is that even though the datasheets state it should be
> possbible, there is no mode which uses the network feature with its
> associated time slots in a sane way to do what we need.
>
> Hence, in order to have full 64 bit I2S on the wire, we need to fiddle
> around with the SSP and the timing paramters a lot. There are some
> constants left in the code which can't be replaced by names because the
> true meaning of their registers remains nebulous.
>
> Signed-off-by: Daniel Mack <daniel at caiaq.de>
> Signed-off-by: Tim Ruetz <tim at caiaq.de>
> Cc: Mark Brown <broonie at opensource.wolfsonmicro.com>
> Cc: Liam Girdwood <lrg at kernel.org>
> Cc: Philipp Zabel <philipp.zabel at gmail.com>
> ---
>  sound/soc/pxa/pxa-ssp.c |   39 +++++++++++++++++++++------------------
>  1 files changed, 21 insertions(+), 18 deletions(-)
>
> diff --git a/sound/soc/pxa/pxa-ssp.c b/sound/soc/pxa/pxa-ssp.c
> index 45fb600..97f11d6 100644
> --- a/sound/soc/pxa/pxa-ssp.c
> +++ b/sound/soc/pxa/pxa-ssp.c
> @@ -319,6 +319,7 @@ static int pxa_ssp_set_dai_sysclk(struct snd_soc_dai *cpu_dai,
>                break;
>        case PXA_SSP_CLK_EXT:
>                priv->sysclk = freq;
> +               ssp_set_scr(&priv->dev, 4);

Shouldn't this be somehow set by set_dai_clkdiv instead?

>                sscr0 |= SSCR0_ECS;
>                break;
>        case PXA_SSP_CLK_NET:
> @@ -551,17 +552,13 @@ static int pxa_ssp_set_dai_fmt(struct snd_soc_dai *cpu_dai,
>
>        switch (fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
>        case SND_SOC_DAIFMT_I2S:
> -               sscr0 |= SSCR0_MOD | SSCR0_PSP;
> +               sscr0 |= SSCR0_PSP;
>                sscr1 |= SSCR1_RWOT | SSCR1_TRAIL;
>
>                switch (fmt & SND_SOC_DAIFMT_INV_MASK) {
>                case SND_SOC_DAIFMT_NB_NF:
> -                       sspsp |= SSPSP_FSRT;
>                        break;
>                case SND_SOC_DAIFMT_NB_IF:
> -                       sspsp |= SSPSP_SFRMP | SSPSP_FSRT;
> -                       break;
> -               case SND_SOC_DAIFMT_IB_IF:
>                        sspsp |= SSPSP_SFRMP;
>                        break;

Removal of SSPSP_FSRT from NB/IB selection seems to be correct from the docs.
Can you check if IB could be properly handled by setting SCMODE(1)?

>                default:
> @@ -652,33 +649,39 @@ static int pxa_ssp_hw_params(struct snd_pcm_substream *substream,
>                break;
>        case SNDRV_PCM_FORMAT_S24_LE:
>                sscr0 |= (SSCR0_EDSS | SSCR0_DataSize(8));
> -               /* we must be in network mode (2 slots) for 24 bit stereo */

This is still dubious ...
S24_LE is 24-bit sound LSB-aligned in 32-bit frames, so DataSize
should be 32 here.

>                break;
>        case SNDRV_PCM_FORMAT_S32_LE:
>                sscr0 |= (SSCR0_EDSS | SSCR0_DataSize(16));
> -               /* we must be in network mode (2 slots) for 32 bit stereo */

How is it possible to send 64bit in one frame otherwise?

>                break;
>        }
>        ssp_write_reg(ssp, SSCR0, sscr0);
>
>        switch (priv->dai_fmt & SND_SOC_DAIFMT_FORMAT_MASK) {
>        case SND_SOC_DAIFMT_I2S:
> -               /* Cleared when the DAI format is set */
> -               sspsp = ssp_read_reg(ssp, SSPSP) | SSPSP_SFRMWDTH(width);
> +               sspsp = ssp_read_reg(ssp, SSPSP);
> +
> +               switch (priv->dai_fmt & SND_SOC_DAIFMT_FRAME_FORMAT_MASK) {
> +               case SND_SOC_DAIFMT_FF_I2S_32:
> +                       /* These values are all found out by trying and
> +                        * failing a lot. PXA's SSP is all black magic and
> +                        * does not work like described in any datasheet.
> +                        */
> +                       sspsp |= SSPSP_SFRMWDTH(32);
> +                       sspsp |= SSPSP_SFRMDLY(32 * 2);
> +                       sspsp |= SSPSP_EDMYSTOP(3);
> +                       sspsp |= SSPSP_DMYSTOP(3);
> +                       sspsp |= SSPSP_DMYSTRT(1);

Wha?! Amazing. And this really works?
How the hell can this result in 16 bits of data followed by 16 bits of
zeroes, twice :)

> +                       break;
> +               default:
> +                       /* Cleared when the DAI format is set */
> +                       sspsp |= SSPSP_SFRMWDTH(width);

Not good for DSP_A/B.

> +                       break;
> +               }
>                ssp_write_reg(ssp, SSPSP, sspsp);
> -               break;
>        default:
>                break;
>        }
>
> -       /* We always use a network mode so we always require TDM slots
> -        * - complain loudly and fail if they've not been set up yet.
> -        */
> -       if (!(ssp_read_reg(ssp, SSTSA) & 0xf)) {
> -               dev_err(&ssp->pdev->dev, "No TDM timeslot configured\n");
> -               return -EINVAL;
> -       }
> -
>        dump_registers(ssp);
>
>        return 0;
> --
> 1.6.1.3
>
>

regards
Philipp


More information about the Alsa-devel mailing list