/* * 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" #include "os.hpp" #include "atomics.hpp" #include "ring_buffer.hpp" #include #include #include struct SoundIoOutStreamDummy { struct SoundIoOsThread *thread; struct SoundIoOsCond *cond; atomic_flag abort_flag; int buffer_frame_count; struct SoundIoRingBuffer ring_buffer; int prebuf_frame_count; int prebuf_frames_left; long frames_consumed; double playback_start_time; SoundIoChannelArea areas[SOUNDIO_MAX_CHANNELS]; }; struct SoundIoInStreamDummy { struct SoundIoOsThread *thread; struct SoundIoOsCond *cond; atomic_flag abort_flag; int read_frame_count; int buffer_frame_count; struct SoundIoRingBuffer ring_buffer; int hole_size; SoundIoChannelArea areas[SOUNDIO_MAX_CHANNELS]; }; struct SoundIoDummy { SoundIoOsMutex *mutex; SoundIoOsCond *cond; bool devices_emitted; }; static void playback_thread_run(void *arg) { SoundIoOutStreamPrivate *os = (SoundIoOutStreamPrivate *)arg; SoundIoOutStream *outstream = &os->pub; SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data; double start_time = soundio_os_get_time(); osd->playback_start_time = start_time; osd->frames_consumed = 0; int fill_bytes = soundio_ring_buffer_fill_count(&osd->ring_buffer); int free_bytes = soundio_ring_buffer_capacity(&osd->ring_buffer) - fill_bytes; int fill_frames = fill_bytes / outstream->bytes_per_frame; int free_frames = free_bytes / outstream->bytes_per_frame; osd->prebuf_frames_left = osd->prebuf_frame_count - fill_frames; outstream->write_callback(outstream, free_frames); while (osd->abort_flag.test_and_set()) { double 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); int fill_bytes = soundio_ring_buffer_fill_count(&osd->ring_buffer); int free_bytes = soundio_ring_buffer_capacity(&osd->ring_buffer) - fill_bytes; int fill_frames = fill_bytes / outstream->bytes_per_frame; int free_frames = free_bytes / outstream->bytes_per_frame; if (osd->prebuf_frames_left > 0) { outstream->write_callback(outstream, free_frames); continue; } double total_time = soundio_os_get_time() - osd->playback_start_time; long total_frames = total_time * outstream->sample_rate; int frames_to_kill = total_frames - osd->frames_consumed; int read_count = min(frames_to_kill, fill_frames); int byte_count = read_count * outstream->bytes_per_frame; soundio_ring_buffer_advance_read_ptr(&osd->ring_buffer, byte_count); osd->frames_consumed += read_count; if (frames_to_kill > read_count) { osd->prebuf_frames_left = osd->prebuf_frame_count; outstream->underflow_callback(outstream); } else if (free_frames > 0) { outstream->write_callback(outstream, free_frames); } } } static void capture_thread_run(void *arg) { SoundIoInStreamPrivate *is = (SoundIoInStreamPrivate *)arg; SoundIoInStream *instream = &is->pub; SoundIoInStreamDummy *isd = (SoundIoInStreamDummy *)is->backend_data; long frames_consumed = 0; double start_time = soundio_os_get_time(); while (isd->abort_flag.test_and_set()) { double now = soundio_os_get_time(); double total_time = now - start_time; long total_frames = total_time * instream->sample_rate; int frames_to_kill = total_frames - frames_consumed; int free_bytes = soundio_ring_buffer_free_count(&isd->ring_buffer); int free_frames = free_bytes / instream->bytes_per_frame; int write_count = min(frames_to_kill, free_frames); int frames_left = max(frames_to_kill - write_count, 0); int byte_count = write_count * instream->bytes_per_frame; soundio_ring_buffer_advance_write_ptr(&isd->ring_buffer, byte_count); frames_consumed += write_count; isd->hole_size += frames_left; if (write_count > 0) instream->read_callback(instream, write_count); now = soundio_os_get_time(); double time_passed = now - start_time; double next_period = start_time + ceil(time_passed / instream->period_duration) * instream->period_duration; double relative_time = next_period - now; soundio_os_cond_timed_wait(isd->cond, nullptr, relative_time); } } static void destroy_dummy(SoundIoPrivate *si) { SoundIoDummy *sid = (SoundIoDummy *)si->backend_data; if (!sid) return; if (sid->cond) soundio_os_cond_destroy(sid->cond); if (sid->mutex) soundio_os_mutex_destroy(sid->mutex); destroy(sid); si->backend_data = nullptr; } static void flush_events(SoundIoPrivate *si) { SoundIo *soundio = &si->pub; SoundIoDummy *sid = (SoundIoDummy *)si->backend_data; if (sid->devices_emitted) return; sid->devices_emitted = true; soundio->on_devices_change(soundio); } static void wait_events(SoundIoPrivate *si) { SoundIoDummy *sid = (SoundIoDummy *)si->backend_data; flush_events(si); soundio_os_cond_wait(sid->cond, nullptr); } static void wakeup(SoundIoPrivate *si) { SoundIoDummy *sid = (SoundIoDummy *)si->backend_data; soundio_os_cond_signal(sid->cond, nullptr); } static void outstream_destroy_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) { SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data; if (!osd) return; if (osd->thread) { osd->abort_flag.clear(); soundio_os_cond_signal(osd->cond, nullptr); soundio_os_thread_destroy(osd->thread); osd->thread = nullptr; } soundio_os_cond_destroy(osd->cond); osd->cond = nullptr; soundio_ring_buffer_deinit(&osd->ring_buffer); destroy(osd); os->backend_data = nullptr; } static int outstream_open_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) { SoundIoOutStream *outstream = &os->pub; SoundIoDevice *device = outstream->device; if (outstream->buffer_duration == 0.0) outstream->buffer_duration = clamp(device->buffer_duration_min, 1.0, device->buffer_duration_max); if (outstream->period_duration == 0.0) { outstream->period_duration = clamp(device->period_duration_min, outstream->buffer_duration / 2.0, device->period_duration_max); } SoundIoOutStreamDummy *osd = create(); if (!osd) { outstream_destroy_dummy(si, os); return SoundIoErrorNoMem; } os->backend_data = osd; int err; int buffer_size = outstream->bytes_per_frame * outstream->sample_rate * outstream->buffer_duration; if ((err = soundio_ring_buffer_init(&osd->ring_buffer, buffer_size))) { outstream_destroy_dummy(si, os); return err; } int actual_capacity = soundio_ring_buffer_capacity(&osd->ring_buffer); osd->buffer_frame_count = actual_capacity / outstream->bytes_per_frame; outstream->buffer_duration = osd->buffer_frame_count / (double) outstream->sample_rate; if (outstream->prebuf_duration == -1.0) outstream->prebuf_duration = outstream->buffer_duration; outstream->prebuf_duration = min(outstream->prebuf_duration, outstream->buffer_duration); osd->prebuf_frame_count = ceil(outstream->prebuf_duration * (double) outstream->sample_rate); osd->cond = soundio_os_cond_create(); if (!osd->cond) { outstream_destroy_dummy(si, os); return SoundIoErrorNoMem; } return 0; } 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) { return outstream_pause_dummy(si, os, false); } static int outstream_begin_write_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os, SoundIoChannelArea **out_areas, int *frame_count) { *out_areas = nullptr; SoundIoOutStream *outstream = &os->pub; SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data; assert(*frame_count >= 0); assert(*frame_count <= osd->buffer_frame_count); 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; } *out_areas = osd->areas; } return 0; } static int outstream_end_write_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os, int frame_count) { SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data; SoundIoOutStream *outstream = &os->pub; int byte_count = frame_count * outstream->bytes_per_frame; soundio_ring_buffer_advance_write_ptr(&osd->ring_buffer, byte_count); if (osd->prebuf_frames_left > 0) { osd->prebuf_frames_left -= frame_count; if (osd->prebuf_frames_left <= 0) { osd->playback_start_time = soundio_os_get_time(); osd->frames_consumed = 0; } } return 0; } static int outstream_clear_buffer_dummy(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) { SoundIoOutStreamDummy *osd = (SoundIoOutStreamDummy *)os->backend_data; soundio_ring_buffer_clear(&osd->ring_buffer); osd->prebuf_frames_left = osd->prebuf_frame_count; return 0; } static void instream_destroy_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) { SoundIoInStreamDummy *isd = (SoundIoInStreamDummy *)is->backend_data; if (!isd) return; if (isd->thread) { isd->abort_flag.clear(); soundio_os_cond_signal(isd->cond, nullptr); soundio_os_thread_destroy(isd->thread); isd->thread = nullptr; } soundio_os_cond_destroy(isd->cond); isd->cond = nullptr; soundio_ring_buffer_deinit(&isd->ring_buffer); destroy(isd); is->backend_data = nullptr; } static int instream_open_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) { SoundIoInStream *instream = &is->pub; SoundIoDevice *device = instream->device; if (instream->buffer_duration == 0.0) instream->buffer_duration = clamp(device->buffer_duration_min, 1.0, device->buffer_duration_max); if (instream->period_duration == 0.0) { instream->period_duration = clamp(instream->device->period_duration_min, instream->buffer_duration / 8.0, instream->device->period_duration_max); } SoundIoInStreamDummy *isd = create(); if (!isd) { instream_destroy_dummy(si, is); return SoundIoErrorNoMem; } is->backend_data = isd; int err; int buffer_size = instream->bytes_per_frame * instream->sample_rate * instream->buffer_duration; if ((err = soundio_ring_buffer_init(&isd->ring_buffer, buffer_size))) { instream_destroy_dummy(si, is); return err; } int actual_capacity = soundio_ring_buffer_capacity(&isd->ring_buffer); isd->buffer_frame_count = actual_capacity / instream->bytes_per_frame; instream->buffer_duration = isd->buffer_frame_count / (double) instream->sample_rate; isd->cond = soundio_os_cond_create(); if (!isd->cond) { instream_destroy_dummy(si, is); return SoundIoErrorNoMem; } return 0; } static int instream_pause_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is, bool pause) { SoundIoInStreamDummy *isd = (SoundIoInStreamDummy *)is->backend_data; if (pause) { if (isd->thread) { isd->abort_flag.clear(); soundio_os_cond_signal(isd->cond, nullptr); soundio_os_thread_destroy(isd->thread); isd->thread = nullptr; } } else { if (!isd->thread) { isd->abort_flag.test_and_set(); int err; if ((err = soundio_os_thread_create(capture_thread_run, is, true, &isd->thread))) { return err; } } } return 0; } static int instream_start_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) { return instream_pause_dummy(si, is, false); } static int instream_begin_read_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is, SoundIoChannelArea **out_areas, int *frame_count) { SoundIoInStream *instream = &is->pub; SoundIoInStreamDummy *isd = (SoundIoInStreamDummy *)is->backend_data; assert(*frame_count >= 0); assert(*frame_count <= isd->buffer_frame_count); if (isd->hole_size > 0) { *out_areas = nullptr; isd->read_frame_count = min(isd->hole_size, *frame_count); *frame_count = isd->read_frame_count; return 0; } int fill_byte_count = soundio_ring_buffer_fill_count(&isd->ring_buffer); int fill_frame_count = fill_byte_count / instream->bytes_per_frame; isd->read_frame_count = min(*frame_count, fill_frame_count); *frame_count = isd->read_frame_count; if (fill_frame_count) { char *read_ptr = soundio_ring_buffer_read_ptr(&isd->ring_buffer); for (int ch = 0; ch < instream->layout.channel_count; ch += 1) { isd->areas[ch].ptr = read_ptr + instream->bytes_per_sample * ch; isd->areas[ch].step = instream->bytes_per_frame; } *out_areas = isd->areas; } return 0; } static int instream_end_read_dummy(SoundIoPrivate *si, SoundIoInStreamPrivate *is) { SoundIoInStreamDummy *isd = (SoundIoInStreamDummy *)is->backend_data; SoundIoInStream *instream = &is->pub; if (isd->hole_size > 0) { isd->hole_size -= isd->read_frame_count; return 0; } int byte_count = isd->read_frame_count * instream->bytes_per_frame; soundio_ring_buffer_advance_write_ptr(&isd->ring_buffer, byte_count); return 0; } static int set_all_device_formats(SoundIoDevice *device) { device->format_count = 18; device->formats = allocate(device->format_count); if (!device->formats) return SoundIoErrorNoMem; 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; return 0; } static int set_all_device_channel_layouts(SoundIoDevice *device) { device->layout_count = soundio_channel_layout_builtin_count(); device->layouts = allocate(device->layout_count); if (!device->layouts) return SoundIoErrorNoMem; for (int i = 0; i < device->layout_count; i += 1) device->layouts[i] = *soundio_channel_layout_get_builtin(i); return 0; } int soundio_dummy_init(SoundIoPrivate *si) { SoundIo *soundio = &si->pub; assert(!si->backend_data); SoundIoDummy *sid = create(); if (!sid) { destroy_dummy(si); return SoundIoErrorNoMem; } si->backend_data = sid; sid->mutex = soundio_os_mutex_create(); if (!sid->mutex) { destroy_dummy(si); return SoundIoErrorNoMem; } sid->cond = soundio_os_cond_create(); if (!sid->cond) { destroy_dummy(si); return SoundIoErrorNoMem; } assert(!si->safe_devices_info); si->safe_devices_info = create(); if (!si->safe_devices_info) { destroy_dummy(si); return SoundIoErrorNoMem; } si->safe_devices_info->default_input_index = 0; si->safe_devices_info->default_output_index = 0; // create output device { SoundIoDevice *device = create(); if (!device) { destroy_dummy(si); return SoundIoErrorNoMem; } device->ref_count = 1; device->soundio = soundio; device->name = strdup("dummy-out"); device->description = strdup("Dummy Output Device"); if (!device->name || !device->description) { soundio_device_unref(device); destroy_dummy(si); return SoundIoErrorNoMem; } int err; if ((err = set_all_device_channel_layouts(device))) { soundio_device_unref(device); destroy_dummy(si); return err; } if ((err = set_all_device_formats(device))) { soundio_device_unref(device); destroy_dummy(si); return err; } device->buffer_duration_min = 0.01; device->buffer_duration_max = 4; device->buffer_duration_current = 0.1; device->sample_rate_min = 2; device->sample_rate_max = 5644800; device->sample_rate_current = 48000; device->period_duration_min = 0.01; device->period_duration_max = 2; device->period_duration_current = 0.05; device->purpose = SoundIoDevicePurposeOutput; if (si->safe_devices_info->output_devices.append(device)) { soundio_device_unref(device); destroy_dummy(si); return SoundIoErrorNoMem; } } // create input device { SoundIoDevice *device = create(); if (!device) { destroy_dummy(si); return SoundIoErrorNoMem; } device->ref_count = 1; device->soundio = soundio; device->name = strdup("dummy-in"); device->description = strdup("Dummy input device"); if (!device->name || !device->description) { soundio_device_unref(device); destroy_dummy(si); return SoundIoErrorNoMem; } int err; if ((err = set_all_device_channel_layouts(device))) { soundio_device_unref(device); destroy_dummy(si); return err; } if ((err = set_all_device_formats(device))) { soundio_device_unref(device); destroy_dummy(si); return err; } device->buffer_duration_min = 0.01; device->buffer_duration_max = 4; device->buffer_duration_current = 0.1; device->sample_rate_min = 2; device->sample_rate_max = 5644800; device->sample_rate_current = 48000; device->period_duration_min = 0.01; device->period_duration_max = 2; device->period_duration_current = 0.05; device->purpose = SoundIoDevicePurposeInput; if (si->safe_devices_info->input_devices.append(device)) { soundio_device_unref(device); destroy_dummy(si); return SoundIoErrorNoMem; } } si->destroy = destroy_dummy; si->flush_events = flush_events; si->wait_events = wait_events; si->wakeup = wakeup; si->outstream_open = outstream_open_dummy; si->outstream_destroy = outstream_destroy_dummy; si->outstream_start = outstream_start_dummy; si->outstream_begin_write = outstream_begin_write_dummy; si->outstream_end_write = outstream_end_write_dummy; si->outstream_clear_buffer = outstream_clear_buffer_dummy; si->outstream_pause = outstream_pause_dummy; si->instream_open = instream_open_dummy; si->instream_destroy = instream_destroy_dummy; si->instream_start = instream_start_dummy; si->instream_begin_read = instream_begin_read_dummy; si->instream_end_read = instream_end_read_dummy; si->instream_pause = instream_pause_dummy; return 0; }