Changed ALSA unit from bytes to frames. Splitted mmap control structs. Better midlevel interrupt handler

This commit is contained in:
Abramo Bagnara 2000-06-10 12:39:52 +00:00
parent 0ac6348583
commit 5cc8a510e9

View file

@ -72,6 +72,7 @@ static int direction = SND_PCM_OPEN_PLAYBACK;
static int stream = SND_PCM_STREAM_PLAYBACK; static int stream = SND_PCM_STREAM_PLAYBACK;
static int mmap_flag = 0; static int mmap_flag = 0;
static snd_pcm_mmap_control_t *mmap_control = NULL; static snd_pcm_mmap_control_t *mmap_control = NULL;
static snd_pcm_mmap_status_t *mmap_status = NULL;
static char *mmap_data = NULL; static char *mmap_data = NULL;
static int noplugin = 0; static int noplugin = 0;
static int nonblock = 0; static int nonblock = 0;
@ -82,6 +83,8 @@ static int buffer_size = -1;
static int frag_length = 250; static int frag_length = 250;
static int show_setup = 0; static int show_setup = 0;
static int buffer_pos = 0; static int buffer_pos = 0;
static size_t bits_per_sample, bits_per_frame;
static size_t buffer_bytes;
static int count; static int count;
static int vocmajor, vocminor; static int vocmajor, vocminor;
@ -624,26 +627,26 @@ static void set_params(void)
memset(&params, 0, sizeof(params)); memset(&params, 0, sizeof(params));
params.mode = mode; params.mode = mode;
params.stream = stream; params.stream = stream;
memcpy(&params.format, &format, sizeof(format)); params.format = format;
if (stream == SND_PCM_STREAM_PLAYBACK) { if (stream == SND_PCM_STREAM_PLAYBACK) {
params.start_mode = SND_PCM_START_FULL; params.start_mode = SND_PCM_START_FULL;
} else { } else {
params.start_mode = SND_PCM_START_DATA; params.start_mode = SND_PCM_START_DATA;
} }
params.xrun_mode = SND_PCM_XRUN_FLUSH; params.xrun_mode = SND_PCM_XRUN_FLUSH;
params.frag_size = snd_pcm_format_bytes_per_second(&format) / 1000.0 * frag_length; params.frag_size = format.rate * frag_length / 1000;
params.buffer_size = params.frag_size * 4; params.buffer_size = params.frag_size * 4;
params.bytes_min = 0; params.frames_min = format.rate / 16;
params.bytes_xrun_max = 0; params.frames_xrun_max = 0;
params.fill_mode = SND_PCM_FILL_SILENCE; params.fill_mode = SND_PCM_FILL_SILENCE;
params.bytes_fill_max = 1024; params.frames_fill_max = 1024;
params.bytes_xrun_max = 0; params.frames_xrun_max = 0;
if (snd_pcm_stream_params(pcm_handle, &params) < 0) { if (snd_pcm_stream_params(pcm_handle, &params) < 0) {
fprintf(stderr, "%s: unable to set stream params\n", command); fprintf(stderr, "%s: unable to set stream params\n", command);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
if (mmap_flag) { if (mmap_flag) {
if (snd_pcm_mmap(pcm_handle, stream, &mmap_control, (void **)&mmap_data)<0) { if (snd_pcm_mmap(pcm_handle, stream, &mmap_status, &mmap_control, (void **)&mmap_data)<0) {
fprintf(stderr, "%s: unable to mmap memory\n", command); fprintf(stderr, "%s: unable to mmap memory\n", command);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
@ -663,7 +666,10 @@ static void set_params(void)
snd_pcm_dump_setup(pcm_handle, stream, stderr); snd_pcm_dump_setup(pcm_handle, stream, stderr);
buffer_size = setup.frag_size; buffer_size = setup.frag_size;
audiobuf = (char *)realloc(audiobuf, buffer_size > 1024 ? buffer_size : 1024); bits_per_sample = snd_pcm_format_physical_width(setup.format.format);
bits_per_frame = bits_per_sample * setup.format.channels;
buffer_bytes = buffer_size * bits_per_frame / 8;
audiobuf = malloc(buffer_bytes);
if (audiobuf == NULL) { if (audiobuf == NULL) {
fprintf(stderr, "%s: not enough memory\n", command); fprintf(stderr, "%s: not enough memory\n", command);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
@ -684,7 +690,7 @@ void playback_underrun(void)
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
if (status.state == SND_PCM_STATE_XRUN) { if (status.state == SND_PCM_STATE_XRUN) {
printf("underrun at position %u!!!\n", status.byte_io); fprintf(stderr, "underrun at position %u!!!\n", status.frame_io);
if (snd_pcm_stream_prepare(pcm_handle, SND_PCM_STREAM_PLAYBACK)<0) { if (snd_pcm_stream_prepare(pcm_handle, SND_PCM_STREAM_PLAYBACK)<0) {
fprintf(stderr, "underrun: playback stream prepare error\n"); fprintf(stderr, "underrun: playback stream prepare error\n");
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
@ -710,7 +716,7 @@ void capture_overrun(void)
if (status.state == SND_PCM_STATE_RUNNING) if (status.state == SND_PCM_STATE_RUNNING)
return; /* everything is ok, but the driver is waiting for data */ return; /* everything is ok, but the driver is waiting for data */
if (status.state == SND_PCM_STATE_XRUN) { if (status.state == SND_PCM_STATE_XRUN) {
printf("overrun at position %u!!!\n", status.byte_io); fprintf(stderr, "overrun at position %u!!!\n", status.frame_io);
if (snd_pcm_stream_prepare(pcm_handle, SND_PCM_STREAM_CAPTURE)<0) { if (snd_pcm_stream_prepare(pcm_handle, SND_PCM_STREAM_CAPTURE)<0) {
fprintf(stderr, "overrun: capture stream prepare error\n"); fprintf(stderr, "overrun: capture stream prepare error\n");
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
@ -732,7 +738,7 @@ static ssize_t pcm_write(u_char *data, size_t count)
if (mode == SND_PCM_MODE_FRAGMENT && if (mode == SND_PCM_MODE_FRAGMENT &&
count != buffer_size) { count != buffer_size) {
snd_pcm_format_set_silence(format.format, data + count, buffer_size - count); snd_pcm_format_set_silence(format.format, data + count * bits_per_frame / 8, buffer_size * format.channels);
count = buffer_size; count = buffer_size;
} }
while (count > 0) { while (count > 0) {
@ -751,7 +757,7 @@ static ssize_t pcm_write(u_char *data, size_t count)
if (r > 0) { if (r > 0) {
result += r; result += r;
count -= r; count -= r;
data += r; data += r * bits_per_frame / 8;
} }
} }
return result; return result;
@ -765,19 +771,19 @@ static ssize_t pcm_writev(u_char **data, unsigned int channels, size_t count)
if (mode == SND_PCM_MODE_FRAGMENT && if (mode == SND_PCM_MODE_FRAGMENT &&
count != buffer_size) { count != buffer_size) {
unsigned int channel; unsigned int channel;
size_t offset = count / channels; size_t offset = count;
size_t remaining = (buffer_size - count) / channels; size_t remaining = buffer_size - count;
for (channel = 0; channel < channels; channel++) for (channel = 0; channel < channels; channel++)
snd_pcm_format_set_silence(format.format, data[channel] + offset, remaining); snd_pcm_format_set_silence(format.format, data[channel] + offset * bits_per_sample / 8, remaining);
count = buffer_size; count = buffer_size;
} }
while (count > 0) { while (count > 0) {
unsigned int channel; unsigned int channel;
struct iovec vec[channels]; struct iovec vec[channels];
size_t offset = result / channels; size_t offset = result;
size_t remaining = count / channels; size_t remaining = count;
for (channel = 0; channel < channels; channel++) { for (channel = 0; channel < channels; channel++) {
vec[channel].iov_base = data[channel] + offset; vec[channel].iov_base = data[channel] + offset * bits_per_sample / 8;
vec[channel].iov_len = remaining; vec[channel].iov_len = remaining;
} }
r = writev_func(pcm_handle, vec, channels); r = writev_func(pcm_handle, vec, channels);
@ -825,7 +831,7 @@ static ssize_t pcm_read(u_char *data, size_t count)
if (r > 0) { if (r > 0) {
result += r; result += r;
count -= r; count -= r;
data += r; data += r * bits_per_frame / 8;
} }
} }
return result; return result;
@ -839,10 +845,10 @@ static ssize_t pcm_readv(u_char **data, unsigned int channels, size_t count)
while (count > 0) { while (count > 0) {
unsigned int channel; unsigned int channel;
struct iovec vec[channels]; struct iovec vec[channels];
size_t offset = result / channels; size_t offset = result;
size_t remaining = count / channels; size_t remaining = count;
for (channel = 0; channel < channels; channel++) { for (channel = 0; channel < channels; channel++) {
vec[channel].iov_base = data[channel] + offset; vec[channel].iov_base = data[channel] + offset * bits_per_sample / 8;
vec[channel].iov_len = remaining; vec[channel].iov_len = remaining;
} }
r = readv_func(pcm_handle, vec, channels); r = readv_func(pcm_handle, vec, channels);
@ -876,13 +882,13 @@ static ssize_t voc_pcm_write(u_char *data, size_t count)
while (count > 0) { while (count > 0) {
size = count; size = count;
if (size > buffer_size - buffer_pos) if (size > buffer_bytes - buffer_pos)
size = buffer_size - buffer_pos; size = buffer_bytes - buffer_pos;
memcpy(audiobuf + buffer_pos, data, size); memcpy(audiobuf + buffer_pos, data, size);
data += size; data += size;
count -= size; count -= size;
buffer_pos += size; buffer_pos += size;
if (buffer_pos == buffer_size) { if (buffer_pos == buffer_bytes) {
if ((r = pcm_write(audiobuf, buffer_size)) != buffer_size) if ((r = pcm_write(audiobuf, buffer_size)) != buffer_size)
return r; return r;
buffer_pos = 0; buffer_pos = 0;
@ -896,12 +902,12 @@ static void voc_write_silence(unsigned x)
unsigned l; unsigned l;
char *buf; char *buf;
buf = (char *) malloc(buffer_size); buf = (char *) malloc(buffer_bytes);
if (buf == NULL) { if (buf == NULL) {
fprintf(stderr, "%s: can allocate buffer for silence\n", command); fprintf(stderr, "%s: can allocate buffer for silence\n", command);
return; /* not fatal error */ return; /* not fatal error */
} }
snd_pcm_format_set_silence(format.format, buf, buffer_size); snd_pcm_format_set_silence(format.format, buf, buffer_size * format.channels);
while (x > 0) { while (x > 0) {
l = x; l = x;
if (l > buffer_size) if (l > buffer_size)
@ -917,12 +923,15 @@ static void voc_write_silence(unsigned x)
static void voc_pcm_flush(void) static void voc_pcm_flush(void)
{ {
if (buffer_pos > 0) { if (buffer_pos > 0) {
size_t b;
if (mode == SND_PCM_MODE_FRAGMENT) { if (mode == SND_PCM_MODE_FRAGMENT) {
if (snd_pcm_format_set_silence(format.format, audiobuf + buffer_pos, buffer_size - buffer_pos) < 0) if (snd_pcm_format_set_silence(format.format, audiobuf + buffer_pos, buffer_bytes - buffer_pos * 8 / bits_per_sample) < 0)
fprintf(stderr, "voc_pcm_flush - silence error\n"); fprintf(stderr, "voc_pcm_flush - silence error\n");
buffer_pos = buffer_size; b = buffer_size;
} else {
b = buffer_pos * 8 / bits_per_frame;
} }
if (pcm_write(audiobuf, buffer_pos) != buffer_pos) if (pcm_write(audiobuf, b) != b)
fprintf(stderr, "voc_pcm_flush error\n"); fprintf(stderr, "voc_pcm_flush error\n");
} }
snd_pcm_stream_flush(pcm_handle, SND_PCM_STREAM_PLAYBACK); snd_pcm_stream_flush(pcm_handle, SND_PCM_STREAM_PLAYBACK);
@ -954,12 +963,12 @@ static void voc_play(int fd, int ofs, char *name)
fprintf(stderr, "Playing Creative Labs Channel file '%s'...\n", name); fprintf(stderr, "Playing Creative Labs Channel file '%s'...\n", name);
} }
/* first we waste the rest of header, ugly but we don't need seek */ /* first we waste the rest of header, ugly but we don't need seek */
while (ofs > buffer_size) { while (ofs > buffer_bytes) {
if (read(fd, buf, buffer_size) != buffer_size) { if (read(fd, buf, buffer_bytes) != buffer_bytes) {
fprintf(stderr, "%s: read error\n", command); fprintf(stderr, "%s: read error\n", command);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
ofs -= buffer_size; ofs -= buffer_bytes;
} }
if (ofs) { if (ofs) {
if (read(fd, buf, ofs) != ofs) { if (read(fd, buf, ofs) != ofs) {
@ -980,7 +989,7 @@ static void voc_play(int fd, int ofs, char *name)
if (in_buffer) if (in_buffer)
memcpy(buf, data, in_buffer); memcpy(buf, data, in_buffer);
data = buf; data = buf;
if ((l = read(fd, buf + in_buffer, buffer_size - in_buffer)) > 0) if ((l = read(fd, buf + in_buffer, buffer_bytes - in_buffer)) > 0)
in_buffer += l; in_buffer += l;
else if (!in_buffer) { else if (!in_buffer) {
/* the file is truncated, so simulate 'Terminator' /* the file is truncated, so simulate 'Terminator'
@ -1358,11 +1367,11 @@ void playback_go(int fd, size_t loaded, size_t count, int rtype, char *name)
header(rtype, name); header(rtype, name);
set_params(); set_params();
while (loaded > buffer_size && written < count) { while (loaded > buffer_bytes && written < count) {
if (pcm_write(audiobuf + written, buffer_size) <= 0) if (pcm_write(audiobuf + written, buffer_size) <= 0)
return; return;
written += buffer_size; written += buffer_bytes;
loaded -= buffer_size; loaded -= buffer_bytes;
} }
if (written > 0 && loaded > 0) if (written > 0 && loaded > 0)
memmove(audiobuf, audiobuf + written, loaded); memmove(audiobuf, audiobuf + written, loaded);
@ -1371,8 +1380,8 @@ void playback_go(int fd, size_t loaded, size_t count, int rtype, char *name)
while (written < count) { while (written < count) {
do { do {
c = count - written; c = count - written;
if (c > buffer_size) if (c > buffer_bytes)
c = buffer_size; c = buffer_bytes;
c -= l; c -= l;
if (c == 0) if (c == 0)
@ -1385,10 +1394,12 @@ void playback_go(int fd, size_t loaded, size_t count, int rtype, char *name)
if (r == 0) if (r == 0)
break; break;
l += r; l += r;
} while (mode != SND_PCM_MODE_FRAME && l < buffer_size); } while (mode != SND_PCM_MODE_FRAME && l < buffer_bytes);
l = l * 8 / bits_per_frame;
r = pcm_write(audiobuf, l); r = pcm_write(audiobuf, l);
if (r != l) if (r != l)
break; break;
r = r * bits_per_frame / 8;
written += r; written += r;
l = 0; l = 0;
} }
@ -1407,10 +1418,12 @@ void capture_go(int fd, size_t count, int rtype, char *name)
while (count > 0) { while (count > 0) {
c = count; c = count;
if (c > buffer_size) if (c > buffer_bytes)
c = buffer_size; c = buffer_bytes;
c = c * 8 / bits_per_frame;
if ((r = pcm_read(audiobuf, c)) != c) if ((r = pcm_read(audiobuf, c)) != c)
break; break;
r = r * bits_per_frame / 8;
if (write(fd, audiobuf, r) != r) { if (write(fd, audiobuf, r) != r) {
perror(name); perror(name);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
@ -1514,7 +1527,7 @@ void playbackv_go(int* fds, unsigned int channels, size_t loaded, size_t count,
header(rtype, names[0]); header(rtype, names[0]);
set_params(); set_params();
vsize = buffer_size / channels; vsize = buffer_bytes / channels;
// Not yet implemented // Not yet implemented
assert(loaded == 0); assert(loaded == 0);
@ -1543,9 +1556,11 @@ void playbackv_go(int* fds, unsigned int channels, size_t loaded, size_t count,
break; break;
c += r; c += r;
} while (mode != SND_PCM_MODE_FRAME && c < expected); } while (mode != SND_PCM_MODE_FRAME && c < expected);
r = pcm_writev(bufs, channels, c * channels); c = c * 8 / bits_per_sample;
if (r != c * channels) r = pcm_writev(bufs, channels, c);
if (r != c)
break; break;
r = r * bits_per_frame / 8;
count -= r; count -= r;
} }
snd_pcm_stream_flush(pcm_handle, SND_PCM_STREAM_PLAYBACK); snd_pcm_stream_flush(pcm_handle, SND_PCM_STREAM_PLAYBACK);
@ -1562,7 +1577,7 @@ void capturev_go(int* fds, unsigned int channels, size_t count, int rtype, char
header(rtype, names[0]); header(rtype, names[0]);
set_params(); set_params();
vsize = buffer_size / channels; vsize = buffer_bytes / channels;
for (channel = 0; channel < channels; ++channel) for (channel = 0; channel < channels; ++channel)
bufs[channel] = audiobuf + vsize * channel; bufs[channel] = audiobuf + vsize * channel;
@ -1570,17 +1585,19 @@ void capturev_go(int* fds, unsigned int channels, size_t count, int rtype, char
while (count > 0) { while (count > 0) {
size_t rv; size_t rv;
c = count; c = count;
if (c > buffer_size) if (c > buffer_bytes)
c = buffer_size; c = buffer_bytes;
c = c * 8 / bits_per_frame;
if ((r = pcm_readv(bufs, channels, c)) != c) if ((r = pcm_readv(bufs, channels, c)) != c)
break; break;
rv = r / channels; rv = r * bits_per_sample / 8;
for (channel = 0; channel < channels; ++channel) { for (channel = 0; channel < channels; ++channel) {
if (write(fds[channel], bufs[channel], rv) != rv) { if (write(fds[channel], bufs[channel], rv) != rv) {
perror(names[channel]); perror(names[channel]);
exit(EXIT_FAILURE); exit(EXIT_FAILURE);
} }
} }
r = r * bits_per_frame / 8;
count -= r; count -= r;
} }
} }