typedef enum { SND_PCM_TYPE_HW, SND_PCM_TYPE_PLUG, SND_PCM_TYPE_MULTI } snd_pcm_type_t;
+#if 0
typedef struct {
snd_pcm_t *handle;
snd_timestamp_t tstamp;
#define SND_PCM_SYNCHRO_MODE_NORMAL SND_PCM_SYNC_MODE_NORMAL
#define SND_PCM_SYNCHRO_MODE_HARDWARE SND_PCM_SYNC_MODE_HARDWARE
#define SND_PCM_SYNCHRO_MODE_RELAXED SND_PCM_SYNC_MODE_RELAXED
+int snd_pcm_synchro(snd_pcm_synchro_cmd_t cmd,
+ unsigned int reqs_count, snd_pcm_synchro_request_t *reqs,
+ snd_pcm_synchro_mode_t mode);
+#endif
int snd_pcm_hw_open_subdevice(snd_pcm_t **handle, int card, int device, int subdevice, int stream, int mode);
int snd_pcm_status(snd_pcm_t *handle, snd_pcm_status_t *status);
int snd_pcm_prepare(snd_pcm_t *handle);
int snd_pcm_go(snd_pcm_t *handle);
-int snd_pcm_synchro(snd_pcm_synchro_cmd_t cmd,
- unsigned int reqs_count, snd_pcm_synchro_request_t *reqs,
- snd_pcm_synchro_mode_t mode);
int snd_pcm_drain(snd_pcm_t *handle);
int snd_pcm_flush(snd_pcm_t *handle);
int snd_pcm_pause(snd_pcm_t *handle, int enable);
ssize_t snd_pcm_readv(snd_pcm_t *handle, const struct iovec *vector, unsigned long count);
int snd_pcm_dump_setup(snd_pcm_t *handle, FILE *fp);
int snd_pcm_dump(snd_pcm_t *handle, FILE *fp);
+int snd_pcm_link(snd_pcm_t *handle1, snd_pcm_t *handle2);
+int snd_pcm_unlink(snd_pcm_t *handle);
int snd_pcm_channels_mask(snd_pcm_t *handle, bitset_t *client_vmask);
return handle->fast_ops->go(handle->fast_op_arg);
}
+#if 0
int snd_pcm_synchro(snd_pcm_synchro_cmd_t cmd,
unsigned int reqs_count, snd_pcm_synchro_request_t *reqs,
snd_pcm_synchro_mode_t mode)
}
return ret;
}
-
+#endif
int snd_pcm_drain(snd_pcm_t *handle)
{
return handle->fast_ops->readv(handle->fast_op_arg, 0, vector, count);
}
+int snd_pcm_link(snd_pcm_t *handle1, snd_pcm_t *handle2)
+{
+ int fd1, fd2;
+ switch (handle1->type) {
+ case SND_PCM_TYPE_HW:
+ case SND_PCM_TYPE_PLUG:
+ case SND_PCM_TYPE_MULTI:
+ fd1 = snd_pcm_file_descriptor(handle1);
+ break;
+ default:
+ errno = -ENOSYS;
+ return -1;
+ }
+ switch (handle2->type) {
+ case SND_PCM_TYPE_HW:
+ case SND_PCM_TYPE_PLUG:
+ case SND_PCM_TYPE_MULTI:
+ fd2 = snd_pcm_file_descriptor(handle2);
+ break;
+ default:
+ errno = -ENOSYS;
+ return -1;
+ }
+ if (ioctl(fd1, SND_PCM_IOCTL_LINK, fd2) < 0)
+ return -errno;
+ return 0;
+}
+
+int snd_pcm_unlink(snd_pcm_t *handle)
+{
+ int fd;
+ switch (handle->type) {
+ case SND_PCM_TYPE_HW:
+ case SND_PCM_TYPE_PLUG:
+ case SND_PCM_TYPE_MULTI:
+ fd = snd_pcm_file_descriptor(handle);
+ break;
+ default:
+ errno = -ENOSYS;
+ return -1;
+ }
+ if (ioctl(fd, SND_PCM_IOCTL_UNLINK) < 0)
+ return -errno;
+ return 0;
+}
+
int snd_pcm_file_descriptor(snd_pcm_t *handle)
{
assert(handle);
fprintf(fp, "xrun_mode: %s\n", assoc(setup->xrun_mode, xruns));
fprintf(fp, "time: %s\n", assoc(setup->time, onoff));
// ust_time
- // sync
fprintf(fp, "buffer_size: %d\n", setup->buffer_size);
fprintf(fp, "frag_size: %d\n", setup->frag_size);
fprintf(fp, "frags: %d\n", setup->frags);
{
snd_pcm_hw_t *hw = (snd_pcm_hw_t*) private;
int fd = hw->fd;
- if (ioctl(fd, SND_PCM_IOCTL_PAUSE, &enable) < 0)
+ if (ioctl(fd, SND_PCM_IOCTL_PAUSE, enable) < 0)
return -errno;
return 0;
}
err = snd_pcm_close(slave->handle);
if (err < 0)
ret = err;
- }
+ } else
+ snd_pcm_unlink(slave->handle);
if (slave->buf) {
free(slave->buf);
free(slave->areas);
static int snd_pcm_multi_prepare(void *private)
{
snd_pcm_multi_t *multi = (snd_pcm_multi_t*) private;
- unsigned int i;
- for (i = 0; i < multi->slaves_count; ++i) {
- snd_pcm_t *handle = multi->slaves[i].handle;
- int err = snd_pcm_prepare(handle);
- if (err < 0)
- return err;
- }
- return 0;
+ return snd_pcm_prepare(multi->slaves[0].handle);
}
static int snd_pcm_multi_go(void *private)
{
snd_pcm_multi_t *multi = (snd_pcm_multi_t*) private;
- unsigned int i;
- for (i = 0; i < multi->slaves_count; ++i) {
- snd_pcm_t *handle = multi->slaves[i].handle;
- int err = snd_pcm_go(handle);
- if (err < 0)
- return err;
- }
- return 0;
+ return snd_pcm_go(multi->slaves[0].handle);
}
static int snd_pcm_multi_drain(void *private)
{
snd_pcm_multi_t *multi = (snd_pcm_multi_t*) private;
- unsigned int i;
- for (i = 0; i < multi->slaves_count; ++i) {
- snd_pcm_t *handle = multi->slaves[i].handle;
- int err = snd_pcm_drain(handle);
- if (err < 0)
- return err;
- }
- return 0;
+ return snd_pcm_drain(multi->slaves[0].handle);
}
static int snd_pcm_multi_flush(void *private)
{
snd_pcm_multi_t *multi = (snd_pcm_multi_t*) private;
- unsigned int i;
- for (i = 0; i < multi->slaves_count; ++i) {
- snd_pcm_t *handle = multi->slaves[i].handle;
- int err = snd_pcm_flush(handle);
- if (err < 0)
- return err;
- }
- return 0;
+ return snd_pcm_flush(multi->slaves[0].handle);
}
static int snd_pcm_multi_pause(void *private, int enable)
{
snd_pcm_multi_t *multi = (snd_pcm_multi_t*) private;
- unsigned int i;
- for (i = 0; i < multi->slaves_count; ++i) {
- snd_pcm_t *handle = multi->slaves[i].handle;
- int err = snd_pcm_pause(handle, enable);
- if (err < 0)
- return err;
- }
- return 0;
+ return snd_pcm_pause(multi->slaves[0].handle, enable);
}
static int snd_pcm_multi_channel_setup(void *private, snd_pcm_channel_setup_t *setup)
slave->handle = slaves_handle[i];
slave->channels_total = slaves_channels_count[i];
slave->close_slave = close_slaves;
+ if (i != 0)
+ snd_pcm_link(slaves_handle[i-1], slaves_handle[i]);
}
for (i = 0; i < binds_count; ++i) {
snd_pcm_multi_bind_t *bind = &multi->binds[i];