ASoC: kirkwood-i2s: fix mute handling
authorRussell King <rmk+kernel@arm.linux.org.uk>
Thu, 26 Jun 2014 14:23:05 +0000 (15:23 +0100)
committerMark Brown <broonie@linaro.org>
Sat, 28 Jun 2014 12:18:28 +0000 (13:18 +0100)
The spec requires that the mute bits must be set while the channel
is disabled.  Ensure that this is the case by providing a helper
which ensures that the appropriate mute bit is set while the enable
bit is clear.

Signed-off-by: Russell King <rmk+kernel@arm.linux.org.uk>
Tested-by: Andrew Lunn <andrew@lunn.ch>
Signed-off-by: Mark Brown <broonie@linaro.org>
sound/soc/kirkwood/kirkwood-i2s.c

index ef1a164d870387ba9fa0c4f7458ed0e99357a3f3..b601ad680d7b6bf9b13d46715b5f9524d4147bb7 100644 (file)
@@ -222,6 +222,15 @@ static int kirkwood_i2s_hw_params(struct snd_pcm_substream *substream,
        return 0;
 }
 
+static unsigned kirkwood_i2s_play_mute(unsigned ctl)
+{
+       if (!(ctl & KIRKWOOD_PLAYCTL_I2S_EN))
+               ctl |= KIRKWOOD_PLAYCTL_I2S_MUTE;
+       if (!(ctl & KIRKWOOD_PLAYCTL_SPDIF_EN))
+               ctl |= KIRKWOOD_PLAYCTL_SPDIF_MUTE;
+       return ctl;
+}
+
 static int kirkwood_i2s_play_trigger(struct snd_pcm_substream *substream,
                                int cmd, struct snd_soc_dai *dai)
 {
@@ -257,7 +266,7 @@ static int kirkwood_i2s_play_trigger(struct snd_pcm_substream *substream,
                        ctl &= ~KIRKWOOD_PLAYCTL_SPDIF_EN;      /* i2s */
                else
                        ctl &= ~KIRKWOOD_PLAYCTL_I2S_EN;        /* spdif */
-
+               ctl = kirkwood_i2s_play_mute(ctl);
                value = ctl & ~KIRKWOOD_PLAYCTL_ENABLE_MASK;
                writel(value, priv->io + KIRKWOOD_PLAYCTL);
 
@@ -296,6 +305,7 @@ static int kirkwood_i2s_play_trigger(struct snd_pcm_substream *substream,
        case SNDRV_PCM_TRIGGER_PAUSE_RELEASE:
                ctl &= ~(KIRKWOOD_PLAYCTL_PAUSE | KIRKWOOD_PLAYCTL_I2S_MUTE |
                                KIRKWOOD_PLAYCTL_SPDIF_MUTE);
+               ctl = kirkwood_i2s_play_mute(ctl);
                writel(ctl, priv->io + KIRKWOOD_PLAYCTL);
                break;