1. 数据结构:
AVInputFormat为FFMPEG的解复用器对象,通过调用av_register_all(),FFMPEG所有的解复用器保存在以first_iformat为链表头的链表中,且还有个链表尾指针last_iformat。
以ff_srt_demuxer为例子来看看该结构体的初始化流程。
先看ff_srt_demuxer的定义:
AVInputFormat ff_srt_demuxer = { .name = "srt", .long_name = NULL_IF_CONFIG_SMALL("SubRip subtitle"), .priv_data_size = sizeof(SRTContext), .read_probe = srt_probe, .read_header = srt_read_header, .read_packet = srt_read_packet, .read_seek2 = srt_read_seek, .read_close = srt_read_close, };易知,name成员为解复用器的名称,long_name为对应的文件格式,priv_data_size为和解复用器关联的对象(SRTContext)的大小,read_probe,read_header,read_packet,read_seek2,read_close分别为探测函数,读头函数,读包函数,seek函数及close函数指针。先看SRTContext定义,再看看这几个具体的函数。
SRTContext定义:
typedef struct { AVPacket *subs; //字幕包数组 int nb_subs; //字幕包条数 int allocated_size; //(每条)字幕的大小 int current_sub_idx; ///当前字幕索引 enum sub_sort sort; ///字幕排序方式 } FFDemuxSubtitlesQueue;
从代码(比如srt_read_packet)易知,解复用器的关联对象指针存储在AVFormatContext的priv_data成员中。
"%*d:%*2d:%*2d%*1[,.]%*3d --> %*d:%*2d:%*2d%*1[,.]%3d"即srt的格式要为:
1 00:00:02,436 --> 00:00:06,505即一行索引+“开始时间-->结束时间”的形式。
srt_read_header()是读头函数(可以看作是初始化函数),该函数实际上把字幕文件中多有的字幕都解析了出来放在FFDemuxSubtitlesQueue对象的队列中。(AVStream???)
注意:1. 该函数会创建流(AVStream),并将流保存到AVFormatContext的streams数组中。
2. 该函数读写文件内容时用的是AVFormat的pb成员(即AVIOContext)。
3. 该函数最后调用了ff_subtitles_queue_finalize()对字幕包进行排序。
srt_read_packet()是读包函数,该函数实际上是从FFDemuxSubtitlesQueue中将当前字幕索引指向的字幕拷贝到传入的包中。(AVPacket???)
srt_read_seek()是搜索函数。
srt_read_close()是关闭函数,主要是释放队列成员及队列本身所占的内存空间。
2. 函数调用:
即:
1. init_input()先调用avio_open2()创建并打开一个AVIOContext对象,用于文件读写;
2. init_input()然后调用av_probe_input_buffer2()探测解复用器类型;
3. av_probe_input_buffer2()包含三个步骤:avio_read()读入探测数据(AVProbeData),然后调用av_probe_input_format2()探测合适的解复用器,最后调用ffio_rewind_witdh_probe_data()将探测数据返回给AVIOContext的缓冲buffer。
av_probe_input_format2()调用av_probe_input_format3(),将得到的匹配分数与要求的匹配值相比较,如果匹配分数>匹配值,这返回得到的解复用器,否则返回NULL。
av_probe_input_format3(),该函数遍历所有的解复用器,调用它们的read_probe()函数计算匹配得分,如果解复用器定义了文件扩展名,还会比较输匹配数据跟扩展名的匹配得分。函数最终返回计算找到的最匹配的解复用器,并将匹配分数也返回。