FFmpeg Principle学习new_video_stream添加视频输出流
new_video_stream() 函数流程
new_video_stream()
函数的流程相对来说比较简单,主要的逻辑如下:
1, 调 new_output_stream()
函数来创建 OutputStream
输出流,以及 AVCodecContext
编码器上下文。
new_output_stream()
是一个公共函数,创建 音频流,数据流,字幕流都用了它。
new_output_stream()
会把命令行的一些公共参数赋值给 OutputStream
跟 AVCodecContext
。
这些公共参数是指音频,视频,字幕都可能会有的参数。因为 new_output_stream()
是一个公共函数。
2, 调 MATCH_PER_STREAM_OPT()
宏函数,把 OptionsContext
里面视频相关的参数,赋值给 给 OutputStream
跟 AVCodecContext
。
流程图
可以看到,实际上就两步,new_video_stream()
肯定会创建视频的输出流,还有视频的编码器实例。
公共参数,就在 new_output_stream()
函数 里面赋值了。
视频相关的参数,就在 new_video_stream()
函数再赋值。
new_video_stream()
跟 new_output_stream()
函数都调用了多次 MATCH_PER_STREAM_OPT()
宏函数来提取 OptionsContext
的内容,
MATCH_PER_STREAM_OPT()
其实是 MATCH_PER_TYPE_OPT()
的兄弟函数,
#define MATCH_PER_TYPE_OPT(name, type, outvar, fmtctx, mediatype)\ {\ int i;\ for (i = 0; i < o->nb_ ## name; i++) {\ char *spec = o->name[i].specifier;\ if (!strcmp(spec, mediatype))\ outvar = o->name[i].u.type;\ }\ }
#define MATCH_PER_STREAM_OPT(name, type, outvar, fmtctx, st)\ {\ int i, ret, matches = 0;\ SpecifierOpt *so;\ for (i = 0; i < o->nb_ ## name; i++) {\ char *spec = o->name[i].specifier;\ if ((ret = check_stream_specifier(fmtctx, st, spec)) > 0) {\ outvar = o->name[i].u.type;\ so = &o->name[i];\ matches++;\ } else if (ret < 0)\ exit_program(1);\ }\ if (matches > 1)\ WARN_MULTIPLE_OPT_USAGE(name, type, so, st);\ }
这两个函数,只有最后一个参数,第四个参数是不一样的。
mediatype 通常是 a 或者 v,也就是根据 a 还是 v 字符来提取 OptionsContext
里面音频或者视频的选项。
st 是 AVStream,所以如果 AVStream
是音频,就提取 OptionsContext
里面的音频选项,如果是视频就提取视频。
这两个函数的宏实现看起来有点复杂,但他们的区别就是这么一点区别。
至此,new_video_stream()
函数分析完毕。new_audio_stream()
跟 new_video_stream()
类似,里面都调了 new_output_stream()
。
new_audio_stream()
主要是提取OptionsContext
里面音频选项,对 OutputStream
输出流,以及 AVCodecContext
编码器 进行赋值操作。
补充一点:虽然 new_video_stream()
里创建了 编码器实例,但是还没真正打开编码器的。打开编码器,需要等到解码出第一帧 AVFrame。才会打开编码器。
原因解析
因为 ffmpeg.exe 的逻辑,是只有在解码出第一帧AVFrame的时候,才去用 avfilter_graph_config()
打开 FilterGragh ,这样才能从出口滤镜读取到 输出的宽高是多少。
ffmpeg.exe
比较谨慎,他可能不太相信容器层记录的宽度,也有可能有些容器根本没记录宽高,所以他必须等到解码出 AVFrame,才能确定输入的宽高,确定了输入的宽高,才能创建 buffer入口滤镜,创建了入口滤镜,才能打开 FilterGragh
。
TODO:这个逻辑非常重要,在本章结尾的时候再重复讲一次。
滤镜出口里面获取宽高
最后是在 init_output_stream_encode()
里面,从滤镜出口里面获取的宽高,如下:
enc_ctx->width = av_buffersink_get_w(ost->filter->filter); enc_ctx->height = av_buffersink_get_h(ost->filter->filter); enc_ctx->sample_aspect_ratio = ost->st->sample_aspect_ratio
最后,推荐一下 clion 的 Call Hierarchy 功能,可以看到函数的调用流程,如下:
大部分的 集成开发环境都有这个功能,你只需用 “工具名称” + Call Hierarchy 关键词,即可搜索到相关教程。
以上就是FFmpeg Principle学习new_video_stream添加视频输出流的详细内容,更多关于FFmpeg Principle new_video_stream的资料请关注脚本之家其它相关文章!
相关文章
Android编程实现可滑动的开关效果(附demo源码下载)
这篇文章主要介绍了Android编程实现可滑动的开关效果,涉及Android的布局与控件设置技巧,并附带demo源码供读者下载参考,需要的朋友可以参考下2016-04-04使用RadioButton+Fragment实现底部导航栏效果
这篇文章主要为大家详细介绍了使用RadioButton+Fragment实现底部导航栏效果,具有一定的参考价值,感兴趣的小伙伴们可以参考一下2018-06-06Android通过自定义Activity实现悬浮的Dialog详解
这篇文章主要给大家介绍了关于Android通过自定义Activity实现悬浮的Dialog的相关资料,文中给出了详细的示例代码供大家参考学习,对大家具有一定的参考学习价值,感兴趣的朋友们下面来一起看看吧。2017-05-05Android使用IntentService进行apk更新示例代码
这篇文章主要介绍了Android使用IntentService进行apk更新示例代码,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧2018-01-01
最新评论