Cog Audio: Made ring buffer locking mechanism more secure and/or smarter

CQTexperiment
Christopher Snowhill 2022-01-23 19:36:33 -08:00
parent 1b4b22501b
commit 494ad84ea7
6 changed files with 35 additions and 39 deletions

View File

@ -16,8 +16,6 @@
@interface Node : NSObject { @interface Node : NSObject {
VirtualRingBuffer *buffer; VirtualRingBuffer *buffer;
Semaphore *semaphore; Semaphore *semaphore;
NSLock *readLock;
NSLock *writeLock;
id __weak previousNode; id __weak previousNode;
id __weak controller; id __weak controller;
@ -41,9 +39,6 @@
- (void)setShouldReset:(BOOL)s; - (void)setShouldReset:(BOOL)s;
- (BOOL)shouldReset; - (BOOL)shouldReset;
- (NSLock *)readLock;
- (NSLock *)writeLock;
- (void)setPreviousNode:(id)p; - (void)setPreviousNode:(id)p;
- (id)previousNode; - (id)previousNode;

View File

@ -20,8 +20,6 @@
{ {
buffer = [[VirtualRingBuffer alloc] initWithLength:BUFFER_SIZE]; buffer = [[VirtualRingBuffer alloc] initWithLength:BUFFER_SIZE];
semaphore = [[Semaphore alloc] init]; semaphore = [[Semaphore alloc] init];
readLock = [[NSLock alloc] init];
writeLock = [[NSLock alloc] init];
initialBufferFilled = NO; initialBufferFilled = NO;
@ -41,7 +39,6 @@
int amountToCopy, availOutput; int amountToCopy, availOutput;
int amountLeft = amount; int amountLeft = amount;
[writeLock lock];
while (shouldContinue == YES && amountLeft > 0) while (shouldContinue == YES && amountLeft > 0)
{ {
availOutput = [buffer lengthAvailableToWriteReturningPointer:&writePtr]; availOutput = [buffer lengthAvailableToWriteReturningPointer:&writePtr];
@ -55,9 +52,13 @@
if (availOutput == 0 || shouldReset) if (availOutput == 0 || shouldReset)
{ {
[writeLock unlock]; if (availOutput)
{
// must unlock buffer before waiting, may as well write silence
memset(writePtr, 0, availOutput);
[buffer didWriteLength:availOutput];
}
[semaphore wait]; [semaphore wait];
[writeLock lock];
} }
else else
{ {
@ -66,15 +67,11 @@
amountToCopy = amountLeft; amountToCopy = amountLeft;
memcpy(writePtr, &((char *)ptr)[amount - amountLeft], amountToCopy); memcpy(writePtr, &((char *)ptr)[amount - amountLeft], amountToCopy);
if (amountToCopy > 0) [buffer didWriteLength:amountToCopy];
{
[buffer didWriteLength:amountToCopy];
}
amountLeft -= amountToCopy; amountLeft -= amountToCopy;
} }
} }
[writeLock unlock];
return (amount - amountLeft); return (amount - amountLeft);
} }
@ -103,7 +100,6 @@
return 0; return 0;
} }
[readLock lock];
availInput = [[previousNode buffer] lengthAvailableToReadReturningPointer:&readPtr]; availInput = [[previousNode buffer] lengthAvailableToReadReturningPointer:&readPtr];
/* if (availInput <= 0) { /* if (availInput <= 0) {
@ -114,14 +110,11 @@
} }
*/ */
if ([previousNode shouldReset] == YES) { if ([previousNode shouldReset] == YES) {
[writeLock lock];
[buffer empty]; [buffer empty];
shouldReset = YES; shouldReset = YES;
[previousNode setShouldReset: NO]; [previousNode setShouldReset: NO];
[writeLock unlock];
[[previousNode semaphore] signal]; [[previousNode semaphore] signal];
} }
@ -134,13 +127,12 @@
memcpy(ptr, readPtr, amountToCopy); memcpy(ptr, readPtr, amountToCopy);
[[previousNode buffer] didReadLength:amountToCopy];
if (amountToCopy > 0) if (amountToCopy > 0)
{ {
[[previousNode buffer] didReadLength:amountToCopy];
[[previousNode semaphore] signal]; [[previousNode semaphore] signal];
} }
[readLock unlock];
return amountToCopy; return amountToCopy;
} }
@ -179,22 +171,10 @@
{ {
shouldReset = YES; //Will reset on next write. shouldReset = YES; //Will reset on next write.
if (previousNode == nil) { if (previousNode == nil) {
[readLock lock];
[buffer empty]; [buffer empty];
[readLock unlock];
} }
} }
- (NSLock *)readLock
{
return readLock;
}
- (NSLock *)writeLock
{
return writeLock;
}
- (Semaphore *)semaphore - (Semaphore *)semaphore
{ {
return semaphore; return semaphore;

View File

@ -21,8 +21,6 @@
// This special node should be able to handle up to four buffers // This special node should be able to handle up to four buffers
buffer = [[VirtualRingBuffer alloc] initWithLength:BUFFER_SIZE * 4]; buffer = [[VirtualRingBuffer alloc] initWithLength:BUFFER_SIZE * 4];
semaphore = [[Semaphore alloc] init]; semaphore = [[Semaphore alloc] init];
readLock = [[NSLock alloc] init];
writeLock = [[NSLock alloc] init];
initialBufferFilled = NO; initialBufferFilled = NO;

View File

@ -97,6 +97,8 @@ static OSStatus renderCallback( void *inRefCon, AudioUnitRenderActionFlags *ioAc
atomic_fetch_add(&_self->bytesRendered, amountRead); atomic_fetch_add(&_self->bytesRendered, amountRead);
[_self->writeSemaphore signal]; [_self->writeSemaphore signal];
} }
else
[[_self->outputController buffer] didReadLength:0];
// Try repeatedly! Buffer wraps can cause a slight data shortage, as can // Try repeatedly! Buffer wraps can cause a slight data shortage, as can
// unexpected track changes. // unexpected track changes.
@ -117,6 +119,7 @@ static OSStatus renderCallback( void *inRefCon, AudioUnitRenderActionFlags *ioAc
[_self->writeSemaphore signal]; [_self->writeSemaphore signal];
} }
else { else {
[[_self->outputController buffer] didReadLength:0];
[_self->readSemaphore timedWait:500]; [_self->readSemaphore timedWait:500];
} }
} }
@ -250,8 +253,8 @@ default_device_changed(AudioObjectID inObjectID, UInt32 inNumberAddresses, const
toWrite = CHUNK_SIZE; toWrite = CHUNK_SIZE;
if (toWrite) if (toWrite)
bytesRead = [outputController readData:writePtr amount:toWrite]; bytesRead = [outputController readData:writePtr amount:toWrite];
[[outputController buffer] didWriteLength:bytesRead];
if (bytesRead) { if (bytesRead) {
[[outputController buffer] didWriteLength:bytesRead];
[readSemaphore signal]; [readSemaphore signal];
continue; continue;
} }

View File

@ -57,6 +57,8 @@
atomic_int readPointer; atomic_int readPointer;
atomic_int writePointer; atomic_int writePointer;
atomic_int bufferFilled; atomic_int bufferFilled;
NSLock *accessLock;
} }
- (id)initWithLength:(UInt32)length; - (id)initWithLength:(UInt32)length;

View File

@ -47,6 +47,8 @@ static void deallocateVirtualBuffer(void *buffer, UInt32 bufferLength);
atomic_init(&writePointer, 0); atomic_init(&writePointer, 0);
atomic_init(&bufferFilled, 0); atomic_init(&bufferFilled, 0);
accessLock = [[NSLock alloc] init];
return self; return self;
} }
@ -60,10 +62,11 @@ static void deallocateVirtualBuffer(void *buffer, UInt32 bufferLength);
{ {
// Assumption: // Assumption:
// No one is reading or writing from the buffer, in any thread, when this method is called. // No one is reading or writing from the buffer, in any thread, when this method is called.
[accessLock lock];
atomic_init(&readPointer, 0); atomic_init(&readPointer, 0);
atomic_init(&writePointer, 0); atomic_init(&writePointer, 0);
atomic_init(&bufferFilled, 0); atomic_init(&bufferFilled, 0);
[accessLock unlock];
} }
- (BOOL)isEmpty - (BOOL)isEmpty
@ -96,6 +99,7 @@ static void deallocateVirtualBuffer(void *buffer, UInt32 bufferLength);
{ {
// Assumptions: // Assumptions:
// returnedReadPointer != NULL // returnedReadPointer != NULL
[accessLock lock];
UInt32 length; UInt32 length;
// Read this pointer exactly once, so we're safe in case it is changed in another thread // Read this pointer exactly once, so we're safe in case it is changed in another thread
@ -109,6 +113,10 @@ static void deallocateVirtualBuffer(void *buffer, UInt32 bufferLength);
// Depending on out-of-order execution and memory storage, either one of these may be NULL when the buffer is empty. So we must check both. // Depending on out-of-order execution and memory storage, either one of these may be NULL when the buffer is empty. So we must check both.
*returnedReadPointer = buffer + localReadPointer; *returnedReadPointer = buffer + localReadPointer;
if (!length)
[accessLock unlock];
return length; return length;
} }
@ -118,10 +126,13 @@ static void deallocateVirtualBuffer(void *buffer, UInt32 bufferLength);
// [self lengthAvailableToReadReturningPointer:] currently returns a value >= length // [self lengthAvailableToReadReturningPointer:] currently returns a value >= length
// length > 0 // length > 0
if (atomic_fetch_add(&readPointer, length) + length >= bufferLength) if (atomic_fetch_add(&readPointer, length) + length >= bufferLength)
atomic_fetch_sub(&readPointer, bufferLength); atomic_fetch_sub(&readPointer, bufferLength);
atomic_fetch_sub(&bufferFilled, length); atomic_fetch_sub(&bufferFilled, length);
[accessLock unlock];
} }
@ -133,6 +144,7 @@ static void deallocateVirtualBuffer(void *buffer, UInt32 bufferLength);
{ {
// Assumptions: // Assumptions:
// returnedWritePointer != NULL // returnedWritePointer != NULL
[accessLock lock];
UInt32 length; UInt32 length;
// Read this pointer exactly once, so we're safe in case it is changed in another thread // Read this pointer exactly once, so we're safe in case it is changed in another thread
@ -144,6 +156,10 @@ static void deallocateVirtualBuffer(void *buffer, UInt32 bufferLength);
length = bufferLength - localWritePointer; length = bufferLength - localWritePointer;
*returnedWritePointer = buffer + localWritePointer; *returnedWritePointer = buffer + localWritePointer;
if (!length)
[accessLock unlock];
return length; return length;
} }
@ -157,6 +173,8 @@ static void deallocateVirtualBuffer(void *buffer, UInt32 bufferLength);
atomic_fetch_sub(&writePointer, bufferLength); atomic_fetch_sub(&writePointer, bufferLength);
atomic_fetch_add(&bufferFilled, length); atomic_fetch_add(&bufferFilled, length);
[accessLock unlock];
} }
@end @end