| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574 |
- // QGMP4FrameHWDecoder.m
- // Tencent is pleased to support the open source community by making vap available.
- //
- // Copyright (C) 2020 THL A29 Limited, a Tencent company. All rights reserved.
- //
- // Licensed under the MIT License (the "License"); you may not use this file except in
- // compliance with the License. You may obtain a copy of the License at
- //
- // http://opensource.org/licenses/MIT
- //
- // Unless required by applicable law or agreed to in writing, software distributed under the License is
- // distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND,
- // either express or implied. See the License for the specific language governing permissions and
- // limitations under the License.
- #import "QGMP4FrameHWDecoder.h"
- #import "QGVAPWeakProxy.h"
- #import "QGMP4AnimatedImageFrame.h"
- #import "QGBaseAnimatedImageFrame+Displaying.h"
- #import <VideoToolbox/VideoToolbox.h>
- #import "QGHWDMP4OpenGLView.h"
- #import "QGMP4Parser.h"
- #import "QGVAPSafeMutableArray.h"
- #import "NSNotificationCenter+VAPThreadSafe.h"
- #include <sys/sysctl.h>
- #import <AVFoundation/AVFoundation.h>
- @implementation UIDevice (HWD)
- - (BOOL)hwd_isSimulator {
- static dispatch_once_t token;
- static BOOL isSimulator = NO;
- dispatch_once(&token, ^{
- NSString *model = [self machineName];
- if ([model isEqualToString:@"x86_64"] || [model isEqualToString:@"i386"]) {
- isSimulator = YES;
- }
- });
- return isSimulator;
- }
- - (NSString *)machineName {
- static dispatch_once_t token;
- static NSString *name;
- dispatch_once(&token, ^{
- size_t size;
- sysctlbyname("hw.machine", NULL, &size, NULL, 0);
- char *machineName = malloc(size);
- sysctlbyname("hw.machine", machineName, &size, NULL, 0);
- name = [NSString stringWithUTF8String:machineName];
- free(machineName);
- });
- return name;
- }
- @end
- @interface NSArray (SafeOperation)
- @end
- @implementation NSArray (SafeOperation)
- - (id)safeObjectAtIndex:(NSUInteger)index
- {
- if (index >= self.count) {
- NSAssert(0, @"Error: access to array index which is beyond bounds! ");
- return nil;
- }
-
- return self[index];
- }
- @end
- @interface QGMP4FrameHWDecoder() {
-
- NSMutableArray *_buffers;
-
- int _videoStream;
- int _outputWidth, _outputHeight;
- OSStatus _status;
- BOOL _isFinish;
- VTDecompressionSessionRef _mDecodeSession;
- CMFormatDescriptionRef _mFormatDescription;
- NSInteger _finishFrameIndex;
- NSError *_constructErr;
- QGMP4ParserProxy *_mp4Parser;
-
- int _invalidRetryCount;
- }
- @property (atomic, strong) dispatch_queue_t decodeQueue; //dispatch decode task
- @property (nonatomic, strong) NSData *ppsData; //Picture Parameter Set
- @property (nonatomic, strong) NSData *spsData; //Sequence Parameter Set
- /** Video Parameter Set */
- @property (nonatomic, strong) NSData *vpsData;
- @property (atomic, assign) NSInteger lastDecodeFrame;
- @end
- NSString *const QGMP4HWDErrorDomain = @"QGMP4HWDErrorDomain";
- @implementation QGMP4FrameHWDecoder
- + (NSString *)errorDescriptionForCode:(QGMP4HWDErrorCode)errorCode {
-
- NSArray *errorDescs = @[@"文件不存在",@"非法文件格式",@"无法获取视频流信息",@"无法获取视频流",@"VTB创建desc失败",@"VTB创建session失败"];
- NSString *desc = @"";
- switch (errorCode) {
- case QGMP4HWDErrorCode_FileNotExist:
- desc = [errorDescs safeObjectAtIndex:0];
- break;
- case QGMP4HWDErrorCode_InvalidMP4File:
- desc = [errorDescs safeObjectAtIndex:1];
- break;
- case QGMP4HWDErrorCode_CanNotGetStreamInfo:
- desc = [errorDescs safeObjectAtIndex:2];
- break;
- case QGMP4HWDErrorCode_CanNotGetStream:
- desc = [errorDescs safeObjectAtIndex:3];
- break;
- case QGMP4HWDErrorCode_ErrorCreateVTBDesc:
- desc = [errorDescs safeObjectAtIndex:4];
- break;
- case QGMP4HWDErrorCode_ErrorCreateVTBSession:
- desc = [errorDescs safeObjectAtIndex:5];
- break;
-
- default:
- break;
- }
- return desc;
- }
- - (instancetype)initWith:(QGMP4HWDFileInfo *)fileInfo error:(NSError *__autoreleasing *)error{
-
- if (self = [super initWith:fileInfo error:error]) {
- _decodeQueue = dispatch_queue_create("com.qgame.vap.decode", DISPATCH_QUEUE_SERIAL);
- _lastDecodeFrame = -1;
- _mp4Parser = fileInfo.mp4Parser;
- BOOL isOpenSuccess = [self onInputStart];
- if (!isOpenSuccess) {
- VAP_Event(kQGVAPModuleCommon, @"onInputStart fail!");
- *error = _constructErr;
- self = nil;
- return nil;
- }
- [self registerNotification];
- }
- return self;
- }
- - (void)registerNotification {
- }
- - (void)hwd_didReceiveEnterBackgroundNotification:(NSNotification *)notification {
-
- }
- - (void)decodeFrame:(NSInteger)frameIndex buffers:(NSMutableArray *)buffers {
-
- if (frameIndex == self.currentDecodeFrame) {
- VAP_Event(kQGVAPModuleCommon, @"already in decode");
- return ;
- }
- self.currentDecodeFrame = frameIndex;
- _buffers = buffers;
- dispatch_async(self.decodeQueue, ^{
- if (frameIndex != self.lastDecodeFrame + 1) {
- // 必须是依次增大,否则解出来的画面会异常
- return;
- }
- [self _decodeFrame:frameIndex drop:NO];
- });
- }
- - (void)_decodeFrame:(NSInteger)frameIndex drop:(BOOL)dropFlag {
- if (_isFinish) {
- return ;
- }
-
- if (!_buffers) {
- return ;
- }
-
- if (self.spsData == nil || self.ppsData == nil) {
- return ;
- }
-
- //解码开始时间
- NSDate *startDate = [NSDate date];
- NSData *packetData = [_mp4Parser readPacketOfSample:frameIndex];
- if (!packetData.length) {
- _finishFrameIndex = frameIndex;
- [self _onInputEnd];
- return;
- }
-
- // 获取当前帧pts,pts是在parse mp4 box时得到的
- uint64_t currentPts = [_mp4Parser.videoSamples[frameIndex] pts];
-
- CVPixelBufferRef outputPixelBuffer = NULL;
- // 4. get NALUnit payload into a CMBlockBuffer,
- CMBlockBufferRef blockBuffer = NULL;
-
- _status = CMBlockBufferCreateWithMemoryBlock(kCFAllocatorDefault,
- (void *)packetData.bytes,
- packetData.length,
- kCFAllocatorNull, NULL, 0,
- packetData.length, 0,
- &blockBuffer);
- // 6. create a CMSampleBuffer.
- CMSampleBufferRef sampleBuffer = NULL;
- const size_t sampleSizeArray[] = {packetData.length};
- _status = CMSampleBufferCreateReady(kCFAllocatorDefault,
- blockBuffer,
- _mFormatDescription,
- 1, 0, NULL, 1, sampleSizeArray,
- &sampleBuffer);
-
- if (blockBuffer) {
- CFRelease(blockBuffer);
- }
-
- // 7. use VTDecompressionSessionDecodeFrame
- if (@available(iOS 9.0, *)) {
- __typeof(self) __weak weakSelf = self;
- VTDecodeFrameFlags flags = 0;
- VTDecodeInfoFlags flagOut = 0;
- OSStatus status = VTDecompressionSessionDecodeFrameWithOutputHandler(_mDecodeSession, sampleBuffer, flags, &flagOut, ^(OSStatus status, VTDecodeInfoFlags infoFlags, CVImageBufferRef _Nullable imageBuffer, CMTime presentationTimeStamp, CMTime presentationDuration) {
- __typeof(self) strongSelf = weakSelf;
- if (strongSelf == nil) {
- return;
- }
-
- [strongSelf handleDecodePixelBuffer:imageBuffer
- sampleBuffer:sampleBuffer
- frameIndex:frameIndex
- currentPts:currentPts
- startDate:startDate
- status:status
- needDrop:dropFlag];
- });
-
- if (status == kVTInvalidSessionErr) {
- CFRelease(sampleBuffer);
-
- // 防止陷入死循环
- if (_invalidRetryCount >= 3) {
- return;
- }
-
- [self resetDecoder];
- // 从最近I帧一直解码到当前帧,中间帧丢弃
- [self findKeyFrameAndDecodeToCurrent:frameIndex];
- } else {
- _invalidRetryCount = 0;
- }
-
- } else {
- // 7. use VTDecompressionSessionDecodeFrame
- VTDecodeFrameFlags flags = 0;
- VTDecodeInfoFlags flagOut = 0;
- _status = VTDecompressionSessionDecodeFrame(_mDecodeSession, sampleBuffer, flags, &outputPixelBuffer, &flagOut);
-
- if (_status == kVTInvalidSessionErr) {
- CFRelease(sampleBuffer);
- // 防止陷入死循环
- if (_invalidRetryCount >= 3) {
- return;
- }
-
- [self resetDecoder];
- // 从最近I帧一直解码到当前帧,中间帧丢弃
- [self findKeyFrameAndDecodeToCurrent:frameIndex];
-
- return;
- } else {
- _invalidRetryCount = 0;
- }
-
- [self handleDecodePixelBuffer:outputPixelBuffer
- sampleBuffer:sampleBuffer
- frameIndex:frameIndex
- currentPts:currentPts
- startDate:startDate
- status:_status
- needDrop:dropFlag];
-
- }
- }
- - (void)handleDecodePixelBuffer:(CVPixelBufferRef)pixelBuffer
- sampleBuffer:(CMSampleBufferRef)sampleBuffer
- frameIndex:(NSInteger)frameIndex
- currentPts:(uint64_t)currentPts
- startDate:(NSDate *)startDate
- status:(OSStatus)status
- needDrop:(BOOL)dropFlag {
-
- self.lastDecodeFrame = frameIndex;
-
- CFRelease(sampleBuffer);
-
- if(status == kVTInvalidSessionErr) {
- VAP_Error(kQGVAPModuleCommon, @"decompress fail! frame:%@ kVTInvalidSessionErr error:%@", @(frameIndex), @(status));
- } else if(status == kVTVideoDecoderBadDataErr) {
- VAP_Error(kQGVAPModuleCommon, @"decompress fail! frame:%@ kVTVideoDecoderBadDataErr error:%@", @(frameIndex), @(status));
- } else if(status != noErr) {
- VAP_Error(kQGVAPModuleCommon, @"decompress fail! frame:%@ error:%@", @(frameIndex), @(status));
- }
-
- if (dropFlag) {
- return;
- }
-
- QGMP4AnimatedImageFrame *newFrame = [[QGMP4AnimatedImageFrame alloc] init];
- // imagebuffer会在frame回收时释放
- CVPixelBufferRetain(pixelBuffer);
- newFrame.pixelBuffer = pixelBuffer;
- newFrame.frameIndex = frameIndex; //dts顺序
- NSTimeInterval decodeTime = [[NSDate date] timeIntervalSinceDate:startDate]*1000;
- newFrame.decodeTime = decodeTime;
- newFrame.defaultFps = (int)_mp4Parser.fps;
- newFrame.pts = currentPts;
-
- // 8. insert into buffer
- [_buffers addObject:newFrame];
-
- // 9. sort
- [_buffers sortUsingComparator:^NSComparisonResult(QGMP4AnimatedImageFrame * _Nonnull obj1, QGMP4AnimatedImageFrame * _Nonnull obj2) {
- return [@(obj1.pts) compare:@(obj2.pts)];
- }];
- }
- #pragma mark - override
- - (BOOL)shouldStopDecode:(NSInteger)nextFrameIndex {
- return _isFinish;
- }
- - (BOOL)isFrameIndexBeyondEnd:(NSInteger)frameIndex {
-
- if (_finishFrameIndex > 0) {
- return (frameIndex >= _finishFrameIndex);
- }
- return NO;
- }
- -(void)dealloc {
-
- [[NSNotificationCenter defaultCenter] removeObserver:self];
- [self _onInputEnd];
- self.fileInfo.occupiedCount --;
- if (self.fileInfo.occupiedCount <= 0) {
-
- }
- }
- #pragma mark - private methods
- - (BOOL)onInputStart {
-
- NSFileManager *fileMgr = [NSFileManager defaultManager];
- if (![fileMgr fileExistsAtPath:self.fileInfo.filePath]) {
- _constructErr = [NSError errorWithDomain:QGMP4HWDErrorDomain code:QGMP4HWDErrorCode_FileNotExist userInfo:[self errorUserInfo]];
- return NO;
- }
-
- _isFinish = NO;
- self.vpsData = nil;
- self.spsData = nil;
- self.ppsData = nil;
- _outputWidth = (int)_mp4Parser.picWidth;
- _outputHeight = (int)_mp4Parser.picHeight;
-
- BOOL paramsSetInitSuccess = [self initPPSnSPS];
- return paramsSetInitSuccess;
- }
- - (BOOL)initPPSnSPS {
-
- VAP_Info(kQGVAPModuleCommon, @"initPPSnSPS");
- if (self.spsData && self.ppsData) {
- VAP_Error(kQGVAPModuleCommon, @"sps&pps is already has value.");
- return YES;
- }
-
- self.spsData = _mp4Parser.spsData;
- self.ppsData = _mp4Parser.ppsData;
- self.vpsData = _mp4Parser.vpsData;
-
- // 2. create CMFormatDescription
- if (self.spsData != nil && self.ppsData != nil && _mp4Parser.videoCodecID != QGMP4VideoStreamCodecIDUnknown) {
- if (_mp4Parser.videoCodecID == QGMP4VideoStreamCodecIDH264) {
- const uint8_t* const parameterSetPointers[2] = { (const uint8_t*)[self.spsData bytes], (const uint8_t*)[self.ppsData bytes] };
- const size_t parameterSetSizes[2] = { [self.spsData length], [self.ppsData length] };
-
- _status = CMVideoFormatDescriptionCreateFromH264ParameterSets(kCFAllocatorDefault,
- 2,
- parameterSetPointers,
- parameterSetSizes,
- 4,
- &_mFormatDescription);
- if (_status != noErr) {
- VAP_Event(kQGVAPModuleCommon, @"CMVideoFormatDescription. Creation: %@.", (_status == noErr) ? @"successfully." : @"failed.");
- _constructErr = [NSError errorWithDomain:QGMP4HWDErrorDomain code:QGMP4HWDErrorCode_ErrorCreateVTBDesc userInfo:[self errorUserInfo]];
- return NO;
- }
- } else if (_mp4Parser.videoCodecID == QGMP4VideoStreamCodecIDH265) {
- if (@available(iOS 11.0, *)) {
- if(VTIsHardwareDecodeSupported(kCMVideoCodecType_HEVC)) {
- const uint8_t* const parameterSetPointers[3] = {(const uint8_t*)[self.vpsData bytes], (const uint8_t*)[self.spsData bytes], (const uint8_t*)[self.ppsData bytes]};
- const size_t parameterSetSizes[3] = {[self.vpsData length], [self.spsData length], [self.ppsData length]};
-
- _status = CMVideoFormatDescriptionCreateFromHEVCParameterSets(kCFAllocatorDefault,
- 3, // parameter_set_count
- parameterSetPointers, // ¶meter_set_pointers
- parameterSetSizes, // ¶meter_set_sizes
- 4, // nal_unit_header_length
- NULL,
- &_mFormatDescription);
- if (_status != noErr) {
- VAP_Event(kQGVAPModuleCommon, @"CMVideoFormatDescription. Creation: %@.", (_status == noErr) ? @"successfully." : @"failed.");
- _constructErr = [NSError errorWithDomain:QGMP4HWDErrorDomain code:QGMP4HWDErrorCode_ErrorCreateVTBDesc userInfo:[self errorUserInfo]];
- return NO;
- }
- } else {
- VAP_Event(kQGVAPModuleCommon, @"H.265 decoding is un-supported because of the hardware");
- return NO;
- }
- } else {
- VAP_Event(kQGVAPModuleCommon, @"System version is too low to support H.265 decoding");
- return NO;
- }
- }
- }
-
- // 3. create VTDecompressionSession
- return [self createDecompressionSession];;
- }
- - (BOOL)createDecompressionSession {
- CFDictionaryRef attrs = NULL;
- const void *keys[] = {kCVPixelBufferPixelFormatTypeKey};
- // kCVPixelFormatType_420YpCbCr8Planar is YUV420
- // kCVPixelFormatType_420YpCbCr8BiPlanarFullRange is NV12
- uint32_t v = kCVPixelFormatType_420YpCbCr8BiPlanarFullRange;
- const void *values[] = { CFNumberCreate(NULL, kCFNumberSInt32Type, &v) };
- attrs = CFDictionaryCreate(NULL, keys, values, 1, NULL, NULL);
-
- if ([UIDevice currentDevice].systemVersion.floatValue >= 9.0) {
- _status = VTDecompressionSessionCreate(kCFAllocatorDefault,
- _mFormatDescription,
- NULL,
- attrs,
- NULL,
- &_mDecodeSession);
- if (_status != noErr) {
- CFRelease(attrs);
- _constructErr = [NSError errorWithDomain:QGMP4HWDErrorDomain code:QGMP4HWDErrorCode_ErrorCreateVTBSession userInfo:[self errorUserInfo]];
- return NO;
- }
- } else {
- VTDecompressionOutputCallbackRecord callBackRecord;
- callBackRecord.decompressionOutputCallback = didDecompress;
- callBackRecord.decompressionOutputRefCon = NULL;
-
- _status = VTDecompressionSessionCreate(kCFAllocatorDefault,
- _mFormatDescription,
- NULL, attrs,
- &callBackRecord,
- &_mDecodeSession);
- if (_status != noErr) {
- CFRelease(attrs);
- _constructErr = [NSError errorWithDomain:QGMP4HWDErrorDomain code:QGMP4HWDErrorCode_ErrorCreateVTBSession userInfo:[self errorUserInfo]];
- return NO;
- }
- }
- CFRelease(attrs);
- return YES;
- }
- - (void)resetDecoder {
- // delete
- if (_mDecodeSession) {
- VTDecompressionSessionWaitForAsynchronousFrames(_mDecodeSession);
- VTDecompressionSessionInvalidate(_mDecodeSession);
- CFRelease(_mDecodeSession);
- _mDecodeSession = NULL;
- }
-
- // recreate
- [self createDecompressionSession];
- }
- - (void)findKeyFrameAndDecodeToCurrent:(NSInteger)frameIndex {
-
- [[NSNotificationCenter defaultCenter] postNotificationName:kQGVAPDecoderSeekStart object:self];
-
- NSArray<NSNumber *> *keyframeIndexes = [_mp4Parser videoSyncSampleIndexes];
- NSInteger index = [[keyframeIndexes firstObject] integerValue];
- for(NSNumber *number in keyframeIndexes) {
- if(number.integerValue < frameIndex) {
- index = number.integerValue;
- continue;
- } else {
- break;
- }
- }
-
- // seek to last key frame
- while (index < frameIndex) {
- [self _decodeFrame:index drop:YES];
- index++;
- }
- [self _decodeFrame:frameIndex drop:NO];
-
- [[NSNotificationCenter defaultCenter] postNotificationName:kQGVAPDecoderSeekFinish object:self];
- }
- - (void)_onInputEnd {
- if (_isFinish) {
- return ;
- }
- _isFinish = YES;
- if (_mDecodeSession) {
- VTDecompressionSessionWaitForAsynchronousFrames(_mDecodeSession);
- VTDecompressionSessionInvalidate(_mDecodeSession);
- CFRelease(_mDecodeSession);
- _mDecodeSession = NULL;
- }
- if (self.spsData || self.ppsData || self.vpsData) {
- self.spsData = nil;
- self.ppsData = nil;
- self.vpsData = nil;
- }
- if (_mFormatDescription) {
- CFRelease(_mFormatDescription);
- _mFormatDescription = NULL;
- }
- }
- - (void)onInputEnd {
-
- //为确保任务停止,必须同步执行
- __weak __typeof(self) weakSelf = self;
- if ([NSThread isMainThread]) {
- dispatch_sync(self.decodeQueue, ^{
- [weakSelf _onInputEnd];
- });
- } else {
- dispatch_async(self.decodeQueue, ^{
- [weakSelf _onInputEnd];
- });
- }
- }
- //decode callback
- static void didDecompress(void *decompressionOutputRefCon, void *sourceFrameRefCon, OSStatus status, VTDecodeInfoFlags infoFlags, CVImageBufferRef pixelBuffer, CMTime presentationTimeStamp, CMTime presentationDuration ){
-
- CVPixelBufferRef *outputPixelBuffer = (CVPixelBufferRef *)sourceFrameRefCon;
- *outputPixelBuffer = CVPixelBufferRetain(pixelBuffer);
- }
- - (NSDictionary *)errorUserInfo {
- NSDictionary *userInfo = @{@"location" : self.fileInfo.filePath ? : @""};
- return userInfo;
- }
- @end
|