1. 以解析的结果为参数创建输入输出
- 在解析命令行之后,ffmpeg_parse_options()调用open_files()分别打开输入文件和输出文件。
- OptionGroupList中的每一个OptionGroup元素包含一个文件项,包括文件名及选项。open_files()遍历每个OptionGroup,用它作为选项调用open_input_file()或open_output_file()。
2. InputStream
- 在open_input_file()中创建IputStream。
- open_input_file()调用avformat_open_input()解析输入文件,得到一个AVFormatContext实例,其中包含AVStream信息。open_input_file()还调用avformat_find_stream_info()得到更准确的信息。这部分和在ffplay中的处理几乎相同。唯一不同的是,ffplay只处理一个AVStream,而ffmpeg处理多个。
- 这里确定demuxer为ff_mov_demuxer,确定decoder为ff_h264_decoder。
- 之后open_input_file()一对一创建InputStream来容纳AVStream,并保存到input_streams[]列表中。Input_streams[]是ffmpeg转换过程的输入项。
d)同时还创建InputFile容纳AVFormatContext,并保存在input_files[]列表中。
3. OutputStream
- 在open_output_file()中创建OutputStream。
- open_output_file()先创建OutputFile,并将它保存在output_files[]列表中。
- 继续创建AVFormatContext。不同的format需要的私有空间大小不同,所以需要调用av_guess_format()确定format(即muxer)。这里根据文件名确定muxer为ff_mp4_muxer。
- 确定Ff_mp4_muxer的同时也确定了encoder类型。有两种类型可选,一种是AV_CODEC_ID_MPEG4,是ffmpeg自己支持的,一种是AV_CODEC_ID_H264,需要外部的x264库的支持,后者优先。
- 选择哪一种encoder是在ffmpeg的配置阶段确定的。这里因为指定了--enable-libx264选项,所以选择了AV_CODEC_ID_H264。
- 然后创建AVStream和容纳它的OutputStream,并将后者保存在output_streams[]中。output_streams[]是ffmpeg转换过程的输出项。
- open_output_file()会遍历input_streams[],为每个InputStream创建一个OutputStream。OutputStream会记住自己对应的源InputStream。
- 继续初始化OutputStream,主要是根据encoder类型创建encoder。这里是ff_libx264_encoder。
4. FilterGraph
- 在Open_output_file()还要创建FilterGraph。FilterGraph是InputStream,Filter和OutputStream的容器。
- init_simple_filtergraph()为每组Stream(OutputStream和它的源InputStream)创建一个FilterGraph来容纳连接它们的Filters。
- 作为初始化OutputStream的最后步骤,assert_file_overwrite()检查是否目标文件是否存在,询问用户是否覆盖等。如果不存在,或用户选择覆盖,avio_open2()初始化底层写组件。
- FilterGraph涉及到的类如下图。
- 有两种filtergraph。simple filtergraph和complex filtergraph。 前者是单输入单输出, 后者多输入多输出。这里的例子是前者。
- simple FilterGraph与Stream组是一对一的关系。FilterGraph创建了InputFilter与InputStream对接,创建了OutputFilter与OutStream对接。
- 图中还标出了类实例的数量映射关系。在图中,filtergraph与InputFilter和OutputFilter是1:n的关系。对simple filtergraph,n就是1。
5. 连接InputStream,Filter和OutputStream
- Configure_filtergraph()负责创建Filter,并把InputStream,Filter和OutputStream连接起来。
- 首先调用avfilter_graph_alloc()创建AVFilterGraph,用它初始化FilterGraph。
- 调用avfilter_graph_parse2()创建Filters。这是一个Filters串。
- configure_input_filter()连接InputStream和Filters串的第一个。
- configure_output_filter()连接Filters串的最后一个和OutputStream。
- configure_filtergraph()是在处理第一个frame时被调用的。后面会再提到这一点。
- 下面的图描述了上述语法的解析过程。
- 可以看到configure_input_filter()创建了Filter buffer连接InputStream和split, configure_output_filter()创建了buffer_sink连接overlay和OuputStream。
- 为了匹配filter之间的格式和大小,configure_filtergraph()还可能创建其他的filter作为连接。如下图中overlay和buffer_sink之间的format filter。
- avfilter_get_by_name()用于查找指定名字的filter,avfilter_gragh_create_filter()用于创建filter实例,avfilter_link()用于连接两个filter。
相关链接
FFMPEG 3.4.2 - ffmpeg源代码分析 (一)
FFMPEG 3.4.2 - ffmpeg源代码分析 (二)
FFMPEG 3.4.2 - ffmpeg源代码分析 (三)
FFMPEG 3.4.2 - ffmpeg源代码分析 (四)- x264
FFMPEG 3.4.2 - ffplay源代码分析 (一)
FFMPEG 3.4.2 - ffplay源代码分析 (二)
FFMPEG 3.4.2 - ffplay源代码分析 (三)