[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
parent
e9230a080c
commit
0997ca2c93
|
@ -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 {
|
||||||
|
|
Loading…
Reference in New Issue