openh264 编码器源码分析:AnalyzePictureComplexity 函数

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

介绍

  1. 文件位置:
    • openh264/codec/processing/src/complexityanalysis/ComplexityAnalysis.cpp
  2. 功能:
    • 作为CWelsPreProcess类中一个方法,用来分析当前图像与参考图像之间的复杂度关系,以便编码策略。
  3. 原型:
void CWelsPreProcess::AnalyzePictureComplexity (sWelsEncCtx* pCtx, SPicture* pCurPicture, SPicture* pRefPicture,const int32_t kiDependencyId, const bool bCalculateBGD)
  1. 参数:
    • sWelsEncCtx* pCtx:指向sWelsEncCtx结构的指针,这通常是一个上下文结构,包含了编码过程中所需的状态信息。
    • SPicture* pCurPicture:指向当前待编码图片的指针。
    • SPicture* pRefPicture:指向参考图片的指针,用于与当前图片进行比较。
    • const int32_t kiDependencyId:一个整数常量,表示依赖层的ID,可能用于确定当前图片所属的编码层。
    • const bool bCalculateBGD:一个布尔值常量,指示是否需要计算背景差分(Background Difference)。

函数关系图

在这里插入图片描述

原理

桌面图像

  1. process 函数
  • 如果是IDR帧或参考图像为空,则调用GomComplexityAnalysisIntra函数执行内部复杂度分析;
  • 如果滚动标志为假或滚动偏移量在X和Y方向上都为0,则调用GomComplexityAnalysisInter函数执行无滚动的交互复杂度分析;
  • 调用GomComplexityAnalysisInter函数执行有滚动的交互复杂度分析;
  1. GomComplexityAnalysisIntra函数
  • 变量初始化;
  • 嵌套循环分析每个宏块;
    - 检查是否有可用的垂直或水平参考宏块,如果有,则使用相应的预测函数m_pIntraFunc和SAD计算函数m_pSadFunc来计算SAD值iBlockSadH、iBlockSadV;
    • 如果垂直或水平参考宏块存在,则累加中iBlockSadH、iBlockSadV的最小值到iGomSad中;
      • 如果当前宏块是一行中的最后一个宏块,或者已经到达了宏块组的底部,将iGomSad的值存储到类成员变量m_ComplexityAnalysisParam.pGomComplexity中,并更新帧复杂度m_ComplexityAnalysisParam.iFrameComplexity;
      • 更新索引iIdx,将变量iGomSad置 0,为下一个宏块组的复杂度值做准备;
      • 在处理完一行的所有宏块后,更新指针pPtrY到下一行的开始位置;
  • 更新类成员变量m_ComplexityAnalysisParam.iGomNumInFrame,表示帧中宏块组的数量;
  1. GomComplexityAnalysisInter函数
  • 变量初始化;
  • 嵌套 for 循环分析每个宏块;
    • 首先调用m_pSadFunc函数计算当前块的SAD值iInterSad;
    • 如果bScrollFlag为真,并且当前块的SAD值不为0,且滚动偏移量在有效范围内,则调用m_pSadFunc函数计算考虑滚动后的SAD值iScrollSad,并取两者的较小值作为当前块的SAD值iInterSad;
    • 如果当前宏块不是第一行或第一列的宏块,使用类成员预测函数m_pIntraFuncm_pSadFunc计算水平和垂直预测的SAD值iBlockSadH、iBlockSadV;
    • 选择iBlockSadH、iBlockSadV、iInterSad中最小值最为总体 SAD 值iGomSad;
    • 果当前宏块是一行中的最后一个宏块,或者已经到达了宏块组的底部,将iGomSad的值存储到类成员变量m_ComplexityAnalysisParam.pGomComplexity中,并更新帧复杂度m_ComplexityAnalysisParam.iFrameComplexity;
    • 更新索引iIdx,将变量iGomSad置 0,为下一个宏块组的复杂度值做准备;
    • 在处理完一行的所有宏块后,更新指针pPtrY到下一行的开始位置;
  • 更新类成员变量m_ComplexityAnalysisParam.iGomNumInFrame,表示帧中宏块组的数量;
  1. m_pSadFunc函数
  • 指向WelsSampleSad16x16_c函数
  1. m_pIntraFunc[0]函数
  • 指向WelsI16x16LumaPredV_c函数
  1. m_pIntraFunc[1]函数
  • 指向WelsI16x16LumaPredH_c函数

摄像镜头图像

  1. process 函数
  • iComplexityAnalysisMode等于FRAME_SAD时,调用AnalyzeFrameComplexityViaSad函数分析图像复杂度;
  • iComplexityAnalysisMode等于GOM_SAD时,调用AnalyzeGomComplexityViaSad函数分析图像复杂度;
  • iComplexityAnalysisMode等于GOM_VAR时,调用AnalyzeGomComplexityViaVar函数分析图像复杂度;
  1. AnalyzeFrameComplexityViaSad函数
  • 将pCalcResult赋值给pVaaCalcResults;
  • 将pVaaCalcResults中的iFrameSad赋值给m_sComplexityAnalysisParam.iFrameComplexity,作为当前帧的复杂度;
  • 如果iCalcBgd,即有背景检测,则调用GetFrameSadExcludeBackground函数计算复杂度,赋值给m_sComplexityAnalysisParam.iFrameComplexity,作为当前帧的复杂度;
  • GetFrameSadExcludeBackground函数:这个函数的目的是计算帧的SAD值,但在计算时排除了背景区域的宏块,这有助于更准确地评估帧中实际的前景复杂度。
    • 初始化变量,计算宏块组iMbNumInGom和宏块数iGomMbNum;
    • m_sComplexityAnalysisParam.pCalcResult赋值给pVaaCalcResults;
    • 双层嵌套 for 循环遍历所有宏块,
      • 如果宏块不是背景宏块,增加对应宏块组的前景宏块数量,并累加该宏块的四个8x8块的SAD值到uiFrameSad;
    • 返回 uiFrameSad值;
  1. AnalyzeGomComplexityViaSad函数
  • 初始化变量,计算宏块组iMbNumInGom和宏块数iGomMbNum;
  • 调用InitGomSadFunc函数初始化SAD计算函数(m_pfGomSad),并为每个GOB分配一个SAD变量(uiGomSad)
  • 嵌套双层循环处理每个宏块,
    • 对当前GOB的每一行进行循环,计算SAD;
    • 调用m_pfGomSad函数来累加当前宏块的SAD值uiGomSad;
    • 将计算得到的SAD值uiGomSad存储到pGomComplexity数组中;
    • 更新整个帧的SAD值(uiFrameSad);
  • 将帧的SAD值uiFrameSad存储在m_sComplexityAnalysisParam结构体中。
  1. AnalyzeGomComplexityViaVar函数
  • 初始化变量,计算宏块组iMbNumInGom和宏块数iGomMbNum;
  • 初始化样本和平方和变量(uiSampleSum和uiSquareSum);
  • 双层嵌套循环遍历每一个宏块;
    • 对当前GOB的每一行进行循环,累加样本和uiSampleSum和平方和uiSquareSum;
    • 计算宏块组方差,将结果存到pGomComplexity中;
    • 累加pGomComplexity到帧的 SAD 值uiFrameSad;
  • 将帧的SAD值uiFrameSad存储在m_sComplexityAnalysisParam结构体中。

源码

  • 略。

这篇关于openh264 编码器源码分析:AnalyzePictureComplexity 函数的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java8需要知道的4个函数式接口简单教程

《Java8需要知道的4个函数式接口简单教程》:本文主要介绍Java8中引入的函数式接口,包括Consumer、Supplier、Predicate和Function,以及它们的用法和特点,文中... 目录什么是函数是接口?Consumer接口定义核心特点注意事项常见用法1.基本用法2.结合andThen链

MySQL 日期时间格式化函数 DATE_FORMAT() 的使用示例详解

《MySQL日期时间格式化函数DATE_FORMAT()的使用示例详解》`DATE_FORMAT()`是MySQL中用于格式化日期时间的函数,本文详细介绍了其语法、格式化字符串的含义以及常见日期... 目录一、DATE_FORMAT()语法二、格式化字符串详解三、常见日期时间格式组合四、业务场景五、总结一、

Go使用pprof进行CPU,内存和阻塞情况分析

《Go使用pprof进行CPU,内存和阻塞情况分析》Go语言提供了强大的pprof工具,用于分析CPU、内存、Goroutine阻塞等性能问题,帮助开发者优化程序,提高运行效率,下面我们就来深入了解下... 目录1. pprof 介绍2. 快速上手:启用 pprof3. CPU Profiling:分析 C

MySQL表锁、页面锁和行锁的作用及其优缺点对比分析

《MySQL表锁、页面锁和行锁的作用及其优缺点对比分析》MySQL中的表锁、页面锁和行锁各有特点,适用于不同的场景,表锁锁定整个表,适用于批量操作和MyISAM存储引擎,页面锁锁定数据页,适用于旧版本... 目录1. 表锁(Table Lock)2. 页面锁(Page Lock)3. 行锁(Row Lock

golang panic 函数用法示例详解

《golangpanic函数用法示例详解》在Go语言中,panic用于触发不可恢复的错误,终止函数执行并逐层向上触发defer,最终若未被recover捕获,程序会崩溃,recover用于在def... 目录1. panic 的作用2. 基本用法3. recover 的使用规则4. 错误处理建议5. 常见错

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

Python itertools中accumulate函数用法及使用运用详细讲解

《Pythonitertools中accumulate函数用法及使用运用详细讲解》:本文主要介绍Python的itertools库中的accumulate函数,该函数可以计算累积和或通过指定函数... 目录1.1前言:1.2定义:1.3衍生用法:1.3Leetcode的实际运用:总结 1.1前言:本文将详

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

轻松上手MYSQL之JSON函数实现高效数据查询与操作

《轻松上手MYSQL之JSON函数实现高效数据查询与操作》:本文主要介绍轻松上手MYSQL之JSON函数实现高效数据查询与操作的相关资料,MySQL提供了多个JSON函数,用于处理和查询JSON数... 目录一、jsON_EXTRACT 提取指定数据二、JSON_UNQUOTE 取消双引号三、JSON_KE

MySQL数据库函数之JSON_EXTRACT示例代码

《MySQL数据库函数之JSON_EXTRACT示例代码》:本文主要介绍MySQL数据库函数之JSON_EXTRACT的相关资料,JSON_EXTRACT()函数用于从JSON文档中提取值,支持对... 目录前言基本语法路径表达式示例示例 1: 提取简单值示例 2: 提取嵌套值示例 3: 提取数组中的值注意