AVSampleBufferDisplayLayer----转
http://blog.csdn.net/fernandowei/article/details/52179631
目前大多数iOS端的视频渲染都使用OpenGLES,但如果仅仅为了渲染而不做其他的例如美颜等效果,其实可以使用iOS8.0新出的AVSampleBufferDisplayLayer。对AVSampleBufferDisplayLayer,官方说明中有一句话,“The AVSampleBufferDisplayLayer class is a subclass of CALayer that displays compressed or uncompressed video frames.”,即AVSampleBufferDisplayLayer既可以用来渲染解码后的视频图片,也可以直接把未解码的视频帧送给它,完成先解码再渲染出去的步骤。
由于本人在使用AVSampleBufferDisplayLayer之前已经videotoolbox中相关api完成了h264视频的硬解,所以这里仅仅使用AVSampleBufferDisplayLayer来渲染,即送给它pixelBuffer。
个人选择了UIImageView作为渲染的view(没有直接使用UIView的原因后面会提到),而且也没有重载UIView的layerClass函数来使AVSampleBufferDisplayLayer成为这个view的默认layer(不这么做的原因后面提到)。
具体做法,首先,建立AVSampleBufferDisplayLayer并把它添加成为当前view的子layer:
- self.sampleBufferDisplayLayer = [[AVSampleBufferDisplayLayer alloc] init];
- self.sampleBufferDisplayLayer.frame = self.bounds;
- self.sampleBufferDisplayLayer.position = CGPointMake(CGRectGetMidX(self.bounds), CGRectGetMidY(self.bounds));
- self.sampleBufferDisplayLayer.videoGravity = AVLayerVideoGravityResizeAspect;
- self.sampleBufferDisplayLayer.opaque = YES;
- [self.layer addSublayer:self.sampleBufferDisplayLayer];
其次,把得到的pixelbuffer包装成CMSampleBuffer并设置时间信息:
- //把pixelBuffer包装成samplebuffer送给displayLayer
- - (void)dispatchPixelBuffer:(CVPixelBufferRef) pixelBuffer
- {
- if (!pixelBuffer){
- return;
- }
- @synchronized(self) {
- if (self.previousPixelBuffer){
- CFRelease(self.previousPixelBuffer);
- self.previousPixelBuffer = nil;
- }
- self.previousPixelBuffer = CFRetain(pixelBuffer);
- }
- //不设置具体时间信息
- CMSampleTimingInfo timing = {kCMTimeInvalid, kCMTimeInvalid, kCMTimeInvalid};
- //获取视频信息
- CMVideoFormatDescriptionRef videoInfo = NULL;
- OSStatus result = CMVideoFormatDescriptionCreateForImageBuffer(NULL, pixelBuffer, &videoInfo);
- NSParameterAssert(result == 0 && videoInfo != NULL);
- CMSampleBufferRef sampleBuffer = NULL;
- result = CMSampleBufferCreateForImageBuffer(kCFAllocatorDefault,pixelBuffer, true, NULL, NULL, videoInfo, &timing, &sampleBuffer);
- NSParameterAssert(result == 0 && sampleBuffer != NULL);
- CFRelease(pixelBuffer);
- CFRelease(videoInfo);
- CFArrayRef attachments = CMSampleBufferGetSampleAttachmentsArray(sampleBuffer, YES);
- CFMutableDictionaryRef dict = (CFMutableDictionaryRef)CFArrayGetValueAtIndex(attachments, 0);
- CFDictionarySetValue(dict, kCMSampleAttachmentKey_DisplayImmediately, kCFBooleanTrue);
- [self enqueueSampleBuffer:sampleBuffer toLayer:self.sampleBufferDisplayLayer];
- CFRelease(sampleBuffer);
这里不设置具体时间信息且设置kCMSampleAttachmentKey_DisplayImmediately为true,是因为这里只需要渲染不需要解码,所以不必根据dts设置解码时间、根据pts设置渲染时间。
最后,数据送给AVSampleBufferDisplayLayer渲染就可以了。
- <p class="p1"><pre name="code" class="objc">- (void)enqueueSampleBuffer:(CMSampleBufferRef) sampleBuffer toLayer:(AVSampleBufferDisplayLayer*) layer
- {
- if (sampleBuffer){
- CFRetain(sampleBuffer);
- [layer enqueueSampleBuffer:sampleBuffer];
- CFRelease(sampleBuffer);
- if (layer.status == AVQueuedSampleBufferRenderingStatusFailed){
- NSLog(@"ERROR: %@", layer.error);
- if (-11847 == layer.error.code){
- [self rebuildSampleBufferDisplayLayer];
- }
- }else{
- // NSLog(@"STATUS: %i", (int)layer.status);
- }
- }else{
- NSLog(@"ignore null samplebuffer");
- }
- }
可以看到,使用AVSampleBufferDisplayLayer进行视频渲染比使用OpenGLES简单了许多。不过遗憾的是,这里有一个iOS系统级的bug,AVSampleBufferDisplayLayer会在遇到后台事件等一些打断事件时失效,即如果视频正在渲染,这个时候摁home键或者锁屏键,再回到视频的渲染界面,就会显示渲染失败,错误码就是上述代码中的-11847。
个人在遇到上述问题后,联想到之前使用videotoolbox解码视频时遇到类似后台事件时VTDecompressionSession会失效从而需要撤销当前VTDecompressionSession来重新建立VTDecompressionSession的过程,在AVSampleBufferDisplayLayer失效时,也去撤销当前这个AVSampleBufferDisplayLayer再重建一个;这里说到之前卖的一个关子,如果这个AVSampleBufferDisplayLayer是view的默认layer,这时就没法只撤销layer而不动view,所以把AVSampleBufferDisplayLayer作为view的子layer更方便,撤销重建的过程如下:
- - (void)rebuildSampleBufferDisplayLayer{
- @synchronized(self) {
- [self teardownSampleBufferDisplayLayer];
- [self setupSampleBufferDisplayLayer];
- }
- }
- - (void)teardownSampleBufferDisplayLayer
- {
- if (self.sampleBufferDisplayLayer){
- [self.sampleBufferDisplayLayer stopRequestingMediaData];
- [self.sampleBufferDisplayLayer removeFromSuperlayer];
- self.sampleBufferDisplayLayer = nil;
- }
- }
- - (void)setupSampleBufferDisplayLayer{
- if (!self.sampleBufferDisplayLayer){
- self.sampleBufferDisplayLayer = [[AVSampleBufferDisplayLayer alloc] init];
- self.sampleBufferDisplayLayer.frame = self.bounds;
- self.sampleBufferDisplayLayer.position = CGPointMake(CGRectGetMidX(self.bounds), CGRectGetMidY(self.bounds));
- self.sampleBufferDisplayLayer.videoGravity = AVLayerVideoGravityResizeAspect;
- self.sampleBufferDisplayLayer.opaque = YES;
- [self.layer addSublayer:self.sampleBufferDisplayLayer];
- }else{
- [CATransaction begin];
- [CATransaction setDisableActions:YES];
- self.sampleBufferDisplayLayer.frame = self.bounds;
- self.sampleBufferDisplayLayer.position = CGPointMake(CGRectGetMidX(self.bounds), CGRectGetMidY(self.bounds));
- [CATransaction commit];
- }
- [self addObserver];
- }
当然,需要监听后台事件,如下:
- - (void)addObserver{
- if (!hasAddObserver){
- NSNotificationCenter * notificationCenter = [NSNotificationCenter defaultCenter];
- [notificationCenter addObserver: self selector:@selector(didResignActive) name:UIApplicationWillResignActiveNotification object:nil];
- [notificationCenter addObserver: self selector:@selector(didBecomeActive) name:UIApplicationDidBecomeActiveNotification object:nil];
- hasAddObserver = YES;
- }
- }
做到这里,基本的问题都解决了,视频可以正常渲染了;不过还有一个稍令人不悦的小问题,即app被切到后台再切回来时,由于这个时候AVSampleBufferDisplayLayer已经失效,所以这个时候渲染的view会是黑屏,这会有一到两秒的时间,直到layer重新建立好并开始渲染。那怎么让这个时候不出现黑屏呢?就需要前面提到的UIImageView,做法如下:
首先,对于每个到来的pixelbuffer,要保留它直到下一个pixelbuffer到来,如下函数中粗体所示:
- - (void)dispatchPixelBuffer:(CVPixelBufferRef) pixelBuffer
- {
- if (!pixelBuffer){
- return;
- }
- <strong> @synchronized(self) {
- if (self.previousPixelBuffer){
- CFRelease(self.previousPixelBuffer);
- self.previousPixelBuffer = nil;
- }
- self.previousPixelBuffer = CFRetain(pixelBuffer);
- }</strong>
- ...........略去其他
- }
其次,当切后台事件resignActive事件到来时,用当前最新保存的pixelbuffer去设置UIImageView的image,当然pixelbuffer要先转化成UIImage,方法如下:
- - (UIImage*)getUIImageFromPixelBuffer:(CVPixelBufferRef)pixelBuffer
- {
- UIImage *uiImage = nil;
- if (pixelBuffer){
- CIImage *ciImage = [CIImage imageWithCVPixelBuffer:pixelBuffer];
- uiImage = [UIImage imageWithCIImage:ciImage];
- UIGraphicsBeginImageContext(self.bounds.size);
- [uiImage drawInRect:self.bounds];
- uiImage = UIGraphicsGetImageFromCurrentImageContext();
- UIGraphicsEndImageContext();
- }
- return uiImage;
- }
然后在resignActive事件处理函数中,设置UIImageView的image,如下:
- - (void)didResignActive{
- NSLog(@"resign active");
- [self setupPlayerBackgroundImage];
- }
- - (void) setupPlayerBackgroundImage{
- if (self.isVideoHWDecoderEnable){
- @synchronized(self) {
- if (self.previousPixelBuffer){
- self.image = [self getUIImageFromPixelBuffer:self.previousPixelBuffer];
- CFRelease(self.previousPixelBuffer);
- self.previousPixelBuffer = nil;
- }
- }
- }
- }
这样,切完后台回来前台,在layer还没有重新建立好之前,看到的就是设置的UIImageView的image而不是黑屏了,而这个image就是切后台开始时渲染的最后一帧画面。
对于前面说到的AVSampleBufferDisplayLayer失效后重建导致的黑屏时间,个人通过验证发现,如果这个重建动作,即下面这句代码,
- [[AVSampleBufferDisplayLayer alloc] init]
发生在app刚从后台会到前台就会非常耗时,接近两秒,而如果是正在前台正常播放的过程中执行这句话,只需要十几毫秒;前者如此耗时的原因,经过请教其他iOS开发的同事,可能是这个时候系统优先恢复整个app的UI,其他操作被delay;
AVSampleBufferDisplayLayer----转的更多相关文章
- iOS8系统H264视频硬件编解码说明
公司项目原因,接触了一下视频流H264的编解码知识,之前项目使用的是FFMpeg多媒体库,利用CPU做视频的编码和解码,俗称为软编软解.该方法比较通用,但是占用CPU资源,编解码效率不高.一般系统都会 ...
- 第11月第3天 直播 rtmp yuv
1. LiveVideoCoreSDK AudioUnitRender ==> MicSource::inputCallback ==> GenericAudioMixer::pushBu ...
- 02:H.264学习笔记
H.264组成 1.网络提取层 (Network Abstraction Layer,NAL) 2.视讯编码层 (Video Coding Layer,VCL) a.H.264/AVC影像格式阶层架构 ...
- 01:***VideoToolbox硬编码H.264
最近接触了一些视频流H264的编解码知识,之前项目使用的是FFMpeg多媒体库,利用CPU做视频的编码和解码,俗称为软编软解.该方法比较通用,但是占用CPU资源,编解码效率不高.一般系统都会提供GPU ...
- Direct Access to Video Encoding and Decoding
来源:http://asciiwwdc.com/2014/sessions/513 Direct Access to Video Encoding and Decoding Session 5 ...
- How to decode a H.264 frame on iOS by hardware decoding?
来源:http://stackoverflow.com/questions/25197169/how-to-decode-a-h-264-frame-on-ios-by-hardware-decodi ...
- How to use VideoToolbox to decompress H.264 video stream
来源:http://stackoverflow.com/questions/29525000/how-to-use-videotoolbox-to-decompress-h-264-video-str ...
随机推荐
- Js动态获取iframe子页面的高度////////////////////////zzzz
Js动态获取iframe子页面的高度 Js动态获取iframe子页面的高度总结 问题的缘由 产品有个评论列表引用的是个iframe,高度不固定于是引发这个总结. 方法1:父级页面获取子级页面的高度 ...
- iOS原生APP与H5+JS交互////////////////////zzzz
原生代码中直接加载页面 1. 具体案例 加载本地/网络HTML5作为功能介绍页 2. 代码示例 //本地 -(void)loadLocalPage:(UIWebView*)webView ...
- php生成随机密码的几种方法
方法一: 1.在 33 – 126 中生成一个随机整数,如 35,2.将 35 转换成对应的ASCII码字符,如 35 对应 #3.重复以上 1.2 步骤 n 次,连接成 n 位的密码 该算法主要用到 ...
- Jquery EeasyUI等框架中图标的处理方法
在使用Query EasyUI.Ext等框架开发项目的时候,经常会用到很多小的图标,常见几个图片应用方式总结如下: 一.在jQuery Easyui中添加小图标 1.添加图标的两小步: 先到theme ...
- 走进spring之springmvc实战篇(二)
本篇继篇一之后加入了jdbc并使用了注解 篇一进行了对spingmvc的基础配置http://www.cnblogs.com/liuyanhao/p/4798989.html 首先让我们先了解下注解的 ...
- System.map文件【转】
转自:http://blog.csdn.net/david104/article/details/7194185 当运行GNU链接器gld(ld)时若使用了"-M"选项,或者使用n ...
- HTML如何让文本两端对齐
<p style="text-align:justify; text-justify:inter-ideograph;>日本驻华大使丹羽宇一郎:日中关系比夫妻还紧密日本驻华大使丹 ...
- angular-select绑定之后option不能更新问题
使用ng-option-- http://jsfiddle.net/sseletskyy/uky9m/1/ 以及select中加入自定义指令 convert-to-number .directive ...
- HTML转移字符对照表
body { margin: 0; padding: 0; background: #FFF; color: #000; font-family: "宋体", arial; fon ...
- 关于sass的安装
关于sass的安装真是费了九牛二虎之力,这么说一点都不夸张,好了我就不多浪费口水了,直接进入正题 1.首先要安装ruby,这个大家可以去度娘上查询,很好安装的,相信大家的智慧与实力都是可以安装成功的 ...