mirror of
https://github.com/Ryujinx/libsoundio.git
synced 2025-01-18 15:17:09 +00:00
various code adjustments
* doc clarifications * examples compile with c99 not c11 * fix pulseaudio on_backend_disconnected not firing only during flush events * make wait events more efficient * fix alsa devices race condition * fix backend disconnected code handling * add overflow test * fix on_events_signal not called at correct times * refactor pulseaudio device scanning * fix SoundIoErrorNoSuchDevice string value
This commit is contained in:
parent
dd49396429
commit
16437bd357
|
@ -171,7 +171,7 @@ set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -Werror -pedantic")
|
||||||
|
|
||||||
|
|
||||||
set(LIB_CFLAGS "-std=c++11 -fno-exceptions -fno-rtti -fvisibility=hidden -Wall -Werror=strict-prototypes -Werror=old-style-definition -Werror=missing-prototypes -Wno-c99-extensions")
|
set(LIB_CFLAGS "-std=c++11 -fno-exceptions -fno-rtti -fvisibility=hidden -Wall -Werror=strict-prototypes -Werror=old-style-definition -Werror=missing-prototypes -Wno-c99-extensions")
|
||||||
set(EXAMPLE_CFLAGS "-std=c11 -Wall")
|
set(EXAMPLE_CFLAGS "-std=c99 -Wall")
|
||||||
set(EXAMPLE_INCLUDES "${CMAKE_SOURCE_DIR}/src")
|
set(EXAMPLE_INCLUDES "${CMAKE_SOURCE_DIR}/src")
|
||||||
set(TEST_CFLAGS "${LIB_CFLAGS} -fprofile-arcs -ftest-coverage")
|
set(TEST_CFLAGS "${LIB_CFLAGS} -fprofile-arcs -ftest-coverage")
|
||||||
set(TEST_LDFLAGS "-fprofile-arcs -ftest-coverage")
|
set(TEST_LDFLAGS "-fprofile-arcs -ftest-coverage")
|
||||||
|
@ -261,7 +261,6 @@ install(TARGETS sio_microphone DESTINATION ${CMAKE_INSTALL_BINDIR})
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
enable_testing()
|
|
||||||
add_executable(unit_tests ${TEST_SOURCES})
|
add_executable(unit_tests ${TEST_SOURCES})
|
||||||
target_link_libraries(unit_tests LINK_PUBLIC
|
target_link_libraries(unit_tests LINK_PUBLIC
|
||||||
${CMAKE_THREAD_LIBS_INIT}
|
${CMAKE_THREAD_LIBS_INIT}
|
||||||
|
@ -279,7 +278,6 @@ set_target_properties(unit_tests PROPERTIES
|
||||||
LINK_FLAGS ${TEST_LDFLAGS}
|
LINK_FLAGS ${TEST_LDFLAGS}
|
||||||
)
|
)
|
||||||
include_directories(${TEST_INCLUDES})
|
include_directories(${TEST_INCLUDES})
|
||||||
add_test(UnitTests unit_tests)
|
|
||||||
|
|
||||||
add_executable(underflow test/underflow.c)
|
add_executable(underflow test/underflow.c)
|
||||||
set_target_properties(underflow PROPERTIES
|
set_target_properties(underflow PROPERTIES
|
||||||
|
@ -287,7 +285,6 @@ set_target_properties(underflow PROPERTIES
|
||||||
COMPILE_FLAGS ${EXAMPLE_CFLAGS})
|
COMPILE_FLAGS ${EXAMPLE_CFLAGS})
|
||||||
include_directories(${EXAMPLE_INCLUDES})
|
include_directories(${EXAMPLE_INCLUDES})
|
||||||
target_link_libraries(underflow libsoundio_shared)
|
target_link_libraries(underflow libsoundio_shared)
|
||||||
add_test(Underflow underflow)
|
|
||||||
|
|
||||||
add_executable(backend_disconnect_recover test/backend_disconnect_recover.c)
|
add_executable(backend_disconnect_recover test/backend_disconnect_recover.c)
|
||||||
set_target_properties(backend_disconnect_recover PROPERTIES
|
set_target_properties(backend_disconnect_recover PROPERTIES
|
||||||
|
@ -295,7 +292,13 @@ set_target_properties(backend_disconnect_recover PROPERTIES
|
||||||
COMPILE_FLAGS ${EXAMPLE_CFLAGS})
|
COMPILE_FLAGS ${EXAMPLE_CFLAGS})
|
||||||
include_directories(${EXAMPLE_INCLUDES})
|
include_directories(${EXAMPLE_INCLUDES})
|
||||||
target_link_libraries(backend_disconnect_recover libsoundio_shared)
|
target_link_libraries(backend_disconnect_recover libsoundio_shared)
|
||||||
add_test(backend_disconnect_recover backend_disconnect_recover)
|
|
||||||
|
add_executable(overflow test/overflow.c)
|
||||||
|
set_target_properties(overflow PROPERTIES
|
||||||
|
LINKER_LANGUAGE C
|
||||||
|
COMPILE_FLAGS ${EXAMPLE_CFLAGS})
|
||||||
|
include_directories(${EXAMPLE_INCLUDES})
|
||||||
|
target_link_libraries(overflow libsoundio_shared)
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -274,10 +274,7 @@ Then look at `html/index.html` in a browser.
|
||||||
|
|
||||||
## Roadmap
|
## Roadmap
|
||||||
|
|
||||||
0. Ability to "activate" a buffer-flexible outstream by jumping the gun and
|
0. `sio_microphone` with ALSA backend in raw mode quickly causes unrecoverable streaming failure
|
||||||
causing `write_callback` to be called early.
|
|
||||||
- Use the same mechanism when destroying the outstream
|
|
||||||
0. Create a test for input stream overflow handling.
|
|
||||||
0. Create a test for the latency / synchronization API.
|
0. Create a test for the latency / synchronization API.
|
||||||
- Input is an audio file and some events indexed at particular frame - when
|
- Input is an audio file and some events indexed at particular frame - when
|
||||||
listening the events should line up exactly with a beat or visual
|
listening the events should line up exactly with a beat or visual
|
||||||
|
|
|
@ -505,13 +505,22 @@ struct SoundIoOutStream {
|
||||||
/// you will call ::soundio_outstream_clear_buffer when you want to reduce
|
/// you will call ::soundio_outstream_clear_buffer when you want to reduce
|
||||||
/// the latency to 0. On systems that do not support clearing the buffer,
|
/// the latency to 0. On systems that do not support clearing the buffer,
|
||||||
/// this defaults to a reasonable lower latency value.
|
/// this defaults to a reasonable lower latency value.
|
||||||
|
///
|
||||||
|
/// On backends with high latencies (such as 2 seconds), `frame_count_min`
|
||||||
|
/// will be 0, meaning you don't have to fill the entire buffer. In this
|
||||||
|
/// case, the large buffer is there if you want it; you only have to fill
|
||||||
|
/// as much as you want. On backends like JACK, `frame_count_min` will be
|
||||||
|
/// equal to `frame_count_max` and if you don't fill that many frames, you
|
||||||
|
/// will get glitches.
|
||||||
|
///
|
||||||
/// If the device has unknown software latency min and max values, you may
|
/// If the device has unknown software latency min and max values, you may
|
||||||
/// still set this, but you might not get the value you requested.
|
/// still set this, but you might not get the value you requested.
|
||||||
/// For PulseAudio, if you set this value to non-default, it sets
|
/// For PulseAudio, if you set this value to non-default, it sets
|
||||||
/// `PA_STREAM_ADJUST_LATENCY` and is the value used for `maxlength` and
|
/// `PA_STREAM_ADJUST_LATENCY` and is the value used for `maxlength` and
|
||||||
/// `tlength`.
|
/// `tlength`.
|
||||||
|
///
|
||||||
/// For JACK, this value is always equal to
|
/// For JACK, this value is always equal to
|
||||||
/// SoundIoDevice::software_latency_current` of the device.
|
/// SoundIoDevice::software_latency_current of the device.
|
||||||
double software_latency;
|
double software_latency;
|
||||||
|
|
||||||
/// Defaults to NULL. Put whatever you want here.
|
/// Defaults to NULL. Put whatever you want here.
|
||||||
|
@ -589,8 +598,8 @@ struct SoundIoInStream {
|
||||||
/// still set this, but you might not get the value you requested.
|
/// still set this, but you might not get the value you requested.
|
||||||
/// For PulseAudio, if you set this value to non-default, it sets
|
/// For PulseAudio, if you set this value to non-default, it sets
|
||||||
/// `PA_STREAM_ADJUST_LATENCY` and is the value used for `fragsize`.
|
/// `PA_STREAM_ADJUST_LATENCY` and is the value used for `fragsize`.
|
||||||
/// For JACK, this value is always equal to `software_latency_current` of
|
/// For JACK, this value is always equal to
|
||||||
/// the device.
|
/// SoundIoDevice::software_latency_current
|
||||||
double software_latency;
|
double software_latency;
|
||||||
|
|
||||||
/// Defaults to NULL. Put whatever you want here.
|
/// Defaults to NULL. Put whatever you want here.
|
||||||
|
@ -602,6 +611,11 @@ struct SoundIoInStream {
|
||||||
/// `frame_count_min`, the frames will be dropped. `frame_count_max` is how
|
/// `frame_count_min`, the frames will be dropped. `frame_count_max` is how
|
||||||
/// many frames are available to read.
|
/// many frames are available to read.
|
||||||
void (*read_callback)(struct SoundIoInStream *, int frame_count_min, int frame_count_max);
|
void (*read_callback)(struct SoundIoInStream *, int frame_count_min, int frame_count_max);
|
||||||
|
/// This optional callback happens when the sound device buffer is full,
|
||||||
|
/// yet there is more captured audio to put in it.
|
||||||
|
/// This is never fired for PulseAudio.
|
||||||
|
/// This is called from the SoundIoInStream::read_callback thread context.
|
||||||
|
void (*overflow_callback)(struct SoundIoInStream *);
|
||||||
/// Optional callback. `err` is always SoundIoErrorStreaming.
|
/// Optional callback. `err` is always SoundIoErrorStreaming.
|
||||||
/// SoundIoErrorStreaming is an unrecoverable error. The stream is in an
|
/// SoundIoErrorStreaming is an unrecoverable error. The stream is in an
|
||||||
/// invalid state and must be destroyed.
|
/// invalid state and must be destroyed.
|
||||||
|
@ -680,9 +694,15 @@ SOUNDIO_EXPORT enum SoundIoBackend soundio_get_backend(struct SoundIo *soundio,
|
||||||
/// Returns whether libsoundio was compiled with backend.
|
/// Returns whether libsoundio was compiled with backend.
|
||||||
SOUNDIO_EXPORT bool soundio_have_backend(enum SoundIoBackend backend);
|
SOUNDIO_EXPORT bool soundio_have_backend(enum SoundIoBackend backend);
|
||||||
|
|
||||||
|
/// Atomically update information for all connected devices. Note that calling
|
||||||
|
/// this function merely flips a pointer; the actual work of collecting device
|
||||||
|
/// information is done elsewhere. It is performant to call this function many
|
||||||
|
/// times per second.
|
||||||
|
///
|
||||||
/// When you call this, the SoundIo::on_devices_change and
|
/// When you call this, the SoundIo::on_devices_change and
|
||||||
/// SoundIo::on_events_signal callbacks
|
/// SoundIo::on_events_signal callbacks
|
||||||
/// might be called. This is the only time those callbacks will be called.
|
/// might be called. This is the only time those callbacks will be called.
|
||||||
|
///
|
||||||
/// This must be called from the same thread as the thread in which you call
|
/// This must be called from the same thread as the thread in which you call
|
||||||
/// these functions:
|
/// these functions:
|
||||||
/// * ::soundio_input_device_count
|
/// * ::soundio_input_device_count
|
||||||
|
@ -691,6 +711,10 @@ SOUNDIO_EXPORT bool soundio_have_backend(enum SoundIoBackend backend);
|
||||||
/// * ::soundio_get_output_device
|
/// * ::soundio_get_output_device
|
||||||
/// * ::soundio_default_input_device_index
|
/// * ::soundio_default_input_device_index
|
||||||
/// * ::soundio_default_output_device_index
|
/// * ::soundio_default_output_device_index
|
||||||
|
///
|
||||||
|
/// Note that if you do not care about learning about updated devices, you
|
||||||
|
/// might call this function only once ever and never call
|
||||||
|
/// ::soundio_wait_events.
|
||||||
SOUNDIO_EXPORT void soundio_flush_events(struct SoundIo *soundio);
|
SOUNDIO_EXPORT void soundio_flush_events(struct SoundIo *soundio);
|
||||||
|
|
||||||
/// This function calls ::soundio_flush_events then blocks until another event
|
/// This function calls ::soundio_flush_events then blocks until another event
|
||||||
|
@ -719,6 +743,9 @@ SOUNDIO_EXPORT enum SoundIoChannelId soundio_parse_channel_id(const char *str, i
|
||||||
/// Returns the number of builtin channel layouts.
|
/// Returns the number of builtin channel layouts.
|
||||||
SOUNDIO_EXPORT int soundio_channel_layout_builtin_count(void);
|
SOUNDIO_EXPORT int soundio_channel_layout_builtin_count(void);
|
||||||
/// Returns a builtin channel layout. 0 <= `index` < ::soundio_channel_layout_builtin_count
|
/// Returns a builtin channel layout. 0 <= `index` < ::soundio_channel_layout_builtin_count
|
||||||
|
///
|
||||||
|
/// Although `index` is of type `int`, it should be a valid
|
||||||
|
/// #SoundIoChannelLayoutId enum value.
|
||||||
SOUNDIO_EXPORT const struct SoundIoChannelLayout *soundio_channel_layout_get_builtin(int index);
|
SOUNDIO_EXPORT const struct SoundIoChannelLayout *soundio_channel_layout_get_builtin(int index);
|
||||||
|
|
||||||
/// Get the default builtin channel layout for the given number of channels.
|
/// Get the default builtin channel layout for the given number of channels.
|
||||||
|
@ -1044,8 +1071,10 @@ SOUNDIO_EXPORT int soundio_instream_pause(struct SoundIoInStream *instream, bool
|
||||||
// Ring Buffer
|
// Ring Buffer
|
||||||
struct SoundIoRingBuffer;
|
struct SoundIoRingBuffer;
|
||||||
/// `requested_capacity` in bytes.
|
/// `requested_capacity` in bytes.
|
||||||
/// See also ::soundio_ring_buffer_destroy
|
|
||||||
/// Returns `NULL` if and only if memory could not be allocated.
|
/// Returns `NULL` if and only if memory could not be allocated.
|
||||||
|
/// Use ::soundio_ring_buffer_capacity to get the actual capacity, which might
|
||||||
|
/// be greater for alignment purposes.
|
||||||
|
/// See also ::soundio_ring_buffer_destroy
|
||||||
SOUNDIO_EXPORT struct SoundIoRingBuffer *soundio_ring_buffer_create(struct SoundIo *soundio, int requested_capacity);
|
SOUNDIO_EXPORT struct SoundIoRingBuffer *soundio_ring_buffer_create(struct SoundIo *soundio, int requested_capacity);
|
||||||
SOUNDIO_EXPORT void soundio_ring_buffer_destroy(struct SoundIoRingBuffer *ring_buffer);
|
SOUNDIO_EXPORT void soundio_ring_buffer_destroy(struct SoundIoRingBuffer *ring_buffer);
|
||||||
|
|
||||||
|
|
127
src/alsa.cpp
127
src/alsa.cpp
|
@ -42,6 +42,8 @@ static void destroy_alsa(SoundIoPrivate *si) {
|
||||||
soundio_os_thread_destroy(sia->thread);
|
soundio_os_thread_destroy(sia->thread);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sia->pending_files.deinit();
|
||||||
|
|
||||||
if (sia->cond)
|
if (sia->cond)
|
||||||
soundio_os_cond_destroy(sia->cond);
|
soundio_os_cond_destroy(sia->cond);
|
||||||
|
|
||||||
|
@ -450,6 +452,12 @@ static int refresh_devices(SoundIoPrivate *si) {
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoAlsa *sia = &si->backend_data.alsa;
|
SoundIoAlsa *sia = &si->backend_data.alsa;
|
||||||
|
|
||||||
|
int err;
|
||||||
|
if ((err = snd_config_update_free_global()) < 0)
|
||||||
|
return SoundIoErrorSystemResources;
|
||||||
|
if ((err = snd_config_update()) < 0)
|
||||||
|
return SoundIoErrorSystemResources;
|
||||||
|
|
||||||
SoundIoDevicesInfo *devices_info = allocate<SoundIoDevicesInfo>(1);
|
SoundIoDevicesInfo *devices_info = allocate<SoundIoDevicesInfo>(1);
|
||||||
if (!devices_info)
|
if (!devices_info)
|
||||||
return SoundIoErrorNoMem;
|
return SoundIoErrorNoMem;
|
||||||
|
@ -685,7 +693,7 @@ static int refresh_devices(SoundIoPrivate *si) {
|
||||||
soundio_os_mutex_lock(sia->mutex);
|
soundio_os_mutex_lock(sia->mutex);
|
||||||
soundio_destroy_devices_info(sia->ready_devices_info);
|
soundio_destroy_devices_info(sia->ready_devices_info);
|
||||||
sia->ready_devices_info = devices_info;
|
sia->ready_devices_info = devices_info;
|
||||||
sia->have_devices_flag.store(true);
|
sia->have_devices_flag = true;
|
||||||
soundio_os_cond_signal(sia->cond, sia->mutex);
|
soundio_os_cond_signal(sia->cond, sia->mutex);
|
||||||
soundio->on_events_signal(soundio);
|
soundio->on_events_signal(soundio);
|
||||||
soundio_os_mutex_unlock(sia->mutex);
|
soundio_os_mutex_unlock(sia->mutex);
|
||||||
|
@ -697,13 +705,28 @@ static void shutdown_backend(SoundIoPrivate *si, int err) {
|
||||||
SoundIoAlsa *sia = &si->backend_data.alsa;
|
SoundIoAlsa *sia = &si->backend_data.alsa;
|
||||||
soundio_os_mutex_lock(sia->mutex);
|
soundio_os_mutex_lock(sia->mutex);
|
||||||
sia->shutdown_err = err;
|
sia->shutdown_err = err;
|
||||||
|
soundio_os_cond_signal(sia->cond, sia->mutex);
|
||||||
soundio->on_events_signal(soundio);
|
soundio->on_events_signal(soundio);
|
||||||
soundio_os_mutex_unlock(sia->mutex);
|
soundio_os_mutex_unlock(sia->mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool copy_str(char *dest, const char *src, int buf_len) {
|
||||||
|
for (;;) {
|
||||||
|
buf_len -= 1;
|
||||||
|
if (buf_len <= 0)
|
||||||
|
return false;
|
||||||
|
*dest = *src;
|
||||||
|
dest += 1;
|
||||||
|
src += 1;
|
||||||
|
if (!*src)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
*dest = '\0';
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
static void device_thread_run(void *arg) {
|
static void device_thread_run(void *arg) {
|
||||||
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
||||||
SoundIo *soundio = &si->pub;
|
|
||||||
SoundIoAlsa *sia = &si->backend_data.alsa;
|
SoundIoAlsa *sia = &si->backend_data.alsa;
|
||||||
|
|
||||||
// Some systems cannot read integer variables if they are not
|
// Some systems cannot read integer variables if they are not
|
||||||
|
@ -749,6 +772,12 @@ static void device_thread_run(void *arg) {
|
||||||
assert(errno != EINVAL);
|
assert(errno != EINVAL);
|
||||||
assert(errno != EIO);
|
assert(errno != EIO);
|
||||||
assert(errno != EISDIR);
|
assert(errno != EISDIR);
|
||||||
|
if (errno == EBADF || errno == EFAULT || errno == EINVAL ||
|
||||||
|
errno == EIO || errno == EISDIR)
|
||||||
|
{
|
||||||
|
shutdown_backend(si, SoundIoErrorSystemResources);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// catches EINTR and EAGAIN
|
// catches EINTR and EAGAIN
|
||||||
|
@ -759,7 +788,7 @@ static void device_thread_run(void *arg) {
|
||||||
for (char *ptr = buf; ptr < buf + len; ptr += sizeof(struct inotify_event) + event->len) {
|
for (char *ptr = buf; ptr < buf + len; ptr += sizeof(struct inotify_event) + event->len) {
|
||||||
event = (const struct inotify_event *) ptr;
|
event = (const struct inotify_event *) ptr;
|
||||||
|
|
||||||
if (!((event->mask & IN_CREATE) || (event->mask & IN_DELETE)))
|
if (!((event->mask & IN_CLOSE_WRITE) || (event->mask & IN_DELETE) || (event->mask & IN_CREATE)))
|
||||||
continue;
|
continue;
|
||||||
if (event->mask & IN_ISDIR)
|
if (event->mask & IN_ISDIR)
|
||||||
continue;
|
continue;
|
||||||
|
@ -771,10 +800,39 @@ static void device_thread_run(void *arg) {
|
||||||
{
|
{
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
if (event->mask & IN_CREATE) {
|
||||||
|
if ((err = sia->pending_files.add_one())) {
|
||||||
|
shutdown_backend(si, SoundIoErrorNoMem);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
SoundIoAlsaPendingFile *pending_file = &sia->pending_files.last();
|
||||||
|
if (!copy_str(pending_file->name, event->name, SOUNDIO_MAX_ALSA_SND_FILE_LEN)) {
|
||||||
|
sia->pending_files.pop();
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (sia->pending_files.length > 0) {
|
||||||
|
// At this point ignore IN_DELETE in favor of waiting until the files
|
||||||
|
// opened with IN_CREATE have their IN_CLOSE_WRITE event.
|
||||||
|
if (!(event->mask & IN_CLOSE_WRITE))
|
||||||
|
continue;
|
||||||
|
for (int i = 0; i < sia->pending_files.length; i += 1) {
|
||||||
|
SoundIoAlsaPendingFile *pending_file = &sia->pending_files.at(i);
|
||||||
|
if (strcmp(pending_file->name, event->name) == 0) {
|
||||||
|
sia->pending_files.swap_remove(i);
|
||||||
|
if (sia->pending_files.length == 0) {
|
||||||
got_rescan_event = true;
|
got_rescan_event = true;
|
||||||
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else if (event->mask & IN_DELETE) {
|
||||||
|
// We are not waiting on created files to be closed, so when
|
||||||
|
// a delete happens we act on it.
|
||||||
|
got_rescan_event = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (fds[1].revents & POLLIN) {
|
if (fds[1].revents & POLLIN) {
|
||||||
got_rescan_event = true;
|
got_rescan_event = true;
|
||||||
|
@ -786,40 +844,29 @@ static void device_thread_run(void *arg) {
|
||||||
assert(errno != EINVAL);
|
assert(errno != EINVAL);
|
||||||
assert(errno != EIO);
|
assert(errno != EIO);
|
||||||
assert(errno != EISDIR);
|
assert(errno != EISDIR);
|
||||||
|
if (errno == EBADF || errno == EFAULT || errno == EINVAL ||
|
||||||
|
errno == EIO || errno == EISDIR)
|
||||||
|
{
|
||||||
|
shutdown_backend(si, SoundIoErrorSystemResources);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if (len <= 0)
|
if (len <= 0)
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (got_rescan_event) {
|
if (got_rescan_event) {
|
||||||
err = refresh_devices(si);
|
if ((err = refresh_devices(si))) {
|
||||||
if (err)
|
|
||||||
shutdown_backend(si, err);
|
shutdown_backend(si, err);
|
||||||
if (!sia->have_devices_flag.exchange(true)) {
|
|
||||||
soundio_os_mutex_lock(sia->mutex);
|
|
||||||
soundio_os_cond_signal(sia->cond, sia->mutex);
|
|
||||||
soundio->on_events_signal(soundio);
|
|
||||||
soundio_os_mutex_unlock(sia->mutex);
|
|
||||||
}
|
|
||||||
if (err)
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void block_until_have_devices(SoundIoAlsa *sia) {
|
|
||||||
if (sia->have_devices_flag.load())
|
|
||||||
return;
|
|
||||||
soundio_os_mutex_lock(sia->mutex);
|
|
||||||
while (!sia->have_devices_flag.load())
|
|
||||||
soundio_os_cond_wait(sia->cond, sia->mutex);
|
|
||||||
soundio_os_mutex_unlock(sia->mutex);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void flush_events(SoundIoPrivate *si) {
|
static void my_flush_events(SoundIoPrivate *si, bool wait) {
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoAlsa *sia = &si->backend_data.alsa;
|
SoundIoAlsa *sia = &si->backend_data.alsa;
|
||||||
block_until_have_devices(sia);
|
|
||||||
|
|
||||||
bool change = false;
|
bool change = false;
|
||||||
bool cb_shutdown = false;
|
bool cb_shutdown = false;
|
||||||
|
@ -827,6 +874,12 @@ static void flush_events(SoundIoPrivate *si) {
|
||||||
|
|
||||||
soundio_os_mutex_lock(sia->mutex);
|
soundio_os_mutex_lock(sia->mutex);
|
||||||
|
|
||||||
|
// block until have devices
|
||||||
|
while (wait || (!sia->have_devices_flag && !sia->shutdown_err)) {
|
||||||
|
soundio_os_cond_wait(sia->cond, sia->mutex);
|
||||||
|
wait = false;
|
||||||
|
}
|
||||||
|
|
||||||
if (sia->shutdown_err && !sia->emitted_shutdown_cb) {
|
if (sia->shutdown_err && !sia->emitted_shutdown_cb) {
|
||||||
sia->emitted_shutdown_cb = true;
|
sia->emitted_shutdown_cb = true;
|
||||||
cb_shutdown = true;
|
cb_shutdown = true;
|
||||||
|
@ -847,15 +900,16 @@ static void flush_events(SoundIoPrivate *si) {
|
||||||
soundio_destroy_devices_info(old_devices_info);
|
soundio_destroy_devices_info(old_devices_info);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wait_events(SoundIoPrivate *si) {
|
static void flush_events_alsa(SoundIoPrivate *si) {
|
||||||
SoundIoAlsa *sia = &si->backend_data.alsa;
|
my_flush_events(si, false);
|
||||||
flush_events(si);
|
|
||||||
soundio_os_mutex_lock(sia->mutex);
|
|
||||||
soundio_os_cond_wait(sia->cond, sia->mutex);
|
|
||||||
soundio_os_mutex_unlock(sia->mutex);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wakeup(SoundIoPrivate *si) {
|
static void wait_events_alsa(SoundIoPrivate *si) {
|
||||||
|
my_flush_events(si, false);
|
||||||
|
my_flush_events(si, true);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void wakeup_alsa(SoundIoPrivate *si) {
|
||||||
SoundIoAlsa *sia = &si->backend_data.alsa;
|
SoundIoAlsa *sia = &si->backend_data.alsa;
|
||||||
soundio_os_mutex_lock(sia->mutex);
|
soundio_os_mutex_lock(sia->mutex);
|
||||||
soundio_os_cond_signal(sia->cond, sia->mutex);
|
soundio_os_cond_signal(sia->cond, sia->mutex);
|
||||||
|
@ -899,10 +953,12 @@ static int os_xrun_recovery(SoundIoOutStreamPrivate *os, int err) {
|
||||||
}
|
}
|
||||||
|
|
||||||
static int instream_xrun_recovery(SoundIoInStreamPrivate *is, int err) {
|
static int instream_xrun_recovery(SoundIoInStreamPrivate *is, int err) {
|
||||||
|
SoundIoInStream *instream = &is->pub;
|
||||||
SoundIoInStreamAlsa *isa = &is->backend_data.alsa;
|
SoundIoInStreamAlsa *isa = &is->backend_data.alsa;
|
||||||
// TODO do something with this overflow
|
|
||||||
if (err == -EPIPE) {
|
if (err == -EPIPE) {
|
||||||
err = snd_pcm_prepare(isa->handle);
|
err = snd_pcm_prepare(isa->handle);
|
||||||
|
if (err >= 0)
|
||||||
|
instream->overflow_callback(instream);
|
||||||
} else if (err == -ESTRPIPE) {
|
} else if (err == -ESTRPIPE) {
|
||||||
while ((err = snd_pcm_resume(isa->handle)) == -EAGAIN) {
|
while ((err = snd_pcm_resume(isa->handle)) == -EAGAIN) {
|
||||||
// wait until suspend flag is released
|
// wait until suspend flag is released
|
||||||
|
@ -910,6 +966,8 @@ static int instream_xrun_recovery(SoundIoInStreamPrivate *is, int err) {
|
||||||
}
|
}
|
||||||
if (err < 0)
|
if (err < 0)
|
||||||
err = snd_pcm_prepare(isa->handle);
|
err = snd_pcm_prepare(isa->handle);
|
||||||
|
if (err >= 0)
|
||||||
|
instream->overflow_callback(instream);
|
||||||
}
|
}
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
@ -1651,7 +1709,6 @@ int soundio_alsa_init(SoundIoPrivate *si) {
|
||||||
|
|
||||||
sia->notify_fd = -1;
|
sia->notify_fd = -1;
|
||||||
sia->notify_wd = -1;
|
sia->notify_wd = -1;
|
||||||
sia->have_devices_flag.store(false);
|
|
||||||
sia->abort_flag.test_and_set();
|
sia->abort_flag.test_and_set();
|
||||||
|
|
||||||
sia->mutex = soundio_os_mutex_create();
|
sia->mutex = soundio_os_mutex_create();
|
||||||
|
@ -1681,7 +1738,7 @@ int soundio_alsa_init(SoundIoPrivate *si) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
sia->notify_wd = inotify_add_watch(sia->notify_fd, "/dev/snd", IN_CREATE | IN_DELETE);
|
sia->notify_wd = inotify_add_watch(sia->notify_fd, "/dev/snd", IN_CREATE | IN_CLOSE_WRITE | IN_DELETE);
|
||||||
if (sia->notify_wd == -1) {
|
if (sia->notify_wd == -1) {
|
||||||
err = errno;
|
err = errno;
|
||||||
assert(err != EACCES);
|
assert(err != EACCES);
|
||||||
|
@ -1714,9 +1771,9 @@ int soundio_alsa_init(SoundIoPrivate *si) {
|
||||||
}
|
}
|
||||||
|
|
||||||
si->destroy = destroy_alsa;
|
si->destroy = destroy_alsa;
|
||||||
si->flush_events = flush_events;
|
si->flush_events = flush_events_alsa;
|
||||||
si->wait_events = wait_events;
|
si->wait_events = wait_events_alsa;
|
||||||
si->wakeup = wakeup;
|
si->wakeup = wakeup_alsa;
|
||||||
|
|
||||||
si->outstream_open = outstream_open_alsa;
|
si->outstream_open = outstream_open_alsa;
|
||||||
si->outstream_destroy = outstream_destroy_alsa;
|
si->outstream_destroy = outstream_destroy_alsa;
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
#include "soundio_private.h"
|
#include "soundio_private.h"
|
||||||
#include "os.h"
|
#include "os.h"
|
||||||
#include "atomics.hpp"
|
#include "atomics.hpp"
|
||||||
|
#include "list.hpp"
|
||||||
|
|
||||||
#include <alsa/asoundlib.h>
|
#include <alsa/asoundlib.h>
|
||||||
|
|
||||||
|
@ -18,6 +19,11 @@ int soundio_alsa_init(struct SoundIoPrivate *si);
|
||||||
|
|
||||||
struct SoundIoDeviceAlsa { };
|
struct SoundIoDeviceAlsa { };
|
||||||
|
|
||||||
|
#define SOUNDIO_MAX_ALSA_SND_FILE_LEN 16
|
||||||
|
struct SoundIoAlsaPendingFile {
|
||||||
|
char name[SOUNDIO_MAX_ALSA_SND_FILE_LEN];
|
||||||
|
};
|
||||||
|
|
||||||
struct SoundIoAlsa {
|
struct SoundIoAlsa {
|
||||||
SoundIoOsMutex *mutex;
|
SoundIoOsMutex *mutex;
|
||||||
SoundIoOsCond *cond;
|
SoundIoOsCond *cond;
|
||||||
|
@ -26,8 +32,9 @@ struct SoundIoAlsa {
|
||||||
atomic_flag abort_flag;
|
atomic_flag abort_flag;
|
||||||
int notify_fd;
|
int notify_fd;
|
||||||
int notify_wd;
|
int notify_wd;
|
||||||
atomic_bool have_devices_flag;
|
bool have_devices_flag;
|
||||||
int notify_pipe_fd[2];
|
int notify_pipe_fd[2];
|
||||||
|
SoundIoList<SoundIoAlsaPendingFile> pending_files;
|
||||||
|
|
||||||
// this one is ready to be read with flush_events. protected by mutex
|
// this one is ready to be read with flush_events. protected by mutex
|
||||||
struct SoundIoDevicesInfo *ready_devices_info;
|
struct SoundIoDevicesInfo *ready_devices_info;
|
||||||
|
|
|
@ -754,7 +754,6 @@ static int refresh_devices(struct SoundIoPrivate *si) {
|
||||||
soundio_os_mutex_lock(sica->mutex);
|
soundio_os_mutex_lock(sica->mutex);
|
||||||
soundio_destroy_devices_info(sica->ready_devices_info);
|
soundio_destroy_devices_info(sica->ready_devices_info);
|
||||||
sica->ready_devices_info = rd.devices_info;
|
sica->ready_devices_info = rd.devices_info;
|
||||||
soundio->on_events_signal(soundio);
|
|
||||||
soundio_os_mutex_unlock(sica->mutex);
|
soundio_os_mutex_unlock(sica->mutex);
|
||||||
|
|
||||||
rd.devices_info = nullptr;
|
rd.devices_info = nullptr;
|
||||||
|
@ -769,21 +768,20 @@ static void shutdown_backend(SoundIoPrivate *si, int err) {
|
||||||
SoundIoCoreAudio *sica = &si->backend_data.coreaudio;
|
SoundIoCoreAudio *sica = &si->backend_data.coreaudio;
|
||||||
soundio_os_mutex_lock(sica->mutex);
|
soundio_os_mutex_lock(sica->mutex);
|
||||||
sica->shutdown_err = err;
|
sica->shutdown_err = err;
|
||||||
soundio->on_events_signal(soundio);
|
sica->have_devices_flag.store(true);
|
||||||
soundio_os_mutex_unlock(sica->mutex);
|
soundio_os_mutex_unlock(sica->mutex);
|
||||||
}
|
soundio_os_cond_signal(sica->cond, nullptr);
|
||||||
|
soundio_os_cond_signal(sica->have_devices_cond, nullptr);
|
||||||
static void block_until_have_devices(SoundIoCoreAudio *sica) {
|
soundio->on_events_signal(soundio);
|
||||||
if (sica->have_devices_flag.load())
|
|
||||||
return;
|
|
||||||
while (!sica->have_devices_flag.load())
|
|
||||||
soundio_os_cond_wait(sica->have_devices_cond, nullptr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void flush_events_ca(struct SoundIoPrivate *si) {
|
static void flush_events_ca(struct SoundIoPrivate *si) {
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoCoreAudio *sica = &si->backend_data.coreaudio;
|
SoundIoCoreAudio *sica = &si->backend_data.coreaudio;
|
||||||
block_until_have_devices(sica);
|
|
||||||
|
// block until have devices
|
||||||
|
while (!sica->have_devices_flag.load())
|
||||||
|
soundio_os_cond_wait(sica->have_devices_cond, nullptr);
|
||||||
|
|
||||||
bool change = false;
|
bool change = false;
|
||||||
bool cb_shutdown = false;
|
bool cb_shutdown = false;
|
||||||
|
@ -837,15 +835,14 @@ static void device_thread_run(void *arg) {
|
||||||
}
|
}
|
||||||
if (sica->device_scan_queued.exchange(false)) {
|
if (sica->device_scan_queued.exchange(false)) {
|
||||||
err = refresh_devices(si);
|
err = refresh_devices(si);
|
||||||
if (err)
|
if (err) {
|
||||||
shutdown_backend(si, err);
|
shutdown_backend(si, err);
|
||||||
if (!sica->have_devices_flag.exchange(true)) {
|
|
||||||
soundio_os_cond_signal(sica->have_devices_cond, nullptr);
|
|
||||||
soundio->on_events_signal(soundio);
|
|
||||||
}
|
|
||||||
if (err)
|
|
||||||
return;
|
return;
|
||||||
|
}
|
||||||
|
if (!sica->have_devices_flag.exchange(true))
|
||||||
|
soundio_os_cond_signal(sica->have_devices_cond, nullptr);
|
||||||
soundio_os_cond_signal(sica->cond, nullptr);
|
soundio_os_cond_signal(sica->cond, nullptr);
|
||||||
|
soundio->on_events_signal(soundio);
|
||||||
}
|
}
|
||||||
soundio_os_cond_wait(sica->scan_devices_cond, nullptr);
|
soundio_os_cond_wait(sica->scan_devices_cond, nullptr);
|
||||||
}
|
}
|
||||||
|
@ -1053,10 +1050,9 @@ static int outstream_clear_buffer_ca(struct SoundIoPrivate *si, struct SoundIoOu
|
||||||
static OSStatus on_instream_device_overload(AudioObjectID in_object_id, UInt32 in_number_addresses,
|
static OSStatus on_instream_device_overload(AudioObjectID in_object_id, UInt32 in_number_addresses,
|
||||||
const AudioObjectPropertyAddress in_addresses[], void *in_client_data)
|
const AudioObjectPropertyAddress in_addresses[], void *in_client_data)
|
||||||
{
|
{
|
||||||
//SoundIoInStreamPrivate *os = (SoundIoInStreamPrivate *)in_client_data;
|
SoundIoInStreamPrivate *os = (SoundIoInStreamPrivate *)in_client_data;
|
||||||
//SoundIoInStream *instream = &os->pub;
|
SoundIoInStream *instream = &os->pub;
|
||||||
fprintf(stderr, "TODO overflow\n");
|
instream->overflow_callback(instream);
|
||||||
//instream->underflow_callback(instream);
|
|
||||||
return noErr;
|
return noErr;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -88,7 +88,7 @@ static void capture_thread_run(void *arg) {
|
||||||
frames_consumed += write_count;
|
frames_consumed += write_count;
|
||||||
|
|
||||||
if (frames_to_kill > free_frames) {
|
if (frames_to_kill > free_frames) {
|
||||||
// TODO overflow callback
|
instream->overflow_callback(instream);
|
||||||
frames_consumed = 0;
|
frames_consumed = 0;
|
||||||
start_time = soundio_os_get_time();
|
start_time = soundio_os_get_time();
|
||||||
}
|
}
|
||||||
|
@ -109,7 +109,7 @@ static void destroy_dummy(SoundIoPrivate *si) {
|
||||||
soundio_os_mutex_destroy(sid->mutex);
|
soundio_os_mutex_destroy(sid->mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void flush_events(SoundIoPrivate *si) {
|
static void flush_events_dummy(SoundIoPrivate *si) {
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoDummy *sid = &si->backend_data.dummy;
|
SoundIoDummy *sid = &si->backend_data.dummy;
|
||||||
if (sid->devices_emitted)
|
if (sid->devices_emitted)
|
||||||
|
@ -118,13 +118,13 @@ static void flush_events(SoundIoPrivate *si) {
|
||||||
soundio->on_devices_change(soundio);
|
soundio->on_devices_change(soundio);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wait_events(SoundIoPrivate *si) {
|
static void wait_events_dummy(SoundIoPrivate *si) {
|
||||||
SoundIoDummy *sid = &si->backend_data.dummy;
|
SoundIoDummy *sid = &si->backend_data.dummy;
|
||||||
flush_events(si);
|
flush_events_dummy(si);
|
||||||
soundio_os_cond_wait(sid->cond, nullptr);
|
soundio_os_cond_wait(sid->cond, nullptr);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wakeup(SoundIoPrivate *si) {
|
static void wakeup_dummy(SoundIoPrivate *si) {
|
||||||
SoundIoDummy *sid = &si->backend_data.dummy;
|
SoundIoDummy *sid = &si->backend_data.dummy;
|
||||||
soundio_os_cond_signal(sid->cond, nullptr);
|
soundio_os_cond_signal(sid->cond, nullptr);
|
||||||
}
|
}
|
||||||
|
@ -505,9 +505,9 @@ int soundio_dummy_init(SoundIoPrivate *si) {
|
||||||
|
|
||||||
|
|
||||||
si->destroy = destroy_dummy;
|
si->destroy = destroy_dummy;
|
||||||
si->flush_events = flush_events;
|
si->flush_events = flush_events_dummy;
|
||||||
si->wait_events = wait_events;
|
si->wait_events = wait_events_dummy;
|
||||||
si->wakeup = wakeup;
|
si->wakeup = wakeup_dummy;
|
||||||
|
|
||||||
si->outstream_open = outstream_open_dummy;
|
si->outstream_open = outstream_open_dummy;
|
||||||
si->outstream_destroy = outstream_destroy_dummy;
|
si->outstream_destroy = outstream_destroy_dummy;
|
||||||
|
|
23
src/jack.cpp
23
src/jack.cpp
|
@ -291,7 +291,7 @@ static int refresh_devices(SoundIoPrivate *si) {
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void flush_events_jack(struct SoundIoPrivate *si) {
|
static void my_flush_events(struct SoundIoPrivate *si, bool wait) {
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoJack *sij = &si->backend_data.jack;
|
SoundIoJack *sij = &si->backend_data.jack;
|
||||||
int err;
|
int err;
|
||||||
|
@ -300,6 +300,9 @@ static void flush_events_jack(struct SoundIoPrivate *si) {
|
||||||
|
|
||||||
soundio_os_mutex_lock(sij->mutex);
|
soundio_os_mutex_lock(sij->mutex);
|
||||||
|
|
||||||
|
if (wait)
|
||||||
|
soundio_os_cond_wait(sij->cond, sij->mutex);
|
||||||
|
|
||||||
if (sij->is_shutdown && !sij->emitted_shutdown_cb) {
|
if (sij->is_shutdown && !sij->emitted_shutdown_cb) {
|
||||||
sij->emitted_shutdown_cb = true;
|
sij->emitted_shutdown_cb = true;
|
||||||
cb_shutdown = true;
|
cb_shutdown = true;
|
||||||
|
@ -320,12 +323,13 @@ static void flush_events_jack(struct SoundIoPrivate *si) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void flush_events_jack(struct SoundIoPrivate *si) {
|
||||||
|
my_flush_events(si, false);
|
||||||
|
}
|
||||||
|
|
||||||
static void wait_events_jack(struct SoundIoPrivate *si) {
|
static void wait_events_jack(struct SoundIoPrivate *si) {
|
||||||
SoundIoJack *sij = &si->backend_data.jack;
|
my_flush_events(si, false);
|
||||||
flush_events_jack(si);
|
my_flush_events(si, true);
|
||||||
soundio_os_mutex_lock(sij->mutex);
|
|
||||||
soundio_os_cond_wait(sij->cond, sij->mutex);
|
|
||||||
soundio_os_mutex_unlock(sij->mutex);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wakeup_jack(struct SoundIoPrivate *si) {
|
static void wakeup_jack(struct SoundIoPrivate *si) {
|
||||||
|
@ -559,9 +563,9 @@ static void instream_destroy_jack(struct SoundIoPrivate *si, struct SoundIoInStr
|
||||||
}
|
}
|
||||||
|
|
||||||
static int instream_xrun_callback(void *arg) {
|
static int instream_xrun_callback(void *arg) {
|
||||||
// SoundIoInStreamPrivate *is = (SoundIoInStreamPrivate *)arg;
|
SoundIoInStreamPrivate *is = (SoundIoInStreamPrivate *)arg;
|
||||||
// SoundIoInStream *instream = &is->pub;
|
SoundIoInStream *instream = &is->pub;
|
||||||
// TODO do something with this overflow
|
instream->overflow_callback(instream);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -794,6 +798,7 @@ static void shutdown_callback(void *arg) {
|
||||||
SoundIoJack *sij = &si->backend_data.jack;
|
SoundIoJack *sij = &si->backend_data.jack;
|
||||||
soundio_os_mutex_lock(sij->mutex);
|
soundio_os_mutex_lock(sij->mutex);
|
||||||
sij->is_shutdown = true;
|
sij->is_shutdown = true;
|
||||||
|
soundio_os_cond_signal(sij->cond, sij->mutex);
|
||||||
soundio->on_events_signal(soundio);
|
soundio->on_events_signal(soundio);
|
||||||
soundio_os_mutex_unlock(sij->mutex);
|
soundio_os_mutex_unlock(sij->mutex);
|
||||||
}
|
}
|
||||||
|
|
11
src/list.hpp
11
src/list.hpp
|
@ -83,6 +83,17 @@ struct SoundIoList {
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
T swap_remove(int index) {
|
||||||
|
assert(index >= 0);
|
||||||
|
assert(index < length);
|
||||||
|
T last = pop();
|
||||||
|
if (index == length)
|
||||||
|
return last;
|
||||||
|
T item = items[index];
|
||||||
|
items[index] = last;
|
||||||
|
return item;
|
||||||
|
}
|
||||||
|
|
||||||
T * items;
|
T * items;
|
||||||
int length;
|
int length;
|
||||||
int capacity;
|
int capacity;
|
||||||
|
|
|
@ -17,9 +17,11 @@ static void subscribe_callback(pa_context *context,
|
||||||
pa_subscription_event_type_t event_bits, uint32_t index, void *userdata)
|
pa_subscription_event_type_t event_bits, uint32_t index, void *userdata)
|
||||||
{
|
{
|
||||||
SoundIoPrivate *si = (SoundIoPrivate *)userdata;
|
SoundIoPrivate *si = (SoundIoPrivate *)userdata;
|
||||||
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
||||||
sipa->device_scan_queued = true;
|
sipa->device_scan_queued = true;
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
||||||
|
soundio->on_events_signal(soundio);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int subscribe_to_events(SoundIoPrivate *si) {
|
static int subscribe_to_events(SoundIoPrivate *si) {
|
||||||
|
@ -55,11 +57,14 @@ static void context_state_callback(pa_context *context, void *userdata) {
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
||||||
return;
|
return;
|
||||||
case PA_CONTEXT_FAILED: // The connection failed or was disconnected.
|
case PA_CONTEXT_FAILED: // The connection failed or was disconnected.
|
||||||
|
if (sipa->ready_flag) {
|
||||||
|
sipa->connection_err = SoundIoErrorBackendDisconnected;
|
||||||
|
} else {
|
||||||
sipa->connection_err = SoundIoErrorInitAudioBackend;
|
sipa->connection_err = SoundIoErrorInitAudioBackend;
|
||||||
if (sipa->ready_flag.exchange(true)) {
|
sipa->ready_flag = true;
|
||||||
soundio->on_backend_disconnect(soundio, SoundIoErrorBackendDisconnected);
|
|
||||||
}
|
}
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
||||||
|
soundio->on_events_signal(soundio);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -70,12 +75,12 @@ static void destroy_pa(SoundIoPrivate *si) {
|
||||||
if (sipa->main_loop)
|
if (sipa->main_loop)
|
||||||
pa_threaded_mainloop_stop(sipa->main_loop);
|
pa_threaded_mainloop_stop(sipa->main_loop);
|
||||||
|
|
||||||
soundio_destroy_devices_info(sipa->current_devices_info);
|
|
||||||
soundio_destroy_devices_info(sipa->ready_devices_info);
|
|
||||||
|
|
||||||
pa_context_disconnect(sipa->pulse_context);
|
pa_context_disconnect(sipa->pulse_context);
|
||||||
pa_context_unref(sipa->pulse_context);
|
pa_context_unref(sipa->pulse_context);
|
||||||
|
|
||||||
|
soundio_destroy_devices_info(sipa->current_devices_info);
|
||||||
|
soundio_destroy_devices_info(sipa->ready_devices_info);
|
||||||
|
|
||||||
if (sipa->main_loop)
|
if (sipa->main_loop)
|
||||||
pa_threaded_mainloop_free(sipa->main_loop);
|
pa_threaded_mainloop_free(sipa->main_loop);
|
||||||
|
|
||||||
|
@ -214,24 +219,193 @@ static int perform_operation(SoundIoPrivate *si, pa_operation *op) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void finish_device_query(SoundIoPrivate *si) {
|
static void sink_info_callback(pa_context *pulse_context, const pa_sink_info *info, int eol, void *userdata) {
|
||||||
|
SoundIoPrivate *si = (SoundIoPrivate *)userdata;
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
||||||
|
int err;
|
||||||
|
if (eol) {
|
||||||
|
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (sipa->device_query_err)
|
||||||
|
return;
|
||||||
|
|
||||||
if (!sipa->have_sink_list ||
|
SoundIoDevicePrivate *dev = allocate<SoundIoDevicePrivate>(1);
|
||||||
!sipa->have_source_list ||
|
if (!dev) {
|
||||||
!sipa->have_default_sink)
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
{
|
return;
|
||||||
|
}
|
||||||
|
SoundIoDevice *device = &dev->pub;
|
||||||
|
|
||||||
|
device->ref_count = 1;
|
||||||
|
device->soundio = soundio;
|
||||||
|
device->id = strdup(info->name);
|
||||||
|
device->name = strdup(info->description);
|
||||||
|
if (!device->id || !device->name) {
|
||||||
|
soundio_device_unref(device);
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (sipa->device_query_err) {
|
device->sample_rate_current = info->sample_spec.rate;
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
// PulseAudio performs resampling, so any value is valid. Let's pick
|
||||||
soundio->on_events_signal(soundio);
|
// some reasonable min and max values.
|
||||||
soundio->on_backend_disconnect(soundio, sipa->device_query_err);
|
device->sample_rate_count = 1;
|
||||||
|
device->sample_rates = &dev->prealloc_sample_rate_range;
|
||||||
|
device->sample_rates[0].min = min(SOUNDIO_MIN_SAMPLE_RATE, device->sample_rate_current);
|
||||||
|
device->sample_rates[0].max = max(SOUNDIO_MAX_SAMPLE_RATE, device->sample_rate_current);
|
||||||
|
|
||||||
|
device->current_format = from_pulseaudio_format(info->sample_spec);
|
||||||
|
// PulseAudio performs sample format conversion, so any PulseAudio
|
||||||
|
// value is valid.
|
||||||
|
if ((err = set_all_device_formats(device))) {
|
||||||
|
soundio_device_unref(device);
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
set_from_pulseaudio_channel_map(info->channel_map, &device->current_layout);
|
||||||
|
// PulseAudio does channel layout remapping, so any channel layout is valid.
|
||||||
|
if ((err = set_all_device_channel_layouts(device))) {
|
||||||
|
soundio_device_unref(device);
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
device->aim = SoundIoDeviceAimOutput;
|
||||||
|
|
||||||
|
if (sipa->current_devices_info->output_devices.append(device)) {
|
||||||
|
soundio_device_unref(device);
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void source_info_callback(pa_context *pulse_context, const pa_source_info *info, int eol, void *userdata) {
|
||||||
|
SoundIoPrivate *si = (SoundIoPrivate *)userdata;
|
||||||
|
SoundIo *soundio = &si->pub;
|
||||||
|
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (eol) {
|
||||||
|
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (sipa->device_query_err)
|
||||||
|
return;
|
||||||
|
|
||||||
|
SoundIoDevicePrivate *dev = allocate<SoundIoDevicePrivate>(1);
|
||||||
|
if (!dev) {
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
SoundIoDevice *device = &dev->pub;
|
||||||
|
|
||||||
|
device->ref_count = 1;
|
||||||
|
device->soundio = soundio;
|
||||||
|
device->id = strdup(info->name);
|
||||||
|
device->name = strdup(info->description);
|
||||||
|
if (!device->id || !device->name) {
|
||||||
|
soundio_device_unref(device);
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
device->sample_rate_current = info->sample_spec.rate;
|
||||||
|
// PulseAudio performs resampling, so any value is valid. Let's pick
|
||||||
|
// some reasonable min and max values.
|
||||||
|
device->sample_rate_count = 1;
|
||||||
|
device->sample_rates = &dev->prealloc_sample_rate_range;
|
||||||
|
device->sample_rates[0].min = min(8000, device->sample_rate_current);
|
||||||
|
device->sample_rates[0].max = max(5644800, device->sample_rate_current);
|
||||||
|
|
||||||
|
device->current_format = from_pulseaudio_format(info->sample_spec);
|
||||||
|
// PulseAudio performs sample format conversion, so any PulseAudio
|
||||||
|
// value is valid.
|
||||||
|
if ((err = set_all_device_formats(device))) {
|
||||||
|
soundio_device_unref(device);
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
set_from_pulseaudio_channel_map(info->channel_map, &device->current_layout);
|
||||||
|
// PulseAudio does channel layout remapping, so any channel layout is valid.
|
||||||
|
if ((err = set_all_device_channel_layouts(device))) {
|
||||||
|
soundio_device_unref(device);
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
device->aim = SoundIoDeviceAimInput;
|
||||||
|
|
||||||
|
if (sipa->current_devices_info->input_devices.append(device)) {
|
||||||
|
soundio_device_unref(device);
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void server_info_callback(pa_context *pulse_context, const pa_server_info *info, void *userdata) {
|
||||||
|
SoundIoPrivate *si = (SoundIoPrivate *)userdata;
|
||||||
|
assert(si);
|
||||||
|
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
||||||
|
|
||||||
|
assert(!sipa->default_sink_name);
|
||||||
|
assert(!sipa->default_source_name);
|
||||||
|
|
||||||
|
sipa->default_sink_name = strdup(info->default_sink_name);
|
||||||
|
sipa->default_source_name = strdup(info->default_source_name);
|
||||||
|
|
||||||
|
if (!sipa->default_sink_name || !sipa->default_source_name)
|
||||||
|
sipa->device_query_err = SoundIoErrorNoMem;
|
||||||
|
|
||||||
|
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
// always called even when refresh_devices succeeds
|
||||||
|
static void cleanup_refresh_devices(SoundIoPrivate *si) {
|
||||||
|
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
||||||
|
|
||||||
|
soundio_destroy_devices_info(sipa->current_devices_info);
|
||||||
|
sipa->current_devices_info = nullptr;
|
||||||
|
|
||||||
|
free(sipa->default_sink_name);
|
||||||
|
sipa->default_sink_name = nullptr;
|
||||||
|
|
||||||
|
free(sipa->default_source_name);
|
||||||
|
sipa->default_source_name = nullptr;
|
||||||
|
}
|
||||||
|
|
||||||
|
// call this while holding the main loop lock
|
||||||
|
static int refresh_devices(SoundIoPrivate *si) {
|
||||||
|
SoundIo *soundio = &si->pub;
|
||||||
|
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
||||||
|
|
||||||
|
assert(!sipa->current_devices_info);
|
||||||
|
sipa->current_devices_info = allocate<SoundIoDevicesInfo>(1);
|
||||||
|
if (!sipa->current_devices_info)
|
||||||
|
return SoundIoErrorNoMem;
|
||||||
|
|
||||||
|
pa_operation *list_sink_op = pa_context_get_sink_info_list(sipa->pulse_context, sink_info_callback, si);
|
||||||
|
pa_operation *list_source_op = pa_context_get_source_info_list(sipa->pulse_context, source_info_callback, si);
|
||||||
|
pa_operation *server_info_op = pa_context_get_server_info(sipa->pulse_context, server_info_callback, si);
|
||||||
|
|
||||||
|
int err;
|
||||||
|
if ((err = perform_operation(si, list_sink_op))) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
if ((err = perform_operation(si, list_source_op))) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
if ((err = perform_operation(si, server_info_op))) {
|
||||||
|
return err;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sipa->device_query_err) {
|
||||||
|
return sipa->device_query_err;
|
||||||
|
}
|
||||||
|
|
||||||
// based on the default sink name, figure out the default output index
|
// based on the default sink name, figure out the default output index
|
||||||
// if the name doesn't match just pick the first one. if there are no
|
// if the name doesn't match just pick the first one. if there are no
|
||||||
// devices then we need to set it to -1.
|
// devices then we need to set it to -1.
|
||||||
|
@ -262,246 +436,36 @@ static void finish_device_query(SoundIoPrivate *si) {
|
||||||
|
|
||||||
soundio_destroy_devices_info(sipa->ready_devices_info);
|
soundio_destroy_devices_info(sipa->ready_devices_info);
|
||||||
sipa->ready_devices_info = sipa->current_devices_info;
|
sipa->ready_devices_info = sipa->current_devices_info;
|
||||||
sipa->current_devices_info = NULL;
|
sipa->current_devices_info = nullptr;
|
||||||
sipa->have_devices_flag = true;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
||||||
soundio->on_events_signal(soundio);
|
soundio->on_events_signal(soundio);
|
||||||
}
|
|
||||||
|
|
||||||
static void sink_info_callback(pa_context *pulse_context, const pa_sink_info *info, int eol, void *userdata) {
|
|
||||||
SoundIoPrivate *si = (SoundIoPrivate *)userdata;
|
|
||||||
SoundIo *soundio = &si->pub;
|
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
|
||||||
int err;
|
|
||||||
if (eol) {
|
|
||||||
sipa->have_sink_list = true;
|
|
||||||
finish_device_query(si);
|
|
||||||
} else if (!sipa->device_query_err) {
|
|
||||||
SoundIoDevicePrivate *dev = allocate<SoundIoDevicePrivate>(1);
|
|
||||||
if (!dev) {
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
SoundIoDevice *device = &dev->pub;
|
|
||||||
|
|
||||||
device->ref_count = 1;
|
|
||||||
device->soundio = soundio;
|
|
||||||
device->id = strdup(info->name);
|
|
||||||
device->name = strdup(info->description);
|
|
||||||
if (!device->id || !device->name) {
|
|
||||||
soundio_device_unref(device);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
device->sample_rate_current = info->sample_spec.rate;
|
|
||||||
// PulseAudio performs resampling, so any value is valid. Let's pick
|
|
||||||
// some reasonable min and max values.
|
|
||||||
device->sample_rate_count = 1;
|
|
||||||
device->sample_rates = &dev->prealloc_sample_rate_range;
|
|
||||||
device->sample_rates[0].min = min(SOUNDIO_MIN_SAMPLE_RATE, device->sample_rate_current);
|
|
||||||
device->sample_rates[0].max = max(SOUNDIO_MAX_SAMPLE_RATE, device->sample_rate_current);
|
|
||||||
|
|
||||||
device->current_format = from_pulseaudio_format(info->sample_spec);
|
|
||||||
// PulseAudio performs sample format conversion, so any PulseAudio
|
|
||||||
// value is valid.
|
|
||||||
if ((err = set_all_device_formats(device))) {
|
|
||||||
soundio_device_unref(device);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
set_from_pulseaudio_channel_map(info->channel_map, &device->current_layout);
|
|
||||||
// PulseAudio does channel layout remapping, so any channel layout is valid.
|
|
||||||
if ((err = set_all_device_channel_layouts(device))) {
|
|
||||||
soundio_device_unref(device);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
device->aim = SoundIoDeviceAimOutput;
|
|
||||||
|
|
||||||
if (sipa->current_devices_info->output_devices.append(device)) {
|
|
||||||
soundio_device_unref(device);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void source_info_callback(pa_context *pulse_context, const pa_source_info *info, int eol, void *userdata) {
|
|
||||||
SoundIoPrivate *si = (SoundIoPrivate *)userdata;
|
|
||||||
SoundIo *soundio = &si->pub;
|
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
|
||||||
int err;
|
|
||||||
if (eol) {
|
|
||||||
sipa->have_source_list = true;
|
|
||||||
finish_device_query(si);
|
|
||||||
} else if (!sipa->device_query_err) {
|
|
||||||
SoundIoDevicePrivate *dev = allocate<SoundIoDevicePrivate>(1);
|
|
||||||
if (!dev) {
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
SoundIoDevice *device = &dev->pub;
|
|
||||||
|
|
||||||
device->ref_count = 1;
|
|
||||||
device->soundio = soundio;
|
|
||||||
device->id = strdup(info->name);
|
|
||||||
device->name = strdup(info->description);
|
|
||||||
if (!device->id || !device->name) {
|
|
||||||
soundio_device_unref(device);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
device->sample_rate_current = info->sample_spec.rate;
|
|
||||||
// PulseAudio performs resampling, so any value is valid. Let's pick
|
|
||||||
// some reasonable min and max values.
|
|
||||||
device->sample_rate_count = 1;
|
|
||||||
device->sample_rates = &dev->prealloc_sample_rate_range;
|
|
||||||
device->sample_rates[0].min = min(8000, device->sample_rate_current);
|
|
||||||
device->sample_rates[0].max = max(5644800, device->sample_rate_current);
|
|
||||||
|
|
||||||
device->current_format = from_pulseaudio_format(info->sample_spec);
|
|
||||||
// PulseAudio performs sample format conversion, so any PulseAudio
|
|
||||||
// value is valid.
|
|
||||||
if ((err = set_all_device_formats(device))) {
|
|
||||||
soundio_device_unref(device);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
set_from_pulseaudio_channel_map(info->channel_map, &device->current_layout);
|
|
||||||
// PulseAudio does channel layout remapping, so any channel layout is valid.
|
|
||||||
if ((err = set_all_device_channel_layouts(device))) {
|
|
||||||
soundio_device_unref(device);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
device->aim = SoundIoDeviceAimInput;
|
|
||||||
|
|
||||||
if (sipa->current_devices_info->input_devices.append(device)) {
|
|
||||||
soundio_device_unref(device);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void server_info_callback(pa_context *pulse_context, const pa_server_info *info, void *userdata) {
|
|
||||||
SoundIoPrivate *si = (SoundIoPrivate *)userdata;
|
|
||||||
assert(si);
|
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
|
||||||
|
|
||||||
free(sipa->default_sink_name);
|
|
||||||
free(sipa->default_source_name);
|
|
||||||
|
|
||||||
sipa->default_sink_name = strdup(info->default_sink_name);
|
|
||||||
sipa->default_source_name = strdup(info->default_source_name);
|
|
||||||
|
|
||||||
if (!sipa->default_sink_name || !sipa->default_source_name) {
|
|
||||||
free(sipa->default_sink_name);
|
|
||||||
free(sipa->default_source_name);
|
|
||||||
sipa->device_query_err = SoundIoErrorNoMem;
|
|
||||||
}
|
|
||||||
|
|
||||||
sipa->have_default_sink = true;
|
|
||||||
finish_device_query(si);
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
static int scan_devices(SoundIoPrivate *si) {
|
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
|
||||||
|
|
||||||
sipa->have_sink_list = false;
|
|
||||||
sipa->have_default_sink = false;
|
|
||||||
sipa->have_source_list = false;
|
|
||||||
|
|
||||||
soundio_destroy_devices_info(sipa->current_devices_info);
|
|
||||||
sipa->current_devices_info = allocate<SoundIoDevicesInfo>(1);
|
|
||||||
if (!sipa->current_devices_info)
|
|
||||||
return SoundIoErrorNoMem;
|
|
||||||
|
|
||||||
pa_threaded_mainloop_lock(sipa->main_loop);
|
|
||||||
|
|
||||||
pa_operation *list_sink_op = pa_context_get_sink_info_list(sipa->pulse_context, sink_info_callback, si);
|
|
||||||
pa_operation *list_source_op = pa_context_get_source_info_list(sipa->pulse_context, source_info_callback, si);
|
|
||||||
pa_operation *server_info_op = pa_context_get_server_info(sipa->pulse_context, server_info_callback, si);
|
|
||||||
|
|
||||||
int err;
|
|
||||||
if ((err = perform_operation(si, list_sink_op))) {
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
if ((err = perform_operation(si, list_source_op))) {
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
if ((err = perform_operation(si, server_info_op))) {
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
|
||||||
return err;
|
|
||||||
}
|
|
||||||
|
|
||||||
pa_threaded_mainloop_signal(sipa->main_loop, 0);
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void block_until_have_devices(SoundIoPrivate *si) {
|
static void my_flush_events(SoundIoPrivate *si, bool wait) {
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
|
||||||
if (sipa->have_devices_flag)
|
|
||||||
return;
|
|
||||||
pa_threaded_mainloop_lock(sipa->main_loop);
|
|
||||||
while (!sipa->have_devices_flag && !sipa->device_query_err) {
|
|
||||||
pa_threaded_mainloop_wait(sipa->main_loop);
|
|
||||||
}
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void block_until_ready(SoundIoPrivate *si) {
|
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
|
||||||
if (sipa->ready_flag)
|
|
||||||
return;
|
|
||||||
pa_threaded_mainloop_lock(sipa->main_loop);
|
|
||||||
while (!sipa->ready_flag) {
|
|
||||||
pa_threaded_mainloop_wait(sipa->main_loop);
|
|
||||||
}
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void flush_events(SoundIoPrivate *si) {
|
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
|
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
||||||
|
|
||||||
int err;
|
|
||||||
if (sipa->device_scan_queued) {
|
|
||||||
if (!(err = scan_devices(si)))
|
|
||||||
sipa->device_scan_queued = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
SoundIoDevicesInfo *old_devices_info = nullptr;
|
|
||||||
bool change = false;
|
bool change = false;
|
||||||
|
bool cb_shutdown = false;
|
||||||
|
SoundIoDevicesInfo *old_devices_info = nullptr;
|
||||||
|
|
||||||
pa_threaded_mainloop_lock(sipa->main_loop);
|
pa_threaded_mainloop_lock(sipa->main_loop);
|
||||||
|
|
||||||
if (sipa->ready_devices_info) {
|
if (wait)
|
||||||
|
pa_threaded_mainloop_wait(sipa->main_loop);
|
||||||
|
|
||||||
|
if (sipa->device_scan_queued && !sipa->connection_err) {
|
||||||
|
sipa->device_scan_queued = false;
|
||||||
|
sipa->connection_err = refresh_devices(si);
|
||||||
|
cleanup_refresh_devices(si);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (sipa->connection_err && !sipa->emitted_shutdown_cb) {
|
||||||
|
sipa->emitted_shutdown_cb = true;
|
||||||
|
cb_shutdown = true;
|
||||||
|
} else if (sipa->ready_devices_info) {
|
||||||
old_devices_info = si->safe_devices_info;
|
old_devices_info = si->safe_devices_info;
|
||||||
si->safe_devices_info = sipa->ready_devices_info;
|
si->safe_devices_info = sipa->ready_devices_info;
|
||||||
sipa->ready_devices_info = nullptr;
|
sipa->ready_devices_info = nullptr;
|
||||||
|
@ -510,20 +474,21 @@ static void flush_events(SoundIoPrivate *si) {
|
||||||
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
pa_threaded_mainloop_unlock(sipa->main_loop);
|
||||||
|
|
||||||
if (change)
|
if (cb_shutdown)
|
||||||
|
soundio->on_backend_disconnect(soundio, sipa->connection_err);
|
||||||
|
else if (change)
|
||||||
soundio->on_devices_change(soundio);
|
soundio->on_devices_change(soundio);
|
||||||
|
|
||||||
soundio_destroy_devices_info(old_devices_info);
|
soundio_destroy_devices_info(old_devices_info);
|
||||||
|
|
||||||
block_until_have_devices(si);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wait_events(SoundIoPrivate *si) {
|
static void flush_events_pa(SoundIoPrivate *si) {
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
my_flush_events(si, false);
|
||||||
flush_events(si);
|
}
|
||||||
pa_threaded_mainloop_lock(sipa->main_loop);
|
|
||||||
pa_threaded_mainloop_wait(sipa->main_loop);
|
static void wait_events_pa(SoundIoPrivate *si) {
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
my_flush_events(si, false);
|
||||||
|
my_flush_events(si, true);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wakeup(SoundIoPrivate *si) {
|
static void wakeup(SoundIoPrivate *si) {
|
||||||
|
@ -658,6 +623,7 @@ static void outstream_destroy_pa(SoundIoPrivate *si, SoundIoOutStreamPrivate *os
|
||||||
pa_stream_set_write_callback(stream, nullptr, nullptr);
|
pa_stream_set_write_callback(stream, nullptr, nullptr);
|
||||||
pa_stream_set_state_callback(stream, nullptr, nullptr);
|
pa_stream_set_state_callback(stream, nullptr, nullptr);
|
||||||
pa_stream_set_underflow_callback(stream, nullptr, nullptr);
|
pa_stream_set_underflow_callback(stream, nullptr, nullptr);
|
||||||
|
pa_stream_set_overflow_callback(stream, nullptr, nullptr);
|
||||||
pa_stream_disconnect(stream);
|
pa_stream_disconnect(stream);
|
||||||
|
|
||||||
pa_stream_unref(stream);
|
pa_stream_unref(stream);
|
||||||
|
@ -733,8 +699,6 @@ static int outstream_open_pa(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
||||||
size_t writable_size = pa_stream_writable_size(ospa->stream);
|
size_t writable_size = pa_stream_writable_size(ospa->stream);
|
||||||
outstream->software_latency = writable_size / bytes_per_second;
|
outstream->software_latency = writable_size / bytes_per_second;
|
||||||
|
|
||||||
// TODO get the correct software_latency value
|
|
||||||
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
pa_threaded_mainloop_unlock(sipa->main_loop);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -759,6 +723,7 @@ static int outstream_start_pa(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
||||||
pa_operation_unref(op);
|
pa_operation_unref(op);
|
||||||
pa_stream_set_write_callback(ospa->stream, playback_stream_write_callback, os);
|
pa_stream_set_write_callback(ospa->stream, playback_stream_write_callback, os);
|
||||||
pa_stream_set_underflow_callback(ospa->stream, playback_stream_underflow_callback, outstream);
|
pa_stream_set_underflow_callback(ospa->stream, playback_stream_underflow_callback, outstream);
|
||||||
|
pa_stream_set_overflow_callback(ospa->stream, playback_stream_underflow_callback, outstream);
|
||||||
|
|
||||||
pa_threaded_mainloop_unlock(sipa->main_loop);
|
pa_threaded_mainloop_unlock(sipa->main_loop);
|
||||||
|
|
||||||
|
@ -913,7 +878,6 @@ static int instream_open_pa(SoundIoPrivate *si, SoundIoInStreamPrivate *is) {
|
||||||
|
|
||||||
pa_stream_set_state_callback(stream, recording_stream_state_callback, is);
|
pa_stream_set_state_callback(stream, recording_stream_state_callback, is);
|
||||||
pa_stream_set_read_callback(stream, recording_stream_read_callback, is);
|
pa_stream_set_read_callback(stream, recording_stream_read_callback, is);
|
||||||
// TODO handle overflow callback
|
|
||||||
|
|
||||||
ispa->buffer_attr.maxlength = UINT32_MAX;
|
ispa->buffer_attr.maxlength = UINT32_MAX;
|
||||||
ispa->buffer_attr.tlength = UINT32_MAX;
|
ispa->buffer_attr.tlength = UINT32_MAX;
|
||||||
|
@ -1039,9 +1003,7 @@ int soundio_pulseaudio_init(SoundIoPrivate *si) {
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
SoundIoPulseAudio *sipa = &si->backend_data.pulseaudio;
|
||||||
|
|
||||||
sipa->device_scan_queued.store(false);
|
sipa->device_scan_queued = true;
|
||||||
sipa->ready_flag.store(false);
|
|
||||||
sipa->have_devices_flag.store(false);
|
|
||||||
|
|
||||||
sipa->main_loop = pa_threaded_mainloop_new();
|
sipa->main_loop = pa_threaded_mainloop_new();
|
||||||
if (!sipa->main_loop) {
|
if (!sipa->main_loop) {
|
||||||
|
@ -1077,22 +1039,29 @@ int soundio_pulseaudio_init(SoundIoPrivate *si) {
|
||||||
return SoundIoErrorNoMem;
|
return SoundIoErrorNoMem;
|
||||||
}
|
}
|
||||||
|
|
||||||
block_until_ready(si);
|
pa_threaded_mainloop_lock(sipa->main_loop);
|
||||||
|
|
||||||
|
// block until ready
|
||||||
|
while (!sipa->ready_flag)
|
||||||
|
pa_threaded_mainloop_wait(sipa->main_loop);
|
||||||
|
|
||||||
if (sipa->connection_err) {
|
if (sipa->connection_err) {
|
||||||
|
pa_threaded_mainloop_unlock(sipa->main_loop);
|
||||||
destroy_pa(si);
|
destroy_pa(si);
|
||||||
return sipa->connection_err;
|
return sipa->connection_err;
|
||||||
}
|
}
|
||||||
|
|
||||||
sipa->device_scan_queued.store(true);
|
|
||||||
if ((err = subscribe_to_events(si))) {
|
if ((err = subscribe_to_events(si))) {
|
||||||
|
pa_threaded_mainloop_unlock(sipa->main_loop);
|
||||||
destroy_pa(si);
|
destroy_pa(si);
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pa_threaded_mainloop_unlock(sipa->main_loop);
|
||||||
|
|
||||||
si->destroy = destroy_pa;
|
si->destroy = destroy_pa;
|
||||||
si->flush_events = flush_events;
|
si->flush_events = flush_events_pa;
|
||||||
si->wait_events = wait_events;
|
si->wait_events = wait_events_pa;
|
||||||
si->wakeup = wakeup;
|
si->wakeup = wakeup;
|
||||||
|
|
||||||
si->outstream_open = outstream_open_pa;
|
si->outstream_open = outstream_open_pa;
|
||||||
|
|
|
@ -18,10 +18,12 @@ int soundio_pulseaudio_init(struct SoundIoPrivate *si);
|
||||||
struct SoundIoDevicePulseAudio { };
|
struct SoundIoDevicePulseAudio { };
|
||||||
|
|
||||||
struct SoundIoPulseAudio {
|
struct SoundIoPulseAudio {
|
||||||
|
int device_query_err;
|
||||||
int connection_err;
|
int connection_err;
|
||||||
|
bool emitted_shutdown_cb;
|
||||||
|
|
||||||
pa_context *pulse_context;
|
pa_context *pulse_context;
|
||||||
atomic_bool device_scan_queued;
|
bool device_scan_queued;
|
||||||
|
|
||||||
// the one that we're working on building
|
// the one that we're working on building
|
||||||
struct SoundIoDevicesInfo *current_devices_info;
|
struct SoundIoDevicesInfo *current_devices_info;
|
||||||
|
@ -31,13 +33,7 @@ struct SoundIoPulseAudio {
|
||||||
// this one is ready to be read with flush_events. protected by mutex
|
// this one is ready to be read with flush_events. protected by mutex
|
||||||
struct SoundIoDevicesInfo *ready_devices_info;
|
struct SoundIoDevicesInfo *ready_devices_info;
|
||||||
|
|
||||||
int device_query_err;
|
bool ready_flag;
|
||||||
bool have_sink_list;
|
|
||||||
bool have_source_list;
|
|
||||||
bool have_default_sink;
|
|
||||||
|
|
||||||
atomic_bool ready_flag;
|
|
||||||
atomic_bool have_devices_flag;
|
|
||||||
|
|
||||||
pa_threaded_mainloop *main_loop;
|
pa_threaded_mainloop *main_loop;
|
||||||
pa_proplist *props;
|
pa_proplist *props;
|
||||||
|
|
|
@ -70,7 +70,7 @@ const char *soundio_strerror(int error) {
|
||||||
case SoundIoErrorInitAudioBackend: return "unable to initialize audio backend";
|
case SoundIoErrorInitAudioBackend: return "unable to initialize audio backend";
|
||||||
case SoundIoErrorSystemResources: return "system resource not available";
|
case SoundIoErrorSystemResources: return "system resource not available";
|
||||||
case SoundIoErrorOpeningDevice: return "unable to open device";
|
case SoundIoErrorOpeningDevice: return "unable to open device";
|
||||||
case SoundIoErrorNoSuchDevice: return "unable to open device";
|
case SoundIoErrorNoSuchDevice: return "no such device";
|
||||||
case SoundIoErrorInvalid: return "invalid value";
|
case SoundIoErrorInvalid: return "invalid value";
|
||||||
case SoundIoErrorBackendUnavailable: return "backend unavailable";
|
case SoundIoErrorBackendUnavailable: return "backend unavailable";
|
||||||
case SoundIoErrorStreaming: return "unrecoverable streaming failure";
|
case SoundIoErrorStreaming: return "unrecoverable streaming failure";
|
||||||
|
|
|
@ -908,6 +908,8 @@ static int refresh_devices(SoundIoPrivate *si) {
|
||||||
soundio_os_mutex_lock(siw->mutex);
|
soundio_os_mutex_lock(siw->mutex);
|
||||||
soundio_destroy_devices_info(siw->ready_devices_info);
|
soundio_destroy_devices_info(siw->ready_devices_info);
|
||||||
siw->ready_devices_info = rd.devices_info;
|
siw->ready_devices_info = rd.devices_info;
|
||||||
|
siw->have_devices_flag = true;
|
||||||
|
soundio_os_cond_signal(siw->cond, siw->mutex);
|
||||||
soundio->on_events_signal(soundio);
|
soundio->on_events_signal(soundio);
|
||||||
soundio_os_mutex_unlock(siw->mutex);
|
soundio_os_mutex_unlock(siw->mutex);
|
||||||
|
|
||||||
|
@ -923,13 +925,13 @@ static void shutdown_backend(SoundIoPrivate *si, int err) {
|
||||||
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
||||||
soundio_os_mutex_lock(siw->mutex);
|
soundio_os_mutex_lock(siw->mutex);
|
||||||
siw->shutdown_err = err;
|
siw->shutdown_err = err;
|
||||||
|
soundio_os_cond_signal(siw->cond, siw->mutex);
|
||||||
soundio->on_events_signal(soundio);
|
soundio->on_events_signal(soundio);
|
||||||
soundio_os_mutex_unlock(siw->mutex);
|
soundio_os_mutex_unlock(siw->mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void device_thread_run(void *arg) {
|
static void device_thread_run(void *arg) {
|
||||||
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
SoundIoPrivate *si = (SoundIoPrivate *)arg;
|
||||||
SoundIo *soundio = &si->pub;
|
|
||||||
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
|
@ -937,10 +939,6 @@ static void device_thread_run(void *arg) {
|
||||||
CLSCTX_ALL, IID_IMMDeviceEnumerator, (void**)&siw->device_enumerator);
|
CLSCTX_ALL, IID_IMMDeviceEnumerator, (void**)&siw->device_enumerator);
|
||||||
if (FAILED(hr)) {
|
if (FAILED(hr)) {
|
||||||
shutdown_backend(si, SoundIoErrorSystemResources);
|
shutdown_backend(si, SoundIoErrorSystemResources);
|
||||||
if (!siw->have_devices_flag.exchange(true)) {
|
|
||||||
soundio_os_cond_signal(siw->cond, nullptr);
|
|
||||||
soundio->on_events_signal(soundio);
|
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -948,10 +946,6 @@ static void device_thread_run(void *arg) {
|
||||||
siw->device_enumerator, &siw->device_events)))
|
siw->device_enumerator, &siw->device_events)))
|
||||||
{
|
{
|
||||||
shutdown_backend(si, SoundIoErrorSystemResources);
|
shutdown_backend(si, SoundIoErrorSystemResources);
|
||||||
if (!siw->have_devices_flag.exchange(true)) {
|
|
||||||
soundio_os_cond_signal(siw->cond, nullptr);
|
|
||||||
soundio->on_events_signal(soundio);
|
|
||||||
}
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -960,15 +954,10 @@ static void device_thread_run(void *arg) {
|
||||||
break;
|
break;
|
||||||
if (siw->device_scan_queued.exchange(false)) {
|
if (siw->device_scan_queued.exchange(false)) {
|
||||||
err = refresh_devices(si);
|
err = refresh_devices(si);
|
||||||
if (err)
|
if (err) {
|
||||||
shutdown_backend(si, err);
|
shutdown_backend(si, err);
|
||||||
if (!siw->have_devices_flag.exchange(true)) {
|
return;
|
||||||
soundio_os_cond_signal(siw->cond, nullptr);
|
|
||||||
soundio->on_events_signal(soundio);
|
|
||||||
}
|
}
|
||||||
if (err)
|
|
||||||
break;
|
|
||||||
soundio_os_cond_signal(siw->cond, nullptr);
|
|
||||||
}
|
}
|
||||||
soundio_os_cond_wait(siw->scan_devices_cond, nullptr);
|
soundio_os_cond_wait(siw->scan_devices_cond, nullptr);
|
||||||
}
|
}
|
||||||
|
@ -978,17 +967,9 @@ static void device_thread_run(void *arg) {
|
||||||
siw->device_enumerator = nullptr;
|
siw->device_enumerator = nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void block_until_have_devices(SoundIoWasapi *siw) {
|
static void my_flush_events(struct SoundIoPrivate *si, bool wait) {
|
||||||
if (siw->have_devices_flag.load())
|
|
||||||
return;
|
|
||||||
while (!siw->have_devices_flag.load())
|
|
||||||
soundio_os_cond_wait(siw->cond, nullptr);
|
|
||||||
}
|
|
||||||
|
|
||||||
static void flush_events_wasapi(struct SoundIoPrivate *si) {
|
|
||||||
SoundIo *soundio = &si->pub;
|
SoundIo *soundio = &si->pub;
|
||||||
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
||||||
block_until_have_devices(siw);
|
|
||||||
|
|
||||||
bool change = false;
|
bool change = false;
|
||||||
bool cb_shutdown = false;
|
bool cb_shutdown = false;
|
||||||
|
@ -996,6 +977,12 @@ static void flush_events_wasapi(struct SoundIoPrivate *si) {
|
||||||
|
|
||||||
soundio_os_mutex_lock(siw->mutex);
|
soundio_os_mutex_lock(siw->mutex);
|
||||||
|
|
||||||
|
// block until have devices
|
||||||
|
while (wait || (!siw->have_devices_flag && !siw->shutdown_err)) {
|
||||||
|
soundio_os_cond_wait(siw->cond, siw->mutex);
|
||||||
|
wait = false;
|
||||||
|
}
|
||||||
|
|
||||||
if (siw->shutdown_err && !siw->emitted_shutdown_cb) {
|
if (siw->shutdown_err && !siw->emitted_shutdown_cb) {
|
||||||
siw->emitted_shutdown_cb = true;
|
siw->emitted_shutdown_cb = true;
|
||||||
cb_shutdown = true;
|
cb_shutdown = true;
|
||||||
|
@ -1016,15 +1003,18 @@ static void flush_events_wasapi(struct SoundIoPrivate *si) {
|
||||||
soundio_destroy_devices_info(old_devices_info);
|
soundio_destroy_devices_info(old_devices_info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void flush_events_wasapi(struct SoundIoPrivate *si) {
|
||||||
|
my_flush_events(si, false);
|
||||||
|
}
|
||||||
|
|
||||||
static void wait_events_wasapi(struct SoundIoPrivate *si) {
|
static void wait_events_wasapi(struct SoundIoPrivate *si) {
|
||||||
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
my_flush_events(si, false);
|
||||||
flush_events_wasapi(si);
|
my_flush_events(si, true);
|
||||||
soundio_os_cond_wait(siw->cond, nullptr);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static void wakeup_wasapi(struct SoundIoPrivate *si) {
|
static void wakeup_wasapi(struct SoundIoPrivate *si) {
|
||||||
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
||||||
soundio_os_cond_signal(siw->cond, nullptr);
|
soundio_os_cond_signal(siw->cond, siw->mutex);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void outstream_thread_deinit(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
static void outstream_thread_deinit(SoundIoPrivate *si, SoundIoOutStreamPrivate *os) {
|
||||||
|
@ -1963,7 +1953,6 @@ int soundio_wasapi_init(SoundIoPrivate *si) {
|
||||||
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
SoundIoWasapi *siw = &si->backend_data.wasapi;
|
||||||
int err;
|
int err;
|
||||||
|
|
||||||
siw->have_devices_flag.store(false);
|
|
||||||
siw->device_scan_queued.store(true);
|
siw->device_scan_queued.store(true);
|
||||||
siw->abort_flag.test_and_set();
|
siw->abort_flag.test_and_set();
|
||||||
|
|
||||||
|
|
|
@ -40,7 +40,7 @@ struct SoundIoWasapi {
|
||||||
atomic_flag abort_flag;
|
atomic_flag abort_flag;
|
||||||
// this one is ready to be read with flush_events. protected by mutex
|
// this one is ready to be read with flush_events. protected by mutex
|
||||||
struct SoundIoDevicesInfo *ready_devices_info;
|
struct SoundIoDevicesInfo *ready_devices_info;
|
||||||
atomic_bool have_devices_flag;
|
bool have_devices_flag;
|
||||||
atomic_bool device_scan_queued;
|
atomic_bool device_scan_queued;
|
||||||
int shutdown_err;
|
int shutdown_err;
|
||||||
bool emitted_shutdown_cb;
|
bool emitted_shutdown_cb;
|
||||||
|
|
|
@ -12,7 +12,6 @@
|
||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <math.h>
|
#include <math.h>
|
||||||
#include <stdatomic.h>
|
|
||||||
#include <unistd.h>
|
#include <unistd.h>
|
||||||
|
|
||||||
__attribute__ ((cold))
|
__attribute__ ((cold))
|
||||||
|
@ -28,18 +27,20 @@ static void panic(const char *format, ...) {
|
||||||
}
|
}
|
||||||
|
|
||||||
static int usage(char *exe) {
|
static int usage(char *exe) {
|
||||||
fprintf(stderr, "Usage: %s [--backend dummy|alsa|pulseaudio|jack|coreaudio|wasapi]\n", exe);
|
fprintf(stderr, "Usage: %s [options]\n"
|
||||||
|
"Options:\n"
|
||||||
|
" [--backend dummy|alsa|pulseaudio|jack|coreaudio|wasapi]\n"
|
||||||
|
" [--timeout seconds]\n", exe);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
static enum SoundIoBackend backend = SoundIoBackendNone;
|
static enum SoundIoBackend backend = SoundIoBackendNone;
|
||||||
|
|
||||||
static atomic_bool severed = ATOMIC_VAR_INIT(false);
|
static bool severed = false;
|
||||||
|
|
||||||
static void on_backend_disconnect(struct SoundIo *soundio, int err) {
|
static void on_backend_disconnect(struct SoundIo *soundio, int err) {
|
||||||
fprintf(stderr, "OK backend disconnected with '%s'.\n", soundio_strerror(err));
|
fprintf(stderr, "OK backend disconnected with '%s'.\n", soundio_strerror(err));
|
||||||
atomic_store(&severed, true);
|
severed = true;
|
||||||
soundio_wakeup(soundio);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char **argv) {
|
int main(int argc, char **argv) {
|
||||||
|
@ -93,7 +94,7 @@ int main(int argc, char **argv) {
|
||||||
fprintf(stderr, "OK connected to %s. Now cause the backend to disconnect.\n",
|
fprintf(stderr, "OK connected to %s. Now cause the backend to disconnect.\n",
|
||||||
soundio_backend_name(soundio->current_backend));
|
soundio_backend_name(soundio->current_backend));
|
||||||
|
|
||||||
while (!atomic_load(&severed))
|
while (!severed)
|
||||||
soundio_wait_events(soundio);
|
soundio_wait_events(soundio);
|
||||||
|
|
||||||
soundio_disconnect(soundio);
|
soundio_disconnect(soundio);
|
||||||
|
|
228
test/overflow.c
Normal file
228
test/overflow.c
Normal file
|
@ -0,0 +1,228 @@
|
||||||
|
/*
|
||||||
|
* Copyright (c) 2015 Andrew Kelley
|
||||||
|
*
|
||||||
|
* This file is part of libsoundio, which is MIT licensed.
|
||||||
|
* See http://opensource.org/licenses/MIT
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <soundio/soundio.h>
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <math.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
static enum SoundIoFormat prioritized_formats[] = {
|
||||||
|
SoundIoFormatFloat32NE,
|
||||||
|
SoundIoFormatFloat32FE,
|
||||||
|
SoundIoFormatS32NE,
|
||||||
|
SoundIoFormatS32FE,
|
||||||
|
SoundIoFormatS24NE,
|
||||||
|
SoundIoFormatS24FE,
|
||||||
|
SoundIoFormatS16NE,
|
||||||
|
SoundIoFormatS16FE,
|
||||||
|
SoundIoFormatFloat64NE,
|
||||||
|
SoundIoFormatFloat64FE,
|
||||||
|
SoundIoFormatU32NE,
|
||||||
|
SoundIoFormatU32FE,
|
||||||
|
SoundIoFormatU24NE,
|
||||||
|
SoundIoFormatU24FE,
|
||||||
|
SoundIoFormatU16NE,
|
||||||
|
SoundIoFormatU16FE,
|
||||||
|
SoundIoFormatS8,
|
||||||
|
SoundIoFormatU8,
|
||||||
|
SoundIoFormatInvalid,
|
||||||
|
};
|
||||||
|
|
||||||
|
__attribute__ ((cold))
|
||||||
|
__attribute__ ((noreturn))
|
||||||
|
__attribute__ ((format (printf, 1, 2)))
|
||||||
|
static void panic(const char *format, ...) {
|
||||||
|
va_list ap;
|
||||||
|
va_start(ap, format);
|
||||||
|
vfprintf(stderr, format, ap);
|
||||||
|
fprintf(stderr, "\n");
|
||||||
|
va_end(ap);
|
||||||
|
abort();
|
||||||
|
}
|
||||||
|
|
||||||
|
static int usage(char *exe) {
|
||||||
|
fprintf(stderr, "Usage: %s [options]\n"
|
||||||
|
"Options:\n"
|
||||||
|
" [--backend dummy|alsa|pulseaudio|jack|coreaudio|wasapi]\n"
|
||||||
|
" [--device id]\n"
|
||||||
|
" [--raw]\n", exe);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
static struct SoundIo *soundio = NULL;
|
||||||
|
static float seconds_offset = 0.0f;
|
||||||
|
static float seconds_end = 9.0f;
|
||||||
|
static bool caused_underflow = false;
|
||||||
|
static int overflow_count = 0;
|
||||||
|
|
||||||
|
static void read_callback(struct SoundIoInStream *instream, int frame_count_min, int frame_count_max) {
|
||||||
|
struct SoundIoChannelArea *areas;
|
||||||
|
float float_sample_rate = instream->sample_rate;
|
||||||
|
float seconds_per_frame = 1.0f / float_sample_rate;
|
||||||
|
int err;
|
||||||
|
|
||||||
|
if (!caused_underflow && seconds_offset >= 3.0f) {
|
||||||
|
fprintf(stderr, "OK sleeping...\n");
|
||||||
|
caused_underflow = true;
|
||||||
|
sleep(3);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (seconds_offset >= seconds_end) {
|
||||||
|
soundio_wakeup(soundio);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
int frames_left = frame_count_max;
|
||||||
|
|
||||||
|
for (;;) {
|
||||||
|
int frame_count = frames_left;
|
||||||
|
|
||||||
|
if ((err = soundio_instream_begin_read(instream, &areas, &frame_count)))
|
||||||
|
panic("begin read error: %s", soundio_strerror(err));
|
||||||
|
|
||||||
|
if (!frame_count)
|
||||||
|
break;
|
||||||
|
|
||||||
|
seconds_offset += seconds_per_frame * frame_count;
|
||||||
|
|
||||||
|
if ((err = soundio_instream_end_read(instream)))
|
||||||
|
panic("end read error: %s", soundio_strerror(err));
|
||||||
|
|
||||||
|
frames_left -= frame_count;
|
||||||
|
if (frames_left <= 0)
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr, "OK received %d frames\n", frame_count_max);
|
||||||
|
}
|
||||||
|
|
||||||
|
static void overflow_callback(struct SoundIoInStream *instream) {
|
||||||
|
fprintf(stderr, "OK overflow %d\n", overflow_count++);
|
||||||
|
}
|
||||||
|
|
||||||
|
int main(int argc, char **argv) {
|
||||||
|
char *exe = argv[0];
|
||||||
|
enum SoundIoBackend backend = SoundIoBackendNone;
|
||||||
|
bool is_raw = false;
|
||||||
|
char *device_id = NULL;
|
||||||
|
for (int i = 1; i < argc; i += 1) {
|
||||||
|
char *arg = argv[i];
|
||||||
|
if (arg[0] == '-' && arg[1] == '-') {
|
||||||
|
if (strcmp(arg, "--raw") == 0) {
|
||||||
|
is_raw = true;
|
||||||
|
} else if (++i >= argc) {
|
||||||
|
return usage(exe);
|
||||||
|
} else if (strcmp(arg, "--device") == 0) {
|
||||||
|
device_id = argv[i];
|
||||||
|
} else if (strcmp(arg, "--backend") == 0) {
|
||||||
|
if (strcmp("dummy", argv[i]) == 0) {
|
||||||
|
backend = SoundIoBackendDummy;
|
||||||
|
} else if (strcmp("alsa", argv[i]) == 0) {
|
||||||
|
backend = SoundIoBackendAlsa;
|
||||||
|
} else if (strcmp("pulseaudio", argv[i]) == 0) {
|
||||||
|
backend = SoundIoBackendPulseAudio;
|
||||||
|
} else if (strcmp("jack", argv[i]) == 0) {
|
||||||
|
backend = SoundIoBackendJack;
|
||||||
|
} else if (strcmp("coreaudio", argv[i]) == 0) {
|
||||||
|
backend = SoundIoBackendCoreAudio;
|
||||||
|
} else if (strcmp("wasapi", argv[i]) == 0) {
|
||||||
|
backend = SoundIoBackendWasapi;
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, "Invalid backend: %s\n", argv[i]);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return usage(exe);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return usage(exe);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr,
|
||||||
|
"Records for 3 seconds, sleeps for 3 seconds, then you should see at least\n"
|
||||||
|
"one buffer overflow message, then records for 3 seconds.\n"
|
||||||
|
"PulseAudio is not expected to pass this test.\n");
|
||||||
|
|
||||||
|
if (!(soundio = soundio_create()))
|
||||||
|
panic("out of memory");
|
||||||
|
|
||||||
|
int err = (backend == SoundIoBackendNone) ?
|
||||||
|
soundio_connect(soundio) : soundio_connect_backend(soundio, backend);
|
||||||
|
|
||||||
|
if (err)
|
||||||
|
panic("error connecting: %s", soundio_strerror(err));
|
||||||
|
|
||||||
|
soundio_flush_events(soundio);
|
||||||
|
|
||||||
|
int selected_device_index = -1;
|
||||||
|
if (device_id) {
|
||||||
|
int device_count = soundio_input_device_count(soundio);
|
||||||
|
for (int i = 0; i < device_count; i += 1) {
|
||||||
|
struct SoundIoDevice *device = soundio_get_input_device(soundio, i);
|
||||||
|
if (strcmp(device->id, device_id) == 0 && device->is_raw == is_raw) {
|
||||||
|
selected_device_index = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
selected_device_index = soundio_default_input_device_index(soundio);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selected_device_index < 0) {
|
||||||
|
fprintf(stderr, "input device not found\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct SoundIoDevice *device = soundio_get_input_device(soundio, selected_device_index);
|
||||||
|
if (!device) {
|
||||||
|
fprintf(stderr, "out of memory\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(stderr, "Input device: %s\n", device->name);
|
||||||
|
|
||||||
|
enum SoundIoFormat *fmt;
|
||||||
|
for (fmt = prioritized_formats; *fmt != SoundIoFormatInvalid; fmt += 1) {
|
||||||
|
if (soundio_device_supports_format(device, *fmt))
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (*fmt == SoundIoFormatInvalid)
|
||||||
|
panic("incompatible sample format");
|
||||||
|
|
||||||
|
struct SoundIoInStream *instream = soundio_instream_create(device);
|
||||||
|
instream->format = *fmt;
|
||||||
|
instream->read_callback = read_callback;
|
||||||
|
instream->overflow_callback = overflow_callback;
|
||||||
|
|
||||||
|
if ((err = soundio_instream_open(instream)))
|
||||||
|
panic("unable to open device: %s", soundio_strerror(err));
|
||||||
|
|
||||||
|
fprintf(stderr, "OK format: %s\n", soundio_format_string(instream->format));
|
||||||
|
|
||||||
|
if ((err = soundio_instream_start(instream)))
|
||||||
|
panic("unable to start device: %s", soundio_strerror(err));
|
||||||
|
|
||||||
|
while (seconds_offset < seconds_end)
|
||||||
|
soundio_wait_events(soundio);
|
||||||
|
|
||||||
|
soundio_instream_destroy(instream);
|
||||||
|
soundio_device_unref(device);
|
||||||
|
soundio_destroy(soundio);
|
||||||
|
|
||||||
|
if (overflow_count > 0) {
|
||||||
|
fprintf(stderr, "OK test passed with %d overflow callbacks\n", overflow_count);
|
||||||
|
return 0;
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, "FAIL no overflow callbacks received\n");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
}
|
|
@ -27,7 +27,7 @@ static void panic(const char *format, ...) {
|
||||||
}
|
}
|
||||||
|
|
||||||
static int usage(char *exe) {
|
static int usage(char *exe) {
|
||||||
fprintf(stderr, "Usage: %s [--dummy] [--alsa] [--pulseaudio] [--jack]\n", exe);
|
fprintf(stderr, "Usage: %s [--backend dummy|alsa|pulseaudio|jack|coreaudio|wasapi]\n", exe);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -98,14 +98,30 @@ int main(int argc, char **argv) {
|
||||||
enum SoundIoBackend backend = SoundIoBackendNone;
|
enum SoundIoBackend backend = SoundIoBackendNone;
|
||||||
for (int i = 1; i < argc; i += 1) {
|
for (int i = 1; i < argc; i += 1) {
|
||||||
char *arg = argv[i];
|
char *arg = argv[i];
|
||||||
if (strcmp("--dummy", arg) == 0) {
|
if (arg[0] == '-' && arg[1] == '-') {
|
||||||
|
i += 1;
|
||||||
|
if (i >= argc) {
|
||||||
|
return usage(exe);
|
||||||
|
} else if (strcmp(arg, "--backend") == 0) {
|
||||||
|
if (strcmp("-dummy", argv[i]) == 0) {
|
||||||
backend = SoundIoBackendDummy;
|
backend = SoundIoBackendDummy;
|
||||||
} else if (strcmp("--alsa", arg) == 0) {
|
} else if (strcmp("alsa", argv[i]) == 0) {
|
||||||
backend = SoundIoBackendAlsa;
|
backend = SoundIoBackendAlsa;
|
||||||
} else if (strcmp("--pulseaudio", arg) == 0) {
|
} else if (strcmp("pulseaudio", argv[i]) == 0) {
|
||||||
backend = SoundIoBackendPulseAudio;
|
backend = SoundIoBackendPulseAudio;
|
||||||
} else if (strcmp("--jack", arg) == 0) {
|
} else if (strcmp("jack", argv[i]) == 0) {
|
||||||
backend = SoundIoBackendJack;
|
backend = SoundIoBackendJack;
|
||||||
|
} else if (strcmp("coreaudio", argv[i]) == 0) {
|
||||||
|
backend = SoundIoBackendCoreAudio;
|
||||||
|
} else if (strcmp("wasapi", argv[i]) == 0) {
|
||||||
|
backend = SoundIoBackendWasapi;
|
||||||
|
} else {
|
||||||
|
fprintf(stderr, "Invalid backend: %s\n", argv[i]);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
return usage(exe);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
return usage(exe);
|
return usage(exe);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue