当前位置: 首页 > news >正文

FFmpeg源码:compute_frame_duration函数分析

一、compute_frame_duration函数的定义

compute_frame_duration函数定义在FFmpeg源码(本文演示用的FFmpeg源码版本为7.0.1)的源文件libavformat/demux.c中:

/*** Return the frame duration in seconds. Return 0 if not available.*/
static void compute_frame_duration(AVFormatContext *s, int *pnum, int *pden,AVStream *st, AVCodecParserContext *pc,AVPacket *pkt)
{FFStream *const sti = ffstream(st);AVRational codec_framerate = sti->avctx->framerate;int frame_size, sample_rate;*pnum = 0;*pden = 0;switch (st->codecpar->codec_type) {case AVMEDIA_TYPE_VIDEO:if (st->r_frame_rate.num && (!pc || !codec_framerate.num)) {*pnum = st->r_frame_rate.den;*pden = st->r_frame_rate.num;} else if ((s->iformat->flags & AVFMT_NOTIMESTAMPS) &&!codec_framerate.num &&st->avg_frame_rate.num && st->avg_frame_rate.den) {*pnum = st->avg_frame_rate.den;*pden = st->avg_frame_rate.num;} else if (st->time_base.num * 1000LL > st->time_base.den) {*pnum = st->time_base.num;*pden = st->time_base.den;} else if (codec_framerate.den * 1000LL > codec_framerate.num) {int ticks_per_frame = (sti->codec_desc &&(sti->codec_desc->props & AV_CODEC_PROP_FIELDS)) ? 2 : 1;av_reduce(pnum, pden,codec_framerate.den,codec_framerate.num * (int64_t)ticks_per_frame,INT_MAX);if (pc && pc->repeat_pict) {av_reduce(pnum, pden,(*pnum) * (1LL + pc->repeat_pict),(*pden),INT_MAX);}/* If this codec can be interlaced or progressive then we need* a parser to compute duration of a packet. Thus if we have* no parser in such case leave duration undefined. */if (sti->codec_desc &&(sti->codec_desc->props & AV_CODEC_PROP_FIELDS) && !pc)*pnum = *pden = 0;}break;case AVMEDIA_TYPE_AUDIO:if (sti->avctx_inited) {frame_size  = av_get_audio_frame_duration(sti->avctx, pkt->size);sample_rate = sti->avctx->sample_rate;} else {frame_size  = av_get_audio_frame_duration2(st->codecpar, pkt->size);sample_rate = st->codecpar->sample_rate;}if (frame_size <= 0 || sample_rate <= 0)break;*pnum = frame_size;*pden = sample_rate;break;default:break;}
}

该函数的作用是:计算某路视频流或音频流的packet占用的时间值。

形参s:输入型参数。指向一个AVFormatContext结构体。

形参pnum:输出型参数。执行compute_frame_duration函数后,如果该路流为视频,(*pnum)÷(*pden)会变为一帧视频pakcet占用的时间,单位为秒;如果该路流为音频,*pnum会变为该音频packet(形参pkt指向的packet)占用的以AVStream的time_base为单位的时间值,(*pnum)÷(*pden)会变为该音频packet占用的以秒为单位的时间值。

形参pden:输出型参数。执行compute_frame_duration函数后,如果该路流为视频,(*pnum)÷(*pden)会变为一帧视频pakcet占用的时间,单位为秒;如果该路流为音频,*pden会变为该音频的采样频率(单位为Hz),(*pnum)÷(*pden)会变为该音频packet占用的以秒为单位的时间值。

形参st:输入型参数。指向一个AVStream结构体。

形参pc:输入型参数。指向一个AVCodecParserContext结构体。

形参pkt:输入型参数。指向一个AVPacket结构体。

返回值:无。

二、compute_frame_duration函数的内部实现分析

compute_frame_duration函数内部首先会通过switch、case语句判断该路流是视频还是音频:

switch (st->codecpar->codec_type) {

(一)情况一:该路流为视频

该路流为视频时,以视频压缩编码格式为H.264为例子,相当于执行了下面代码:

case AVMEDIA_TYPE_VIDEO:int ticks_per_frame = (sti->codec_desc &&(sti->codec_desc->props & AV_CODEC_PROP_FIELDS)) ? 2 : 1;av_reduce(pnum, pden,    codec_framerate.den,codec_framerate.num * (int64_t)ticks_per_frame,INT_MAX);if (pc && pc->repeat_pict) {av_reduce(pnum, pden,(*pnum) * (1LL + pc->repeat_pict),(*pden),INT_MAX);}

sti->codec_desc和sti->codec_desc->props是通过avformat_open_input函数中的avcodec_descriptor_get函数得到的,具体可以参考:《FFmpeg源码:avcodec_descriptor_get函数分析》。

int ticks_per_frame = (sti->codec_desc && (sti->codec_desc->props & AV_CODEC_PROP_FIELDS)) ? 2 : 1;

比如对于H.264,通过语句:avcodec_descriptor_get(AV_CODEC_ID_H264)得到的AVCodecDescriptor结构为:

    {.id        = AV_CODEC_ID_H264,.type      = AVMEDIA_TYPE_VIDEO,.name      = "h264",.long_name = "H.264 / AVC / MPEG-4 AVC / MPEG-4 part 10",.props     = 30,.profiles  = ff_h264_profiles,},

所以对于H.264,sti->codec_desc为true,sti->codec_desc->props为30。而宏定义AV_CODEC_PROP_FIELDS为:

/*** Video codec supports separate coding of fields in interlaced frames.*/
#define AV_CODEC_PROP_FIELDS        (1 << 4)

所以满足条件:sti->codec_desc && (sti->codec_desc->props & AV_CODEC_PROP_FIELDS)为true。所以对于H.264,变量ticks_per_frame值为2。

av_reduce函数的用法可以参考:《FFmpeg源码:av_reduce函数分析》。codec_framerate.den为视频帧率的分母,codec_framerate.num为视频帧率的分子,所以对于H.264,执行下面的av_reduce函数后,(*pden) ÷ (*pnum)会变为视频帧率的两倍:

av_reduce(pnum, pden,codec_framerate.den,codec_framerate.num * (int64_t)ticks_per_frame,INT_MAX);

对于H.264,满足条件pc && pc->repeat_pict为true,所以会继续执行av_reduce函数。对于H.264下面语句相当于执行了语句:av_reduce(pnum, pden ,(*pnum) * (2), (*pden), INT_MAX);执行完后(*pden) ÷ (*pnum)会变为视频帧率:

            if (pc && pc->repeat_pict) {av_reduce(pnum, pden,(*pnum) * (1LL + pc->repeat_pict),(*pden),INT_MAX);}

我们都知道视频帧率是视频每秒钟播放的帧数,所以视频帧率的倒数即(*pnum)÷(*pden)就是一帧视频pakcet占用的时间,单位为秒。

(二)情况二:该路流为音频

该路流为音频时,相当于执行了下面代码:

case AVMEDIA_TYPE_AUDIO:frame_size  = av_get_audio_frame_duration2(st->codecpar, pkt->size);sample_rate = st->codecpar->sample_rate;*pnum = frame_size;*pden = sample_rate;

 通过av_get_audio_frame_duration2函数得到该音频packet占用的时间值(关于该函数可以参考:《FFmpeg源码:get_audio_frame_duration、av_get_audio_frame_duration2函数分析》)。通过语句:sample_rate = st->codecpar->sample_rate得到该路音频的采样频率:

frame_size  = av_get_audio_frame_duration2(st->codecpar, pkt->size);
sample_rate = st->codecpar->sample_rate;

让*pnum变为该音频packet(形参pkt指向的packet)占用的以AVStream的time_base为单位的时间值,让*pden变为该音频的采样频率(单位为Hz)。(*pnum)÷(*pden)就是该音频packet占用的以秒为单位的时间值:

    *pnum = frame_size;*pden = sample_rate;

相关文章:

FFmpeg源码:compute_frame_duration函数分析

一、compute_frame_duration函数的定义 compute_frame_duration函数定义在FFmpeg源码&#xff08;本文演示用的FFmpeg源码版本为7.0.1&#xff09;的源文件libavformat/demux.c中&#xff1a; /*** Return the frame duration in seconds. Return 0 if not available.*/ stat…...

ARM 异常处理(21)

异常的流程&#xff1a; 首先&#xff1a; 在硬件上阶段&#xff1a; 这里是4大步3小步 然后是 异常处理&#xff1a; 这里主要是保存现场&#xff0c;进行异常处理 然后是 异常返回&#xff1a; 主要指 恢复现场&#xff0c; 再跳转回去。 首先硬件上&#xff…...

我开源了我的新闻网站项目

&#x1f389; 前言 暑假时写了一个Web项目&#xff0c;感觉做的还是有模有样的&#xff0c;不仅做了前端&#xff0c;还加了后端并连了数据库。最近也是将它开源了&#xff0c;一来是为了熟悉一下Github流程和Git使用命令&#xff0c;二来也是想和大家分享一下自己的成果&…...

LlamaIndex 使用 RouterOutputAgentWorkflow

LlamaIndex 中提供了一个 RouterOutputAgentWorkflow 功能&#xff0c;可以集成多个 QueryTool&#xff0c;根据用户的输入判断使用那个 QueryEngine&#xff0c;在做查询的时候&#xff0c;可以从不同的数据源进行查询&#xff0c;例如确定的数据从数据库查询&#xff0c;如果…...

设计模式学习-责任链模式

概念 使多个对象都有机会处理请求&#xff0c;从而避免了请求的发送者和接受者之间的耦合关系。将这些对象连成一条链&#xff0c;并沿着这条链传递该请求&#xff0c;直到有对象处理它为止. 代码编写 using UnityEngine; using System.Collections; public class ChainOfResp…...

【全网最全】2024年数学建模国赛B题31页完整建模过程+成品论文+matlab/python代码等(后续会更新

您的点赞收藏是我继续更新的最大动力&#xff01; 一定要点击如下的卡片&#xff0c;那是获取资料的入口&#xff01; 2024数学建模国赛B题 【全网最全】2024年数学建模国赛B题31页完整建模过程成品论文matlab/python代码等&#xff08;后续会更新「首先来看看目前已有的资料…...

第二十一届华为杯数学建模经验分享之资料分享篇

今天给大家分享一些数学建模的资料&#xff0c;通过这些资料的学习相信你们一定在比赛中获得好的成绩。今天分享的资料包括美赛和国赛的优秀论文集、研赛的优秀论文集、推荐数学建模的相关书籍、智能算法的学习PPT、python机器学习的书籍和数学建模经验分享与总结&#xff0c;其…...

使用 OpenSSL 创建自签名证书

mkdir -p /etc/nginx/conf.d/cert #2、创建私钥 openssl genrsa -des3 -out https.key 1024 提示输入字符&#xff1a; 输入字符&#xff1a;rancher [rootocean-app-1a-01 cert]# openssl genrsa -des3 -out https.key 1024 Generating RSA private key, 1024 bit long modulu…...

EmguCV学习笔记 VB.Net 9.1 VideoCapture类

版权声明&#xff1a;本文为博主原创文章&#xff0c;转载请在显著位置标明本文出处以及作者网名&#xff0c;未经作者允许不得用于商业目的。 EmguCV是一个基于OpenCV的开源免费的跨平台计算机视觉库,它向C#和VB.NET开发者提供了OpenCV库的大部分功能。 教程VB.net版本请访问…...

Rspack 1.0 发布了!

文章来源&#xff5c;Rspack Team 项目地址&#xff5c;https://github.com/web-infra-dev/rspack Rspack 是基于 Rust 编写的下一代 JavaScript 打包工具&#xff0c; 兼容 webpack 的 API 和生态&#xff0c;并提供 10 倍于 webpack 的构建性能。 在 18 个月前&#xff0c;我…...

【全网最全】2024年数学建模国赛E题超详细保奖思路+可视化图表+成品论文+matlab/python代码等(后续会更新

您的点赞收藏是我继续更新的最大动力&#xff01; 一定要点击如下的卡片&#xff0c;那是获取资料的入口&#xff01; 【全网最全】2024年数学建模国赛E题成品论文超详细保奖思路可视化图表matlab/python代码等&#xff08;后续会更新「首先来看看目前已有的资料&#xff0c;还…...

数智转型,看JNPF如何成为企业的必备工具

随着数字化转型的浪潮席卷全球&#xff0c;企业面临着前所未有的挑战与机遇。在这一过程中&#xff0c;低代码开发平台作为一种创新的软件开发方式&#xff0c;正逐渐成为企业实现快速迭代和敏捷开发的关键工具。JNPF作为一款领先的低代码开发平台&#xff0c;凭借其强大的功能…...

ArcGIS Pro 发布松散型切片

使用ArcGIS Pro发布松散型切片问题&#xff0c;有时候会出现切片方案写了松散型&#xff0c;但是自动切片完成后依然是紧凑型的问题&#xff0c;这时候可以采用手动修改然后再切片的方式。 1. 发布切片服务 选择手动切片方式 2. 手动修改服务的切片方案文件 修改cache服务…...

奖项再+1!通义灵码智能编码助手通过可信 AI 智能编码工具评估,获当前最高等级

阿里云的通义灵码智能编码助手参与中国信通院组织的可信AI智能编码工具首轮评估&#xff0c;最终获得 4 级评级&#xff0c;成为国内首批通过该项评估并获得当前最高评级的企业之一。 此次评估以《智能化软件工程技术和应用要求 第 2 部分&#xff1a;智能开发能力》为依据&…...

如何使用 yum 在 CentOS 6 上安装 nginx

前些天发现了一个巨牛的人工智能学习网站&#xff0c;通俗易懂&#xff0c;风趣幽默&#xff0c;忍不住分享一下给大家。点击跳转到网站。 状态 状态&#xff1a; 已弃用 本文涵盖的 CentOS 版本已不再受支持。如果您目前正在运行 CentOS 6 服务器&#xff0c;我们强烈建议升…...

F12抓包05:Network接口测试(抓包篡改请求)

课程大纲 使用线上接口测试网站演示操作&#xff0c;浏览器F12检查工具如何进行简单的接口测试&#xff1a;抓包、复制请求、篡改数据、发送新请求。 测试地址&#xff1a;https://httpbin.org/forms/post ① 抓包&#xff1a;鼠标右键打开“检查”工具&#xff08;F12&#xf…...

OPenCV结构分析与形状描述符(4)计算一个旋转矩形的四个顶点的函数boxPoints()的使用

操作系统&#xff1a;ubuntu22.04 OpenCV版本&#xff1a;OpenCV4.9 IDE:Visual Studio Code 编程语言&#xff1a;C11 算法描述 找到一个旋转矩形的四个顶点。对于绘制旋转矩形很有用。 该函数找到一个旋转矩形的四个顶点。这个函数对于绘制矩形很有帮助。在C中&#xff0c;…...

【Matplotlib】利用Python进行绘图!(python数据分析与可视化)

文章开始前打个小广告——分享一份Python学习大礼包&#xff08;激活码安装包、Python web开发&#xff0c;Python爬虫&#xff0c;Python数据分析&#xff0c;人工智能、自动化办公等学习教程&#xff09;点击领取&#xff0c;100%免费&#xff01; 【Matplotlib】 教程&…...

第二百二十节 JPA教程 - JPA 实体管理器删除示例

JPA教程 - JPA 实体管理器删除示例 我们可以使用JPA中的EntityManager来删除一个实体。 在下面的代码中&#xff0c;我们首先通过使用EntityManager中的find方法从数据库获取person对象&#xff0c;然后调用remove方法并传递person对象引用。 Person emp em.find(Person.cla…...

[⁠TypeError⁠]‍ {message: “Cannot read property ‘‘ of undefined“}

11:11:25.500 ‍[⁠TypeError⁠]‍ {message: “Cannot read property ‘’ of undefined”} 11:11:25.586 [Vue warn]: Unhandled error during execution of render function \n at \nat \nat \nat \nat \nat \nat <V uniapp 使用报错 解决方法 页面加 v-if 来判断这个字…...

C++27异常处理增强配置(ISO/IEC 14882:2027草案第12.8节深度解密)

第一章&#xff1a;C27异常处理增强配置的标准化演进脉络C27标准委员会在异常处理机制上引入了关键性配置抽象&#xff0c;旨在统一跨编译器、跨平台的异常行为语义。核心演进方向聚焦于将异常传播策略、栈展开控制与诊断信息生成三者解耦&#xff0c;并通过标准化属性和编译期…...

通义千问1.5-1.8B-Chat商业应用:企业智能助手快速落地方案

通义千问1.5-1.8B-Chat商业应用&#xff1a;企业智能助手快速落地方案 1. 企业智能助手市场现状与需求 当前企业运营面临人力成本上升、服务标准化不足、数据分析需求激增等挑战。传统解决方案往往需要投入大量资源进行定制开发&#xff0c;而基于大模型的智能助手提供了快速…...

IceC:面向嵌入式平台的轻量级ICE兼容中间件

1. IceC&#xff1a;面向资源受限嵌入式平台的轻量级ZeroC ICE兼容中间件 1.1 设计定位与工程必要性 IceC并非ZeroC ICE的全功能移植&#xff0c;而是在AVR&#xff08;如ATmega328P&#xff09;和ESP8266等典型资源受限平台约束下&#xff0c;对ICE通信模型进行深度裁剪与重构…...

【Linux基础】文件编辑压缩解压

文件编辑&&压缩解压 实验环境准备 [rootCentOS-Harvy /]# cd tmp/ [rootCentOS-Harvy tmp]# mkdir -p lab_319[rootCentOS-Harvy tmp]# ll | grep *319 drwxr-xr-x. 2 root root 6 3月 19 16:59 lab_319实验1&#xff1a;Vim 文本编辑 1.1 创建文件并写入内容…...

基于Stackelberg博弈与需求响应的光伏用户群内部优化定价模型研究

MATLAB代码&#xff1a;基于Stackelberg博弈的光伏用户群优化定价模型关键词&#xff1a;光伏用户群 内部电价 需求响应 斯塔克伯格博弈 参考文档&#xff1a;《基于Stackelberg博弈的光伏用户群优化定价模型》王程 刘念 仿真平台&#xff1a;MATLAB Cplex 主要内容&#xff1…...

2026毕业论文降AI工具指南:实测4款高通过率方案

答辩前三天被通知AI率超标要重改的焦虑&#xff0c;我至今印象深刻。去年帮二十多位同专业学弟学妹调整过毕业论文的AI检测问题&#xff0c;整理出的实用经验今天全部分享给大家。 先说结论&#xff1a;SpeedAI科研小助手和思笔AI是我最推荐的两款。前者性价比极高且全平台适配…...

【C++27协程调试终极指南】:20年专家亲授5大不可外泄的断点追踪黑科技

第一章&#xff1a;C27协程调试的底层模型与认知重构 C27将首次将协程&#xff08;coroutine&#xff09;纳入核心语言调试规范&#xff0c;其调试模型不再依赖于传统栈帧回溯&#xff0c;而是围绕可恢复执行上下文&#xff08;resumable execution context&#xff09;、挂起点…...

大模型剪枝(二)Wanda实战:如何在不重训练的情况下高效压缩LLM

1. Wanda剪枝方法的核心原理 Wanda方法的创新点在于它巧妙地结合了权重幅度和输入激活信息来决定剪枝策略。传统的大模型剪枝往往只关注权重本身的绝对值大小&#xff0c;而忽略了这些权重在实际推理过程中所起的作用。这就好比修剪果树时只根据树枝粗细做决定&#xff0c;却不…...

API统一管控平台:new-api、one-api、Grok2API、Quotio、UniAPI、Sub2API、OpenAI Router

之前写过类似文章&#xff0c;请参考LLM系列之API聚合平台&#xff1a;OpenRouter、TogetherAI、LiteLLM。 本文继续汇总几个开源API统一管控平台。 one-api 官网&#xff0c;开源&#xff08;GitHub&#xff0c;30.1K Star&#xff0c;5.8K Fork&#xff09;平台&#xff0…...

3大突破!网盘下载加速工具让你的文件获取效率倍增

3大突破&#xff01;网盘下载加速工具让你的文件获取效率倍增 【免费下载链接】Online-disk-direct-link-download-assistant 一个基于 JavaScript 的网盘文件下载地址获取工具。基于【网盘直链下载助手】修改 &#xff0c;支持 百度网盘 / 阿里云盘 / 中国移动云盘 / 天翼云盘…...