[alsa-devel] [PATCH] ALSA: oxygen: Fix S/PDIF muting
Laurence Darby
ldarby at tuffmail.com
Sun Jul 5 20:56:50 CEST 2015
On Tue, 30 Jun 2015 11:43:52 +0200 Clemens Ladisch wrote:
> Laurence Darby wrote:
> > On Sun, 28 Jun 2015 20:44:26 +0200 Clemens Ladisch wrote:
> >> Laurence Darby wrote:
> >>> The S/PDIF output was muted whenever audio wasn't playing which
> >>> resulted in a clicking noise from the DAC when resuming
> >>
> >> It indeed appears that there are too many oxygen_clear_bits32()
> >> calls. However, I don't trust the hardware; please confirm that
> >> the S/PDIF output, without an active stream, play zeros and not
> >> the last sample.
> >
> > Well, to test that I created a wav file of about a quarter of a sine
> > wave (at about -15dB), consisting of 5520 samples, so it matched the
> > alsa buffer size, otherwise aplay pads the buffer with silence.
> >
> > The chip does continue playing the last sample value, then the
> > speakers pop later when something else is played. Is this really
> > an issue though? I just found my intel hda chip behaves the same
> > way, and that chip's driver leaves it un-muted.
>
> The current problem is that the output gets disabled, then later, when
> the driver is about to start the new stream, it enables the output
> again (which outputs the last old sample), and then the new stream
> starts (usually with zero samples). Going from disabled(=zero) to
> old-sample to new-sample(=zero) is an extra pop.
>
> In any case, putting a DC offset on the speakers is a bad idea.
>
> > If this is still going to prevent fixing the popping noise for the
> > oxygen chip, what about writing a single 0x00 sample to it in the
> > driver instead of muting it?
>
> This is likely to work (but the zero sample has to go through DMA).
Ok, I eventually got this to work, with this in oxygen_spdif_hw_free():
memset (runtime->dma_area, 0, runtime->dma_bytes);
oxygen_trigger(substream,SNDRV_PCM_TRIGGER_START);
msleep (1);
oxygen_trigger(substream,SNDRV_PCM_TRIGGER_STOP);
instead of disabling OXYGEN_SPDIF_OUT_ENABLE. The 1ms sleep is
necessary otherwise it intermittently still leaves DC on the output.
If I create a proper patch for that (it should probably use
snd_pcm_format_set_silence() instead of memset) would that be
acceptable?
> At the moment, oxygen_hw_free() pokes at the OXYGEN_DMA_FLUSH
> register. Please check if there is an improvement if it also pokes at
> the OXYGEN_DMA_RESET register.
Unfortunately not. I did some archeology and found your CVS commit
from 2007: http://sourceforge.net/p/alsa/mailman/message/15439076/
which has this code:
+ if (chip->CMI8788IC_revision == CMI8788IC_Revision1)
+ DMARestRegister = PCI_DMA_Reset;
+ if (chip->CMI8788IC_revision == CMI8788IC_Revision2)
+ DMARestRegister = PCI_DMA_FLUSH;
...
+ /* Reset DMA Channel*/
+ reset = snd_cmipci_read_b(chip, DMARestRegister);
+ reset |= cmi_subs->DMA_chan_reset; /* set bit */
So I understand why you asked if the RESET works :)
Regards,
Laurence
More information about the Alsa-devel
mailing list