关于GPUImage

本文内容参考自以下文章:KM文章一张图教你看懂GPUImageiOS 图像处理系列 - GPUImage源码解读(一),博文GPUImage文集。文中不再详细标注引用来源。

关于GPUImage

GPUImage框架是一个开源iOS类库,基于OpenGL进行图像和视频处理,借助GPU加速实现各种滤镜效果,并支持摄像头拍摄实时渲染。

它具有以下特性:

·             对OpenGL资源创建和滤镜做统一封装,接口简单易用,内置cache模块进行framebuffer管理。

·             框架设计灵活,支持上下级滤镜的串联、并联,以实现复杂的滤镜响应链。

·             有丰富的输入、输出组件及封装好的内置滤镜,可满足大多数场景的应用。

·             可继承GPUImageFilter实现自定义滤镜效果,扩展非常方便。

GPUImage框架

GPUImage功能的实现依赖四个部分:资源管理、Sources、Filters、Outputs。核心架构用下图概括:

1.     资源管理

GLProgram是GPUImage对OpenGL ES中的program的封装,可用vertexShader和fragmentShader创建GLProgram对象,实现自定义滤镜效果。

GPUImageContext是GPUImage对OpenGL ES上下文的封装,并添加GPUImge相关的上下文,如上下文队列(contextQueue,解决OpenGLContext不是多线程安全的问题)、CV纹理缓存(coreVideoTexttureCache)、framebuffer缓存、program使用的缓存等。

GPUImageFramebuffer是管理纹理缓存格式、帧缓存的buffer,完成缓存创建、指定渲染目标、调整视口、解锁等。其中,newCGImageFromFramebufferContents函数从缓存中获取图像数据,创建CGImageRef并返回。

GPUImageFramebufferCache是GPUImageFramebuffer的管理类。

2.    Sources

Sources目录结构如上图所示。GPUImageOutput是其他输入源的基类,输入组件将图像或视频帧数据转换成OpenGL纹理传递给滤镜组件。下面以滤镜视频拍摄这一场景下使用的GPUImageVideoCamera为例进行介绍。

GUPImagVideoCamera利用AVCaptureSession从AV输入设备采集数据。

流程概述如下:

1) 找到摄像头设备_inoutCamera、麦克风_microphone,创建摄像头输入videoInput和麦克风输入audioInput,设置为_captureSession的输入。

2) 创建摄像头输出videoOutput和麦克风输出audioOutput,设置为_captureSession的输出,设置二者的输出delegate。

3)     调用[_captureSessionstartRunning]开始获取音视频数据。

4)      音频数据到达后发送给设置的audioEncordingTarget处理。

5)     视频数据到达后利用下面的函数处理回调:

摄像头默认采样格式为YUV420,Y通道数据和UV通道数据分别存放在两个plane中,processVideoSampleBuffer:函数将两个plane的数据取出,处理后在convertYUVToRGBOutput函数中通过颜色转换的shader,得到了RGB格式的GPU纹理提供给下一级滤镜组件,完成数据输入工作。

3.     Filters

GPUImageFilter是各滤镜类的基类。它实现了GPUImageInput协议,可以作为输出对象。GPUImageInput定义了组件作为响应链中的target(上一级组件的输出对象)需要实现的接口。

另一方面GPUImageFilter继承自Sources中的GPUImageOutput,因此一个滤镜组件也可以作为输入对象。基于这种设计,输入组件、滤镜组件、输出组件可以链式串联起来,推动输入数据的处理与传递。

GPUImageFilter及子类接受一个或多个输入,调用关联的GLProgram进行渲染,将结果输出到targets属性保存的对象列表中的每一个对象。输入、输出的纹理由GPUImageFrameBuffer管理。

输入数据通过setInputFramebuffer:接口传入,由响应链中的上一级组件调用。

读入输入数据后,下图接口实现了渲染和数据向下级组件的传递。

其中,renderToTextureWithVertices:函数实现滤镜渲染,将渲染结果输出到outputframebuffer指定的缓存。

informTargetsAboutNewFrameAtTime:将自己获得的渲染结果数据传递给下一级组件,并推动其处理数据。

4.     Outputs

Outputs组件是响应链的终点,负责将处理结果输出。这组组件同样实现GPUImageInput协议,处理上一级组件的输出结果。GPUImageView是UIView的子类,用于实时将滤波结果显示在屏幕上,GPUImageMovieWriter将滤镜视频保存在本地,GPUImageTextureOutput输出GPU纹理,GPUImageRawDataOutput提供输出完成的delegate回调和读区像素数据的接口。录制滤镜视频时需要用到GPUImageView和GPUImageMovieWriter。

GPUImageMovieWriter将视频输出到磁盘,通过设置、使用AVAssetWriter,在newFrameReadyAtTime:中实现功能。

GPUImageView内部将自己的CALayer申明为CAEAGLayer,在初始化后调用createDisplayFramebuffer方法:

将renderBuffer和CALayer关联,newFrameReadyAtTime:渲染后图像会输出至CALayer上。

录制滤镜视频

下面的例子利用GPUImage录制视频、实现实时的滤镜效果渲染并将视频文件保存到本地。

通过GPUImageVideoCamera采集视频和音频数据,音频直接传递至GPUImageMovieWriter;视频传入滤镜链,经过滤镜处理后,输出的渲染结果传递给GPUImageMovieWriter准备写入本地文件,并通过GPUImageView显示在屏幕上。

下图是视频录制页面,使用内置滤镜GPUImageSwirlFilter,屏幕下方滑动条可以调整滤波核参数,实时的改变滤镜渲染的效果(本例表现为螺旋形的角度)。点击左上方按钮启动/结束录制,结束录制后视频文件保存在本地相册。

录制界面

设置响应链:

点击录制按钮开始录制视频,用NSURL及尺寸初始化_movieWriter:

结束录制:

其他滤镜效果:

玻璃球                                       阴影边框

泊松图像融合                            左图中的水印原图

 形态学开运算                               形态学闭运算

风格化之素描画                                卡通

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏AI派

Pandas基本功能详解 | 轻松玩转Pandas(2)

在 Pandas数据结构详解 | 轻松玩转Pandas(1) 介绍了 Pandas 中常用的两种数据结构 Series 以及 DataFrame,这里来看下这...

962
来自专栏Linux驱动

25.QT-模型视图

在Qt中,不管模型以什么结构组织数据,都必须为每个数据提供不同的索引值,使得视图能通过索引值访问模型中的具体数据

752
来自专栏华仔的技术笔记

SceneKitScene Kit 概要节点 (Nodes)光照动画开始用 Scene Kit 写游戏扩展默认渲染流程延时着色

3428
来自专栏腾讯Bugly的专栏

Android自绘动画实现与优化实战——以Tencent OS录音机波形动画为实例

前言 我们所熟知的,Android 的图形绘制主要是基于 View 这个类实现。 每个 View 的绘制都需要经过 onMeasure、onLayout、onD...

3044
来自专栏生信技能树

(11)仿写bowtie-生信菜鸟团博客2周年精选文章集

然后仿写了bowtie,对我的编程技术提高非常有帮助。目录如下: 自己动手写bowtie第一讲:BWT算法详解并建立索引 自己动手写bowtie第二讲:优化索引...

2966
来自专栏Golang语言社区

【Go 语言社区】把Go程序变小的办法

把Go程序变小的办法是: go build -ldflags “-s -w” (go install类似) -s去掉符号表(然后panic时候的stack tr...

3506
来自专栏数据结构与算法

2729:Blah数集

2729:Blah数集 查看 提交 统计 提问 总时间限制:3000ms内存限制:65536kB描述大数学家高斯小时候偶然间发现一种有趣的自然数集合Blah,对...

2654
来自专栏贾志刚-OpenCV学堂

OpenCV中图像形态学操作

图像形态学是图像处理的分支学科,在二值图像处理中占有重要地位、OpenCV中实现了图像形态学如下常见操作: -膨胀操作 -腐蚀操作 -开操作 -闭操作 -击中击...

3505
来自专栏落影的专栏

GPUImage详细解析(七)文字水印和动态图像水印

回顾 GPUImage源码解析、图片模糊、视频滤镜、视频水印都已经介绍过,这次带来的是给视频添加文字水印、动态图像水印。 效果展示 “我是水印”的文字,还有心形...

3575
来自专栏C语言C++游戏编程

无形细节最为致命,C语言中与零值比较那些小事儿

无论是float还是double类型的变量,都有精度限制。所以一定要避免将浮点变量用“==”或“!=”与数字比较,应该设法转化成“>=”或“。小编给大家推荐一个...

1283

扫码关注云+社区