[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