/* * QEMU OS X CoreAudio audio driver * * Copyright (c) 2005 Mike Kronenberg * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN * THE SOFTWARE. */ #include "qemu/osdep.h" #include #include /* pthread_X */ #include "qemu/main-loop.h" #include "qemu/module.h" #include "qemu/error-report.h" #include "qemu/audio.h" #include "qom/object.h" #include "audio_int.h" #define TYPE_AUDIO_COREAUDIO "audio-coreaudio" OBJECT_DECLARE_SIMPLE_TYPE(AudioCoreaudio, AUDIO_COREAUDIO) struct AudioCoreaudio { AudioMixengBackend parent_obj; }; typedef struct coreaudioVoiceOut { HWVoiceOut hw; pthread_mutex_t buf_mutex; AudioDeviceID outputDeviceID; int frameSizeSetting; uint32_t bufferCount; UInt32 audioDevicePropertyBufferFrameSize; AudioDeviceIOProcID ioprocid; bool enabled; } coreaudioVoiceOut; static const AudioObjectPropertyAddress voice_addr = { kAudioHardwarePropertyDefaultOutputDevice, kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMain }; static OSStatus coreaudio_get_voice(AudioDeviceID *id) { UInt32 size = sizeof(*id); return AudioObjectGetPropertyData(kAudioObjectSystemObject, &voice_addr, 0, NULL, &size, id); } static OSStatus coreaudio_get_framesizerange(AudioDeviceID id, AudioValueRange *framerange) { UInt32 size = sizeof(*framerange); AudioObjectPropertyAddress addr = { kAudioDevicePropertyBufferFrameSizeRange, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMain }; return AudioObjectGetPropertyData(id, &addr, 0, NULL, &size, framerange); } static OSStatus coreaudio_get_framesize(AudioDeviceID id, UInt32 *framesize) { UInt32 size = sizeof(*framesize); AudioObjectPropertyAddress addr = { kAudioDevicePropertyBufferFrameSize, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMain }; return AudioObjectGetPropertyData(id, &addr, 0, NULL, &size, framesize); } static OSStatus coreaudio_set_framesize(AudioDeviceID id, UInt32 *framesize) { UInt32 size = sizeof(*framesize); AudioObjectPropertyAddress addr = { kAudioDevicePropertyBufferFrameSize, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMain }; return AudioObjectSetPropertyData(id, &addr, 0, NULL, size, framesize); } static OSStatus coreaudio_set_streamformat(AudioDeviceID id, AudioStreamBasicDescription *d) { UInt32 size = sizeof(*d); AudioObjectPropertyAddress addr = { kAudioDevicePropertyStreamFormat, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMain }; return AudioObjectSetPropertyData(id, &addr, 0, NULL, size, d); } static OSStatus coreaudio_get_isrunning(AudioDeviceID id, UInt32 *result) { UInt32 size = sizeof(*result); AudioObjectPropertyAddress addr = { kAudioDevicePropertyDeviceIsRunning, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMain }; return AudioObjectGetPropertyData(id, &addr, 0, NULL, &size, result); } static void coreaudio_logstatus(OSStatus status) { const char *str = "BUG"; switch (status) { case kAudioHardwareNoError: str = "kAudioHardwareNoError"; break; case kAudioHardwareNotRunningError: str = "kAudioHardwareNotRunningError"; break; case kAudioHardwareUnspecifiedError: str = "kAudioHardwareUnspecifiedError"; break; case kAudioHardwareUnknownPropertyError: str = "kAudioHardwareUnknownPropertyError"; break; case kAudioHardwareBadPropertySizeError: str = "kAudioHardwareBadPropertySizeError"; break; case kAudioHardwareIllegalOperationError: str = "kAudioHardwareIllegalOperationError"; break; case kAudioHardwareBadDeviceError: str = "kAudioHardwareBadDeviceError"; break; case kAudioHardwareBadStreamError: str = "kAudioHardwareBadStreamError"; break; case kAudioHardwareUnsupportedOperationError: str = "kAudioHardwareUnsupportedOperationError"; break; case kAudioDeviceUnsupportedFormatError: str = "kAudioDeviceUnsupportedFormatError"; break; case kAudioDevicePermissionsError: str = "kAudioDevicePermissionsError"; break; default: error_printf(" Reason: status code %" PRId32, (int32_t)status); return; } error_printf(" Reason: %s", str); } static void G_GNUC_PRINTF(2, 3) coreaudio_logerr(OSStatus status, const char *fmt, ...) { va_list ap; error_printf("coreaudio: "); va_start(ap, fmt); error_vprintf(fmt, ap); va_end(ap); coreaudio_logstatus(status); error_printf("\n"); } static void G_GNUC_PRINTF(3, 4) coreaudio_logerr2(OSStatus status, const char *typ, const char *fmt, ...) { va_list ap; error_printf("coreaudio: Could not initialize %s: ", typ); va_start(ap, fmt); error_vprintf(fmt, ap); va_end(ap); coreaudio_logstatus(status); error_printf("\n"); } #define coreaudio_playback_logerr(status, ...) \ coreaudio_logerr2(status, "playback", __VA_ARGS__) static int coreaudio_buf_lock(coreaudioVoiceOut *core, const char *fn_name) { int err; err = pthread_mutex_lock(&core->buf_mutex); if (err) { error_report("coreaudio: Could not lock voice for %s: %s", fn_name, strerror(err)); return -1; } return 0; } static int coreaudio_buf_unlock(coreaudioVoiceOut *core, const char *fn_name) { int err; err = pthread_mutex_unlock(&core->buf_mutex); if (err) { error_report("coreaudio: Could not unlock voice for %s: %s", fn_name, strerror(err)); return -1; } return 0; } #define COREAUDIO_WRAPPER_FUNC(name, ret_type, args_decl, args) \ static ret_type glue(coreaudio_, name)args_decl \ { \ coreaudioVoiceOut *core = (coreaudioVoiceOut *) hw; \ ret_type ret; \ \ if (coreaudio_buf_lock(core, "coreaudio_" #name)) { \ return 0; \ } \ \ ret = glue(audio_generic_, name)args; \ \ coreaudio_buf_unlock(core, "coreaudio_" #name); \ return ret; \ } COREAUDIO_WRAPPER_FUNC(buffer_get_free, size_t, (HWVoiceOut *hw), (hw)) COREAUDIO_WRAPPER_FUNC(get_buffer_out, void *, (HWVoiceOut *hw, size_t *size), (hw, size)) COREAUDIO_WRAPPER_FUNC(put_buffer_out, size_t, (HWVoiceOut *hw, void *buf, size_t size), (hw, buf, size)) COREAUDIO_WRAPPER_FUNC(write, size_t, (HWVoiceOut *hw, void *buf, size_t size), (hw, buf, size)) #undef COREAUDIO_WRAPPER_FUNC /* * callback to feed audiooutput buffer. called without BQL. * allowed to lock "buf_mutex", but disallowed to have any other locks. */ static OSStatus audioDeviceIOProc( AudioDeviceID inDevice, const AudioTimeStamp *inNow, const AudioBufferList *inInputData, const AudioTimeStamp *inInputTime, AudioBufferList *outOutputData, const AudioTimeStamp *inOutputTime, void *hwptr) { UInt32 frameCount, pending_frames; void *out = outOutputData->mBuffers[0].mData; HWVoiceOut *hw = hwptr; coreaudioVoiceOut *core = (coreaudioVoiceOut *) hwptr; size_t len; if (coreaudio_buf_lock (core, "audioDeviceIOProc")) { inInputTime = 0; return 0; } if (inDevice != core->outputDeviceID) { coreaudio_buf_unlock (core, "audioDeviceIOProc(old device)"); return 0; } frameCount = core->audioDevicePropertyBufferFrameSize; pending_frames = hw->pending_emul / hw->info.bytes_per_frame; /* if there are not enough samples, set signal and return */ if (pending_frames < frameCount) { inInputTime = 0; coreaudio_buf_unlock (core, "audioDeviceIOProc(empty)"); return 0; } len = frameCount * hw->info.bytes_per_frame; while (len) { size_t write_len, start; start = audio_ring_posb(hw->pos_emul, hw->pending_emul, hw->size_emul); assert(start < hw->size_emul); write_len = MIN(MIN(hw->pending_emul, len), hw->size_emul - start); memcpy(out, hw->buf_emul + start, write_len); hw->pending_emul -= write_len; len -= write_len; out += write_len; } coreaudio_buf_unlock (core, "audioDeviceIOProc"); return 0; } static OSStatus init_out_device(coreaudioVoiceOut *core) { OSStatus status; AudioValueRange frameRange; AudioStreamBasicDescription streamBasicDescription = { .mBitsPerChannel = audio_format_bits(core->hw.info.af), .mBytesPerFrame = core->hw.info.bytes_per_frame, .mBytesPerPacket = core->hw.info.bytes_per_frame, .mChannelsPerFrame = core->hw.info.nchannels, .mFormatFlags = kLinearPCMFormatFlagIsFloat, .mFormatID = kAudioFormatLinearPCM, .mFramesPerPacket = 1, .mSampleRate = core->hw.info.freq }; status = coreaudio_get_voice(&core->outputDeviceID); if (status != kAudioHardwareNoError) { coreaudio_playback_logerr(status, "Could not get default output device"); return status; } if (core->outputDeviceID == kAudioDeviceUnknown) { error_report("coreaudio: Could not initialize playback: " "Unknown audio device"); return status; } /* get minimum and maximum buffer frame sizes */ status = coreaudio_get_framesizerange(core->outputDeviceID, &frameRange); if (status == kAudioHardwareBadObjectError) { return 0; } if (status != kAudioHardwareNoError) { coreaudio_playback_logerr(status, "Could not get device buffer frame range"); return status; } if (frameRange.mMinimum > core->frameSizeSetting) { core->audioDevicePropertyBufferFrameSize = (UInt32) frameRange.mMinimum; warn_report("coreaudio: Upsizing buffer frames to %f", frameRange.mMinimum); } else if (frameRange.mMaximum < core->frameSizeSetting) { core->audioDevicePropertyBufferFrameSize = (UInt32) frameRange.mMaximum; warn_report("coreaudio: Downsizing buffer frames to %f", frameRange.mMaximum); } else { core->audioDevicePropertyBufferFrameSize = core->frameSizeSetting; } /* set Buffer Frame Size */ status = coreaudio_set_framesize(core->outputDeviceID, &core->audioDevicePropertyBufferFrameSize); if (status == kAudioHardwareBadObjectError) { return 0; } if (status != kAudioHardwareNoError) { coreaudio_playback_logerr(status, "Could not set device buffer frame size %" PRIu32, (uint32_t)core->audioDevicePropertyBufferFrameSize); return status; } /* get Buffer Frame Size */ status = coreaudio_get_framesize(core->outputDeviceID, &core->audioDevicePropertyBufferFrameSize); if (status == kAudioHardwareBadObjectError) { return 0; } if (status != kAudioHardwareNoError) { coreaudio_playback_logerr(status, "Could not get device buffer frame size"); return status; } core->hw.samples = core->bufferCount * core->audioDevicePropertyBufferFrameSize; /* set Samplerate */ status = coreaudio_set_streamformat(core->outputDeviceID, &streamBasicDescription); if (status == kAudioHardwareBadObjectError) { return 0; } if (status != kAudioHardwareNoError) { coreaudio_playback_logerr(status, "Could not set samplerate %lf", streamBasicDescription.mSampleRate); core->outputDeviceID = kAudioDeviceUnknown; return status; } /* * set Callback. * * On macOS 11.3.1, Core Audio calls AudioDeviceIOProc after calling an * internal function named HALB_Mutex::Lock(), which locks a mutex in * HALB_IOThread::Entry(void*). HALB_Mutex::Lock() is also called in * AudioObjectGetPropertyData, which is called by coreaudio driver. * Therefore, the specified callback must be designed to avoid a deadlock * with the callers of AudioObjectGetPropertyData. */ core->ioprocid = NULL; status = AudioDeviceCreateIOProcID(core->outputDeviceID, audioDeviceIOProc, &core->hw, &core->ioprocid); if (status == kAudioHardwareBadDeviceError) { return 0; } if (status != kAudioHardwareNoError || core->ioprocid == NULL) { coreaudio_playback_logerr(status, "Could not set IOProc"); core->outputDeviceID = kAudioDeviceUnknown; return status; } return 0; } static void fini_out_device(coreaudioVoiceOut *core) { OSStatus status; UInt32 isrunning; /* stop playback */ status = coreaudio_get_isrunning(core->outputDeviceID, &isrunning); if (status != kAudioHardwareBadObjectError) { if (status != kAudioHardwareNoError) { coreaudio_logerr(status, "Could not determine whether device is playing"); } if (isrunning) { status = AudioDeviceStop(core->outputDeviceID, core->ioprocid); if (status != kAudioHardwareBadDeviceError && status != kAudioHardwareNoError) { coreaudio_logerr(status, "Could not stop playback"); } } } /* remove callback */ status = AudioDeviceDestroyIOProcID(core->outputDeviceID, core->ioprocid); if (status != kAudioHardwareBadDeviceError && status != kAudioHardwareNoError) { coreaudio_logerr(status, "Could not remove IOProc"); } core->outputDeviceID = kAudioDeviceUnknown; } static void update_device_playback_state(coreaudioVoiceOut *core) { OSStatus status; UInt32 isrunning; status = coreaudio_get_isrunning(core->outputDeviceID, &isrunning); if (status != kAudioHardwareNoError) { if (status != kAudioHardwareBadObjectError) { coreaudio_logerr(status, "Could not determine whether device is playing"); } return; } if (core->enabled) { /* start playback */ if (!isrunning) { status = AudioDeviceStart(core->outputDeviceID, core->ioprocid); if (status != kAudioHardwareBadDeviceError && status != kAudioHardwareNoError) { coreaudio_logerr(status, "Could not resume playback"); } } } else { /* stop playback */ if (isrunning) { status = AudioDeviceStop(core->outputDeviceID, core->ioprocid); if (status != kAudioHardwareBadDeviceError && status != kAudioHardwareNoError) { coreaudio_logerr(status, "Could not pause playback"); } } } } /* called without BQL. */ static OSStatus handle_voice_change( AudioObjectID in_object_id, UInt32 in_number_addresses, const AudioObjectPropertyAddress *in_addresses, void *in_client_data) { coreaudioVoiceOut *core = in_client_data; bql_lock(); if (core->outputDeviceID) { fini_out_device(core); } if (!init_out_device(core)) { update_device_playback_state(core); } bql_unlock(); return 0; } static int coreaudio_init_out(HWVoiceOut *hw, struct audsettings *as) { OSStatus status; coreaudioVoiceOut *core = (coreaudioVoiceOut *) hw; int err; Audiodev *dev = hw->s->dev; AudiodevCoreaudioPerDirectionOptions *cpdo = dev->u.coreaudio.out; struct audsettings obt_as; /* create mutex */ err = pthread_mutex_init(&core->buf_mutex, NULL); if (err) { error_report("coreaudio: Could not create mutex: %s", strerror(err)); return -1; } obt_as = *as; as = &obt_as; as->fmt = AUDIO_FORMAT_F32; audio_pcm_init_info (&hw->info, as); core->frameSizeSetting = audio_buffer_frames( qapi_AudiodevCoreaudioPerDirectionOptions_base(cpdo), as, 11610); core->bufferCount = cpdo->has_buffer_count ? cpdo->buffer_count : 4; status = AudioObjectAddPropertyListener(kAudioObjectSystemObject, &voice_addr, handle_voice_change, core); if (status != kAudioHardwareNoError) { coreaudio_playback_logerr(status, "Could not listen to voice property change"); return -1; } if (init_out_device(core)) { status = AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &voice_addr, handle_voice_change, core); if (status != kAudioHardwareNoError) { coreaudio_playback_logerr(status, "Could not remove voice property change listener"); } return -1; } return 0; } static void coreaudio_fini_out (HWVoiceOut *hw) { OSStatus status; int err; coreaudioVoiceOut *core = (coreaudioVoiceOut *) hw; status = AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &voice_addr, handle_voice_change, core); if (status != kAudioHardwareNoError) { coreaudio_logerr(status, "Could not remove voice property change listener"); } fini_out_device(core); /* destroy mutex */ err = pthread_mutex_destroy(&core->buf_mutex); if (err) { error_report("coreaudio: Could not destroy mutex: %s", strerror(err)); } } static void coreaudio_enable_out(HWVoiceOut *hw, bool enable) { coreaudioVoiceOut *core = (coreaudioVoiceOut *) hw; core->enabled = enable; update_device_playback_state(core); } static void audio_coreaudio_class_init(ObjectClass *klass, const void *data) { AudioMixengBackendClass *k = AUDIO_MIXENG_BACKEND_CLASS(klass); k->max_voices_out = 1; k->max_voices_in = 0; k->voice_size_out = sizeof(coreaudioVoiceOut); k->voice_size_in = 0; k->init_out = coreaudio_init_out; k->fini_out = coreaudio_fini_out; /* wrapper for audio_generic_write */ k->write = coreaudio_write; /* wrapper for audio_generic_buffer_get_free */ k->buffer_get_free = coreaudio_buffer_get_free; /* wrapper for audio_generic_get_buffer_out */ k->get_buffer_out = coreaudio_get_buffer_out; /* wrapper for audio_generic_put_buffer_out */ k->put_buffer_out = coreaudio_put_buffer_out; k->enable_out = coreaudio_enable_out; } static const TypeInfo audio_types[] = { { .name = TYPE_AUDIO_COREAUDIO, .parent = TYPE_AUDIO_MIXENG_BACKEND, .instance_size = sizeof(AudioCoreaudio), .class_init = audio_coreaudio_class_init, }, }; DEFINE_TYPES(audio_types) module_obj(TYPE_AUDIO_COREAUDIO);