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

相关文章

Olingo分析和实践之EDM 辅助序列化器详解(最佳实践)

《Olingo分析和实践之EDM辅助序列化器详解(最佳实践)》EDM辅助序列化器是ApacheOlingoOData框架中无需完整EDM模型的智能序列化工具,通过运行时类型推断实现灵活数据转换,适用... 目录概念与定义什么是 EDM 辅助序列化器?核心概念设计目标核心特点1. EDM 信息可选2. 智能类

Olingo分析和实践之OData框架核心组件初始化(关键步骤)

《Olingo分析和实践之OData框架核心组件初始化(关键步骤)》ODataSpringBootService通过初始化OData实例和服务元数据,构建框架核心能力与数据模型结构,实现序列化、URI... 目录概述第一步:OData实例创建1.1 OData.newInstance() 详细分析1.1.1

Olingo分析和实践之ODataImpl详细分析(重要方法详解)

《Olingo分析和实践之ODataImpl详细分析(重要方法详解)》ODataImpl.java是ApacheOlingoOData框架的核心工厂类,负责创建序列化器、反序列化器和处理器等组件,... 目录概述主要职责类结构与继承关系核心功能分析1. 序列化器管理2. 反序列化器管理3. 处理器管理重要方

SpringBoot中六种批量更新Mysql的方式效率对比分析

《SpringBoot中六种批量更新Mysql的方式效率对比分析》文章比较了MySQL大数据量批量更新的多种方法,指出REPLACEINTO和ONDUPLICATEKEY效率最高但存在数据风险,MyB... 目录效率比较测试结构数据库初始化测试数据批量修改方案第一种 for第二种 case when第三种

解决1093 - You can‘t specify target table报错问题及原因分析

《解决1093-Youcan‘tspecifytargettable报错问题及原因分析》MySQL1093错误因UPDATE/DELETE语句的FROM子句直接引用目标表或嵌套子查询导致,... 目录报js错原因分析具体原因解决办法方法一:使用临时表方法二:使用JOIN方法三:使用EXISTS示例总结报错原

MySQL中的LENGTH()函数用法详解与实例分析

《MySQL中的LENGTH()函数用法详解与实例分析》MySQLLENGTH()函数用于计算字符串的字节长度,区别于CHAR_LENGTH()的字符长度,适用于多字节字符集(如UTF-8)的数据验证... 目录1. LENGTH()函数的基本语法2. LENGTH()函数的返回值2.1 示例1:计算字符串

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

怎样通过分析GC日志来定位Java进程的内存问题

《怎样通过分析GC日志来定位Java进程的内存问题》:本文主要介绍怎样通过分析GC日志来定位Java进程的内存问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、GC 日志基础配置1. 启用详细 GC 日志2. 不同收集器的日志格式二、关键指标与分析维度1.

MySQL中的表连接原理分析

《MySQL中的表连接原理分析》:本文主要介绍MySQL中的表连接原理分析,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、环境3、表连接原理【1】驱动表和被驱动表【2】内连接【3】外连接【4编程】嵌套循环连接【5】join buffer4、总结1、背景

python中Hash使用场景分析

《python中Hash使用场景分析》Python的hash()函数用于获取对象哈希值,常用于字典和集合,不可变类型可哈希,可变类型不可,常见算法包括除法、乘法、平方取中和随机数哈希,各有优缺点,需根... 目录python中的 Hash除法哈希算法乘法哈希算法平方取中法随机数哈希算法小结在Python中,