On Mon, 7 Jan 2008, Lennart Poettering wrote:
On Mon, 07.01.08 12:07, Takashi Iwai (tiwai@suse.de) wrote:
In PulseAudio I want to schedule on my own when I need to write audio data into the device and when not. To achieve that I want to be notified via poll() whenever a period boundary is passed (i.e. when an IRQ happens), but only then. That's different from the usual mode where you are notified via poll() whether there is space in the playback buffer that needs to be filled up.
On OSS the mmap() mode enables a mode like I described above. After enabling mmap() the application can decide by itself what it considers full and what empty in the dma buffer, and use GETOPTR to query the playback position. poll() on the OSS fd will directly reflect the sound card IRQs and is not influenced if you ever wrote data to device or not.
I assume that I can enable a mode like that with one of the SW params. But quite frankly the docs for it are not enlighening at all.
Set the stop_threshould sw_params to the boundary size.
snd_pcm_sw_params_get_boundary(sw_params, &boundary); snd_pcm_sw_params_set_stop_threshold(pcm, sw_params, boundary);
then the driver behaves in the "freewheel" mode. The dmix plugin uses this technique.
That's not what I was looking for. This will only disable automatic stopping on buffer underrun. I am using that already in PA (however I pass -1 as stop threshold, which should work, too, shouldn't it?)
What I am really looking for is a way to disable that ALSA reports via poll() the buffer fill level, but instead only reports whether an interrupt happened.
Note that you can control fill level using snd_pcm_forward/rewind without any R/W calls (of course if supported in whole chain). ALSA supports only "controlled" I/O not dumb I/O as OSS driver for mmap.
If you look for timing source, use timer API - you may try alsa-lib/test/timer.c:
./timer class=3 card=0 device=0 subdevice=0
The background why I want this is this: As mentioned I am now scheduling audio in PA mostly based on system timers. To be able to do that I need to be able to translate timespans from the sound card clock to the system clock. Which requires me to get the sample time from the sound card from time to time and filter it through some code that estimates how the sound card clock and the system clock deviates. I'd prefer to do that only once or maybe twice everytime the playback buffer is fully played, and only shortly after an IRQ happened, under the assumption that this is the best time to get the most accurate timing information from the sound card.
It's not really necessary. You can use only one timing source (system timer) and use position timestamps to do corrections.
But your example does not explain, why you don't move r/w pointer in the ring buffer (use mmap_commit), thus why you don't fullfill the avail_min requirement for poll wakeup. It seems to me that you're trying to do some crazy things with the ring buffer which are not allowed.
Jaroslav
----- Jaroslav Kysela perex@perex.cz Linux Kernel Sound Maintainer ALSA Project, Red Hat, Inc.