diff --git a/modules/audio_output/auhal.c b/modules/audio_output/auhal.c index 872834610044ed885175569d82d04e6dab0f1bde..dba14015fc4261cc87dcbac1844c98f9cc5cbc69 100644 --- a/modules/audio_output/auhal.c +++ b/modules/audio_output/auhal.c @@ -38,6 +38,7 @@ #import <AudioUnit/AudioUnit.h> // AudioUnit #import <CoreAudio/CoreAudio.h> // AudioDeviceID #import <AudioToolbox/AudioFormat.h> // AudioFormatGetProperty +#import <CoreServices/CoreServices.h> #import "TPCircularBuffer.h" @@ -69,10 +70,10 @@ struct aout_sys_t { aout_packet_t packet; - AudioDeviceID i_default_dev; /* DeviceID of defaultOutputDevice */ - AudioDeviceID i_selected_dev; /* DeviceID of the selected device */ + AudioObjectID i_default_dev; /* DeviceID of defaultOutputDevice */ + AudioObjectID i_selected_dev; /* DeviceID of the selected device */ + bool b_selected_dev_is_digital; AudioDeviceIOProcID i_procID; /* DeviceID of current device */ - UInt32 i_devices; /* Number of CoreAudio Devices */ bool b_digital; /* Are we running in digital mode? */ mtime_t clock_diff; /* Difference between VLC clock and Device clock */ @@ -100,9 +101,19 @@ struct aout_sys_t mtime_t i_played_length; /* how much did we play already */ mtime_t i_last_sample_time; /* last sample time played by the AudioUnit */ + struct audio_device_t *devices; + vlc_mutex_t lock; }; +struct audio_device_t +{ + struct audio_device_t *next; + UInt32 deviceid; + char *name; + bool digital; +}; + /***************************************************************************** * Local prototypes. *****************************************************************************/ @@ -116,7 +127,9 @@ static void FlushAnalog (audio_output_t *, bool); static void PauseAnalog (audio_output_t *, bool, mtime_t); static int TimeGetAnalog (audio_output_t *, mtime_t *); -static void Probe (audio_output_t *); +static int DeviceList (audio_output_t *p_aout, char ***namesp, char ***descsp); +static void RebuildDeviceList (audio_output_t *); +static int SwitchAudioDevice (audio_output_t *p_aout, const char *name); static int AudioDeviceHasOutput (AudioDeviceID); static int AudioDeviceSupportsDigital(audio_output_t *, AudioDeviceID); @@ -129,8 +142,6 @@ static OSStatus RenderCallbackSPDIF (AudioDeviceID, const AudioTimeStamp *, AudioBufferList *, const AudioTimeStamp *, void *); static OSStatus HardwareListener (AudioObjectID, UInt32, const AudioObjectPropertyAddress *, void *); static OSStatus StreamListener (AudioObjectID, UInt32, const AudioObjectPropertyAddress *, void *); -static int AudioDeviceCallback (vlc_object_t *, const char *, - vlc_value_t, vlc_value_t, void *); static int VolumeSet (audio_output_t *, float); static int MuteSet (audio_output_t *, bool); @@ -139,11 +150,6 @@ static int MuteSet (audio_output_t *, bool); /***************************************************************************** * Module descriptor *****************************************************************************/ -#define ADEV_TEXT N_("Audio Device") -#define ADEV_LONGTEXT N_("Choose a number corresponding to the number of an " \ - "audio device, as listed in your 'Audio Device' menu. This device will " \ - "then be used by default for audio playback.") - #define VOLUME_TEXT N_("Audio volume") #define VOLUME_LONGTEXT VOLUME_TEXT @@ -154,10 +160,10 @@ vlc_module_begin () set_category(CAT_AUDIO) set_subcategory(SUBCAT_AUDIO_AOUT) set_callbacks(Open, Close) - add_integer("macosx-audio-device", 0, ADEV_TEXT, ADEV_LONGTEXT, false) add_integer("auhal-volume", AOUT_VOLUME_DEFAULT, VOLUME_TEXT, VOLUME_LONGTEXT, true) change_integer_range(0, AOUT_VOLUME_MAX) + add_obsolete_integer("macosx-audio-device") /* since 2.1.0 */ vlc_module_end () /***************************************************************************** @@ -168,16 +174,12 @@ static int Start(audio_output_t *p_aout, audio_sample_format_t *restrict fmt) OSStatus err = noErr; UInt32 i_param_size = 0; struct aout_sys_t *p_sys = NULL; - vlc_value_t val; /* Use int here, to match kAudioDevicePropertyDeviceIsAlive * property size */ int b_alive = false; p_sys = p_aout->sys; - p_sys->i_default_dev = 0; - p_sys->i_selected_dev = 0; - p_sys->i_devices = 0; p_sys->b_digital = false; p_sys->au_component = NULL; p_sys->au_unit = NULL; @@ -190,25 +192,11 @@ static int Start(audio_output_t *p_aout, audio_sample_format_t *restrict fmt) aout_FormatPrint(p_aout, "VLC is looking for:", fmt); - /* Persistent device variable */ - if (var_Type(p_aout->p_libvlc, "macosx-audio-device") == 0) - var_Create(p_aout->p_libvlc, "macosx-audio-device", VLC_VAR_INTEGER | VLC_VAR_DOINHERIT); - - /* Build a list of devices */ - if (var_Type(p_aout, "audio-device") == 0) - Probe(p_aout); - - /* What device do we want? */ - if (var_Get(p_aout, "audio-device", &val) < 0) { - msg_Err(p_aout, "audio-device var does not exist. device probe failed."); - goto error; - } - - p_sys->i_selected_dev = val.i_int & ~AOUT_VAR_SPDIF_FLAG; /* remove SPDIF flag to get the true DeviceID */ - bool b_supports_digital = (val.i_int & AOUT_VAR_SPDIF_FLAG); - if (b_supports_digital) + if (p_sys->b_selected_dev_is_digital) msg_Dbg(p_aout, "audio device supports digital output"); + msg_Dbg(p_aout, "attempting to use device %i", p_sys->i_selected_dev); + /* Check if the desired device is alive and usable */ i_param_size = sizeof(b_alive); AudioObjectPropertyAddress audioDeviceAliveAddress = { kAudioDevicePropertyDeviceIsAlive, @@ -256,11 +244,8 @@ static int Start(audio_output_t *p_aout, audio_sample_format_t *restrict fmt) goto error; } - /* If we change the device we want to use, we should renegotiate the audio chain */ - var_AddCallback(p_aout, "audio-device", AudioDeviceCallback, NULL); - /* Check for Digital mode or Analog output mode */ - if (AOUT_FMT_SPDIF (fmt) && b_supports_digital) { + if (AOUT_FMT_SPDIF (fmt) && p_sys->b_selected_dev_is_digital) { if (OpenSPDIF (p_aout, fmt)) { msg_Dbg(p_aout, "digital output successfully opened"); return VLC_SUCCESS; @@ -275,7 +260,6 @@ static int Start(audio_output_t *p_aout, audio_sample_format_t *restrict fmt) error: /* If we reach this, this aout has failed */ msg_Err(p_aout, "opening the auhal output failed"); - var_Destroy(p_aout, "audio-device"); return VLC_EGENERIC; } @@ -320,7 +304,7 @@ static int OpenAnalog(audio_output_t *p_aout, audio_sample_format_t *fmt) kAudioUnitScope_Global, 0, &p_sys->i_selected_dev, - sizeof(AudioDeviceID)); + sizeof(AudioObjectID)); if (err != noErr) { msg_Warn(p_aout, "we cannot select the audio device"); @@ -332,7 +316,7 @@ static int OpenAnalog(audio_output_t *p_aout, audio_sample_format_t *fmt) err = AudioUnitGetProperty(p_sys->au_unit, kAudioUnitProperty_StreamFormat, - kAudioUnitScope_Input, + kAudioUnitScope_Output, 0, &DeviceFormat, &i_param_size); @@ -376,7 +360,12 @@ static int OpenAnalog(audio_output_t *p_aout, audio_sample_format_t *fmt) layout)); } - msg_Dbg(p_aout, "layout of AUHAL has %d channels" , (int)layout->mNumberChannelDescriptions); + msg_Dbg(p_aout, "layout of AUHAL has %i channels" , layout->mNumberChannelDescriptions); + + if (layout->mNumberChannelDescriptions == 0) { + msg_Err(p_aout, "insufficient number of output channels"); + return false; + } /* Initialize the VLC core channel count */ fmt->i_physical_channels = 0; @@ -584,7 +573,6 @@ static int OpenAnalog(audio_output_t *p_aout, audio_sample_format_t *fmt) p_sys->clock_diff += mdate(); /* setup circular buffer */ - msg_Dbg( p_aout, "suggested buffer size: %i, default %i", DeviceFormat.mBytesPerPacket * 4, kBufferLength); TPCircularBufferInit(&p_sys->circular_buffer, kBufferLength); p_sys->b_got_first_sample = false; @@ -899,8 +887,6 @@ static void Stop(audio_output_t *p_aout) msg_Err(p_aout, "Could not release hogmode: [%4.4s]", (char *)&err); } - var_DelCallback(p_aout, "audio-device", AudioDeviceCallback, NULL); - p_sys->i_played_length = 0; p_sys->i_last_sample_time = 0; @@ -911,136 +897,150 @@ static void Stop(audio_output_t *p_aout) aout_PacketDestroy(p_aout); } -/***************************************************************************** - * Probe: Check which devices the OS has, and add them to our audio-device menu - *****************************************************************************/ -static void Probe(audio_output_t * p_aout) +static int DeviceList(audio_output_t *p_aout, char ***namesp, char ***descsp) +{ + struct aout_sys_t *p_sys = p_aout->sys; + char **names, **descs; + unsigned n = 0; + + for (struct audio_device_t *device = p_sys->devices; device != NULL; device = device->next) + n++; + + *namesp = names = xmalloc(sizeof(*names) * n); + *descsp = descs = xmalloc(sizeof(*descs) * n); + + for (struct audio_device_t *device = p_sys->devices; device != NULL; device = device->next) { + *(names++) = strdup(device->name); + sprintf(*(descs++), "%d", device->deviceid); + } + + msg_Dbg(p_aout, "returning a list of %i devices", n); + + return n; +} + +static void add_device_to_list(audio_output_t * p_aout, UInt32 i_id, char *name, bool b_digital) +{ + struct aout_sys_t *p_sys = p_aout->sys; + + msg_Dbg(p_aout, "adding device %i (%s) to list", i_id, name); + + struct audio_device_t *device = malloc(sizeof(*device)); + if (unlikely(device == NULL)) + return; + + device->next = p_sys->devices; + device->deviceid = i_id; + device->name = strdup(name); + device->digital = b_digital; + + p_sys->devices = device; +} + +static void RebuildDeviceList(audio_output_t * p_aout) { OSStatus err = noErr; - UInt32 i_param_size = 0; - AudioDeviceID devid_def = 0; - AudioDeviceID *p_devices = NULL; - vlc_value_t val, text; + UInt32 propertySize = 0; + AudioObjectID defaultDeviceID = 0; + AudioObjectID *deviceIDs; + UInt32 numberOfDevices; struct aout_sys_t *p_sys = p_aout->sys; + if (p_sys->devices) + free(p_sys->devices); + /* Get number of devices */ AudioObjectPropertyAddress audioDevicesAddress = { kAudioHardwarePropertyDevices, kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster }; - err = AudioObjectGetPropertyDataSize(kAudioObjectSystemObject, &audioDevicesAddress, 0, NULL, &i_param_size); + err = AudioObjectGetPropertyDataSize(kAudioObjectSystemObject, &audioDevicesAddress, 0, NULL, &propertySize); if (err != noErr) { msg_Err(p_aout, "Could not get number of devices: [%s]", (char *)&err); - goto error; + return; } - p_sys->i_devices = i_param_size / sizeof(AudioDeviceID); + numberOfDevices = propertySize / sizeof(AudioDeviceID); - if (p_sys->i_devices < 1) { + if (numberOfDevices < 1) { msg_Err(p_aout, "No audio output devices were found."); - goto error; + return; } - msg_Dbg(p_aout, "found %u audio device(s)", (unsigned)p_sys->i_devices); + msg_Dbg(p_aout, "found %i audio device(s)", numberOfDevices); /* Allocate DeviceID array */ - p_devices = (AudioDeviceID*)malloc(sizeof(AudioDeviceID) * p_sys->i_devices); - if (p_devices == NULL) - goto error; + deviceIDs = (AudioDeviceID *)calloc(numberOfDevices, sizeof(AudioDeviceID)); + if (deviceIDs == NULL) + return; /* Populate DeviceID array */ - err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &audioDevicesAddress, 0, NULL, &i_param_size, p_devices); + err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &audioDevicesAddress, 0, NULL, &propertySize, deviceIDs); if (err != noErr) { msg_Err(p_aout, "could not get the device IDs: [%s]", (char *)&err); - goto error; + return; } /* Find the ID of the default Device */ AudioObjectPropertyAddress defaultDeviceAddress = { kAudioHardwarePropertyDefaultOutputDevice, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster }; - i_param_size = sizeof(AudioDeviceID); - err= AudioObjectGetPropertyData(kAudioObjectSystemObject, &defaultDeviceAddress, 0, NULL, &i_param_size, &devid_def); + propertySize = sizeof(AudioObjectID); + err= AudioObjectGetPropertyData(kAudioObjectSystemObject, &defaultDeviceAddress, 0, NULL, &propertySize, &defaultDeviceID); if (err != noErr) { msg_Err(p_aout, "could not get default audio device: [%s]", (char *)&err); - goto error; + return; } - p_sys->i_default_dev = devid_def; - - var_Create(p_aout, "audio-device", VLC_VAR_INTEGER|VLC_VAR_HASCHOICE); - text.psz_string = (char*)_("Audio Device"); - var_Change(p_aout, "audio-device", VLC_VAR_SETTEXT, &text, NULL); + p_sys->i_default_dev = defaultDeviceID; - AudioObjectPropertyAddress deviceNameAddress = { kAudioDevicePropertyDeviceName, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster }; + AudioObjectPropertyAddress deviceNameAddress = { kAudioDevicePropertyDeviceName, kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster }; - for (unsigned int i = 0; i < p_sys->i_devices; i++) { + for (unsigned int i = 0; i < numberOfDevices; i++) { char *psz_name; - i_param_size = 0; + bool b_digital = false; + UInt32 i_id = deviceIDs[i]; /* Retrieve the length of the device name */ - err = AudioObjectGetPropertyDataSize(p_devices[i], &deviceNameAddress, 0, NULL, &i_param_size); - if (err != noErr) - goto error; + err = AudioObjectGetPropertyDataSize(deviceIDs[i], &deviceNameAddress, 0, NULL, &propertySize); + if (err != noErr) { + msg_Dbg(p_aout, "failed to get name size for device %i", deviceIDs[i]); + continue; + } /* Retrieve the name of the device */ - psz_name = (char *)malloc(i_param_size); - err = AudioObjectGetPropertyData(p_devices[i], &deviceNameAddress, 0, NULL, &i_param_size, psz_name); - if (err != noErr) - goto error; + psz_name = (char *)malloc(propertySize); + err = AudioObjectGetPropertyData(deviceIDs[i], &deviceNameAddress, 0, NULL, &propertySize, psz_name); + if (err != noErr) { + msg_Dbg(p_aout, "failed to get name for device %i", deviceIDs[i]); + continue; + } - msg_Dbg(p_aout, "DevID: %u DevName: %s", (unsigned)p_devices[i], psz_name); + msg_Dbg(p_aout, "DevID: %i DevName: %s", deviceIDs[i], psz_name); - if (!AudioDeviceHasOutput(p_devices[i])) { + if (!AudioDeviceHasOutput(deviceIDs[i])) { msg_Dbg(p_aout, "this device is INPUT only. skipping..."); free(psz_name); continue; } - /* Add the menu entries */ - val.i_int = (int)p_devices[i]; - text.psz_string = psz_name; - var_Change(p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val, &text); - text.psz_string = NULL; - if (p_sys->i_default_dev == p_devices[i]) { - /* The default device is the selected device normally */ - var_Change(p_aout, "audio-device", VLC_VAR_SETDEFAULT, &val, NULL); - var_Set(p_aout, "audio-device", val); - } - - if (AudioDeviceSupportsDigital(p_aout, p_devices[i])) { - val.i_int = (int)p_devices[i] | AOUT_VAR_SPDIF_FLAG; - if (asprintf(&text.psz_string, _("%s (Encoded Output)"), psz_name) != -1) { - var_Change(p_aout, "audio-device", VLC_VAR_ADDCHOICE, &val, &text); - free(text.psz_string); - if (p_sys->i_default_dev == p_devices[i] && var_InheritBool(p_aout, "spdif")) { - /* We selected to prefer SPDIF output if available - * then this "dummy" entry should be selected */ - var_Change(p_aout, "audio-device", VLC_VAR_SETDEFAULT, &val, NULL); - var_Set(p_aout, "audio-device", val); - } - } + if (AudioDeviceSupportsDigital(p_aout, deviceIDs[i])) { + b_digital = true; + msg_Dbg(p_aout, "this device supports digital"); + asprintf(&psz_name, _("%s (Encoded Output)"), psz_name); } + add_device_to_list(p_aout, i_id, psz_name, b_digital); free(psz_name); } - /* If a device is already "preselected", then use this device */ - var_Get(p_aout->p_libvlc, "macosx-audio-device", &val); - if (val.i_int > 0) { - msg_Dbg(p_aout, "using preselected output device %#"PRIx64, val.i_int); - var_Change(p_aout, "audio-device", VLC_VAR_SETDEFAULT, &val, NULL); - var_Set(p_aout, "audio-device", val); - } - /* Attach a Listener so that we are notified of a change in the Device setup */ err = AudioObjectAddPropertyListener(kAudioObjectSystemObject, &audioDevicesAddress, HardwareListener, (void *)p_aout); - if (err != noErr) { + if (err != noErr) msg_Warn(p_aout, "failed to add listener for audio device configuration (%i)", err); - goto error; - } - free(p_devices); - return; + free(deviceIDs); +} -error: - msg_Warn(p_aout, "audio device already in use"); - free(p_devices); - return; +static int SwitchAudioDevice(audio_output_t *p_aout, const char *name) +{ + msg_Warn(p_aout, "we should switch to device '%s'", name); + + return 0; } /***************************************************************************** @@ -1069,7 +1069,7 @@ static int AudioDeviceSupportsDigital(audio_output_t *p_aout, AudioDeviceID i_de UInt32 i_param_size = 0; AudioStreamID *p_streams = NULL; int i_streams = 0; - bool b_return = false; + bool b_return = false; /* Retrieve all the output streams */ AudioObjectPropertyAddress streamsAddress = { kAudioDevicePropertyStreams, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster }; @@ -1429,8 +1429,7 @@ static OSStatus HardwareListener(AudioObjectID inObjectID, UInt32 inNumberAddre } #endif - var_TriggerCallback(p_aout, "audio-device"); - var_Destroy(p_aout, "audio-device"); + RebuildDeviceList(p_aout); return err; } @@ -1456,19 +1455,6 @@ static OSStatus StreamListener(AudioObjectID inObjectID, UInt32 inNumberAddress return err; } -/***************************************************************************** - * AudioDeviceCallback: Callback triggered when the audio-device variable is changed - *****************************************************************************/ -static int AudioDeviceCallback(vlc_object_t *p_this, const char *psz_variable, - vlc_value_t old_val, vlc_value_t new_val, void *param) -{ - audio_output_t *p_aout = (audio_output_t *)p_this; - var_Set(p_aout->p_libvlc, "macosx-audio-device", new_val); - msg_Dbg(p_aout, "Set Device: %#"PRIx64, new_val.i_int); - return aout_ChannelsRestart(p_this, psz_variable, old_val, new_val, param); -} - - /***************************************************************************** * VolumeSet: Implements volume_set(). Update the CoreAudio AU volume immediately. *****************************************************************************/ @@ -1530,6 +1516,11 @@ static int Open(vlc_object_t *obj) aout->stop = Stop; aout->volume_set = VolumeSet; aout->mute_set = MuteSet; + aout->device_enum = DeviceList; + aout->sys->devices = NULL; + aout->device_select = SwitchAudioDevice; + + RebuildDeviceList(aout); /* remember the volume */ aout_VolumeReport(aout, var_InheritInteger(aout, "auhal-volume") / (float)AOUT_VOLUME_DEFAULT);