mirror of
https://github.com/mpv-player/mpv.git
synced 2025-12-28 05:33:14 +00:00
This may or may not fix some issues with the format switching code. Actually, it seems somewhat unlikely, but then checking the stream type isn't incorrect either, and is probably something the API user should always be doing.
376 lines
12 KiB
C
376 lines
12 KiB
C
/*
|
|
* CoreAudio audio output driver for Mac OS X
|
|
*
|
|
* original copyright (C) Timothy J. Wood - Aug 2000
|
|
* ported to MPlayer libao2 by Dan Christiansen
|
|
*
|
|
* Chris Roccati
|
|
* Stefano Pigozzi
|
|
*
|
|
* The S/PDIF part of the code is based on the auhal audio output
|
|
* module from VideoLAN:
|
|
* Copyright (c) 2006 Derk-Jan Hartman <hartman at videolan dot org>
|
|
*
|
|
* This file is part of mpv.
|
|
*
|
|
* mpv is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* mpv is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License along
|
|
* with mpv. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
/*
|
|
* The MacOS X CoreAudio framework doesn't mesh as simply as some
|
|
* simpler frameworks do. This is due to the fact that CoreAudio pulls
|
|
* audio samples rather than having them pushed at it (which is nice
|
|
* when you are wanting to do good buffering of audio).
|
|
*/
|
|
|
|
#include <CoreAudio/HostTime.h>
|
|
|
|
#include "config.h"
|
|
#include "ao.h"
|
|
#include "internal.h"
|
|
#include "audio/format.h"
|
|
#include "osdep/timer.h"
|
|
#include "osdep/atomics.h"
|
|
#include "options/m_option.h"
|
|
#include "common/msg.h"
|
|
#include "audio/out/ao_coreaudio_properties.h"
|
|
#include "audio/out/ao_coreaudio_utils.h"
|
|
|
|
struct priv {
|
|
AudioDeviceID device; // selected device
|
|
|
|
bool paused;
|
|
|
|
// audio render callback
|
|
AudioDeviceIOProcID render_cb;
|
|
|
|
// pid set for hog mode, (-1) means that hog mode on the device was
|
|
// released. hog mode is exclusive access to a device
|
|
pid_t hog_pid;
|
|
|
|
AudioStreamID stream;
|
|
|
|
// stream index in an AudioBufferList
|
|
int stream_idx;
|
|
|
|
// format we changed the stream to, and the original format to restore
|
|
AudioStreamBasicDescription stream_asbd;
|
|
AudioStreamBasicDescription original_asbd;
|
|
|
|
bool changed_mixing;
|
|
|
|
atomic_bool reload_requested;
|
|
|
|
uint32_t hw_latency_us;
|
|
};
|
|
|
|
static OSStatus property_listener_cb(
|
|
AudioObjectID object, uint32_t n_addresses,
|
|
const AudioObjectPropertyAddress addresses[],
|
|
void *data)
|
|
{
|
|
struct ao *ao = data;
|
|
struct priv *p = ao->priv;
|
|
|
|
// Check whether we need to reset the compressed output stream.
|
|
AudioStreamBasicDescription f;
|
|
OSErr err = CA_GET(p->stream, kAudioStreamPropertyPhysicalFormat, &f);
|
|
CHECK_CA_WARN("could not get stream format");
|
|
if (err != noErr || !ca_asbd_equals(&p->stream_asbd, &f)) {
|
|
if (atomic_compare_exchange_strong(&p->reload_requested,
|
|
&(bool){false}, true))
|
|
{
|
|
ao_request_reload(ao);
|
|
MP_INFO(ao, "Stream format changed! Reloading.\n");
|
|
}
|
|
}
|
|
|
|
return noErr;
|
|
}
|
|
|
|
static OSStatus enable_property_listener(struct ao *ao, bool enabled)
|
|
{
|
|
struct priv *p = ao->priv;
|
|
|
|
uint32_t selectors[] = {kAudioDevicePropertyDeviceHasChanged,
|
|
kAudioHardwarePropertyDevices};
|
|
AudioDeviceID devs[] = {p->device,
|
|
kAudioObjectSystemObject};
|
|
assert(MP_ARRAY_SIZE(selectors) == MP_ARRAY_SIZE(devs));
|
|
|
|
OSStatus status = noErr;
|
|
for (int n = 0; n < MP_ARRAY_SIZE(devs); n++) {
|
|
AudioObjectPropertyAddress addr = {
|
|
.mScope = kAudioObjectPropertyScopeGlobal,
|
|
.mElement = kAudioObjectPropertyElementMaster,
|
|
.mSelector = selectors[n],
|
|
};
|
|
AudioDeviceID device = devs[n];
|
|
|
|
OSStatus status2;
|
|
if (enabled) {
|
|
status2 = AudioObjectAddPropertyListener(
|
|
device, &addr, property_listener_cb, ao);
|
|
} else {
|
|
status2 = AudioObjectRemovePropertyListener(
|
|
device, &addr, property_listener_cb, ao);
|
|
}
|
|
if (status == noErr)
|
|
status = status2;
|
|
}
|
|
|
|
return status;
|
|
}
|
|
|
|
static OSStatus render_cb_compressed(
|
|
AudioDeviceID device, const AudioTimeStamp *ts,
|
|
const void *in_data, const AudioTimeStamp *in_ts,
|
|
AudioBufferList *out_data, const AudioTimeStamp *out_ts, void *ctx)
|
|
{
|
|
struct ao *ao = ctx;
|
|
struct priv *p = ao->priv;
|
|
AudioBuffer buf = out_data->mBuffers[p->stream_idx];
|
|
int requested = buf.mDataByteSize;
|
|
|
|
int pseudo_frames = requested / ao->sstride;
|
|
|
|
// we expect the callback to read full frames, which are aligned accordingly
|
|
if (pseudo_frames * ao->sstride != requested) {
|
|
MP_ERR(ao, "Unsupported unaligned read of %d bytes.\n", requested);
|
|
return kAudioHardwareUnspecifiedError;
|
|
}
|
|
|
|
int64_t end = mp_time_us();
|
|
end += p->hw_latency_us + ca_get_latency(ts)
|
|
+ ca_frames_to_us(ao, pseudo_frames);
|
|
|
|
ao_read_data(ao, &buf.mData, pseudo_frames, end);
|
|
|
|
return noErr;
|
|
}
|
|
|
|
static int init(struct ao *ao)
|
|
{
|
|
struct priv *p = ao->priv;
|
|
|
|
OSStatus err = ca_select_device(ao, ao->device, &p->device);
|
|
CHECK_CA_ERROR_L(coreaudio_error_nounlock, "failed to select device");
|
|
|
|
ao->format = af_fmt_from_planar(ao->format);
|
|
|
|
if (!AF_FORMAT_IS_IEC61937(ao->format)) {
|
|
MP_ERR(ao, "Only compressed formats are supported.\n");
|
|
goto coreaudio_error_nounlock;
|
|
}
|
|
|
|
if (!ca_device_supports_compressed(ao, p->device)) {
|
|
MP_ERR(ao, "selected device doesn't support compressed formats\n");
|
|
goto coreaudio_error_nounlock;
|
|
}
|
|
|
|
// Build ASBD for the input format
|
|
AudioStreamBasicDescription asbd;
|
|
ca_fill_asbd(ao, &asbd);
|
|
|
|
uint32_t is_alive = 1;
|
|
err = CA_GET(p->device, kAudioDevicePropertyDeviceIsAlive, &is_alive);
|
|
CHECK_CA_WARN("could not check whether device is alive");
|
|
|
|
if (!is_alive)
|
|
MP_WARN(ao , "device is not alive\n");
|
|
|
|
err = ca_lock_device(p->device, &p->hog_pid);
|
|
CHECK_CA_WARN("failed to set hogmode");
|
|
|
|
err = ca_disable_mixing(ao, p->device, &p->changed_mixing);
|
|
CHECK_CA_WARN("failed to disable mixing");
|
|
|
|
AudioStreamID *streams;
|
|
size_t n_streams;
|
|
|
|
/* Get a list of all the streams on this device. */
|
|
err = CA_GET_ARY_O(p->device, kAudioDevicePropertyStreams,
|
|
&streams, &n_streams);
|
|
|
|
CHECK_CA_ERROR("could not get number of streams");
|
|
|
|
for (int i = 0; i < n_streams && p->stream_idx < 0; i++) {
|
|
uint32_t direction;
|
|
err = CA_GET(streams[i], kAudioStreamPropertyDirection, &direction);
|
|
CHECK_CA_ERROR("could not get stream direction");
|
|
if (direction != 0) {
|
|
MP_VERBOSE(ao, "Substream %d is not an output stream.\n", i);
|
|
continue;
|
|
}
|
|
|
|
bool compressed = ca_stream_supports_compressed(ao, streams[i]);
|
|
|
|
if (compressed) {
|
|
AudioStreamRangedDescription *formats;
|
|
size_t n_formats;
|
|
|
|
err = CA_GET_ARY(streams[i],
|
|
kAudioStreamPropertyAvailablePhysicalFormats,
|
|
&formats, &n_formats);
|
|
|
|
if (!CHECK_CA_WARN("could not get number of stream formats"))
|
|
continue; // try next one
|
|
|
|
int req_rate_format = -1;
|
|
int max_rate_format = -1;
|
|
|
|
p->stream = streams[i];
|
|
p->stream_idx = i;
|
|
|
|
for (int j = 0; j < n_formats; j++)
|
|
if (ca_formatid_is_compressed(formats[j].mFormat.mFormatID)) {
|
|
// select the compressed format that has exactly the same
|
|
// samplerate. If an exact match cannot be found, select
|
|
// the format with highest samplerate as backup.
|
|
if (formats[j].mFormat.mSampleRate == asbd.mSampleRate) {
|
|
req_rate_format = j;
|
|
break;
|
|
} else if (max_rate_format < 0 ||
|
|
formats[j].mFormat.mSampleRate >
|
|
formats[max_rate_format].mFormat.mSampleRate)
|
|
max_rate_format = j;
|
|
}
|
|
|
|
if (req_rate_format >= 0)
|
|
p->stream_asbd = formats[req_rate_format].mFormat;
|
|
else
|
|
p->stream_asbd = formats[max_rate_format].mFormat;
|
|
|
|
talloc_free(formats);
|
|
}
|
|
}
|
|
|
|
talloc_free(streams);
|
|
|
|
if (p->stream_idx < 0) {
|
|
MP_WARN(ao , "can't find any compressed output stream format\n");
|
|
goto coreaudio_error;
|
|
}
|
|
|
|
err = CA_GET(p->stream, kAudioStreamPropertyPhysicalFormat,
|
|
&p->original_asbd);
|
|
CHECK_CA_ERROR("could not get stream's original physical format");
|
|
|
|
if (!ca_change_physical_format_sync(ao, p->stream, p->stream_asbd))
|
|
goto coreaudio_error;
|
|
|
|
err = enable_property_listener(ao, true);
|
|
CHECK_CA_ERROR("cannot install format change listener during init");
|
|
|
|
if (p->stream_asbd.mFormatFlags & kAudioFormatFlagIsBigEndian)
|
|
MP_WARN(ao, "stream has non-native byte order, output may fail\n");
|
|
|
|
ao->samplerate = p->stream_asbd.mSampleRate;
|
|
ao->bps = ao->samplerate *
|
|
(p->stream_asbd.mBytesPerPacket /
|
|
p->stream_asbd.mFramesPerPacket);
|
|
|
|
uint32_t latency_frames = 0;
|
|
uint32_t latency_properties[] = {
|
|
kAudioDevicePropertyLatency,
|
|
kAudioDevicePropertyBufferFrameSize,
|
|
kAudioDevicePropertySafetyOffset,
|
|
};
|
|
for (int n = 0; n < MP_ARRAY_SIZE(latency_properties); n++) {
|
|
uint32_t temp;
|
|
err = CA_GET_O(p->device, latency_properties[n], &temp);
|
|
CHECK_CA_WARN("cannot get device latency");
|
|
if (err == noErr)
|
|
latency_frames += temp;
|
|
}
|
|
|
|
p->hw_latency_us = ca_frames_to_us(ao, latency_frames);
|
|
MP_VERBOSE(ao, "base latency: %d microseconds\n", (int)p->hw_latency_us);
|
|
|
|
err = AudioDeviceCreateIOProcID(p->device,
|
|
(AudioDeviceIOProc)render_cb_compressed,
|
|
(void *)ao,
|
|
&p->render_cb);
|
|
CHECK_CA_ERROR("failed to register audio render callback");
|
|
|
|
return CONTROL_TRUE;
|
|
|
|
coreaudio_error:
|
|
err = enable_property_listener(ao, false);
|
|
CHECK_CA_WARN("can't remove format change listener");
|
|
err = ca_unlock_device(p->device, &p->hog_pid);
|
|
CHECK_CA_WARN("can't release hog mode");
|
|
coreaudio_error_nounlock:
|
|
return CONTROL_ERROR;
|
|
}
|
|
|
|
static void uninit(struct ao *ao)
|
|
{
|
|
struct priv *p = ao->priv;
|
|
OSStatus err = noErr;
|
|
|
|
err = enable_property_listener(ao, false);
|
|
CHECK_CA_WARN("can't remove device listener, this may cause a crash");
|
|
|
|
err = AudioDeviceStop(p->device, p->render_cb);
|
|
CHECK_CA_WARN("failed to stop audio device");
|
|
|
|
err = AudioDeviceDestroyIOProcID(p->device, p->render_cb);
|
|
CHECK_CA_WARN("failed to remove device render callback");
|
|
|
|
if (!ca_change_physical_format_sync(ao, p->stream, p->original_asbd))
|
|
MP_WARN(ao, "can't revert to original device format");
|
|
|
|
err = ca_enable_mixing(ao, p->device, p->changed_mixing);
|
|
CHECK_CA_WARN("can't re-enable mixing");
|
|
|
|
err = ca_unlock_device(p->device, &p->hog_pid);
|
|
CHECK_CA_WARN("can't release hog mode");
|
|
}
|
|
|
|
static void audio_pause(struct ao *ao)
|
|
{
|
|
struct priv *p = ao->priv;
|
|
|
|
OSStatus err = AudioDeviceStop(p->device, p->render_cb);
|
|
CHECK_CA_WARN("can't stop audio device");
|
|
}
|
|
|
|
static void audio_resume(struct ao *ao)
|
|
{
|
|
struct priv *p = ao->priv;
|
|
|
|
OSStatus err = AudioDeviceStart(p->device, p->render_cb);
|
|
CHECK_CA_WARN("can't start audio device");
|
|
}
|
|
|
|
#define OPT_BASE_STRUCT struct priv
|
|
|
|
const struct ao_driver audio_out_coreaudio_exclusive = {
|
|
.description = "CoreAudio Exclusive Mode",
|
|
.name = "coreaudio_exclusive",
|
|
.uninit = uninit,
|
|
.init = init,
|
|
.pause = audio_pause,
|
|
.resume = audio_resume,
|
|
.list_devs = ca_get_device_list,
|
|
.priv_size = sizeof(struct priv),
|
|
.priv_defaults = &(const struct priv){
|
|
.hog_pid = -1,
|
|
.stream = 0,
|
|
.stream_idx = -1,
|
|
.changed_mixing = false,
|
|
},
|
|
};
|