当前位置:网站首页>Analysis of ffplay video playback principle
Analysis of ffplay video playback principle
2022-08-03 16:08:00 【Baidu Geek says】
作者 | 赵家祝
FFmpeg Framework is composed of command line tools and libraries, ffplay Is one of them command line tools,Provides the function of play audio and video files,Not only can play local multimedia files,Can also play network streaming media files.本文从 ffplay The whole play process of,Draw lessons from its design idea,Learn how to design a simple player.
一、Player workflow
在学习 ffplay 源码之前,为了方便理解,We first introduce the macro players in the working process of the broadcast media file.
解协议:When the media files on the network transmission,Need to go through streaming protocol will media data is segmented into several packets,So you can meet the needs of users to download and watch,Without the need for all media files such as the download is complete to watch.常见的流媒体协议有 RTMP、HTTP、HLS、MPEG-DASH、MSS、HDS 等.Since streaming protocol not only contain the media data,Also contains control broadcast signaling data.因此,Solution protocol is to remove the signaling data,The output audio and video data encapsulation format.
解封装:Encapsulation format is also called container,Is to have good coding compression streaming video and audio streams according to certain format in a file,常见的封装格式有 MP4、FLV、MPEG2-TS、AVI、MKV、MOV 等.Decapsulation is to encapsulate data format of the audio compression coding data and streaming video compression coding data separation,Convenient in decoding phase using different decoder decoding.
解码:Compression data is based on the original data using the data from different coding compression,And decoding is the reverse coding operation.Common video compression coding standard are H.264/H.265 、MPEG-2 、AV1 、V8/9 等,Audio compression coding standard have AAC 、MP3 等.After decompression for video image data is YUV 或 RGB ,Audio sampling data is PCM .
音视频同步:After decoding the video data and audio data is independent of the,Before to graphics and sound card play,Need to video and audio sync,Avoid inconsistent play schedule.
二、main函数
ffplay 的使用非常简单,以ffplay -i input.mp4 -loop 2为例,表示使用 ffplay A player loop input.mp4 File twice.执行该命令时,对应的源码在 fftools/ffplay.c 中,程序入口函数是 main 函数 .
注:本文 ffplay 源码基于 ffmpeg 4.4.
2.1 环境初始化
Initialization section, call the following function mainly:
init_dynload:调用SetDllDirectory("")删除 动态链接库(DLL)In the search path in the current working directory,是 Windows Platform is a safety precaution.
av_log_set_flag:设置 log Print the markedAV_LOG_SKIP_REPEATED,The skip duplicate messages.
parse_loglevel:解析 log 的级别,Will match the command of-loglevel字段.If the command to add-report,Will play the log output into a file.
avdevice_register_all:Registration of special equipment package library.
avformat_network_init:Initializes the network resources,Can pull from network flow.
parse_options:解析命令行参数,示例中的-i input.mp4和-loop 2It is through this function resolved,Support options defined inoptions静态数组中.Parsing the file name、File format stored in global variables respectivelyinput_filename和file_iformat中.
2.2 SDL初始化
SDL的全称是 Simple DirectMedia Layer,Is a cross-platform multimedia development library,支持 Linux、Windows、Mac OS等多个平台,实际上是对 DirectX、OpenGL、Xlib再封装,On different operating system provides the same functions.ffplay Broadcast shows through SDL 实现的.
main Function calls to the following three main SDL 函数:
SDL_Init:初始化 SDL 库,传入的参数 flags,The default support for video、Audio and timer,If the command configuration-anDisables the audio,配置了-vnDisables video.
SDL_CreateWindow:Create a play video window,The function can specify the location of the window、大小,默认是 640*480 大小.
SDL_CreateRenderer:Create the renderer context for the specified window,对应的结构体是 SDL_Render .We can use the renderer to create texture,也可以渲染视图.
2.3 Parsing media stream
stream_open函数是 ffplay The starting point of start playing process,该函数传入两个参数,分别是文件名input_filename和文件格式file_iformat.Below is the function of internal process:
(1) 初始化VideoState:VideoState 是 ffplay The largest of the struct,All the video information defined in it.初始化 VideoState 时,先定义 VideoState The local variable structure pointer typeis,分配堆内存.然后初始化结构体中的变量,例如视频流、音频流、Subtitle stream index,And assignment function into the reference filename 和 iformat.
(2)初始化FrameQueue:FrameQueue Is after decoding Frame 队列, Frame Is after decoding数据,For example, after the video decoding is YUV 或 RGB 数据,After the audio decoding is PCM 数据.初始化 FrameQueue 时,会对 VideoState 中的 pictq(视频帧队列)、subpq(Subtitles frame queue)、sampq(音频帧队列)依次调用frame_queue_init函数进行初始化.FrameQueue Internal is achieved by an array of ring buffer area for a first in first out,windex 是写指针,Used by the decoding thread;rindex 是读指针,Used by the broadcast thread.The benefits of using circular buffer is,Elements within a buffer will be removed after,Other elements do not need to move,Applicable to know in advance of buffer capacity scene.
(3)初始化PacketQueue:PacketQueue 是解码前的 Packet 队列,Used to save data after the decapsulation.初始化 PacketQueue 时,会对 VideoState 中的 videoq(视频包队列)、audio(音频包队列)、subtitleq(Subtitle packet queue)依次调用packet_queue_init函数进行初始化.不同于 FrameQueue , PacketQueue With the method of linked list implementation queue.Due to decode the packet size before uncontrollable,Unable to clear the buffer capacity,If you use a ring buffer,Easy to trigger buffer capacity,Need to move data in buffer.因此,Using a linked list implementation queue more appropriate.
(4)初始化Clock:Clock 是时钟,In the audio and video synchronization phase,有三种同步方法:视频同步到音频,音频同步到视频,As well as audio and video synchronization to the external clock.初始化 Clock 时,会对 VideoState 中的 vidclk(视频时钟)、audclk(音频时钟)、extclk(外部时钟)依次调用init_clock函数进行初始化.
(5)Limit the volume range:To limit the volume range in 0~100 之间,然后再根据 SDL The volume of the scope for further restrictions.
(6)设置音视频同步方式:ffplay 默认采用AV_SYNC_AUDIO_MASTER,The video synchronous audio.
(7)创建读线程:调用SDL_CreateThread创建读线程,At the same time set the thread to create successful correctionread_threadFunction and receiving parametersis( stream_open Function started to create VideoState Pointer type of local variables).如果线程创建失败,则调用stream_closeDo destroy logic.
(8)返回值:将局部变量is作为函数返回值返回,Used to handle the following all kinds of SDL 事件.
2.4 SDL事件处理
event_loopFunction is a for 循环,使用 SDL Monitor the user's keyboard key events、鼠标点击事件、窗口事件、Exit event, etc.
三、read_thread函数
read_thread() function from disk or network access flow,Including audio stream、视频流和字幕流,Based on availability then create corresponding stream decoding thread.因此read_threadThe thread actually have understanding agreement/解封装的作用.The core process can be divided into the following steps:
3.1创建AVFormatContext
AVFormatContext Is encapsulated context,Describe the composition of the media file or media flow and basic information.avformat_alloc_contextFunction is used to allocate memory to create AVFormatContext 对象ic.
拿到 AVFormatContext 对象后,在调用avformat_open_inputFunction to open the file before,You need to set up the interrupt the callback function,Used to check whether should break IO 操作.
ic->interrupt_callback.callback = decode_interrupt_cb;ic->interrupt_callback.opaque = is;
decode_interrupt_cb内部返回了一个 VideoState 的 abort_request 变量,The variable in the callingstream_closeFunction off flow will be set to1.
3.2 打开输入文件
In front of the ready after some assignment operation,You can start according to the filename 打开文件了.avformat_open_input函数用于打开一个文件,并对文件进行解析.If the file is a network link,则发起网络请求,After the network data back analytic audio stream、Streaming data related.
3.3 搜索流信息
Search flow information useavformat_find_stream_info函数,The read several packages from media file,Then from the search flow related information,Finally to search to the flow of information on theic->streams指针数组中,数组的大小为ic->nb_streams.
Because in the actual process of play,The user can specify whether to disable the audio stream、视频流、字幕流.So before decoding to deal with the flow,Will determine whether the corresponding flow under unavailable,If it is available, callav_find_best_streamFunction to find the corresponding flow index,并保存在st_index数组中.
3.4 设置窗口大小
If find the video stream index,You will need to render the video images.Due to the size of the form generally use default value 640*480 ,The value and the size of the video frame really may not be the same.In order to correctly display bearing video images form,Need to compute a video frame aspect ratio.调用av_guess_sample_aspect_rationFunction of speculation ratio of high to width of the frame sample,调用set_default_window_sizeFunction to set the size of the display window and aspect ratio.
3.5 创建解码线程
根据st_indexDetermine audio stream、视频流、Subtitle stream index is found,If found, in turn, callstream_component_open创建对应流的解码线程.
3.6 解封装处理
接下来是一个 for(;;) 循环:
(1)响应中断停止、暂停/继续、Seek操作;
(2)判断 PacketQueue 队列是否满了,If full sleep10ms,继续循环;
(3)调用av_read_frameRead from the code stream several audio frame or a video frame;
(4)从输入文件中读取一个 AVPacket ,判断当前 AVPacket Whether within the scope of the playing time,如果是则调用packet_queue_put函数,According to the type of put it in audio/视频/字幕的 PacketQueue 中.
四、stream_component_open函数
3.5Section when it comes to,stream_component_openThe decoding function is responsible for creating different flow thread.So how does it create decoding thread?
4.1 创建AVCodecContext
AVCodecContext是编解码器上下文,Save the audio and video codec related information.使用avcodec_alloc_context3函数分配空间 ,使用avcodec_free_context函数释放空间.
4.2 查找解码器
According to the decoderid ,调用avcodec_find_decoder函数,查找对应的解码器.Similarly a function isavcodec_find_encoder,用于查找 FFmpeg 的编码器.Two () function returns the structure is AVCodec .
If you specify the name of the decoder,则需要调用avcodec_find_decoder_by_nameFunction finds decoder.
Either way finding decoder,If not found decoder,Will throw exception exit process.
4.3 解码器初始化
Find the decoder after,Need to open the decoder,And the decoder initialization,对应的函数是avcodec_open2,This function will also support the initialization of encoder.
4.4 创建解码线程
Determine the decoding type,Create different decoding thread.
switch (avctx->codec_type) {case AVMEDIA_TYPE_AUDIO: // 音频 ... if ((ret = decoder_init(&is->auddec, avctx, &is->audioq, is->continue_read_thread)) < 0) goto fail; ... if ((ret = decoder_start(&is->auddec, audio_thread, "audio_decoder", is)) < 0) goto out; ...case AVMEDIA_TYPE_VIDEO: // 视频 ... if ((ret = decoder_init(&is->viddec, avctx, &is->videoq, is->continue_read_thread)) < 0) goto fail; if ((ret = decoder_start(&is->viddec, video_thread, "video_decoder", is)) < 0) goto out; ...case AVMEDIA_TYPE_SUBTITLE: // 字幕 ... if ((ret = decoder_init(&is->subdec, avctx, &is->subtitleq, is->continue_read_thread)) < 0) goto fail; if ((ret = decoder_start(&is->subdec, subtitle_thread, "subtitle_decoder", is)) < 0) goto out; ...}
Thread creation indecoder_start函数中,依然使用 SDL 创建线程的方式,调用SDL_CreateThread函数.
五、video_thread函数
Video decoding thread from the video PacketQueue 中不断读取 AVPacket ,After the completion of the decoding will AVFrame 放入视频 FrameQueue .Audio video decoding implementation similar to,Here only the introduction of the decoding process of video.
5.1创建AVFrame
AVFrame Describe the decoding of original audio or video data,通过av_frame_alloc函数分配内存,通过av_frame_free函数释放内存.
5.2 视频解码
开启 for(;;) 循环,不断调用get_video_frameFunction to decode a video frame.This function mainly calldecoder_decode_frame函数解码,decoder_decode_frameFunction of audio、视频、Subtitles are disposed,主要依靠 FFmpeg 的avcodec_receive_frameFunction to obtain decoder to decode the output data.
After get decoded video frames,Depending on the audio and video synchronization mode and command line-framedrop选项,Determine whether need to throw away out of sync video frames.
The command line with-framedrop选项,No matter what kind of audio and video synchronization mechanism,Video frame is discarded out of sync.
The command line with-noframedrop选项,No matter what kind of audio and video synchronization mechanism,Don't throw out of sync video frame.
The command line without-framedrop或-noframedrop选项,If the audio and video synchronization mechanism for synchronous video,则不丢弃失去同步的视频帧,Otherwise it will cast out of sync video frames.
5.3 放入FrameQueue
调用queue_picture函数,将 AVFrame 放入 FrameQueue .该函数内部调用了frame_queue_push函数,Adopting the processing mode of the ring buffer,The write pointerwindex累加.
static void frame_queue_push(FrameQueue *f){ if (++f->windex == f->max_size) f->windex = 0; SDL_LockMutex(f->mutex); f->size++; SDL_CondSignal(f->cond); SDL_UnlockMutex(f->mutex);}
六、音视频同步
ffplay The default with the method of video synchronous to the audio,分以下三种情况:
If the video and audio schedule consistent,不需要同步;
如果视频落后音频,则丢弃当前帧直接播放下一帧,The human eye feel jump frame;
如果视频超前音频,Then repeat show on a frame,等待音频,The human eye feel video screen stopped,But a voice in the broadcast;
ffplay Video synchronization to the logic of the audio in video playback functionvideo_refresh中实现.The function invocation chain is:main()->event_loop()->refresh_loop_wait_event()->video_refresh.
6.1 Judge play finish
调用frame_queue_nb_remaingFunction to calculate the residual did not show the number of frames is equal to0,如果是,You don't need to walk the rest of the steps.计算过程比较简单,用 FrameQueue 的 size - rindex_shown , size 是 FrameQueue 的大小, rindex_shown 表示 rindex Point to the node does have shown,If have shown as1,否则为0.
6.2 Play the sequence matching**
分别调用frame_queue_peek_last和frame_queue_peek函数从 FrameQueue Get on a frame and current frame,Shows the frame of a frame is the last time on,The current frame is display the current frame.
(1)Compare the current frame and the current PacketQueue The broadcast sequence ofserial是否相等:
如果不等,Retry logic of video broadcast;
如果相等,则进入(2)流程判断;
注:
serial
Is used to distinguish whether continuous data,如果发生了 seek ,Start a new playback sequence,
(2)Comparison on a frame and current frame playback sequenceserial是否相等:
如果不相等,则将frame_timer更新为当前时间;
如果相等,Don't handle and into the next process
6.3 Determine whether to repeat the a frame
(1)Will be on a framelastvp和当前帧vp传入vp_duration函数,通过vp->pts - lastvp->ptsThe calculation on a frame of time.
注:
pts
全称是 Presentation Time Stamp ,显示时间戳,Said after decoding the frame display in the form of time.
(2)在compute_target_delay函数中,调用get_clockFunction to obtain video clock,调用get_master_clockFunction for synchronous clock,Calculated two clock difference,According to the difference between computing needs delay 的时间.
(3)If the current frame moment(is->frame_timer + delay)大于当前时刻(time),According to the current frame of broadcast time hasn't arrived yet,Equivalent to the current video audio in advance,You need to will play on a frame again.
last_duration = vp_duration(is, lastvp, vp);delay = compute_target_delay(last_duration, is);time= av_gettime_relative()/1000000.0;if (time < is->frame_timer + delay) { *remaining_time = FFMIN(is->frame_timer + delay - time, *remaining_time); goto display;}
6.4 Determine whether to discard not broadcast frames
If the current queue is greater than the number of frames1,Need to consider to throw the frame,Only when a frame does not consider frame lose.
(1)调用frame_queue_peek_nextFunction to obtain the next frame(Next to display the frame),According to the current frame and the next frame calculation of the current frame playback length,计算过程和6.3相同.
(2)满足以下条件时,开始丢帧:
The current mode is not stepping mode;
丢帧策略生效:framedrop>0,Or the current audio and video synchronization strategy is not audio to the video.
当前帧vpHaven't had time to play,But the next frame play time(is->frame_timer + duration)Is less than the current system time(time)了.
(3)丢帧时,将is->frame_drops_late++,并调用frame_queue_nextFunction will be deleted on a frame,更新 FrameQueue 的读指针 rindex 和 size .
if (frame_queue_nb_remaining(&is->pictq) > 1) { Frame *nextvp = frame_queue_peek_next(&is->pictq); duration = vp_duration(is, vp, nextvp); if(!is->step && (framedrop>0 || (framedrop && get_master_sync_type(is) != AV_SYNC_VIDEO_MASTER)) && time > is->frame_timer + duration){ is->frame_drops_late++; frame_queue_next(&is->pictq); goto retry; }}
七、渲染
ffplay The final image rendering is a SDL 完成的,在 video_display 中调用了 SDL_RenderPresent(render) 函数,其中 render The parameter is started in main 函数中创建的.在渲染之前,Need to decode the video frame data is converted to a SDL 支持的图像格式.转换过程在 upload_texture 函数中实现,Details here analysis.
Audio similar,If the decoding the audio can't be SDL 支持,Need to re-sampling audio,Converts audio frame format SDL 支持的格式.
八、小结
This article embarks from the whole play process,介绍了 ffplay Players play media files main process,Don't stuck code details.同时,对 FFmpeg Some commonly used functions have some understand,For our own written a simple player has a lot of help.
---------- END ----------
推荐阅读【技术加油站】系列:
Baidu engineers in the eyes of cloud native observability tracking technology
边栏推荐
猜你喜欢
ModelWhale 云端运行 WRF 中尺度数值气象模式,随时随地即开即用的一体化工作流
leetcode:899. 有序队列【思维题】
[Unity Getting Started Plan] Basic Concepts (8) - Tile Map TileMap 01
Awesome!Coroutines are finally here!Thread is about to be in the past
ffplay视频播放原理分析
JS basics--judgment
Basic knowledge points in js - events
QT QT 】 【 to have developed a good program for packaging into a dynamic library
为教育插上数字化的翅膀,网易云信发布「互联网+教育」整体解决方案
【Unity入门计划】基本概念(7)-Input Manager&Input类
随机推荐
NodeJs - cross domain
语音识别新一轮竞争打响,自然对话会是下一个制高点吗?
ffplay视频播放原理分析
如何选择合适的损失函数,请看......
1、实例开启无锁表结构变更以后,在任务编排中通过“单实例SQL”节点进行的结构变更,是优先采用无锁表
一个文件管理系统的软硬件配置清单
高压直流输电(HVDC)的最优潮流(OPF)(Matlab代码实现)
Small Tools (4) integrated Seata1.5.2 distributed transactions
window.open不显示favicon.icon
泰山OFFICE技术讲座:文字边框高度研究
ECCV 2022 | Relational Query-Based Temporal Action Detection Methods
开源一夏 | 阿里云物联网平台之极速体验
How much do you know about the intelligent operation and maintenance service of data warehouse based on DMS?
JS basics--judgment
一文看懂推荐系统:概要02:推荐系统的链路,从召回粗排,到精排,到重排,最终推荐展示给用户
字典表(还需要输入2个字)
使用Make/CMake编译ARM裸机程序(基于HT32F52352 Cortex-M0+)
分享一款免费OPC UA服务器
Ark server open tool, server tutorial win
如何分析周活跃率?