Core Audio output: Default device setting now tracks system device changes, and output assigns a speaker mapping depending on the channel count
parent
378aaf23ae
commit
8435416cd7
|
@ -45,7 +45,6 @@
|
||||||
- (void)launchThreads;
|
- (void)launchThreads;
|
||||||
|
|
||||||
- (InputNode *)inputNode;
|
- (InputNode *)inputNode;
|
||||||
- (InputNode *)stealInputNode;
|
|
||||||
|
|
||||||
- (id)finalNode;
|
- (id)finalNode;
|
||||||
|
|
||||||
|
|
|
@ -23,6 +23,8 @@
|
||||||
BOOL stopping;
|
BOOL stopping;
|
||||||
BOOL stopped;
|
BOOL stopped;
|
||||||
|
|
||||||
|
BOOL listenerapplied;
|
||||||
|
|
||||||
float volume;
|
float volume;
|
||||||
|
|
||||||
AudioDeviceID outputDeviceID;
|
AudioDeviceID outputDeviceID;
|
||||||
|
|
|
@ -28,6 +28,8 @@
|
||||||
buffers = NULL;
|
buffers = NULL;
|
||||||
numberOfBuffers = 0;
|
numberOfBuffers = 0;
|
||||||
volume = 1.0;
|
volume = 1.0;
|
||||||
|
outputDeviceID = -1;
|
||||||
|
listenerapplied = NO;
|
||||||
|
|
||||||
[[NSUserDefaultsController sharedUserDefaultsController] addObserver:self forKeyPath:@"values.outputDevice" options:0 context:NULL];
|
[[NSUserDefaultsController sharedUserDefaultsController] addObserver:self forKeyPath:@"values.outputDevice" options:0 context:NULL];
|
||||||
}
|
}
|
||||||
|
@ -83,6 +85,13 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
AudioQueueEnqueueBuffer(queue, buffer, 0, NULL);
|
AudioQueueEnqueueBuffer(queue, buffer, 0, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static OSStatus
|
||||||
|
default_device_changed(AudioObjectID inObjectID, UInt32 inNumberAddresses, const AudioObjectPropertyAddress *inAddresses, void *inUserData)
|
||||||
|
{
|
||||||
|
OutputCoreAudio *this = (__bridge OutputCoreAudio *) inUserData;
|
||||||
|
return [this setOutputDeviceByID:-1];
|
||||||
|
}
|
||||||
|
|
||||||
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
|
- (void)observeValueForKeyPath:(NSString *)keyPath ofObject:(id)object change:(NSDictionary *)change context:(void *)context
|
||||||
{
|
{
|
||||||
if ([keyPath isEqualToString:@"values.outputDevice"]) {
|
if ([keyPath isEqualToString:@"values.outputDevice"]) {
|
||||||
|
@ -97,6 +106,7 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
- (OSStatus)setOutputDeviceByID:(AudioDeviceID)deviceID
|
- (OSStatus)setOutputDeviceByID:(AudioDeviceID)deviceID
|
||||||
{
|
{
|
||||||
OSStatus err;
|
OSStatus err;
|
||||||
|
BOOL defaultDevice = NO;
|
||||||
UInt32 thePropSize;
|
UInt32 thePropSize;
|
||||||
AudioObjectPropertyAddress theAddress = {
|
AudioObjectPropertyAddress theAddress = {
|
||||||
.mSelector = kAudioHardwarePropertyDefaultOutputDevice,
|
.mSelector = kAudioHardwarePropertyDefaultOutputDevice,
|
||||||
|
@ -105,6 +115,7 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
};
|
};
|
||||||
|
|
||||||
if (deviceID == -1) {
|
if (deviceID == -1) {
|
||||||
|
defaultDevice = YES;
|
||||||
UInt32 size = sizeof(AudioDeviceID);
|
UInt32 size = sizeof(AudioDeviceID);
|
||||||
err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &theAddress, 0, NULL, &size, &deviceID);
|
err = AudioObjectGetPropertyData(kAudioObjectSystemObject, &theAddress, 0, NULL, &size, &deviceID);
|
||||||
|
|
||||||
|
@ -113,17 +124,25 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
|
|
||||||
return err;
|
return err;
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
outputDeviceID = deviceID;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
printf("DEVICE: %i\n", deviceID);
|
|
||||||
outputDeviceID = deviceID;
|
|
||||||
|
|
||||||
if (audioQueue) {
|
if (audioQueue) {
|
||||||
|
if (outputDeviceID == deviceID)
|
||||||
|
return noErr;
|
||||||
|
|
||||||
|
AudioObjectPropertyAddress defaultDeviceAddress = theAddress;
|
||||||
|
|
||||||
|
if (listenerapplied && !defaultDevice) {
|
||||||
|
AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &defaultDeviceAddress, default_device_changed, (__bridge void * _Nullable)(self));
|
||||||
|
listenerapplied = NO;
|
||||||
|
}
|
||||||
|
|
||||||
|
printf("DEVICE: %i\n", deviceID);
|
||||||
|
outputDeviceID = deviceID;
|
||||||
|
|
||||||
CFStringRef theDeviceUID;
|
CFStringRef theDeviceUID;
|
||||||
theAddress.mSelector = kAudioDevicePropertyDeviceUID;
|
theAddress.mSelector = kAudioDevicePropertyDeviceUID;
|
||||||
|
theAddress.mScope = kAudioDevicePropertyScopeOutput;
|
||||||
thePropSize = sizeof(theDeviceUID);
|
thePropSize = sizeof(theDeviceUID);
|
||||||
err = AudioObjectGetPropertyData(outputDeviceID, &theAddress, 0, NULL, &thePropSize, &theDeviceUID);
|
err = AudioObjectGetPropertyData(outputDeviceID, &theAddress, 0, NULL, &thePropSize, &theDeviceUID);
|
||||||
|
|
||||||
|
@ -143,6 +162,11 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
CFRelease(theDeviceUID);
|
CFRelease(theDeviceUID);
|
||||||
if (running)
|
if (running)
|
||||||
[self start];
|
[self start];
|
||||||
|
|
||||||
|
if (!listenerapplied && defaultDevice) {
|
||||||
|
AudioObjectAddPropertyListener(kAudioObjectSystemObject, &defaultDeviceAddress, default_device_changed, (__bridge void * _Nullable)(self));
|
||||||
|
listenerapplied = YES;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else if (outputUnit) {
|
else if (outputUnit) {
|
||||||
err = AudioUnitSetProperty(outputUnit,
|
err = AudioUnitSetProperty(outputUnit,
|
||||||
|
@ -280,6 +304,7 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
|
|
||||||
stopping = NO;
|
stopping = NO;
|
||||||
stopped = NO;
|
stopped = NO;
|
||||||
|
outputDeviceID = -1;
|
||||||
|
|
||||||
AudioComponentDescription desc;
|
AudioComponentDescription desc;
|
||||||
OSStatus err;
|
OSStatus err;
|
||||||
|
@ -367,6 +392,41 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
[self setOutputDeviceWithDeviceDict:nil];
|
[self setOutputDeviceWithDeviceDict:nil];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* Set the channel layout for the audio queue */
|
||||||
|
AudioChannelLayout layout = {0};
|
||||||
|
switch (deviceFormat.mChannelsPerFrame) {
|
||||||
|
case 1:
|
||||||
|
layout.mChannelLayoutTag = kAudioChannelLayoutTag_Mono;
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
layout.mChannelLayoutTag = kAudioChannelLayoutTag_Stereo;
|
||||||
|
break;
|
||||||
|
case 3:
|
||||||
|
layout.mChannelLayoutTag = kAudioChannelLayoutTag_DVD_4;
|
||||||
|
break;
|
||||||
|
case 4:
|
||||||
|
layout.mChannelLayoutTag = kAudioChannelLayoutTag_Quadraphonic;
|
||||||
|
break;
|
||||||
|
case 5:
|
||||||
|
layout.mChannelLayoutTag = kAudioChannelLayoutTag_MPEG_5_0_A;
|
||||||
|
break;
|
||||||
|
case 6:
|
||||||
|
layout.mChannelLayoutTag = kAudioChannelLayoutTag_MPEG_5_1_A;
|
||||||
|
break;
|
||||||
|
case 7:
|
||||||
|
layout.mChannelLayoutTag = kAudioChannelLayoutTag_MPEG_6_1_A;
|
||||||
|
break;
|
||||||
|
case 8:
|
||||||
|
layout.mChannelLayoutTag = kAudioChannelLayoutTag_MPEG_7_1_A;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
if (layout.mChannelLayoutTag != 0) {
|
||||||
|
err = AudioQueueSetProperty(audioQueue, kAudioQueueProperty_ChannelLayout, &layout, sizeof(layout));
|
||||||
|
if (err != noErr) {
|
||||||
|
return NO;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
numberOfBuffers = 4;
|
numberOfBuffers = 4;
|
||||||
bufferByteSize = deviceFormat.mBytesPerPacket * 512;
|
bufferByteSize = deviceFormat.mBytesPerPacket * 512;
|
||||||
|
|
||||||
|
@ -426,20 +486,26 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
- (void)stop
|
- (void)stop
|
||||||
{
|
{
|
||||||
stopping = YES;
|
stopping = YES;
|
||||||
if (outputUnit)
|
if (listenerapplied) {
|
||||||
{
|
AudioObjectPropertyAddress theAddress = {
|
||||||
|
.mSelector = kAudioHardwarePropertyDefaultOutputDevice,
|
||||||
|
.mScope = kAudioObjectPropertyScopeGlobal,
|
||||||
|
.mElement = kAudioObjectPropertyElementMaster
|
||||||
|
};
|
||||||
|
AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &theAddress, default_device_changed, (__bridge void * _Nullable)(self));
|
||||||
|
listenerapplied = NO;
|
||||||
|
}
|
||||||
|
if (outputUnit) {
|
||||||
AudioUnitUninitialize (outputUnit);
|
AudioUnitUninitialize (outputUnit);
|
||||||
AudioComponentInstanceDispose(outputUnit);
|
AudioComponentInstanceDispose(outputUnit);
|
||||||
outputUnit = NULL;
|
outputUnit = NULL;
|
||||||
}
|
}
|
||||||
if (audioQueue && buffers)
|
if (audioQueue && buffers) {
|
||||||
{
|
|
||||||
AudioQueuePause(audioQueue);
|
AudioQueuePause(audioQueue);
|
||||||
AudioQueueStop(audioQueue, true);
|
AudioQueueStop(audioQueue, true);
|
||||||
running = NO;
|
running = NO;
|
||||||
|
|
||||||
for (UInt32 i = 0; i < numberOfBuffers; ++i)
|
for (UInt32 i = 0; i < numberOfBuffers; ++i) {
|
||||||
{
|
|
||||||
if (buffers[i])
|
if (buffers[i])
|
||||||
AudioQueueFreeBuffer(audioQueue, buffers[i]);
|
AudioQueueFreeBuffer(audioQueue, buffers[i]);
|
||||||
buffers[i] = NULL;
|
buffers[i] = NULL;
|
||||||
|
@ -447,8 +513,7 @@ static void Sound_Renderer(void *userData, AudioQueueRef queue, AudioQueueBuffer
|
||||||
free(buffers);
|
free(buffers);
|
||||||
buffers = NULL;
|
buffers = NULL;
|
||||||
}
|
}
|
||||||
if (audioQueue)
|
if (audioQueue) {
|
||||||
{
|
|
||||||
AudioQueueDispose(audioQueue, true);
|
AudioQueueDispose(audioQueue, true);
|
||||||
audioQueue = NULL;
|
audioQueue = NULL;
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue