GoPlay是一款基于 FFmpeg/OpenGL ES 2.0 的 iOS 播放器。支持 FFmpeg 内嵌的所有格式。而且可以自定义各种滤镜, 包括 VR、水印、贴图等。
关于 iOS 视频播放,苹果提供的 AVPlayer 性能是非常出色的,但是有个缺点,就是支持播放的格式并不多,仅仅支持 mp4/mov/ts 等有限的几种格式。显然业界中比较知名的 jikplayer 确实弥补了这种缺陷,然而 ijkplayer 是在 FFmpeg/ffplay 的基础上进行开发的,最终是通过 SDL2.0 进行显示。在当前大环境下,VR、水印、贴图、九宫格等滤镜盛行,在 ijkplayer 中默认是支持 avfilter 滤镜的,但是并没有支持 GPU 滤镜;那么有没有一种办法可以播放 AVPlayer 不支持格式的视频,又能够在视频上无限制的加滤镜,例如 GPUImage 那么方便那么丝滑的做法呢?上面两个痛点也就是 GoPlay 解决的问题。
关于格式支持,采用了业界比较出名的 FFmpeg 解封装不同的视频格式;在解码阶段,如果开启了 VideoToolBox 硬解码,那么就采用 iOS 的硬解码方式,否则自动切换到 FFmpeg 的软解码方式。
为了方便滤镜的接入,滤镜包括滤镜链的实现都采用了 GPUImage 类似的做法,如果使用过 GPUImage,那么就可以无缝的切换到 GoPlay,同时可以根据 GPUImage 的已有滤镜自定义滤镜,无限扩展自己的滤镜库。GoPlay 和 GPUImage 的滤镜类比如下表。
GOPlay GPUImage
输入 FFMovie GPUImageMovie
滤镜 FFFilter GPUImageFilter
输出显示 FFView GPUImageView
GoPlay 主要有 5 个线程(包括主线程),其中 OpenGL ES 渲染、滤镜都是在一个统一的异步线程中处理,在这方面与 GPUImage 的处理稍有不同。异步线程可以防止阻塞 UI 界面,串行可以防止线程间加锁从而导致的性能损耗。线程模型如下。
解封装线程 -- FFmpeg 解封装,读取 packet,分发到视频解码线程和音频解码线程
视频解码线程 -- 将 packet 解码成 frame,并保存到队列缓存中
音频解码线程 -- 将 packet 解码成 frame,并保存到队列缓存中
OpenGL ES 渲染、滤镜处理线程
在业界中,普遍没有认识到音频视频两者的同步算法是控制学的问题,而仅仅停留在谁快谁慢的问题上。在现实中,音频和视频的 PTS 的误差是客观存在的,我们需要通过一种控制学算法实现音频和视频的相对同步,需要考虑到累积误差的存在,在相对范围内,同步算法是具有自我调节能力的,当超出某个范围了,那么就需要丢帧了,否则会影响观感。在这么多开源项目中,FFmpeg/ffplay 实现了这种思想。
如果真正理解了音视频同步算法,那么丢帧的做法就很简单了,当超出了音视频同步算法的调节范围,而且是视频帧慢于音频帧很多,那么此时就需要丢调视频帧。
全景图像是将一张平面图片映射到一个球面上。本质上也是一种滤镜处理,即要处理好顶点坐标和纹理坐标的映射关系。
ArcBall 本质上是将二维平面上的滑动转换成三维立体球的转动,具体的做法以屏幕中心为球心,画一个球。在屏幕中滑动时,就将滑动的点映射到球面上,如果滑动的范围超出了球的范围,那么就映射到最靠近球面的点上。根据起始点的四元数和终点的四元数的差值(求逆),就可以得到旋转的角度。
滤镜链的思路来源于 GPUImage,但多路滤镜的处理情况并没有沿袭 GPUImageTwoInput 之流的做法。在多路滤镜的处理上,水印滤镜中进行了一种尝试。
关于 GoPlay 的相关原理基本上到这里结束了。感兴趣的可以在GoPlay中找到相关的实现,当然也可以提 BUG 一起讨论。