2015-07-01 08:02:44 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2015 Andrew Kelley
|
|
|
|
*
|
|
|
|
* This file is part of libsoundio, which is MIT licensed.
|
|
|
|
* See http://opensource.org/licenses/MIT
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "dummy.hpp"
|
|
|
|
#include "soundio.hpp"
|
2015-07-01 09:37:51 +00:00
|
|
|
#include "os.hpp"
|
2015-07-02 09:45:02 +00:00
|
|
|
#include "atomics.hpp"
|
2015-07-06 08:05:22 +00:00
|
|
|
#include "ring_buffer.hpp"
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <string.h>
|
2015-07-20 20:16:52 +00:00
|
|
|
#include <math.h>
|
2015-07-01 09:37:51 +00:00
|
|
|
|
2015-07-10 09:42:29 +00:00
|
|
|
struct SoundIoOutStreamDummy {
|
2015-07-01 09:37:51 +00:00
|
|
|
struct SoundIoOsThread *thread;
|
|
|
|
struct SoundIoOsCond *cond;
|
|
|
|
atomic_flag abort_flag;
|
|
|
|
int buffer_size;
|
2015-07-06 08:05:22 +00:00
|
|
|
struct SoundIoRingBuffer ring_buffer;
|
2015-07-16 03:57:00 +00:00
|
|
|
SoundIoChannelArea areas[SOUNDIO_MAX_CHANNELS];
|
2015-07-01 09:37:51 +00:00
|
|
|
};
|
|
|
|
|
2015-07-10 09:42:29 +00:00
|
|
|
struct SoundIoInStreamDummy {
|
2015-07-01 09:37:51 +00:00
|
|
|
// TODO
|
|
|
|
};
|
|
|
|
|
|
|
|
struct SoundIoDummy {
|
|
|
|
SoundIoOsMutex *mutex;
|
|
|
|
SoundIoOsCond *cond;
|
|
|
|
bool devices_emitted;
|
|
|
|
};
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
static void playback_thread_run(void *arg) {
|
2015-07-13 16:17:20 +00:00
|
|
|
SoundIoOutStreamPrivate *os = (SoundIoOutStreamPrivate *)arg;
|
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data;
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
long frames_consumed = 0;
|
2015-07-21 03:13:35 +00:00
|
|
|
double start_time = soundio_os_get_time();
|
2015-07-20 20:16:52 +00:00
|
|
|
while (osd->abort_flag.test_and_set()) {
|
2015-07-01 08:02:44 +00:00
|
|
|
double now = soundio_os_get_time();
|
|
|
|
double total_time = now - start_time;
|
2015-07-21 03:13:35 +00:00
|
|
|
long total_frames = total_time * outstream->sample_rate;
|
2015-07-01 08:02:44 +00:00
|
|
|
int frames_to_kill = total_frames - frames_consumed;
|
2015-07-10 09:42:29 +00:00
|
|
|
int fill_count = soundio_ring_buffer_fill_count(&osd->ring_buffer);
|
2015-07-13 16:17:20 +00:00
|
|
|
int frames_in_buffer = fill_count / outstream->bytes_per_frame;
|
2015-07-01 08:02:44 +00:00
|
|
|
int read_count = min(frames_to_kill, frames_in_buffer);
|
|
|
|
int frames_left = frames_to_kill - read_count;
|
2015-07-13 16:17:20 +00:00
|
|
|
int byte_count = read_count * outstream->bytes_per_frame;
|
2015-07-10 09:42:29 +00:00
|
|
|
soundio_ring_buffer_advance_read_ptr(&osd->ring_buffer, byte_count);
|
2015-07-01 08:02:44 +00:00
|
|
|
frames_consumed += read_count;
|
|
|
|
|
|
|
|
if (frames_left > 0) {
|
2015-07-16 03:57:00 +00:00
|
|
|
outstream->error_callback(outstream, SoundIoErrorUnderflow);
|
2015-07-21 03:13:35 +00:00
|
|
|
// simulate filling with silence
|
|
|
|
int free_count = soundio_ring_buffer_free_count(&osd->ring_buffer);
|
|
|
|
soundio_ring_buffer_advance_write_ptr(&osd->ring_buffer, free_count);
|
2015-07-01 08:02:44 +00:00
|
|
|
} else if (read_count > 0) {
|
2015-07-13 16:17:20 +00:00
|
|
|
outstream->write_callback(outstream, read_count);
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
2015-07-20 20:16:52 +00:00
|
|
|
now = soundio_os_get_time();
|
|
|
|
double time_passed = now - start_time;
|
|
|
|
double next_period = start_time +
|
|
|
|
ceil(time_passed / outstream->period_duration) * outstream->period_duration;
|
|
|
|
double relative_time = next_period - now;
|
|
|
|
soundio_os_cond_timed_wait(osd->cond, nullptr, relative_time);
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
static void recording_thread_run(void *arg) {
|
2015-07-13 16:17:20 +00:00
|
|
|
SoundIoInStream *instream = (SoundIoInStream *)arg;
|
|
|
|
SoundIoDevice *device = instream->device;
|
2015-07-01 08:02:44 +00:00
|
|
|
SoundIo *soundio = device->soundio;
|
|
|
|
// TODO
|
|
|
|
}
|
|
|
|
*/
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void destroy_dummy(SoundIoPrivate *si) {
|
|
|
|
SoundIoDummy *sid = (SoundIoDummy *)si->backend_data;
|
2015-07-01 09:37:51 +00:00
|
|
|
if (!sid)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (sid->cond)
|
|
|
|
soundio_os_cond_destroy(sid->cond);
|
|
|
|
|
|
|
|
if (sid->mutex)
|
|
|
|
soundio_os_mutex_destroy(sid->mutex);
|
|
|
|
|
2015-07-01 09:53:53 +00:00
|
|
|
destroy(sid);
|
2015-07-13 16:17:20 +00:00
|
|
|
si->backend_data = nullptr;
|
2015-07-01 09:37:51 +00:00
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void flush_events(SoundIoPrivate *si) {
|
|
|
|
SoundIo *soundio = &si->pub;
|
|
|
|
SoundIoDummy *sid = (SoundIoDummy *)si->backend_data;
|
2015-07-01 09:37:51 +00:00
|
|
|
if (sid->devices_emitted)
|
|
|
|
return;
|
|
|
|
sid->devices_emitted = true;
|
|
|
|
soundio->on_devices_change(soundio);
|
2015-07-01 08:20:26 +00:00
|
|
|
}
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void wait_events(SoundIoPrivate *si) {
|
|
|
|
SoundIoDummy *sid = (SoundIoDummy *)si->backend_data;
|
|
|
|
flush_events(si);
|
2015-07-02 09:45:02 +00:00
|
|
|
soundio_os_cond_wait(sid->cond, nullptr);
|
2015-07-01 09:37:51 +00:00
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void wakeup(SoundIoPrivate *si) {
|
|
|
|
SoundIoDummy *sid = (SoundIoDummy *)si->backend_data;
|
2015-07-02 09:45:02 +00:00
|
|
|
soundio_os_cond_signal(sid->cond, nullptr);
|
2015-07-01 09:37:51 +00:00
|
|
|
}
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void outstream_destroy_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
|
|
|
SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data;
|
2015-07-10 09:42:29 +00:00
|
|
|
if (!osd)
|
2015-07-04 09:57:06 +00:00
|
|
|
return;
|
|
|
|
|
2015-07-10 09:42:29 +00:00
|
|
|
if (osd->thread) {
|
2015-07-16 20:37:41 +00:00
|
|
|
osd->abort_flag.clear();
|
|
|
|
soundio_os_cond_signal(osd->cond, nullptr);
|
|
|
|
soundio_os_thread_destroy(osd->thread);
|
|
|
|
osd->thread = nullptr;
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
2015-07-10 09:42:29 +00:00
|
|
|
soundio_os_cond_destroy(osd->cond);
|
|
|
|
osd->cond = nullptr;
|
2015-07-04 09:57:06 +00:00
|
|
|
|
2015-07-10 09:42:29 +00:00
|
|
|
soundio_ring_buffer_deinit(&osd->ring_buffer);
|
2015-07-04 09:57:06 +00:00
|
|
|
|
2015-07-10 09:42:29 +00:00
|
|
|
destroy(osd);
|
2015-07-13 16:17:20 +00:00
|
|
|
os->backend_data = nullptr;
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-14 04:30:37 +00:00
|
|
|
static int outstream_open_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
2015-07-13 16:17:20 +00:00
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
2015-07-10 09:42:29 +00:00
|
|
|
SoundIoOutStreamDummy *osd = create<SoundIoOutStreamDummy>();
|
|
|
|
if (!osd) {
|
2015-07-13 16:17:20 +00:00
|
|
|
outstream_destroy_dummy(si, os);
|
2015-07-04 09:57:06 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-07-13 16:17:20 +00:00
|
|
|
os->backend_data = osd;
|
2015-07-04 09:57:06 +00:00
|
|
|
|
2015-07-20 07:35:46 +00:00
|
|
|
osd->buffer_size = outstream->bytes_per_frame * outstream->sample_rate * outstream->buffer_duration;
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-10 09:42:29 +00:00
|
|
|
soundio_ring_buffer_init(&osd->ring_buffer, osd->buffer_size);
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-10 09:42:29 +00:00
|
|
|
osd->cond = soundio_os_cond_create();
|
|
|
|
if (!osd->cond) {
|
2015-07-13 16:17:20 +00:00
|
|
|
outstream_destroy_dummy(si, os);
|
2015-07-01 08:02:44 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-07-20 20:16:52 +00:00
|
|
|
static int outstream_pause_dummy(struct SoundIoPrivate *si, struct SoundIoOutStreamPrivate *os, bool pause) {
|
|
|
|
SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data;
|
|
|
|
if (pause) {
|
|
|
|
if (osd->thread) {
|
|
|
|
osd->abort_flag.clear();
|
|
|
|
soundio_os_cond_signal(osd->cond, nullptr);
|
|
|
|
soundio_os_thread_destroy(osd->thread);
|
|
|
|
osd->thread = nullptr;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
if (!osd->thread) {
|
|
|
|
osd->abort_flag.test_and_set();
|
|
|
|
int err;
|
|
|
|
if ((err = soundio_os_thread_create(playback_thread_run, os, true, &osd->thread))) {
|
|
|
|
return err;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
static int outstream_start_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
2015-07-13 16:17:20 +00:00
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data;
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
soundio_outstream_fill_with_silence(outstream);
|
2015-07-10 09:42:29 +00:00
|
|
|
assert(soundio_ring_buffer_fill_count(&osd->ring_buffer) == osd->buffer_size);
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-20 20:16:52 +00:00
|
|
|
return outstream_pause_dummy(si, os, false);
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-20 20:16:52 +00:00
|
|
|
static int outstream_free_count_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
2015-07-13 16:17:20 +00:00
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data;
|
2015-07-10 09:42:29 +00:00
|
|
|
int fill_count = soundio_ring_buffer_fill_count(&osd->ring_buffer);
|
|
|
|
int bytes_free_count = osd->buffer_size - fill_count;
|
2015-07-13 16:17:20 +00:00
|
|
|
return bytes_free_count / outstream->bytes_per_frame;
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-16 03:57:00 +00:00
|
|
|
static int outstream_begin_write_dummy(SoundIoPrivate *si,
|
|
|
|
SoundIoOutStreamPrivate *os, SoundIoChannelArea **out_areas, int *frame_count)
|
2015-07-01 08:02:44 +00:00
|
|
|
{
|
2015-07-16 03:57:00 +00:00
|
|
|
*out_areas = nullptr;
|
2015-07-13 16:17:20 +00:00
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data;
|
2015-07-01 08:02:44 +00:00
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
int byte_count = *frame_count * outstream->bytes_per_frame;
|
2015-07-10 09:42:29 +00:00
|
|
|
assert(byte_count <= osd->buffer_size);
|
2015-07-16 03:57:00 +00:00
|
|
|
|
2015-07-20 07:35:46 +00:00
|
|
|
int free_byte_count = soundio_ring_buffer_free_count(&osd->ring_buffer);
|
|
|
|
int free_frame_count = free_byte_count / outstream->bytes_per_frame;
|
|
|
|
*frame_count = min(*frame_count, free_frame_count);
|
|
|
|
|
|
|
|
if (free_frame_count) {
|
|
|
|
char *write_ptr = soundio_ring_buffer_write_ptr(&osd->ring_buffer);
|
|
|
|
for (int ch = 0; ch < outstream->layout.channel_count; ch += 1) {
|
|
|
|
osd->areas[ch].ptr = write_ptr + outstream->bytes_per_sample * ch;
|
|
|
|
osd->areas[ch].step = outstream->bytes_per_frame;
|
|
|
|
}
|
2015-07-16 03:57:00 +00:00
|
|
|
|
2015-07-20 07:35:46 +00:00
|
|
|
*out_areas = osd->areas;
|
|
|
|
}
|
2015-07-16 03:57:00 +00:00
|
|
|
return 0;
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-16 03:57:00 +00:00
|
|
|
static int outstream_write_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os, int frame_count) {
|
2015-07-13 16:17:20 +00:00
|
|
|
SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data;
|
|
|
|
SoundIoOutStream *outstream = &os->pub;
|
|
|
|
int byte_count = frame_count * outstream->bytes_per_frame;
|
2015-07-10 09:42:29 +00:00
|
|
|
soundio_ring_buffer_advance_write_ptr(&osd->ring_buffer, byte_count);
|
2015-07-16 03:57:00 +00:00
|
|
|
return 0;
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void outstream_clear_buffer_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
|
|
|
SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data;
|
2015-07-10 09:42:29 +00:00
|
|
|
soundio_ring_buffer_clear(&osd->ring_buffer);
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-14 04:30:37 +00:00
|
|
|
static int instream_open_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) {
|
2015-07-07 09:55:32 +00:00
|
|
|
soundio_panic("TODO");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void instream_destroy_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) {
|
2015-07-07 09:55:32 +00:00
|
|
|
soundio_panic("TODO");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static int instream_start_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) {
|
2015-07-07 09:55:32 +00:00
|
|
|
soundio_panic("TODO");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void instream_peek_dummy(SoundIoPrivate *si,
|
|
|
|
SoundIoInStreamPrivate *is, const char **data, int *frame_count)
|
2015-07-01 08:02:44 +00:00
|
|
|
{
|
2015-07-07 09:55:32 +00:00
|
|
|
soundio_panic("TODO");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void instream_drop_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) {
|
2015-07-07 09:55:32 +00:00
|
|
|
soundio_panic("TODO");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
static void instream_clear_buffer_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) {
|
2015-07-07 09:55:32 +00:00
|
|
|
soundio_panic("TODO");
|
2015-07-01 08:02:44 +00:00
|
|
|
}
|
|
|
|
|
2015-07-16 20:37:41 +00:00
|
|
|
static int instream_pause_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is, bool pause) {
|
|
|
|
soundio_panic("TODO");
|
|
|
|
}
|
|
|
|
|
2015-07-10 09:21:47 +00:00
|
|
|
static int set_all_device_formats(SoundIoDevice *device) {
|
|
|
|
device->format_count = 18;
|
|
|
|
device->formats = allocate<SoundIoFormat>(device->format_count);
|
|
|
|
if (!device->formats)
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
|
2015-07-14 04:30:37 +00:00
|
|
|
device->formats[0] = SoundIoFormatFloat32NE;
|
|
|
|
device->formats[1] = SoundIoFormatFloat32FE;
|
|
|
|
device->formats[2] = SoundIoFormatS32NE;
|
|
|
|
device->formats[3] = SoundIoFormatS32FE;
|
|
|
|
device->formats[4] = SoundIoFormatU32NE;
|
|
|
|
device->formats[5] = SoundIoFormatU32FE;
|
|
|
|
device->formats[6] = SoundIoFormatS24NE;
|
|
|
|
device->formats[7] = SoundIoFormatS24FE;
|
|
|
|
device->formats[8] = SoundIoFormatU24NE;
|
|
|
|
device->formats[9] = SoundIoFormatU24FE;
|
|
|
|
device->formats[10] = SoundIoFormatFloat64NE;
|
|
|
|
device->formats[11] = SoundIoFormatFloat64FE;
|
|
|
|
device->formats[12] = SoundIoFormatS16NE;
|
|
|
|
device->formats[13] = SoundIoFormatS16FE;
|
|
|
|
device->formats[14] = SoundIoFormatU16NE;
|
|
|
|
device->formats[15] = SoundIoFormatU16FE;
|
|
|
|
device->formats[16] = SoundIoFormatS8;
|
|
|
|
device->formats[17] = SoundIoFormatU8;
|
2015-07-10 09:21:47 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
int soundio_dummy_init(SoundIoPrivate *si) {
|
|
|
|
SoundIo *soundio = &si->pub;
|
|
|
|
assert(!si->backend_data);
|
2015-07-01 09:37:51 +00:00
|
|
|
SoundIoDummy *sid = create<SoundIoDummy>();
|
|
|
|
if (!sid) {
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 09:37:51 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-07-13 16:17:20 +00:00
|
|
|
si->backend_data = sid;
|
2015-07-01 09:37:51 +00:00
|
|
|
|
|
|
|
sid->mutex = soundio_os_mutex_create();
|
|
|
|
if (!sid->mutex) {
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 09:37:51 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
|
|
|
sid->cond = soundio_os_cond_create();
|
|
|
|
if (!sid->cond) {
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 09:37:51 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
assert(!si->safe_devices_info);
|
|
|
|
si->safe_devices_info = create<SoundIoDevicesInfo>();
|
|
|
|
if (!si->safe_devices_info) {
|
|
|
|
destroy_dummy(si);
|
2015-07-01 08:20:26 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
si->safe_devices_info->default_input_index = 0;
|
|
|
|
si->safe_devices_info->default_output_index = 0;
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
// create output device
|
|
|
|
{
|
|
|
|
SoundIoDevice *device = create<SoundIoDevice>();
|
2015-07-01 09:37:51 +00:00
|
|
|
if (!device) {
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 08:02:44 +00:00
|
|
|
return SoundIoErrorNoMem;
|
2015-07-01 09:37:51 +00:00
|
|
|
}
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
device->ref_count = 1;
|
|
|
|
device->soundio = soundio;
|
|
|
|
device->name = strdup("dummy-out");
|
2015-07-10 09:21:47 +00:00
|
|
|
device->description = strdup("Dummy Output Device");
|
2015-07-01 08:02:44 +00:00
|
|
|
if (!device->name || !device->description) {
|
2015-07-10 09:21:47 +00:00
|
|
|
soundio_device_unref(device);
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 08:02:44 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-07-20 07:35:46 +00:00
|
|
|
device->layout_count = soundio_channel_layout_builtin_count();
|
|
|
|
device->layouts = allocate<SoundIoChannelLayout>(device->layout_count);
|
2015-07-13 16:17:20 +00:00
|
|
|
if (!device->layouts) {
|
|
|
|
soundio_device_unref(device);
|
|
|
|
destroy_dummy(si);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-07-20 07:35:46 +00:00
|
|
|
for (int i = 0; i < device->layout_count; i += 1)
|
|
|
|
device->layouts[i] = *soundio_channel_layout_get_builtin(i);
|
2015-07-13 16:17:20 +00:00
|
|
|
|
2015-07-10 09:21:47 +00:00
|
|
|
int err;
|
|
|
|
if ((err = set_all_device_formats(device))) {
|
|
|
|
soundio_device_unref(device);
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-10 09:21:47 +00:00
|
|
|
return err;
|
|
|
|
}
|
|
|
|
|
2015-07-14 04:30:37 +00:00
|
|
|
device->buffer_duration_min = 0.01;
|
|
|
|
device->buffer_duration_max = 4;
|
|
|
|
device->buffer_duration_current = 0.1;
|
2015-07-10 09:21:47 +00:00
|
|
|
device->sample_rate_min = 2;
|
|
|
|
device->sample_rate_max = 5644800;
|
|
|
|
device->sample_rate_current = 48000;
|
2015-07-16 06:45:21 +00:00
|
|
|
device->period_duration_min = 0.01;
|
|
|
|
device->period_duration_max = 2;
|
|
|
|
device->period_duration_current = 0.05;
|
2015-07-01 08:02:44 +00:00
|
|
|
device->purpose = SoundIoDevicePurposeOutput;
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
if (si->safe_devices_info->output_devices.append(device)) {
|
2015-07-01 08:29:35 +00:00
|
|
|
soundio_device_unref(device);
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 08:02:44 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// create input device
|
|
|
|
{
|
|
|
|
SoundIoDevice *device = create<SoundIoDevice>();
|
2015-07-01 09:37:51 +00:00
|
|
|
if (!device) {
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 08:02:44 +00:00
|
|
|
return SoundIoErrorNoMem;
|
2015-07-01 09:37:51 +00:00
|
|
|
}
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
device->ref_count = 1;
|
|
|
|
device->soundio = soundio;
|
|
|
|
device->name = strdup("dummy-in");
|
|
|
|
device->description = strdup("Dummy input device");
|
|
|
|
if (!device->name || !device->description) {
|
2015-07-07 09:55:32 +00:00
|
|
|
soundio_device_unref(device);
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 08:02:44 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-07-13 16:17:20 +00:00
|
|
|
|
2015-07-20 07:35:46 +00:00
|
|
|
device->layout_count = soundio_channel_layout_builtin_count();
|
|
|
|
device->layouts = allocate<SoundIoChannelLayout>(device->layout_count);
|
2015-07-13 16:17:20 +00:00
|
|
|
if (!device->layouts) {
|
|
|
|
soundio_device_unref(device);
|
|
|
|
destroy_dummy(si);
|
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
2015-07-20 07:35:46 +00:00
|
|
|
for (int i = 0; i < device->layout_count; i += 1)
|
|
|
|
device->layouts[i] = *soundio_channel_layout_get_builtin(i);
|
2015-07-13 16:17:20 +00:00
|
|
|
|
2015-07-10 09:21:47 +00:00
|
|
|
int err;
|
|
|
|
if ((err = set_all_device_formats(device))) {
|
|
|
|
soundio_device_unref(device);
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-10 09:21:47 +00:00
|
|
|
return err;
|
|
|
|
}
|
2015-07-14 04:30:37 +00:00
|
|
|
device->buffer_duration_min = 0.01;
|
|
|
|
device->buffer_duration_max = 4;
|
|
|
|
device->buffer_duration_current = 0.1;
|
2015-07-10 09:21:47 +00:00
|
|
|
device->sample_rate_min = 2;
|
|
|
|
device->sample_rate_max = 5644800;
|
|
|
|
device->sample_rate_current = 48000;
|
2015-07-16 06:45:21 +00:00
|
|
|
device->period_duration_min = 0.01;
|
|
|
|
device->period_duration_max = 2;
|
|
|
|
device->period_duration_current = 0.05;
|
2015-07-01 08:02:44 +00:00
|
|
|
device->purpose = SoundIoDevicePurposeInput;
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
if (si->safe_devices_info->input_devices.append(device)) {
|
2015-07-01 08:29:35 +00:00
|
|
|
soundio_device_unref(device);
|
2015-07-13 16:17:20 +00:00
|
|
|
destroy_dummy(si);
|
2015-07-01 08:02:44 +00:00
|
|
|
return SoundIoErrorNoMem;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-07-13 16:17:20 +00:00
|
|
|
si->destroy = destroy_dummy;
|
|
|
|
si->flush_events = flush_events;
|
|
|
|
si->wait_events = wait_events;
|
|
|
|
si->wakeup = wakeup;
|
|
|
|
|
2015-07-14 04:30:37 +00:00
|
|
|
si->outstream_open = outstream_open_dummy;
|
2015-07-13 16:17:20 +00:00
|
|
|
si->outstream_destroy = outstream_destroy_dummy;
|
|
|
|
si->outstream_start = outstream_start_dummy;
|
|
|
|
si->outstream_free_count = outstream_free_count_dummy;
|
|
|
|
si->outstream_begin_write = outstream_begin_write_dummy;
|
|
|
|
si->outstream_write = outstream_write_dummy;
|
|
|
|
si->outstream_clear_buffer = outstream_clear_buffer_dummy;
|
2015-07-16 20:37:41 +00:00
|
|
|
si->outstream_pause = outstream_pause_dummy;
|
2015-07-13 16:17:20 +00:00
|
|
|
|
2015-07-14 04:30:37 +00:00
|
|
|
si->instream_open = instream_open_dummy;
|
2015-07-13 16:17:20 +00:00
|
|
|
si->instream_destroy = instream_destroy_dummy;
|
|
|
|
si->instream_start = instream_start_dummy;
|
|
|
|
si->instream_peek = instream_peek_dummy;
|
|
|
|
si->instream_drop = instream_drop_dummy;
|
|
|
|
si->instream_clear_buffer = instream_clear_buffer_dummy;
|
2015-07-16 20:37:41 +00:00
|
|
|
si->instream_pause = instream_pause_dummy;
|
2015-07-01 08:02:44 +00:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|