Fix the memory leak in the video decoder

This commit is contained in:
Cameron Gutman 2014-10-20 12:55:23 -04:00
parent f909371a90
commit f14a2a66dd
2 changed files with 74 additions and 32 deletions

View File

@ -65,9 +65,8 @@ void DrSubmitDecodeUnit(PDECODE_UNIT decodeUnit)
entry = entry->next; entry = entry->next;
} }
// This function will take our buffer
[renderer submitDecodeBuffer:data length:decodeUnit->fullLength]; [renderer submitDecodeBuffer:data length:decodeUnit->fullLength];
free(data);
} }
} }

View File

@ -46,6 +46,35 @@
OSStatus status; OSStatus status;
size_t oldOffset = CMBlockBufferGetDataLength(existingBuffer); size_t oldOffset = CMBlockBufferGetDataLength(existingBuffer);
// If we're at index 1 (first NALU in frame), enqueue this buffer to the memory block
// so it can handle freeing it when the block buffer is destroyed
if (offset == 1) {
int dataLength = nalLength - NALU_START_PREFIX_SIZE;
// Pass the real buffer pointer directly (no offset)
// This will give it to the block buffer to free when it's released.
// All further calls to CMBlockBufferAppendMemoryBlock will do so
// at an offset and will not be asking the buffer to be freed.
status = CMBlockBufferAppendMemoryBlock(existingBuffer, data,
nalLength + 1, // Add 1 for the offset we decremented
kCFAllocatorDefault,
NULL, 0, nalLength + 1, 0);
if (status != noErr) {
NSLog(@"CMBlockBufferReplaceDataBytes failed: %d", (int)status);
return;
}
// Write the length prefix to existing buffer
const uint8_t lengthBytes[] = {(uint8_t)(dataLength >> 24), (uint8_t)(dataLength >> 16),
(uint8_t)(dataLength >> 8), (uint8_t)dataLength};
status = CMBlockBufferReplaceDataBytes(lengthBytes, existingBuffer,
oldOffset, NAL_LENGTH_PREFIX_SIZE);
if (status != noErr) {
NSLog(@"CMBlockBufferReplaceDataBytes failed: %d", (int)status);
return;
}
}
else {
// Append a 4 byte buffer to this block for the length prefix // Append a 4 byte buffer to this block for the length prefix
status = CMBlockBufferAppendMemoryBlock(existingBuffer, NULL, status = CMBlockBufferAppendMemoryBlock(existingBuffer, NULL,
NAL_LENGTH_PREFIX_SIZE, NAL_LENGTH_PREFIX_SIZE,
@ -67,7 +96,7 @@
return; return;
} }
// Append the rest of the data by simply attaching a reference to the buffer // Attach the buffer by reference to the block buffer
status = CMBlockBufferAppendMemoryBlock(existingBuffer, &data[offset+NALU_START_PREFIX_SIZE], status = CMBlockBufferAppendMemoryBlock(existingBuffer, &data[offset+NALU_START_PREFIX_SIZE],
dataLength, dataLength,
kCFAllocatorNull, // Don't deallocate data on free kCFAllocatorNull, // Don't deallocate data on free
@ -76,8 +105,10 @@
NSLog(@"CMBlockBufferReplaceDataBytes failed: %d", (int)status); NSLog(@"CMBlockBufferReplaceDataBytes failed: %d", (int)status);
return; return;
} }
}
} }
// This function must free data
- (void)submitDecodeBuffer:(unsigned char *)data length:(int)length - (void)submitDecodeBuffer:(unsigned char *)data length:(int)length
{ {
unsigned char nalType = data[FRAME_START_PREFIX_SIZE] & 0x1F; unsigned char nalType = data[FRAME_START_PREFIX_SIZE] & 0x1F;
@ -120,21 +151,25 @@
if (status != noErr) { if (status != noErr) {
NSLog(@"Failed to create format description: %d", (int)status); NSLog(@"Failed to create format description: %d", (int)status);
formatDesc = NULL; formatDesc = NULL;
return;
} }
} }
// Free the data buffer
free(data);
// No frame data to submit for these NALUs // No frame data to submit for these NALUs
return; return;
} }
if (formatDesc == NULL) { if (formatDesc == NULL) {
// Can't decode if we haven't gotten our parameter sets yet // Can't decode if we haven't gotten our parameter sets yet
free(data);
return; return;
} }
if (nalType != 0x1 && nalType != 0x5) { if (nalType != 0x1 && nalType != 0x5) {
// Don't submit parameter set data // Don't submit parameter set data
free(data);
return; return;
} }
@ -144,6 +179,7 @@
status = CMBlockBufferCreateEmpty(NULL, 0, 0, &blockBuffer); status = CMBlockBufferCreateEmpty(NULL, 0, 0, &blockBuffer);
if (status != noErr) { if (status != noErr) {
NSLog(@"CMBlockBufferCreateEmpty failed: %d", (int)status); NSLog(@"CMBlockBufferCreateEmpty failed: %d", (int)status);
free(data);
return; return;
} }
@ -166,6 +202,8 @@
[self updateBufferForRange:blockBuffer data:data offset:lastOffset length:length - lastOffset]; [self updateBufferForRange:blockBuffer data:data offset:lastOffset length:length - lastOffset];
} }
// From now on, CMBlockBuffer owns the data pointer and will free it when it's dereferenced
CMSampleBufferRef sampleBuffer; CMSampleBufferRef sampleBuffer;
status = CMSampleBufferCreate(kCFAllocatorDefault, status = CMSampleBufferCreate(kCFAllocatorDefault,
@ -176,6 +214,7 @@
&sampleBuffer); &sampleBuffer);
if (status != noErr) { if (status != noErr) {
NSLog(@"CMSampleBufferCreate failed: %d", (int)status); NSLog(@"CMSampleBufferCreate failed: %d", (int)status);
CFRelease(blockBuffer);
return; return;
} }
@ -197,6 +236,10 @@
} }
[displayLayer enqueueSampleBuffer:sampleBuffer]; [displayLayer enqueueSampleBuffer:sampleBuffer];
// Dereference the buffers
CFRelease(blockBuffer);
CFRelease(sampleBuffer);
} }
@end @end