[alsa-devel] pcm_hw bug when calling snd_pcm_sw_params twice
Jamey Sharp
jamey at minilop.net
Tue Jan 8 00:26:20 CET 2019
I would have reported this to your bug tracker, but I can't find it; the
links on alsa-project.org are dead.
I've attached a small test program that demonstrates that calling
snd_pcm_sw_params twice changes the value of the period_event flag in
the sw_params struct, at least on pcm_hw devices.
In src/pcm/pcm_hw.c, in snd_pcm_hw_sw_params, there's an early
`sw_set_period_event(params, 0)` call. Its effect is not undone if the
function returns early. This is easiest to trigger by calling it with
unchanged parameters, although there are other early-exit paths with the
same problem.
I don't actually care about this bug, but since I noticed it while
reading the alsa-lib source code trying to figure out what a period
event is for, I thought I'd go ahead and report it.
Jamey
-------------- next part --------------
#define _GNU_SOURCE
#include <assert.h>
#include <stdio.h>
#include <stdlib.h>
#include <alsa/asoundlib.h>
#define check(v) if(v < 0) { perror(#v); exit(1); }
int main(void)
{
snd_pcm_t *handle;
snd_pcm_hw_params_t *hw_params;
snd_pcm_sw_params_t *sw_params;
int val = 0;
char *name = getenv("DEVICE");
snd_pcm_hw_params_alloca(&hw_params);
snd_pcm_sw_params_alloca(&sw_params);
if(!name)
name = "hw:0";
printf("opening %s\n", name);
check(snd_pcm_open(&handle, name, SND_PCM_STREAM_PLAYBACK, 0));
check(snd_pcm_hw_params_any(handle, hw_params));
check(snd_pcm_hw_params(handle, hw_params));
check(snd_pcm_sw_params_current(handle, sw_params));
// Enable the period event
check(snd_pcm_sw_params_set_period_event(handle, sw_params, 1));
check(snd_pcm_sw_params_get_period_event(sw_params, &val));
assert(val == 1);
// After applying sw params once, the params struct still says the
// period event is enabled.
check(snd_pcm_sw_params(handle, sw_params));
check(snd_pcm_sw_params_get_period_event(sw_params, &val));
assert(val == 1);
// Applying the same sw params a second time should still leave the
// period event enabled, but this assert fails.
check(snd_pcm_sw_params(handle, sw_params));
check(snd_pcm_sw_params_get_period_event(sw_params, &val));
assert(val == 1);
return 0;
}
More information about the Alsa-devel
mailing list