Android 播放器音视频同步机制

视频是一帧一帧播放的,音频也是一帧一帧(20ms)播放的。播放器都是按照它们的每一帧的PTS来作为参考进行同步的。可以参考FFmpeg Tutorial

前置声明:当前代码分析版本为ijkplayer V0.7.5


本文需要3小节来描述清楚是如何同步的

  • 第一节是 描述清楚ijkplayer 整个线程模型 这个可以提供给大家一个高度的俯瞰效果。知道什么与什么进行同步
  • 第二节是 具体描述是怎么样进行同步的
  • 第三节是 描述 MediaCodec configure Surface 的时候进行渲染的时候
    程序获取不到解码后的数据是如何做到进行同步的。前一篇博文有一些描述MediaCodec API 运用的知识点。

第一节

先来理清一下ijkplayer 的线程模型

JAVA 端调用 prepareAsync 开启了
  • 消息队列线程 ijkmp_msg_loop(ff_msg_loop)
  • 视频渲染线程 video_refresh_thread(ff_vout)
  • 数据解复用线程 read_thread(ff_read)
解复用线程里 read_thread 开启:
  • 音频渲染线程 aout_thread(ff_aout_android)
  • 音频解码线程 audio_thread(ff_audio_dec)
  • 视频解码线程 video_thread(ff_video_dec)

ijkplayer是如果硬件解码器打开失败,则会自动切换至软解。大家需要注意的一点不是硬解失败就自动转到软解,是硬解解码器打开失败,才会自动切换至软解且只有一次机会。
如果配置为硬解解码则 视频解码线程video_thread 中将会在创建一个辅助 MediaCodec input线程用于向MediaCodec 中输入数据。

  • 硬解数据输入线程 enqueue_thread_func(amediacodec_input_thread)

备注:

ijkplayer V0.7.9.1+ 版本中已经添加单线程选项mediacodec decode by single thread


第二节

线程模型已经整理清楚了,接下来就是分析是如何同步的。
音视频同步基本是所有的播放器都是以音频PTS为参考时间。原因如下:

  • 音频是由硬件驱动按照采样率播放,相对来说比较简单,不需要程序做额外调整且漂移误差较小。
  • 如果音频参考其它时间进行同步的话,进行加速或者减慢或者等待都会立刻反应到人的感知中。视频帧影响就非常小了,一帧图片丢弃或者重复都不会造成影响。

不错播放器一般可以自主选择以哪一个时间做为参考基准

  • 外部时间
  • 音频时间
  • 视频时间

我们这里只分析以音频为基准进行同步。一般首先想到就是:
比较时间戳然后视频太慢了就立刻播放,视频太快了就延时播放
看起来是不是很简单呢,事实上实际同步算法确实如上所述那样简单。只不过由于其它影响的因子没有如此的理想化

其实音视频同步里面需要用到的是3个时间线
1、音频播放的时间线
2、视频播放的时间线
3、本地单调增长的时间线 用于记录2次到达同一个代码点流逝的时间

  • av_gettime_relative 获取当前时间,以微妙为单位。这个函数获取的当前时间是从系统(电脑/手机)启动算起的,而不是1970年1月1日0点0分。
  • av_gettime 获取的时间是从1970年1月1日0点0分开始的。也是以微妙为单位。

理论上我有参考音频时间戳不就行了吗,为何还需要一个。其实因为视频播放线程不是高优先级,很容易出现 Sleep 唤醒误差或者因cpu资源卡顿一下,统计这个时间为了在同步的时候计入这个时间。

这里提醒一下音频播放线程一定要设置高优先级,不然很容易出现卡顿的声音的现象,大家可以试一试。

现在已经很清楚明了,可以阅读一下音视频同步核心实现函数:

  • static void video_refresh(FFPlayer *opaque, double *remaining_time)
  • static double compute_target_delay(FFPlayer *ffp, double delay, VideoState *is)

同步音视频算法大致以下3步
1、首先计算当前将要显示视频帧的 duration 如下:

last_duration = vp_duration(is, lastvp, vp);

2、然后根据参照音频时间戳计算当前将要显示帧需要多少delay时间然后显示
delay = compute_target_delay(ffp, last_duration, is);

3、最后引用本地单调增长的时间线来计算真正需要av_usleep多长时间再来显示

是不是过程很简单呢。我们再来探索一下每一步的具体实现。

我们看一下下面这个字段:

is->frame_timer

每当视频开始播放或者seek 后都重现初始化为 就是将要显示第一帧时间

is->frame_timer = av_gettime_relative() / 1000000.0;

经过 compute_target_delay 计算得到当前帧需要delay的时间是多少,然后进行累加

is->frame_timer += delay;

到此软解情况下音视频同步算法描述完毕。接下来介绍一下硬解情况下的同步机制

第三节

描述 MediaCodec configure Surface 进行渲染时同步机制实现。
ijkplayer 有2种类型的队列,一个是av_packet queue 一个是av_frame queue.前者保存解码前的数据,后者保存解码后的数据。问题出现了,MediaCodec 拿不到解码后的数据。那岂不是av_frame queue 队列里没有视频数据。怎么进行同步呢?其实想到聪明的人这里肯定想到一个相关的问题是
软解的时候就算后台播放音频但实际上视频还是进行解码的,av_frame queue队列还是有真实的视频帧数据的。如果硬解时后台播放是需要销毁MediaCodec 的 这样更不可能有视频数据且什么都没有了怎么同步?硬解前后台切换一些问题可以参考(Android 播放器硬解前后台切换黑屏问题

其实大家只要抓住一个关键点就是我们是用时间戳同步的,并不是视频帧数据音频帧数据来进行同步的。不过硬解驱动播放音频自己是按照音频帧数据来自我同步的。

首先直接告诉大家大致的实现机制是:
ijkplayer 播放器硬解的时候是用不包含解码后的视频帧数据的 av_frame queue.只不过保存的时候mediacodec 解码后的缓冲区索引ID和时间PTS。

硬解的时候后台的时候是如何同步的呢?

其实ijkplayer 内部创建了一个MediaCodecDummy 虚拟硬件解码器 用来模拟。只不过这个时候没有任何解码数据,只有PTS 等基本数据用于音视频同步。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容

  • 教程一:视频截图(Tutorial 01: Making Screencaps) 首先我们需要了解视频文件的一些基...
    90后的思维阅读 4,627评论 0 3
  • 随着互联网技术的飞速发展,移动端播放视频的需求如日中天,由此也催生了一批开源/闭源的播放器,但是无论这个播放器功能...
    金山视频云阅读 46,028评论 28 170
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,057评论 25 707
  • 前言 随着音视频领域的火热,在很多领域(教育,游戏,娱乐,体育,跑步,餐饮,音乐等)尝试做音视频直播/点播功能,那...
    passiontim阅读 3,244评论 1 46
  • 昨天,和两个朋友一起去公园走了走。 那是离我们住的地方坐地铁大概有20分钟的小公园,本来计划想早起就出发,结果我们...
    若荷_阅读 345评论 3 1