[Core Audio Output] Improve safety on stopping

Synchronize audio setup and audio stopping on the object's own pointer,
to hopefully prevent race conditions with out of sync calls to the stop
function from both the main and the audio thread.

Signed-off-by: Christopher Snowhill <kode54@gmail.com>
swiftingly
Christopher Snowhill 2022-06-15 21:46:02 -07:00
parent e9230a080c
commit 0997ca2c93
1 changed files with 189 additions and 184 deletions

View File

@ -344,8 +344,7 @@ default_device_changed(AudioObjectID inObjectID, UInt32 inNumberAddresses, const
} }
stopped = YES; stopped = YES;
if(!stopInvoked) [self stop];
[self stop];
} }
- (OSStatus)setOutputDeviceByID:(AudioDeviceID)deviceID { - (OSStatus)setOutputDeviceByID:(AudioDeviceID)deviceID {
@ -623,170 +622,173 @@ default_device_changed(AudioObjectID inObjectID, UInt32 inNumberAddresses, const
if(_au) if(_au)
[self stop]; [self stop];
stopInvoked = NO; @synchronized(self) {
running = NO; stopInvoked = NO;
stopping = NO;
stopped = NO;
paused = NO;
stopNext = NO;
outputDeviceID = -1;
restarted = NO;
downmixer = nil; running = NO;
downmixerForVis = nil; stopping = NO;
stopped = NO;
paused = NO;
stopNext = NO;
outputDeviceID = -1;
restarted = NO;
AudioComponentDescription desc; downmixer = nil;
NSError *err; downmixerForVis = nil;
desc.componentType = kAudioUnitType_Output; AudioComponentDescription desc;
desc.componentSubType = kAudioUnitSubType_HALOutput; NSError *err;
desc.componentManufacturer = kAudioUnitManufacturer_Apple;
desc.componentFlags = 0;
desc.componentFlagsMask = 0;
_au = [[AUAudioUnit alloc] initWithComponentDescription:desc error:&err]; desc.componentType = kAudioUnitType_Output;
if(err != nil) desc.componentSubType = kAudioUnitSubType_HALOutput;
return NO; desc.componentManufacturer = kAudioUnitManufacturer_Apple;
desc.componentFlags = 0;
desc.componentFlagsMask = 0;
// Setup the output device before mucking with settings _au = [[AUAudioUnit alloc] initWithComponentDescription:desc error:&err];
NSDictionary *device = [[[NSUserDefaultsController sharedUserDefaultsController] defaults] objectForKey:@"outputDevice"]; if(err != nil)
if(device) { return NO;
BOOL ok = [self setOutputDeviceWithDeviceDict:device];
if(!ok) { // Setup the output device before mucking with settings
// Ruh roh. NSDictionary *device = [[[NSUserDefaultsController sharedUserDefaultsController] defaults] objectForKey:@"outputDevice"];
if(device) {
BOOL ok = [self setOutputDeviceWithDeviceDict:device];
if(!ok) {
// Ruh roh.
[self setOutputDeviceWithDeviceDict:nil];
[[[NSUserDefaultsController sharedUserDefaultsController] defaults] removeObjectForKey:@"outputDevice"];
}
} else {
[self setOutputDeviceWithDeviceDict:nil]; [self setOutputDeviceWithDeviceDict:nil];
[[[NSUserDefaultsController sharedUserDefaultsController] defaults] removeObjectForKey:@"outputDevice"];
} }
} else {
[self setOutputDeviceWithDeviceDict:nil];
}
_deviceFormat = nil; _deviceFormat = nil;
AudioComponent comp = NULL; AudioComponent comp = NULL;
desc.componentType = kAudioUnitType_Effect; desc.componentType = kAudioUnitType_Effect;
desc.componentSubType = kAudioUnitSubType_GraphicEQ; desc.componentSubType = kAudioUnitSubType_GraphicEQ;
comp = AudioComponentFindNext(comp, &desc); comp = AudioComponentFindNext(comp, &desc);
if(!comp) if(!comp)
return NO; return NO;
OSStatus _err = AudioComponentInstanceNew(comp, &_eq); OSStatus _err = AudioComponentInstanceNew(comp, &_eq);
if(err) if(err)
return NO; return NO;
[self updateDeviceFormat]; [self updateDeviceFormat];
__block AudioUnit eq = _eq; __block AudioUnit eq = _eq;
__block AudioStreamBasicDescription *format = &deviceFormat; __block AudioStreamBasicDescription *format = &deviceFormat;
__block BOOL *eqEnabled = &self->eqEnabled; __block BOOL *eqEnabled = &self->eqEnabled;
__block void *refCon = (__bridge void *)self; __block void *refCon = (__bridge void *)self;
#ifdef OUTPUT_LOG #ifdef OUTPUT_LOG
__block FILE *logFile = _logFile; __block FILE *logFile = _logFile;
#endif #endif
_au.outputProvider = ^AUAudioUnitStatus(AudioUnitRenderActionFlags *_Nonnull actionFlags, const AudioTimeStamp *_Nonnull timestamp, AUAudioFrameCount frameCount, NSInteger inputBusNumber, AudioBufferList *_Nonnull inputData) { _au.outputProvider = ^AUAudioUnitStatus(AudioUnitRenderActionFlags *_Nonnull actionFlags, const AudioTimeStamp *_Nonnull timestamp, AUAudioFrameCount frameCount, NSInteger inputBusNumber, AudioBufferList *_Nonnull inputData) {
// This expects multiple buffers, so: // This expects multiple buffers, so:
int i; int i;
const int channels = format->mChannelsPerFrame; const int channels = format->mChannelsPerFrame;
const int channelsminusone = channels - 1; const int channelsminusone = channels - 1;
float buffers[frameCount * format->mChannelsPerFrame]; float buffers[frameCount * format->mChannelsPerFrame];
uint8_t bufferlistbuffer[sizeof(AudioBufferList) + sizeof(AudioBuffer) * channelsminusone]; uint8_t bufferlistbuffer[sizeof(AudioBufferList) + sizeof(AudioBuffer) * channelsminusone];
AudioBufferList *ioData = (AudioBufferList *)(bufferlistbuffer); AudioBufferList *ioData = (AudioBufferList *)(bufferlistbuffer);
ioData->mNumberBuffers = channels; ioData->mNumberBuffers = channels;
memset(buffers, 0, sizeof(buffers)); memset(buffers, 0, sizeof(buffers));
for(i = 0; i < channels; ++i) { for(i = 0; i < channels; ++i) {
ioData->mBuffers[i].mNumberChannels = 1; ioData->mBuffers[i].mNumberChannels = 1;
ioData->mBuffers[i].mData = buffers + frameCount * i; ioData->mBuffers[i].mData = buffers + frameCount * i;
ioData->mBuffers[i].mDataByteSize = frameCount * sizeof(float); ioData->mBuffers[i].mDataByteSize = frameCount * sizeof(float);
} }
OSStatus ret; OSStatus ret;
if(*eqEnabled) if(*eqEnabled)
ret = AudioUnitRender(eq, actionFlags, timestamp, (UInt32)inputBusNumber, frameCount, ioData); ret = AudioUnitRender(eq, actionFlags, timestamp, (UInt32)inputBusNumber, frameCount, ioData);
else else
ret = renderCallback(refCon, actionFlags, timestamp, (UInt32)inputBusNumber, frameCount, ioData); ret = renderCallback(refCon, actionFlags, timestamp, (UInt32)inputBusNumber, frameCount, ioData);
if(ret) if(ret)
return ret; return ret;
for(i = 0; i < channels; ++i) { for(i = 0; i < channels; ++i) {
float *outBuffer = ((float *)inputData->mBuffers[0].mData) + i; float *outBuffer = ((float *)inputData->mBuffers[0].mData) + i;
const float *inBuffer = ((float *)ioData->mBuffers[i].mData); const float *inBuffer = ((float *)ioData->mBuffers[i].mData);
const int frameCount = ioData->mBuffers[i].mDataByteSize / sizeof(float); const int frameCount = ioData->mBuffers[i].mDataByteSize / sizeof(float);
cblas_scopy(frameCount, inBuffer, 1, outBuffer, channels); cblas_scopy(frameCount, inBuffer, 1, outBuffer, channels);
} }
#ifdef OUTPUT_LOG #ifdef OUTPUT_LOG
if(logFile) { if(logFile) {
fwrite(inputData->mBuffers[0].mData, 1, inputData->mBuffers[0].mDataByteSize, logFile); fwrite(inputData->mBuffers[0].mData, 1, inputData->mBuffers[0].mDataByteSize, logFile);
} }
// memset(inputData->mBuffers[0].mData, 0, inputData->mBuffers[0].mDataByteSize); // memset(inputData->mBuffers[0].mData, 0, inputData->mBuffers[0].mDataByteSize);
#endif #endif
inputData->mBuffers[0].mNumberChannels = channels; inputData->mBuffers[0].mNumberChannels = channels;
#ifdef _DEBUG #ifdef _DEBUG
[BadSampleCleaner cleanSamples:(float *)inputData->mBuffers[0].mData [BadSampleCleaner cleanSamples:(float *)inputData->mBuffers[0].mData
amount:inputData->mBuffers[0].mDataByteSize / sizeof(float) amount:inputData->mBuffers[0].mDataByteSize / sizeof(float)
location:@"final output"]; location:@"final output"];
#endif #endif
return 0; return 0;
}; };
[_au setMaximumFramesToRender:512]; [_au setMaximumFramesToRender:512];
UInt32 value; UInt32 value;
UInt32 size = sizeof(value); UInt32 size = sizeof(value);
value = CHUNK_SIZE; value = CHUNK_SIZE;
AudioUnitSetProperty(_eq, kAudioUnitProperty_MaximumFramesPerSlice, AudioUnitSetProperty(_eq, kAudioUnitProperty_MaximumFramesPerSlice,
kAudioUnitScope_Global, 0, &value, size); kAudioUnitScope_Global, 0, &value, size);
value = 127; value = 127;
AudioUnitSetProperty(_eq, kAudioUnitProperty_RenderQuality, AudioUnitSetProperty(_eq, kAudioUnitProperty_RenderQuality,
kAudioUnitScope_Global, 0, &value, size); kAudioUnitScope_Global, 0, &value, size);
AURenderCallbackStruct callbackStruct; AURenderCallbackStruct callbackStruct;
callbackStruct.inputProcRefCon = (__bridge void *)self; callbackStruct.inputProcRefCon = (__bridge void *)self;
callbackStruct.inputProc = renderCallback; callbackStruct.inputProc = renderCallback;
AudioUnitSetProperty(_eq, kAudioUnitProperty_SetRenderCallback, AudioUnitSetProperty(_eq, kAudioUnitProperty_SetRenderCallback,
kAudioUnitScope_Input, 0, &callbackStruct, sizeof(callbackStruct)); kAudioUnitScope_Input, 0, &callbackStruct, sizeof(callbackStruct));
AudioUnitReset(_eq, kAudioUnitScope_Input, 0); AudioUnitReset(_eq, kAudioUnitScope_Input, 0);
AudioUnitReset(_eq, kAudioUnitScope_Output, 0); AudioUnitReset(_eq, kAudioUnitScope_Output, 0);
AudioUnitReset(_eq, kAudioUnitScope_Global, 0); AudioUnitReset(_eq, kAudioUnitScope_Global, 0);
_err = AudioUnitInitialize(_eq); _err = AudioUnitInitialize(_eq);
if(_err) if(_err)
return NO; return NO;
eqInitialized = YES; eqInitialized = YES;
[self setEqualizerEnabled:[[[[NSUserDefaultsController sharedUserDefaultsController] defaults] objectForKey:@"GraphicEQenable"] boolValue]]; [self setEqualizerEnabled:[[[[NSUserDefaultsController sharedUserDefaultsController] defaults] objectForKey:@"GraphicEQenable"] boolValue]];
[outputController beginEqualizer:_eq]; [outputController beginEqualizer:_eq];
[_au allocateRenderResourcesAndReturnError:&err]; [_au allocateRenderResourcesAndReturnError:&err];
visController = [VisualizationController sharedController]; visController = [VisualizationController sharedController];
[[NSUserDefaultsController sharedUserDefaultsController] addObserver:self forKeyPath:@"values.outputDevice" options:0 context:kOutputCoreAudioContext]; [[NSUserDefaultsController sharedUserDefaultsController] addObserver:self forKeyPath:@"values.outputDevice" options:0 context:kOutputCoreAudioContext];
[[NSUserDefaultsController sharedUserDefaultsController] addObserver:self forKeyPath:@"values.GraphicEQenable" options:0 context:kOutputCoreAudioContext]; [[NSUserDefaultsController sharedUserDefaultsController] addObserver:self forKeyPath:@"values.GraphicEQenable" options:0 context:kOutputCoreAudioContext];
[[NSUserDefaultsController sharedUserDefaultsController] addObserver:self forKeyPath:@"values.eqPreamp" options:(NSKeyValueObservingOptionInitial | NSKeyValueObservingOptionNew) context:kOutputCoreAudioContext]; [[NSUserDefaultsController sharedUserDefaultsController] addObserver:self forKeyPath:@"values.eqPreamp" options:(NSKeyValueObservingOptionInitial | NSKeyValueObservingOptionNew) context:kOutputCoreAudioContext];
observersapplied = YES; observersapplied = YES;
return (err == nil); return (err == nil);
}
} }
- (void)setVolume:(double)v { - (void)setVolume:(double)v {
@ -810,82 +812,85 @@ default_device_changed(AudioObjectID inObjectID, UInt32 inNumberAddresses, const
} }
- (void)stop { - (void)stop {
stopInvoked = YES; @synchronized(self) {
if(observersapplied) { if(stopInvoked) return;
[[NSUserDefaultsController sharedUserDefaultsController] removeObserver:self forKeyPath:@"values.outputDevice" context:kOutputCoreAudioContext]; stopInvoked = YES;
[[NSUserDefaultsController sharedUserDefaultsController] removeObserver:self forKeyPath:@"values.GraphicEQenable" context:kOutputCoreAudioContext]; if(observersapplied) {
[[NSUserDefaultsController sharedUserDefaultsController] removeObserver:self forKeyPath:@"values.eqPreamp" context:kOutputCoreAudioContext]; [[NSUserDefaultsController sharedUserDefaultsController] removeObserver:self forKeyPath:@"values.outputDevice" context:kOutputCoreAudioContext];
observersapplied = NO; [[NSUserDefaultsController sharedUserDefaultsController] removeObserver:self forKeyPath:@"values.GraphicEQenable" context:kOutputCoreAudioContext];
} [[NSUserDefaultsController sharedUserDefaultsController] removeObserver:self forKeyPath:@"values.eqPreamp" context:kOutputCoreAudioContext];
if(stopNext && !paused) { observersapplied = NO;
if(!started) {
// This happens if playback is started on a very short file, and the queue is empty or at the end of the playlist
started = YES;
NSError *err;
[_au startHardwareAndReturnError:&err];
} }
while(![[outputController buffer] isEmpty]) { if(stopNext && !paused) {
[writeSemaphore signal]; if(!started) {
[readSemaphore signal]; // This happens if playback is started on a very short file, and the queue is empty or at the end of the playlist
usleep(500); started = YES;
NSError *err;
[_au startHardwareAndReturnError:&err];
}
while(![[outputController buffer] isEmpty]) {
[writeSemaphore signal];
[readSemaphore signal];
usleep(500);
}
} }
} if(stopNext) {
if(stopNext) { stopNext = NO;
stopNext = NO; [self signalEndOfStream];
[self signalEndOfStream];
}
stopping = YES;
paused = NO;
[writeSemaphore signal];
[readSemaphore signal];
if(listenerapplied) {
AudioObjectPropertyAddress theAddress = {
.mSelector = kAudioHardwarePropertyDefaultOutputDevice,
.mScope = kAudioObjectPropertyScopeGlobal,
.mElement = kAudioObjectPropertyElementMaster
};
AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &theAddress, default_device_changed, (__bridge void *_Nullable)(self));
listenerapplied = NO;
}
if(_au) {
if(started)
[_au stopHardware];
_au = nil;
}
if(running)
while(!stopped) {
stopping = YES;
[readSemaphore signal];
[writeSemaphore timedWait:5000];
} }
if(_eq) { stopping = YES;
[outputController endEqualizer:_eq]; paused = NO;
if(eqInitialized) { [writeSemaphore signal];
AudioUnitUninitialize(_eq); [readSemaphore signal];
eqInitialized = NO; if(listenerapplied) {
AudioObjectPropertyAddress theAddress = {
.mSelector = kAudioHardwarePropertyDefaultOutputDevice,
.mScope = kAudioObjectPropertyScopeGlobal,
.mElement = kAudioObjectPropertyElementMaster
};
AudioObjectRemovePropertyListener(kAudioObjectSystemObject, &theAddress, default_device_changed, (__bridge void *_Nullable)(self));
listenerapplied = NO;
}
if(_au) {
if(started)
[_au stopHardware];
_au = nil;
}
if(running) {
while(!stopped) {
stopping = YES;
[readSemaphore signal];
[writeSemaphore timedWait:5000];
}
}
if(_eq) {
[outputController endEqualizer:_eq];
if(eqInitialized) {
AudioUnitUninitialize(_eq);
eqInitialized = NO;
}
AudioComponentInstanceDispose(_eq);
_eq = NULL;
}
if(downmixer) {
downmixer = nil;
}
if(downmixerForVis) {
downmixerForVis = nil;
} }
AudioComponentInstanceDispose(_eq);
_eq = NULL;
}
if(downmixer) {
downmixer = nil;
}
if(downmixerForVis) {
downmixerForVis = nil;
}
#ifdef OUTPUT_LOG #ifdef OUTPUT_LOG
if(_logFile) { if(_logFile) {
fclose(_logFile); fclose(_logFile);
_logFile = NULL; _logFile = NULL;
} }
#endif #endif
outputController = nil; outputController = nil;
visController = nil; visController = nil;
}
} }
- (void)dealloc { - (void)dealloc {
if(!stopInvoked) [self stop];
[self stop];
} }
- (void)pause { - (void)pause {