FFplay源码分析-rtmp入口

2024-06-24 01:48
文章标签 分析 源码 入口 rtmp ffplay

本文主要是介绍FFplay源码分析-rtmp入口,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

《FFmpeg原理》的社群来了,想加入社群的朋友请购买 VIP 版,VIP 版有更高级的内容与答疑服务。


本系列 以 ffmpeg4.4 源码为准。本文主要讲解 ffplay 的 RTMP 协议解析,播放。本文使用的命令如下:

ffplay -i rtmp://192.168.0.122/live/livestream

先按照之前的博客的教程 配置好 window 10 qt creator + msvc 的调试环境。必须是 msvc ,mingw不行,mingw无法debug 进去 ffmpeg的动态库。本博客暂时没有具体的教程讲解如何 移植 ffplay.c 到 qt msvc 环境,具体可以参考《FFplay源码分析-环境搭建》 跟ffmpeg.c 的 msvc移植教程 《ffmpeg-qt-msvc移植调试》。ffmpeg.c 跟 ffplay.c 的移植大同小异。


从之前的文章知道,RTMP的播放,跟本地文件的播放在 API 使用上,没有多大区别,都是 调 av_read_frame() 就能拿到一个 AVPacket,所以需要首先了解一下 av_read_frame() 的内部实现。在 ffplay.c 的 av_read_frame() 打一个断点,然后 step into 跳进去,因为是 msvc 环境,所以可以跳进去动态库的实现代码。如下图:

发现,windows 的环境并不是那么的美好,很多变量都是 not accessible,看不到具体的数据,所以咱们还是切换到 ubuntu 18的 clion环境进行调试。请阅读 《FFmpeg,Fplay,clion调试环境搭建》


首先提一个问题,ffplay 播放 RTMP 流的时候,肯定需要通过TCP去连接 服务器,然后做 RTMP 握手,协商交互等操作。那 ffpaly 是什么时候跟服务器建立 RTMP 连接的呢?带着这个疑问,开始研究。

猜测在 avformat_open_input() 函数里面,所以在这个函数的入口打个断点,经过研究 avformat_open_input() 函数的流程图如下:

上图主要有以下重点:

重点一: avformat_open_input() 这个函数有两种传参方式。

  1. filename 不为 NULL,这种情况就是 字符串传参,然后他内部判断是什么类似的输入流,MP4,FLV,还是 RTMP,
  2. AVInputFormat 不为 NULL,这种指定格式传参。

filename 字符串传参方式,ffmpeg 内部会读取一部分音视频数据来判断是什么类型,对于 RTMP 输入,就会建立RTMP链接,读取一些数据。

不过两种传参方式在效率上没有太大的区别,虽然字符串传参,会读取数据,但是,这些数据会缓存下来 给 av_read_frame() 函数实现,并不是探测完输入流类型就丢弃了。

不过还是有点区别,如果你指定了 AVInputFormat,ffmpeg 内部会少执行很多判断,因为他内部会把 大部分的 格式都遍历一遍。所以 指定 AVInputFormat可以加快程序运行速度。

av_probe_input_format3() 并没有开始建立 RTMP 链接,只是做一下字符串匹配,看看能不能找到 对应的 AVInputFormat,本文的命令没有找到。所以代码会继续往下走,去探测找到 对应的 AVInputFormat


重点二:ffio_open_whitelist() 为什么有白名单?

这个是 ffmpeg 提供了协议扩展的功能,如果你想用 ffplay 支持一个新的网络协议,除了你要写一个跟 RTMP 类似的模块加进去之外,还要 使用 -protocol_whitelist 命令行参数 把这个协议加进去。


重点三:默认是 探测 AVPROBE_PADDING_SIZE (32字节) 这么多音视频数据,也就是 RTMP 包的 body 数据加起来达到这个数字就停止探测。代码如下:

可以通过命令行参数 -probesize 80 设置探测的数据大小。有时候某些情况,需要更大的数据才能猜到输入流是什么类型。

从上图代码可以看到 函数 init_input() 的注释如下:

/* Open input file and probe the format if necessary. */
static int init_input(AVFormatContext *s, const char *filename,AVDictionary **options)

FFmpeg 项目的注释是非常完善的,任何一个函数,你看下他的注释,debug一下,就能搞懂这个函数主要在干什么。


最后就是 流程图的最后一个节点,也就是 rtmp_open() 。

uc->prot->url_open2() 就是 rtmp_open() ,ffmpeg 的多态模块隔离实现,经常是使用这种函数指针的方式实现的。后续会写一篇文章,讲解如何在 FFmpeg 4.4 的基础上加一个新的传输协议。可以在博客搜索《FFmpeg 传输协议实现》。


现在已经找到 RTMP 建立链接的入口,下一篇文章主要讲解 ffpaly RTMP 的具体实现。


由于笔者的水平有限, 加之编写的同时还要参与开发工作,文中难免会出现一些错误或者不准确的地方,恳请读者批评指正。

这篇关于FFplay源码分析-rtmp入口的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1088852

相关文章

Nginx分布式部署流程分析

《Nginx分布式部署流程分析》文章介绍Nginx在分布式部署中的反向代理和负载均衡作用,用于分发请求、减轻服务器压力及解决session共享问题,涵盖配置方法、策略及Java项目应用,并提及分布式事... 目录分布式部署NginxJava中的代理代理分为正向代理和反向代理正向代理反向代理Nginx应用场景

Redis中的有序集合zset从使用到原理分析

《Redis中的有序集合zset从使用到原理分析》Redis有序集合(zset)是字符串与分值的有序映射,通过跳跃表和哈希表结合实现高效有序性管理,适用于排行榜、延迟队列等场景,其时间复杂度低,内存占... 目录开篇:排行榜背后的秘密一、zset的基本使用1.1 常用命令1.2 Java客户端示例二、zse

Redis中的AOF原理及分析

《Redis中的AOF原理及分析》Redis的AOF通过记录所有写操作命令实现持久化,支持always/everysec/no三种同步策略,重写机制优化文件体积,与RDB结合可平衡数据安全与恢复效率... 目录开篇:从日记本到AOF一、AOF的基本执行流程1. 命令执行与记录2. AOF重写机制二、AOF的

MyBatis Plus大数据量查询慢原因分析及解决

《MyBatisPlus大数据量查询慢原因分析及解决》大数据量查询慢常因全表扫描、分页不当、索引缺失、内存占用高及ORM开销,优化措施包括分页查询、流式读取、SQL优化、批处理、多数据源、结果集二次... 目录大数据量查询慢的常见原因优化方案高级方案配置调优监控与诊断总结大数据量查询慢的常见原因MyBAT

分析 Java Stream 的 peek使用实践与副作用处理方案

《分析JavaStream的peek使用实践与副作用处理方案》StreamAPI的peek操作是中间操作,用于观察元素但不终止流,其副作用风险包括线程安全、顺序混乱及性能问题,合理使用场景有限... 目录一、peek 操作的本质:有状态的中间操作二、副作用的定义与风险场景1. 并行流下的线程安全问题2. 顺

MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决

《MyBatis/MyBatis-Plus同事务循环调用存储过程获取主键重复问题分析及解决》MyBatis默认开启一级缓存,同一事务中循环调用查询方法时会重复使用缓存数据,导致获取的序列主键值均为1,... 目录问题原因解决办法如果是存储过程总结问题myBATis有如下代码获取序列作为主键IdMappe

Java中最全最基础的IO流概述和简介案例分析

《Java中最全最基础的IO流概述和简介案例分析》JavaIO流用于程序与外部设备的数据交互,分为字节流(InputStream/OutputStream)和字符流(Reader/Writer),处理... 目录IO流简介IO是什么应用场景IO流的分类流的超类类型字节文件流应用简介核心API文件输出流应用文

java 恺撒加密/解密实现原理(附带源码)

《java恺撒加密/解密实现原理(附带源码)》本文介绍Java实现恺撒加密与解密,通过固定位移量对字母进行循环替换,保留大小写及非字母字符,由于其实现简单、易于理解,恺撒加密常被用作学习加密算法的入... 目录Java 恺撒加密/解密实现1. 项目背景与介绍2. 相关知识2.1 恺撒加密算法原理2.2 Ja

Nginx屏蔽服务器名称与版本信息方式(源码级修改)

《Nginx屏蔽服务器名称与版本信息方式(源码级修改)》本文详解如何通过源码修改Nginx1.25.4,移除Server响应头中的服务类型和版本信息,以增强安全性,需重新配置、编译、安装,升级时需重复... 目录一、背景与目的二、适用版本三、操作步骤修改源码文件四、后续操作提示五、注意事项六、总结一、背景与

Android实现图片浏览功能的示例详解(附带源码)

《Android实现图片浏览功能的示例详解(附带源码)》在许多应用中,都需要展示图片并支持用户进行浏览,本文主要为大家介绍了如何通过Android实现图片浏览功能,感兴趣的小伙伴可以跟随小编一起学习一... 目录一、项目背景详细介绍二、项目需求详细介绍三、相关技术详细介绍四、实现思路详细介绍五、完整实现代码