diff --git a/src/coreaudio.cpp b/src/coreaudio.cpp index 6060cf6..36af71f 100644 --- a/src/coreaudio.cpp +++ b/src/coreaudio.cpp @@ -20,18 +20,36 @@ static OSStatus on_devices_changed(AudioObjectID in_object_id, UInt32 in_number_ return noErr; } +static OSStatus on_service_restarted(AudioObjectID in_object_id, UInt32 in_number_addresses, + const AudioObjectPropertyAddress in_addresses[], void *in_client_data) +{ + SoundIoPrivate *si = (SoundIoPrivate*)in_client_data; + SoundIoCoreAudio *sica = &si->backend_data.coreaudio; + + sica->service_restarted.store(true); + soundio_os_cond_signal(sica->cond, nullptr); + + return noErr; +} + static void destroy_ca(struct SoundIoPrivate *si) { SoundIoCoreAudio *sica = &si->backend_data.coreaudio; + int err; AudioObjectPropertyAddress prop_address = { kAudioHardwarePropertyDevices, kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster }; - int err = AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &prop_address, + err = AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &prop_address, on_devices_changed, si); assert(!err); + prop_address.mSelector = kAudioHardwarePropertyServiceRestarted; + err = AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &prop_address, + on_service_restarted, si); + assert(!err); + if (sica->thread) { sica->abort_flag.clear(); soundio_os_cond_signal(sica->cond, nullptr); @@ -83,14 +101,6 @@ static int aim_to_scope(SoundIoDeviceAim aim) { @constant kAudioHardwarePropertyDefaultOutputDevice The AudioObjectID of the default output AudioDevice. */ -/* - @constant kAudioHardwarePropertyServiceRestarted - A UInt32 whose value has no meaning. Rather, this property exists so that - clients can be informed when the service has been reset for some reason. - When a reset happens, any state the client has , such as cached data or - added listeners, must be re-established by the client. - */ - /* * @constant kAudioDevicePropertyDeviceHasChanged @@ -295,6 +305,10 @@ static void device_thread_run(void *arg) { for (;;) { if (!sica->abort_flag.test_and_set()) break; + if (sica->service_restarted.load()) { + shutdown_backend(si, SoundIoErrorBackendDisconnected); + return; + } if (sica->device_scan_queued.exchange(false)) { if ((err = refresh_devices(si))) { shutdown_backend(si, err); @@ -374,6 +388,7 @@ int soundio_coreaudio_init(SoundIoPrivate *si) { sica->have_devices_flag.store(false); sica->device_scan_queued.store(true); + sica->service_restarted.store(false); sica->abort_flag.test_and_set(); sica->mutex = soundio_os_mutex_create(); @@ -405,6 +420,13 @@ int soundio_coreaudio_init(SoundIoPrivate *si) { soundio_panic("add prop listener"); } + prop_address.mSelector = kAudioHardwarePropertyServiceRestarted; + if ((err = AudioObjectAddPropertyListener(kAudioObjectSystemObject, &prop_address, + on_service_restarted, si))) + { + soundio_panic("add prop listener 2"); + } + if ((err = soundio_os_thread_create(device_thread_run, si, false, &sica->thread))) { destroy_ca(si); return err; diff --git a/src/coreaudio.hpp b/src/coreaudio.hpp index 5ca92e0..4723c63 100644 --- a/src/coreaudio.hpp +++ b/src/coreaudio.hpp @@ -30,6 +30,7 @@ struct SoundIoCoreAudio { SoundIoOsCond *have_devices_cond; atomic_bool device_scan_queued; + atomic_bool service_restarted; int shutdown_err; bool emitted_shutdown_cb; };