Commit 6471e0ed authored by Felix Paul Kühne's avatar Felix Paul Kühne

auhal: updated to the latest API

the newly used API was introduced in OS X 10.5, while we still tried to use an older one

note that optical audio output is completely untested
parent 925b2255
/***************************************************************************** /*****************************************************************************
* auhal.c: AUHAL and Coreaudio output plugin * auhal.c: AUHAL and Coreaudio output plugin
***************************************************************************** *****************************************************************************
* Copyright (C) 2005 the VideoLAN team * Copyright (C) 2005, 2011 the VideoLAN team
* $Id$ * $Id$
* *
* Authors: Derk-Jan Hartman <hartman at videolan dot org> * Authors: Derk-Jan Hartman <hartman at videolan dot org>
* Felix Paul Kühne <fkuehne -at- videolan dot org>
* *
* This program is free software; you can redistribute it and/or modify * This program is free software; you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
...@@ -35,33 +36,16 @@ ...@@ -35,33 +36,16 @@
#include <vlc_dialog.h> #include <vlc_dialog.h>
#include <vlc_aout.h> #include <vlc_aout.h>
// By pass part of header which compile with some warnings,
// and that we don't require.
#define __MACHINEEXCEPTIONS__
#include <CoreAudio/CoreAudio.h> #include <CoreAudio/CoreAudio.h>
#include <AudioUnit/AudioUnit.h> #include <AudioUnit/AudioUnit.h>
#include <AudioUnit/AudioUnitProperties.h>
#include <AudioUnit/AudioUnitParameters.h>
#include <AudioUnit/AudioOutputUnit.h>
#include <AudioToolbox/AudioFormat.h> #include <AudioToolbox/AudioFormat.h>
#include <CoreServices/CoreServices.h>
#ifndef verify_noerr #ifndef verify_noerr
#define verify_noerr(a) assert((a) == noErr) #define verify_noerr(a) assert((a) == noErr)
#endif #endif
#if AUDIO_UNIT_VERSION < 1060
#define AudioComponent Component
#define AudioComponentDescription ComponentDescription
#define AudioComponentFindNext FindNextComponent
#define AudioComponentInstanceNew OpenAComponent
#define AudioComponentInstanceDispose CloseComponent
#define AudioComponentInstanceNew OpenAComponent
#define AudioComponentInstanceNew OpenAComponent
#else
#include <AudioUnit/AudioComponent.h>
#endif
#define STREAM_FORMAT_MSG( pre, sfm ) \ #define STREAM_FORMAT_MSG( pre, sfm ) \
pre "[%u][%4.4s][%u][%u][%u][%u][%u][%u]", \ pre "[%u][%4.4s][%u][%u][%u][%u][%u][%u]", \
(UInt32)sfm.mSampleRate, (char *)&sfm.mFormatID, \ (UInt32)sfm.mSampleRate, (char *)&sfm.mFormatID, \
...@@ -83,7 +67,6 @@ ...@@ -83,7 +67,6 @@
/* /*
* TODO: * TODO:
* - clean up the debug info * - clean up the debug info
* - clean up C99'isms
* - be better at changing stream setup or devices setup changes while playing. * - be better at changing stream setup or devices setup changes while playing.
* - fix 6.1 and 7.1 * - fix 6.1 and 7.1
*/ */
...@@ -98,13 +81,14 @@ struct aout_sys_t ...@@ -98,13 +81,14 @@ struct aout_sys_t
{ {
AudioDeviceID i_default_dev; /* Keeps DeviceID of defaultOutputDevice */ AudioDeviceID i_default_dev; /* Keeps DeviceID of defaultOutputDevice */
AudioDeviceID i_selected_dev; /* Keeps DeviceID of the selected device */ AudioDeviceID i_selected_dev; /* Keeps DeviceID of the selected device */
AudioDeviceIOProcID i_procID; /* DeviceID of current device */
UInt32 i_devices; /* Number of CoreAudio Devices */ UInt32 i_devices; /* Number of CoreAudio Devices */
bool b_supports_digital;/* Does the currently selected device support digital mode? */ bool b_supports_digital;/* Does the currently selected device support digital mode? */
bool b_digital; /* Are we running in digital mode? */ bool b_digital; /* Are we running in digital mode? */
mtime_t clock_diff; /* Difference between VLC clock and Device clock */ mtime_t clock_diff; /* Difference between VLC clock and Device clock */
/* AUHAL specific */ /* AUHAL specific */
AudioComponent au_component; /* The Audiocomponent we use */ Component au_component; /* The Audiocomponent we use */
AudioUnit au_unit; /* The AudioUnit we use */ AudioUnit au_unit; /* The AudioUnit we use */
uint8_t p_remainder_buffer[BUFSIZE]; uint8_t p_remainder_buffer[BUFSIZE];
uint32_t i_read_bytes; uint32_t i_read_bytes;
...@@ -140,9 +124,8 @@ static OSStatus RenderCallbackAnalog ( vlc_object_t *, AudioUnitRenderActionF ...@@ -140,9 +124,8 @@ static OSStatus RenderCallbackAnalog ( vlc_object_t *, AudioUnitRenderActionF
unsigned int, unsigned int, AudioBufferList *); unsigned int, unsigned int, AudioBufferList *);
static OSStatus RenderCallbackSPDIF ( AudioDeviceID, const AudioTimeStamp *, const void *, const AudioTimeStamp *, static OSStatus RenderCallbackSPDIF ( AudioDeviceID, const AudioTimeStamp *, const void *, const AudioTimeStamp *,
AudioBufferList *, const AudioTimeStamp *, void * ); AudioBufferList *, const AudioTimeStamp *, void * );
static OSStatus HardwareListener ( AudioHardwarePropertyID, void *); static OSStatus HardwareListener ( AudioObjectID, UInt32, const AudioObjectPropertyAddress *, void * );
static OSStatus StreamListener ( AudioStreamID, UInt32, static OSStatus StreamListener ( AudioObjectID, UInt32, const AudioObjectPropertyAddress *, void * );
AudioDevicePropertyID, void * );
static int AudioDeviceCallback ( vlc_object_t *, const char *, static int AudioDeviceCallback ( vlc_object_t *, const char *,
vlc_value_t, vlc_value_t, void * ); vlc_value_t, vlc_value_t, void * );
...@@ -229,13 +212,18 @@ static int Open( vlc_object_t * p_this ) ...@@ -229,13 +212,18 @@ static int Open( vlc_object_t * p_this )
p_sys->i_selected_dev = val.i_int & ~AOUT_VAR_SPDIF_FLAG; /* remove SPDIF flag to get the true DeviceID */ p_sys->i_selected_dev = val.i_int & ~AOUT_VAR_SPDIF_FLAG; /* remove SPDIF flag to get the true DeviceID */
p_sys->b_supports_digital = ( val.i_int & AOUT_VAR_SPDIF_FLAG ) ? true : false; p_sys->b_supports_digital = ( val.i_int & AOUT_VAR_SPDIF_FLAG ) ? true : false;
if( p_sys->b_supports_digital )
msg_Dbg( p_aout, "audio-device supports digital output" );
else
msg_Dbg( p_aout, "audio-device does not support digital output" );
/* Check if the desired device is alive and usable */ /* Check if the desired device is alive and usable */
/* TODO: add a callback to the device to alert us if the device dies */ /* TODO: add a callback to the device to alert us if the device dies */
i_param_size = sizeof( b_alive ); i_param_size = sizeof( b_alive );
err = AudioDeviceGetProperty( p_sys->i_selected_dev, 0, FALSE, AudioObjectPropertyAddress audioDeviceAliveAddress = { kAudioDevicePropertyDeviceIsAlive,
kAudioDevicePropertyDeviceIsAlive, kAudioDevicePropertyScopeOutput,
&i_param_size, &b_alive ); kAudioObjectPropertyElementMaster };
err = AudioObjectGetPropertyData( p_sys->i_selected_dev, &audioDeviceAliveAddress, 0, NULL, &i_param_size, &b_alive );
if( err != noErr ) if( err != noErr )
{ {
...@@ -244,17 +232,19 @@ static int Open( vlc_object_t * p_this ) ...@@ -244,17 +232,19 @@ static int Open( vlc_object_t * p_this )
b_alive = false; b_alive = false;
} }
if( b_alive == false ) if( !b_alive )
{ {
msg_Warn( p_aout, "selected audio device is not alive, switching to default device" ); msg_Warn( p_aout, "selected audio device is not alive, switching to default device" );
p_sys->i_selected_dev = p_sys->i_default_dev; p_sys->i_selected_dev = p_sys->i_default_dev;
} }
else
msg_Dbg( p_aout, "selected audio device is alive" );
AudioObjectPropertyAddress audioDeviceHogModeAddress = { kAudioDevicePropertyHogMode,
kAudioDevicePropertyScopeOutput,
kAudioObjectPropertyElementMaster };
i_param_size = sizeof( p_sys->i_hog_pid ); i_param_size = sizeof( p_sys->i_hog_pid );
err = AudioDeviceGetProperty( p_sys->i_selected_dev, 0, FALSE, err = AudioObjectGetPropertyData( p_sys->i_selected_dev, &audioDeviceHogModeAddress, 0, NULL, &i_param_size, &p_sys->i_hog_pid );
kAudioDevicePropertyHogMode,
&i_param_size, &p_sys->i_hog_pid );
if( err != noErr ) if( err != noErr )
{ {
/* This is not a fatal error. Some drivers simply don't support this property */ /* This is not a fatal error. Some drivers simply don't support this property */
...@@ -271,21 +261,30 @@ static int Open( vlc_object_t * p_this ) ...@@ -271,21 +261,30 @@ static int Open( vlc_object_t * p_this )
"use by another program.") ); "use by another program.") );
goto error; goto error;
} }
else
msg_Dbg( p_aout, "device is free for us to use" );
/* Check for Digital mode or Analog output mode */ /* Check for Digital mode or Analog output mode */
if( AOUT_FMT_NON_LINEAR( &p_aout->output.output ) && p_sys->b_supports_digital ) if( AOUT_FMT_NON_LINEAR( &p_aout->output.output ) && p_sys->b_supports_digital )
{ {
if( OpenSPDIF( p_aout ) ) if( OpenSPDIF( p_aout ) )
{
msg_Dbg( p_aout, "digital output successfully opened" );
return VLC_SUCCESS; return VLC_SUCCESS;
} }
}
else else
{ {
if( OpenAnalog( p_aout ) ) if( OpenAnalog( p_aout ) )
{
msg_Dbg( p_aout, "analog output successfully opened" );
return VLC_SUCCESS; return VLC_SUCCESS;
} }
}
error: error:
/* If we reach this, this aout has failed */ /* If we reach this, this aout has failed */
msg_Err( p_aout, "opening the auhal output failed" );
var_Destroy( p_aout, "audio-device" ); var_Destroy( p_aout, "audio-device" );
free( p_sys ); free( p_sys );
return VLC_EGENERIC; return VLC_EGENERIC;
...@@ -300,7 +299,7 @@ static int OpenAnalog( aout_instance_t *p_aout ) ...@@ -300,7 +299,7 @@ static int OpenAnalog( aout_instance_t *p_aout )
OSStatus err = noErr; OSStatus err = noErr;
UInt32 i_param_size = 0, i = 0; UInt32 i_param_size = 0, i = 0;
int i_original; int i_original;
AudioComponentDescription desc; ComponentDescription desc;
AudioStreamBasicDescription DeviceFormat; AudioStreamBasicDescription DeviceFormat;
AudioChannelLayout *layout; AudioChannelLayout *layout;
AudioChannelLayout new_layout; AudioChannelLayout new_layout;
...@@ -313,14 +312,14 @@ static int OpenAnalog( aout_instance_t *p_aout ) ...@@ -313,14 +312,14 @@ static int OpenAnalog( aout_instance_t *p_aout )
desc.componentFlags = 0; desc.componentFlags = 0;
desc.componentFlagsMask = 0; desc.componentFlagsMask = 0;
p_sys->au_component = AudioComponentFindNext( NULL, &desc ); p_sys->au_component = FindNextComponent( NULL, &desc );
if( p_sys->au_component == NULL ) if( p_sys->au_component == NULL )
{ {
msg_Warn( p_aout, "we cannot find our HAL component" ); msg_Warn( p_aout, "we cannot find our HAL component" );
return false; return false;
} }
err = AudioComponentInstanceNew( p_sys->au_component, &p_sys->au_unit ); err = OpenAComponent( p_sys->au_component, &p_sys->au_unit );
if( err != noErr ) if( err != noErr )
{ {
msg_Warn( p_aout, "we cannot open our HAL component" ); msg_Warn( p_aout, "we cannot open our HAL component" );
...@@ -623,11 +622,11 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -623,11 +622,11 @@ static int OpenSPDIF( aout_instance_t * p_aout )
p_sys->b_digital = true; p_sys->b_digital = true;
/* Hog the device */ /* Hog the device */
AudioObjectPropertyAddress audioDeviceHogModeAddress = { kAudioDevicePropertyHogMode, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
i_param_size = sizeof( p_sys->i_hog_pid ); i_param_size = sizeof( p_sys->i_hog_pid );
p_sys->i_hog_pid = getpid() ; p_sys->i_hog_pid = getpid() ;
err = AudioDeviceSetProperty( p_sys->i_selected_dev, 0, 0, FALSE, err = AudioObjectSetPropertyData( p_sys->i_selected_dev, &audioDeviceHogModeAddress, 0, NULL, i_param_size, &p_sys->i_hog_pid );
kAudioDevicePropertyHogMode, i_param_size, &p_sys->i_hog_pid );
if( err != noErr ) if( err != noErr )
{ {
...@@ -636,17 +635,15 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -636,17 +635,15 @@ static int OpenSPDIF( aout_instance_t * p_aout )
} }
/* Set mixable to false if we are allowed to */ /* Set mixable to false if we are allowed to */
err = AudioDeviceGetPropertyInfo( p_sys->i_selected_dev, 0, FALSE, kAudioDevicePropertySupportsMixing, AudioObjectPropertyAddress audioDeviceSupportsMixingAddress = { kAudioDevicePropertySupportsMixing , kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
&i_param_size, &b_writeable ); b_writeable = AudioObjectHasProperty( p_sys->i_selected_dev, &audioDeviceSupportsMixingAddress );
err = AudioObjectGetPropertyDataSize( p_sys->i_selected_dev, &audioDeviceSupportsMixingAddress, 0, NULL, &i_param_size );
err = AudioDeviceGetProperty( p_sys->i_selected_dev, 0, FALSE, kAudioDevicePropertySupportsMixing, err = AudioObjectGetPropertyData( p_sys->i_selected_dev, &audioDeviceSupportsMixingAddress, 0, NULL, &i_param_size, &b_mix );
&i_param_size, &b_mix );
if( !err && b_writeable ) if( !err && b_writeable )
{ {
b_mix = 0; b_mix = 0;
err = AudioDeviceSetProperty( p_sys->i_selected_dev, 0, 0, FALSE, err = AudioObjectSetPropertyData( p_sys->i_selected_dev, &audioDeviceSupportsMixingAddress, 0, NULL, i_param_size, &b_mix );
kAudioDevicePropertySupportsMixing, i_param_size, &b_mix );
p_sys->b_changed_mixing = true; p_sys->b_changed_mixing = true;
} }
...@@ -657,9 +654,8 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -657,9 +654,8 @@ static int OpenSPDIF( aout_instance_t * p_aout )
} }
/* Get a list of all the streams on this device */ /* Get a list of all the streams on this device */
err = AudioDeviceGetPropertyInfo( p_sys->i_selected_dev, 0, FALSE, AudioObjectPropertyAddress streamsAddress = { kAudioDevicePropertyStreams, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
kAudioDevicePropertyStreams, err = AudioObjectGetPropertyDataSize( p_sys->i_selected_dev, &streamsAddress, 0, NULL, &i_param_size );
&i_param_size, NULL );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not get number of streams: [%4.4s]", (char *)&err ); msg_Err( p_aout, "could not get number of streams: [%4.4s]", (char *)&err );
...@@ -671,9 +667,7 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -671,9 +667,7 @@ static int OpenSPDIF( aout_instance_t * p_aout )
if( p_streams == NULL ) if( p_streams == NULL )
return false; return false;
err = AudioDeviceGetProperty( p_sys->i_selected_dev, 0, FALSE, err = AudioObjectGetPropertyData( p_sys->i_selected_dev, &streamsAddress, 0, NULL, &i_param_size, p_streams );
kAudioDevicePropertyStreams,
&i_param_size, p_streams );
if( err != noErr ) if( err != noErr )
{ {
...@@ -682,6 +676,7 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -682,6 +676,7 @@ static int OpenSPDIF( aout_instance_t * p_aout )
return false; return false;
} }
AudioObjectPropertyAddress physicalFormatsAddress = { kAudioStreamPropertyPhysicalFormats, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
for( i = 0; i < i_streams && p_sys->i_stream_index < 0 ; i++ ) for( i = 0; i < i_streams && p_sys->i_stream_index < 0 ; i++ )
{ {
/* Find a stream with a cac3 stream */ /* Find a stream with a cac3 stream */
...@@ -690,12 +685,10 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -690,12 +685,10 @@ static int OpenSPDIF( aout_instance_t * p_aout )
bool b_digital = false; bool b_digital = false;
/* Retrieve all the stream formats supported by each output stream */ /* Retrieve all the stream formats supported by each output stream */
err = AudioStreamGetPropertyInfo( p_streams[i], 0, err = AudioObjectGetPropertyDataSize( p_streams[i], &physicalFormatsAddress, 0, NULL, &i_param_size );
kAudioStreamPropertyPhysicalFormats,
&i_param_size, NULL );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not get number of streamformats: [%4.4s]", (char *)&err ); msg_Err( p_aout, "OpenSPDIF: could not get number of streamformats: [%s] (%i)", (char *)&err, err );
continue; continue;
} }
...@@ -704,9 +697,7 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -704,9 +697,7 @@ static int OpenSPDIF( aout_instance_t * p_aout )
if( p_format_list == NULL ) if( p_format_list == NULL )
continue; continue;
err = AudioStreamGetProperty( p_streams[i], 0, err = AudioObjectGetPropertyData( p_streams[i], &physicalFormatsAddress, 0, NULL, &i_param_size, p_format_list );
kAudioStreamPropertyPhysicalFormats,
&i_param_size, p_format_list );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not get the list of streamformats: [%4.4s]", (char *)&err ); msg_Err( p_aout, "could not get the list of streamformats: [%4.4s]", (char *)&err );
...@@ -735,14 +726,11 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -735,14 +726,11 @@ static int OpenSPDIF( aout_instance_t * p_aout )
p_sys->i_stream_id = p_streams[i]; p_sys->i_stream_id = p_streams[i];
p_sys->i_stream_index = i; p_sys->i_stream_index = i;
if( p_sys->b_revert == false ) if( !p_sys->b_revert )
{ {
/* Retrieve the original format of this stream first if not done so already */ /* Retrieve the original format of this stream first if not done so already */
i_param_size = sizeof( p_sys->sfmt_revert ); i_param_size = sizeof( p_sys->sfmt_revert );
err = AudioStreamGetProperty( p_sys->i_stream_id, 0, err = AudioObjectGetPropertyData( p_sys->i_stream_id, &physicalFormatsAddress, 0, NULL, &i_param_size, &p_sys->sfmt_revert );
kAudioStreamPropertyPhysicalFormat,
&i_param_size,
&p_sys->sfmt_revert );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not retrieve the original streamformat: [%4.4s]", (char *)&err ); msg_Err( p_aout, "could not retrieve the original streamformat: [%4.4s]", (char *)&err );
...@@ -802,13 +790,13 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -802,13 +790,13 @@ static int OpenSPDIF( aout_instance_t * p_aout )
aout_VolumeNoneInit( p_aout ); aout_VolumeNoneInit( p_aout );
/* Add IOProc callback */ /* Add IOProc callback */
err = AudioDeviceAddIOProc( p_sys->i_selected_dev, err = AudioDeviceCreateIOProcID( p_sys->i_selected_dev,
(AudioDeviceIOProc)RenderCallbackSPDIF, (AudioDeviceIOProc)RenderCallbackSPDIF,
(void *)p_aout ); (void *)p_aout,
&p_sys->i_procID );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "AudioDeviceAddIOProc failed: [%4.4s]", (char *)&err ); msg_Err( p_aout, "AudioDeviceCreateIOProcID failed: [%4.4s]", (char *)&err );
return false; return false;
} }
...@@ -818,16 +806,16 @@ static int OpenSPDIF( aout_instance_t * p_aout ) ...@@ -818,16 +806,16 @@ static int OpenSPDIF( aout_instance_t * p_aout )
p_sys->clock_diff += mdate(); p_sys->clock_diff += mdate();
/* Start device */ /* Start device */
err = AudioDeviceStart( p_sys->i_selected_dev, (AudioDeviceIOProc)RenderCallbackSPDIF ); err = AudioDeviceStart( p_sys->i_selected_dev, p_sys->i_procID );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "AudioDeviceStart failed: [%4.4s]", (char *)&err ); msg_Err( p_aout, "AudioDeviceStart failed: [%4.4s]", (char *)&err );
err = AudioDeviceRemoveIOProc( p_sys->i_selected_dev, err = AudioDeviceDestroyIOProcID( p_sys->i_selected_dev,
(AudioDeviceIOProc)RenderCallbackSPDIF ); p_sys->i_procID );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "AudioDeviceRemoveIOProc failed: [%4.4s]", (char *)&err ); msg_Err( p_aout, "AudioDeviceDestroyIOProcID failed: [%4.4s]", (char *)&err );
} }
return false; return false;
} }
...@@ -850,25 +838,25 @@ static void Close( vlc_object_t * p_this ) ...@@ -850,25 +838,25 @@ static void Close( vlc_object_t * p_this )
{ {
verify_noerr( AudioOutputUnitStop( p_sys->au_unit ) ); verify_noerr( AudioOutputUnitStop( p_sys->au_unit ) );
verify_noerr( AudioUnitUninitialize( p_sys->au_unit ) ); verify_noerr( AudioUnitUninitialize( p_sys->au_unit ) );
verify_noerr( AudioComponentInstanceDispose( p_sys->au_unit ) ); verify_noerr( CloseComponent( p_sys->au_unit ) );
} }
if( p_sys->b_digital ) if( p_sys->b_digital )
{ {
/* Stop device */ /* Stop device */
err = AudioDeviceStop( p_sys->i_selected_dev, err = AudioDeviceStop( p_sys->i_selected_dev,
(AudioDeviceIOProc)RenderCallbackSPDIF ); p_sys->i_procID );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "AudioDeviceStop failed: [%4.4s]", (char *)&err ); msg_Err( p_aout, "AudioDeviceStop failed: [%4.4s]", (char *)&err );
} }
/* Remove IOProc callback */ /* Remove IOProc callback */
err = AudioDeviceRemoveIOProc( p_sys->i_selected_dev, err = AudioDeviceDestroyIOProcID( p_sys->i_selected_dev,
(AudioDeviceIOProc)RenderCallbackSPDIF ); p_sys->i_procID );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "AudioDeviceRemoveIOProc failed: [%4.4s]", (char *)&err ); msg_Err( p_aout, "AudioDeviceDestroyIOProcID failed: [%4.4s]", (char *)&err );
} }
if( p_sys->b_revert ) if( p_sys->b_revert )
...@@ -881,18 +869,15 @@ static void Close( vlc_object_t * p_this ) ...@@ -881,18 +869,15 @@ static void Close( vlc_object_t * p_this )
int b_mix; int b_mix;
Boolean b_writeable; Boolean b_writeable;
/* Revert mixable to true if we are allowed to */ /* Revert mixable to true if we are allowed to */
err = AudioDeviceGetPropertyInfo( p_sys->i_selected_dev, 0, FALSE, kAudioDevicePropertySupportsMixing, AudioObjectPropertyAddress audioDeviceSupportsMixingAddress = { kAudioDevicePropertySupportsMixing , kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
&i_param_size, &b_writeable ); b_writeable = AudioObjectHasProperty( p_sys->i_selected_dev, &audioDeviceSupportsMixingAddress );
err = AudioObjectGetPropertyData( p_sys->i_selected_dev, &audioDeviceSupportsMixingAddress, 0, NULL, &i_param_size, &b_mix );
err = AudioDeviceGetProperty( p_sys->i_selected_dev, 0, FALSE, kAudioDevicePropertySupportsMixing,
&i_param_size, &b_mix );
if( !err && b_writeable ) if( !err && b_writeable )
{ {
msg_Dbg( p_aout, "mixable is: %d", b_mix ); msg_Dbg( p_aout, "mixable is: %d", b_mix );
b_mix = 1; b_mix = 1;
err = AudioDeviceSetProperty( p_sys->i_selected_dev, 0, 0, FALSE, err = AudioObjectSetPropertyData( p_sys->i_selected_dev, &audioDeviceSupportsMixingAddress, 0, NULL, i_param_size, &b_mix );
kAudioDevicePropertySupportsMixing, i_param_size, &b_mix );
} }
if( err != noErr ) if( err != noErr )
...@@ -902,8 +887,8 @@ static void Close( vlc_object_t * p_this ) ...@@ -902,8 +887,8 @@ static void Close( vlc_object_t * p_this )
} }
} }
err = AudioHardwareRemovePropertyListener( kAudioHardwarePropertyDevices, AudioObjectPropertyAddress audioDevicesAddress = { kAudioHardwarePropertyDevices, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
HardwareListener ); err = AudioObjectRemovePropertyListener( kAudioObjectSystemObject, &audioDevicesAddress, HardwareListener, NULL );
if( err != noErr ) if( err != noErr )
{ {
...@@ -914,8 +899,10 @@ static void Close( vlc_object_t * p_this ) ...@@ -914,8 +899,10 @@ static void Close( vlc_object_t * p_this )
{ {
p_sys->i_hog_pid = -1; p_sys->i_hog_pid = -1;
i_param_size = sizeof( p_sys->i_hog_pid ); i_param_size = sizeof( p_sys->i_hog_pid );
err = AudioDeviceSetProperty( p_sys->i_selected_dev, 0, 0, FALSE, AudioObjectPropertyAddress audioDeviceHogModeAddress = { kAudioDevicePropertyHogMode,
kAudioDevicePropertyHogMode, i_param_size, &p_sys->i_hog_pid ); kAudioDevicePropertyScopeOutput,
kAudioObjectPropertyElementMaster };
err = AudioObjectSetPropertyData( p_sys->i_selected_dev, &audioDeviceHogModeAddress, 0, NULL, i_param_size, &p_sys->i_hog_pid );
if( err != noErr ) msg_Err( p_aout, "Could not release hogmode: [%4.4s]", (char *)&err ); if( err != noErr ) msg_Err( p_aout, "Could not release hogmode: [%4.4s]", (char *)&err );
} }
...@@ -945,11 +932,11 @@ static void Probe( aout_instance_t * p_aout ) ...@@ -945,11 +932,11 @@ static void Probe( aout_instance_t * p_aout )
struct aout_sys_t *p_sys = p_aout->output.p_sys; struct aout_sys_t *p_sys = p_aout->output.p_sys;
/* Get number of devices */ /* Get number of devices */
err = AudioHardwareGetPropertyInfo( kAudioHardwarePropertyDevices, AudioObjectPropertyAddress audioDevicesAddress = { kAudioHardwarePropertyDevices, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
&i_param_size, NULL ); err = AudioObjectGetPropertyDataSize(kAudioObjectSystemObject, &audioDevicesAddress, 0, NULL, &i_param_size);
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "Could not get number of devices: [%4.4s]", (char *)&err ); msg_Err( p_aout, "Could not get number of devices: [%s]", (char *)&err );
goto error; goto error;
} }
...@@ -969,21 +956,20 @@ static void Probe( aout_instance_t * p_aout ) ...@@ -969,21 +956,20 @@ static void Probe( aout_instance_t * p_aout )
goto error; goto error;
/* Populate DeviceID array */ /* Populate DeviceID array */
err = AudioHardwareGetProperty( kAudioHardwarePropertyDevices, err = AudioObjectGetPropertyData( kAudioObjectSystemObject, &audioDevicesAddress, 0, NULL, &i_param_size, p_devices );
&i_param_size, p_devices );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not get the device IDs: [%4.4s]", (char *)&err ); msg_Err( p_aout, "could not get the device IDs: [%s]", (char *)&err );
goto error; goto error;
} }
/* Find the ID of the default Device */ /* Find the ID of the default Device */
AudioObjectPropertyAddress defaultDeviceAddress = { kAudioHardwarePropertyDefaultOutputDevice, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
i_param_size = sizeof( AudioDeviceID ); i_param_size = sizeof( AudioDeviceID );
err = AudioHardwareGetProperty( kAudioHardwarePropertyDefaultOutputDevice, err= AudioObjectGetPropertyData( kAudioObjectSystemObject, &defaultDeviceAddress, 0, NULL, &i_param_size, &devid_def );
&i_param_size, &devid_def );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not get default audio device: [%4.4s]", (char *)&err ); msg_Err( p_aout, "could not get default audio device: [%s]", (char *)&err );
goto error; goto error;
} }
p_sys->i_default_dev = devid_def; p_sys->i_default_dev = devid_def;
...@@ -992,24 +978,20 @@ static void Probe( aout_instance_t * p_aout ) ...@@ -992,24 +978,20 @@ static void Probe( aout_instance_t * p_aout )
text.psz_string = (char*)_("Audio Device"); text.psz_string = (char*)_("Audio Device");
var_Change( p_aout, "audio-device", VLC_VAR_SETTEXT, &text, NULL ); var_Change( p_aout, "audio-device", VLC_VAR_SETTEXT, &text, NULL );
AudioObjectPropertyAddress deviceNameAddress = { kAudioDevicePropertyDeviceName, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
for( i = 0; i < p_sys->i_devices; i++ ) for( i = 0; i < p_sys->i_devices; i++ )
{ {
char *psz_name; char *psz_name;
i_param_size = 0; i_param_size = 0;
/* Retrieve the length of the device name */ /* Retrieve the length of the device name */
err = AudioDeviceGetPropertyInfo( err = AudioObjectGetPropertyDataSize( p_devices[i], &deviceNameAddress, 0, NULL, &i_param_size );
p_devices[i], 0, false,
kAudioDevicePropertyDeviceName,
&i_param_size, NULL);
if( err ) goto error; if( err ) goto error;
/* Retrieve the name of the device */ /* Retrieve the name of the device */
psz_name = (char *)malloc( i_param_size ); psz_name = (char *)malloc( i_param_size );
err = AudioDeviceGetProperty( err = AudioObjectGetPropertyData( p_devices[i], &deviceNameAddress, 0, NULL, &i_param_size, psz_name );
p_devices[i], 0, false,
kAudioDevicePropertyDeviceName,
&i_param_size, psz_name);
if( err ) goto error; if( err ) goto error;
msg_Dbg( p_aout, "DevID: %u DevName: %s", p_devices[i], psz_name ); msg_Dbg( p_aout, "DevID: %u DevName: %s", p_devices[i], psz_name );
...@@ -1066,9 +1048,7 @@ static void Probe( aout_instance_t * p_aout ) ...@@ -1066,9 +1048,7 @@ static void Probe( aout_instance_t * p_aout )
var_AddCallback( p_aout, "audio-device", AudioDeviceCallback, NULL ); var_AddCallback( p_aout, "audio-device", AudioDeviceCallback, NULL );
/* Attach a Listener so that we are notified of a change in the Device setup */ /* Attach a Listener so that we are notified of a change in the Device setup */
err = AudioHardwareAddPropertyListener( kAudioHardwarePropertyDevices, err = AudioObjectAddPropertyListener( kAudioObjectSystemObject, &audioDevicesAddress, HardwareListener, (void *)p_aout );
HardwareListener,
(void *)p_aout );
if( err ) if( err )
goto error; goto error;
...@@ -1087,9 +1067,9 @@ error: ...@@ -1087,9 +1067,9 @@ error:
static int AudioDeviceHasOutput( AudioDeviceID i_dev_id ) static int AudioDeviceHasOutput( AudioDeviceID i_dev_id )
{ {
UInt32 dataSize; UInt32 dataSize;
Boolean isWritable;
verify_noerr( AudioDeviceGetPropertyInfo( i_dev_id, 0, FALSE, kAudioDevicePropertyStreams, &dataSize, &isWritable) ); AudioObjectPropertyAddress streamsAddress = { kAudioDevicePropertyStreams, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
verify_noerr( AudioObjectGetPropertyDataSize( i_dev_id, &streamsAddress, 0, NULL, &dataSize ) );
if (dataSize == 0) return FALSE; if (dataSize == 0) return FALSE;
return TRUE; return TRUE;
...@@ -1107,12 +1087,12 @@ static int AudioDeviceSupportsDigital( aout_instance_t *p_aout, AudioDeviceID i_ ...@@ -1107,12 +1087,12 @@ static int AudioDeviceSupportsDigital( aout_instance_t *p_aout, AudioDeviceID i_
bool b_return = false; bool b_return = false;
/* Retrieve all the output streams */ /* Retrieve all the output streams */
err = AudioDeviceGetPropertyInfo( i_dev_id, 0, FALSE, AudioObjectPropertyAddress streamsAddress = { kAudioDevicePropertyStreams, kAudioObjectPropertyScopeGlobal, kAudioObjectPropertyElementMaster };
kAudioDevicePropertyStreams, err = AudioObjectGetPropertyDataSize( i_dev_id, &streamsAddress, 0, NULL, &i_param_size );
&i_param_size, NULL );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not get number of streams: [%4.4s]", (char *)&err ); msg_Err( p_aout, "could not get number of streams: [%s] (%i)", (char *)&err, err );
return false; return false;
} }
...@@ -1121,13 +1101,11 @@ static int AudioDeviceSupportsDigital( aout_instance_t *p_aout, AudioDeviceID i_ ...@@ -1121,13 +1101,11 @@ static int AudioDeviceSupportsDigital( aout_instance_t *p_aout, AudioDeviceID i_
if( p_streams == NULL ) if( p_streams == NULL )
return VLC_ENOMEM; return VLC_ENOMEM;
err = AudioDeviceGetProperty( i_dev_id, 0, FALSE, err = AudioObjectGetPropertyData( i_dev_id, &streamsAddress, 0, NULL, &i_param_size, p_streams );
kAudioDevicePropertyStreams,
&i_param_size, p_streams );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not get number of streams: [%4.4s]", (char *)&err ); msg_Err( p_aout, "could not get list of streams: [%s]", (char *)&err );
return false; return false;
} }
...@@ -1153,12 +1131,16 @@ static int AudioStreamSupportsDigital( aout_instance_t *p_aout, AudioStreamID i_ ...@@ -1153,12 +1131,16 @@ static int AudioStreamSupportsDigital( aout_instance_t *p_aout, AudioStreamID i_
bool b_return = false; bool b_return = false;
/* Retrieve all the stream formats supported by each output stream */ /* Retrieve all the stream formats supported by each output stream */
err = AudioStreamGetPropertyInfo( i_stream_id, 0, AudioObjectPropertyAddress physicalFormatsAddress = { kAudioStreamPropertyAvailablePhysicalFormats, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
kAudioStreamPropertyPhysicalFormats, err = AudioObjectGetPropertyDataSize( i_stream_id, &physicalFormatsAddress, 0, NULL, &i_param_size );
&i_param_size, NULL ); if( err == kAudioHardwareUnknownPropertyError )
if( err != noErr )
{ {
msg_Err( p_aout, "could not get number of streamformats: [%4.4s]", (char *)&err ); msg_Err( p_aout, "audio stream doesn't support query kAudioStreamPropertyAvailablePhysicalFormats" );
return false;
}
else if( err != noErr )
{
msg_Err( p_aout, "AudioStreamSupportsDigital: could not get number of streamformats: [%s] (%i)", (char *)&err, err );
return false; return false;
} }
...@@ -1167,9 +1149,7 @@ static int AudioStreamSupportsDigital( aout_instance_t *p_aout, AudioStreamID i_ ...@@ -1167,9 +1149,7 @@ static int AudioStreamSupportsDigital( aout_instance_t *p_aout, AudioStreamID i_
if( p_format_list == NULL ) if( p_format_list == NULL )
return false; return false;
err = AudioStreamGetProperty( i_stream_id, 0, err = AudioObjectGetPropertyData( i_stream_id, &physicalFormatsAddress, 0, NULL, &i_param_size, p_format_list );
kAudioStreamPropertyPhysicalFormats,
&i_param_size, p_format_list );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "could not get the list of streamformats: [%4.4s]", (char *)&err ); msg_Err( p_aout, "could not get the list of streamformats: [%4.4s]", (char *)&err );
...@@ -1202,6 +1182,8 @@ static int AudioStreamChangeFormat( aout_instance_t *p_aout, AudioStreamID i_str ...@@ -1202,6 +1182,8 @@ static int AudioStreamChangeFormat( aout_instance_t *p_aout, AudioStreamID i_str
UInt32 i_param_size = 0; UInt32 i_param_size = 0;
int i; int i;
AudioObjectPropertyAddress physicalFormatAddress = { kAudioStreamPropertyPhysicalFormat, kAudioDevicePropertyScopeOutput, kAudioObjectPropertyElementMaster };
struct { vlc_mutex_t lock; vlc_cond_t cond; } w; struct { vlc_mutex_t lock; vlc_cond_t cond; } w;
msg_Dbg( p_aout, STREAM_FORMAT_MSG( "setting stream format: ", change_format ) ); msg_Dbg( p_aout, STREAM_FORMAT_MSG( "setting stream format: ", change_format ) );
...@@ -1212,19 +1194,15 @@ static int AudioStreamChangeFormat( aout_instance_t *p_aout, AudioStreamID i_str ...@@ -1212,19 +1194,15 @@ static int AudioStreamChangeFormat( aout_instance_t *p_aout, AudioStreamID i_str
vlc_mutex_lock( &w.lock ); vlc_mutex_lock( &w.lock );
/* Install the callback */ /* Install the callback */
err = AudioStreamAddPropertyListener( i_stream_id, 0, err = AudioObjectAddPropertyListener( i_stream_id, &physicalFormatAddress, StreamListener, (void *)&w );
kAudioStreamPropertyPhysicalFormat,
StreamListener, (void *)&w );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "AudioStreamAddPropertyListener failed: [%4.4s]", (char *)&err ); msg_Err( p_aout, "AudioObjectAddPropertyListener for kAudioStreamPropertyPhysicalFormat failed: [%4.4s]", (char *)&err );
return false; return false;
} }
/* change the format */ /* change the format */
err = AudioStreamSetProperty( i_stream_id, 0, 0, err = AudioObjectSetPropertyData( i_stream_id, &physicalFormatAddress, 0, NULL, sizeof( AudioStreamBasicDescription ),
kAudioStreamPropertyPhysicalFormat,
sizeof( AudioStreamBasicDescription ),
&change_format ); &change_format );
if( err != noErr ) if( err != noErr )
{ {
...@@ -1247,10 +1225,7 @@ static int AudioStreamChangeFormat( aout_instance_t *p_aout, AudioStreamID i_str ...@@ -1247,10 +1225,7 @@ static int AudioStreamChangeFormat( aout_instance_t *p_aout, AudioStreamID i_str
} }
i_param_size = sizeof( AudioStreamBasicDescription ); i_param_size = sizeof( AudioStreamBasicDescription );
err = AudioStreamGetProperty( i_stream_id, 0, err = AudioObjectGetPropertyData( i_stream_id, &physicalFormatAddress, 0, NULL, &i_param_size, &actual_format );
kAudioStreamPropertyPhysicalFormat,
&i_param_size,
&actual_format );
msg_Dbg( p_aout, STREAM_FORMAT_MSG( "actual format in use: ", actual_format ) ); msg_Dbg( p_aout, STREAM_FORMAT_MSG( "actual format in use: ", actual_format ) );
if( actual_format.mSampleRate == change_format.mSampleRate && if( actual_format.mSampleRate == change_format.mSampleRate &&
...@@ -1264,9 +1239,7 @@ static int AudioStreamChangeFormat( aout_instance_t *p_aout, AudioStreamID i_str ...@@ -1264,9 +1239,7 @@ static int AudioStreamChangeFormat( aout_instance_t *p_aout, AudioStreamID i_str
} }
/* Removing the property listener */ /* Removing the property listener */
err = AudioStreamRemovePropertyListener( i_stream_id, 0, err = AudioObjectRemovePropertyListener( i_stream_id, &physicalFormatAddress, StreamListener, NULL );
kAudioStreamPropertyPhysicalFormat,
StreamListener );
if( err != noErr ) if( err != noErr )
{ {
msg_Err( p_aout, "AudioStreamRemovePropertyListener failed: [%4.4s]", (char *)&err ); msg_Err( p_aout, "AudioStreamRemovePropertyListener failed: [%4.4s]", (char *)&err );
...@@ -1435,22 +1408,21 @@ static OSStatus RenderCallbackSPDIF( AudioDeviceID inDevice, ...@@ -1435,22 +1408,21 @@ static OSStatus RenderCallbackSPDIF( AudioDeviceID inDevice,
/***************************************************************************** /*****************************************************************************
* HardwareListener: Warns us of changes in the list of registered devices * HardwareListener: Warns us of changes in the list of registered devices
*****************************************************************************/ *****************************************************************************/
static OSStatus HardwareListener( AudioHardwarePropertyID inPropertyID, static OSStatus HardwareListener( AudioObjectID inObjectID, UInt32 inNumberAddresses, const AudioObjectPropertyAddress inAddresses[], void*inClientData)
void * inClientData )
{ {
OSStatus err = noErr; OSStatus err = noErr;
aout_instance_t *p_aout = (aout_instance_t *)inClientData; aout_instance_t *p_aout = (aout_instance_t *)inClientData;
VLC_UNUSED(inObjectID);
switch( inPropertyID ) for ( unsigned int i = 0; i < inNumberAddresses; i++ )
{ {
case kAudioHardwarePropertyDevices: if( inAddresses[i].mSelector == kAudioHardwarePropertyDevices )
{ {
/* something changed in the list of devices */ /* something changed in the list of devices */
/* We trigger the audio-device's aout_ChannelsRestart callback */ /* We trigger the audio-device's aout_ChannelsRestart callback */
var_TriggerCallback( p_aout, "audio-device" ); var_TriggerCallback( p_aout, "audio-device" );
var_Destroy( p_aout, "audio-device" ); var_Destroy( p_aout, "audio-device" );
} }
break;
} }
return( err ); return( err );
...@@ -1459,27 +1431,20 @@ static OSStatus HardwareListener( AudioHardwarePropertyID inPropertyID, ...@@ -1459,27 +1431,20 @@ static OSStatus HardwareListener( AudioHardwarePropertyID inPropertyID,
/***************************************************************************** /*****************************************************************************
* StreamListener * StreamListener
*****************************************************************************/ *****************************************************************************/
static OSStatus StreamListener( AudioStreamID inStream, static OSStatus StreamListener( AudioObjectID inObjectID, UInt32 inNumberAddresses, const AudioObjectPropertyAddress inAddresses[], void*inClientData)
UInt32 inChannel,
AudioDevicePropertyID inPropertyID,
void * inClientData )
{ {
OSStatus err = noErr; OSStatus err = noErr;
struct { vlc_mutex_t lock; vlc_cond_t cond; } * w = inClientData; struct { vlc_mutex_t lock; vlc_cond_t cond; } * w = inClientData;
VLC_UNUSED(inStream); VLC_UNUSED(inObjectID);
VLC_UNUSED(inChannel);
switch( inPropertyID ) for ( unsigned int i = 0; i < inNumberAddresses; i++ )
{
if( inAddresses[i].mSelector == kAudioStreamPropertyPhysicalFormat )
{ {
case kAudioStreamPropertyPhysicalFormat:
vlc_mutex_lock( &w->lock ); vlc_mutex_lock( &w->lock );
vlc_cond_signal( &w->cond ); vlc_cond_signal( &w->cond );
vlc_mutex_unlock( &w->lock ); }
break;
default:
break;
} }
return( err ); return( err );
} }
...@@ -1492,7 +1457,7 @@ static int AudioDeviceCallback( vlc_object_t *p_this, const char *psz_variable, ...@@ -1492,7 +1457,7 @@ static int AudioDeviceCallback( vlc_object_t *p_this, const char *psz_variable,
{ {
aout_instance_t *p_aout = (aout_instance_t *)p_this; aout_instance_t *p_aout = (aout_instance_t *)p_this;
var_Set( p_aout->p_libvlc, "macosx-audio-device", new_val ); var_Set( p_aout->p_libvlc, "macosx-audio-device", new_val );
msg_Dbg( p_aout, "Set Device: %#x", new_val.i_int ); msg_Dbg( p_aout, "Set Device: %i", new_val.i_int );
return aout_ChannelsRestart( p_this, psz_variable, old_val, new_val, param ); return aout_ChannelsRestart( p_this, psz_variable, old_val, new_val, param );
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment