VVC/H.266 VTM10.0 代码阅读记录 (1. nal_unit_type 与 帧类型)

2023-10-30 22:59

本文主要是介绍VVC/H.266 VTM10.0 代码阅读记录 (1. nal_unit_type 与 帧类型),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

从2019年12月后,由于先后两段实习,没有认真跟过最新的标准和代码,感觉落下挺多东西。
接下来的一段时间,没有了秋招和paper的压力,下定决心重新整理和学习一下视频编解码的东西,包括VVC、AVS3和X264,有时间再看一下AV1。
在此系列,记录一下VVC的学习心得以及VTM10.0的代码阅读记录(可能会比较乱,博客主要是当成草稿纸使用)。

在之前发的博客里面,总结过 NALU提取 和 non-VCLU解码 两部分代码和内容。

VVC/H.266代码阅读(VTM8.0)(一. NALU提取)
VVC/H.266代码阅读(VTM8.0)(二. non-VCLU解码)

但是,现在回顾博客,针对 nal_unit_type 和 帧类型 这块,并没有学习记录也没有理解透彻,现在根据 draft 记录一下这部分的学习内容。
在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

nal_unit_type 指定了 NALU 的类型。应遵循以下规则:

① 对于一个 subpicture 的所有 VCL NAL,nal_unit_type 应该是相同的。
② PPS 里面有一个 pps_mixed_nalu_types_in_pic_flag。该值等于0时,指定引用PPS的每个图片有一个或多个VCL NAL单位,并且引用PPS的每个图片的VCL NAL单位具有相同的nal_unit_type值。该值等于1时,指定引用PPS的每个图片有一个以上的VCL NAL单元,并且VCL NAL单元不具有相同的nal_unit_type值。
③ 如果 pps_mixed_nalu_types_in_pic_flag 等于0, 一幅图像的所有 VCL NAL 单元的 nal_unit_type应当相同的。如果 pps_mixed_nalu_types_in_pic_flag 等于1: ① 图片最少有两个 subpictures。 ② 图片的 VCL NAL 单元应该有两个或更多不同的nal_unit_type 值。③ 图片中没有 nal_unit_type 等于 GDR_NUT 的 VCL NAL 单元。④ 当图片的一个 VCL NAL 单位有nal_unit_type 等于 nalUnitTypeA (IDR_W_RADL, IDR_N_LP或CRA_NUT),图片的其他 VCL NAL 单位 nal_unit_type 都等于 nalUnitTypeA 或 TRAIL_NUT。
④ IRAP 或 GDR AU 中的所有图片的 nal_unit_type 的值应该是相同的。

下表就是 nal_unit_type 对应的 NALU的类型。下面的叙述不区分 picture 和 subpicture,一律用 帧/图像 来统一描述。

在这里插入图片描述

  • 0 - 3,7 - 10 对应的是VCLU,负责高效的视频内容表示。

  • 12 - 25 对应的是non-VCLU,是一些参数集或者信息内容。

  • 0 - 1 后置图像,2 - 3 前置图像,7 - 10 IRAP 帧内随机接入点图像。

  • 0 Trail 是后置接入图像,解码顺序和显示顺序都在 IRAP 和 GDR 帧之后。Trail 图像可以参考相关联的 IRAP 和 GDR 图像以及他们的相关联的后置图像,不可以参考其前置图像或者相关联的非后置图像。

如下图的B33、B34、B35就是I28的后置图像,其解码顺序和显示顺序都在I28 (CRA图像是IRAP图像的一种) 之后。
在这里插入图片描述

  • 1 STSA 是分布时域子层接入图像。在 STSA 图像编码时,STSA 图像能够从同一层的下一层向上切换到包含 STSA 图像的子层。STSA 图像保证 STSA图像自己、同一子层中跟随在此 STSA 之后图像(解码顺序)不参考同层子层超前 STSA 的图像。

① 如下图中的P2就是STSA图像。在P2后,可以从TS0上切换到TS1层,也就说可以保证 “在 STSA 图像编码时,STSA 图像能够从同一层的下一层向上切换到包含 STSA 图像的子层。” 但是,并不可以继续切换到TS2层,因为后面的P3需要参考P1,而P1在P2之前解码,可能是被丢弃。至于可以从TS0切换到TS1,是因为P2无需参考TS1的任何图像,只用参考TS0的图像,而且P2也不用参考在解码顺序中跟随在P2后面的任何TS1子层的图像。
② 此外,其实在P6后,允许TS0直接切换到TS1,乃至更高层TS2。这是因为按照后续参考逻辑来看,TS1乃至TS2层的后续图像都可以找到肯定可以解码的参考图像。在HEVC中,这种图像叫做 STA 图像(时域子层接入图像),但是VVC目前没看到有这个图像的定义。
③ STSA 图像是一种特殊的 Trail 图像。
在这里插入图片描述

  • 2 RADL 是可解码随机接入前置图像。RADL 图像不需要参考任何解码顺序在 IRAP之前的图像。
  • 3 RASL 是跳过随机接入前置图像。RASL 图像需要参考解码器中可能不存在的图像,所以解码器只能跳过这部分图像继续解码处理。

① 前置图像就是指解码顺序在 IRAP 之前,显示顺序在 IRAP 图像之后的图像。
② 如下图的B29和B31就是RADL图像,如果在 I28 (CRA) 随机接入,B29 和 B31 是在 I28 (CRA) 之后进行解码的,但是 B29只参考 I28,B31参考 I28 和 B29,都是可以正常解码的。
③ 如下图的B30就是RASLT图像,B30还需要参考P24,这可能不存在于解码器中,所以B30只能被跳过。
在这里插入图片描述

  • 7 - 8 IDR 即时解码刷新图像,不使用帧间预测,可以是比特流中的首帧或后续帧。在解码顺序上,IDR帧 是 CVS(coded video sequence) 的首帧。IDR帧有两种类型,分别是 IDR_N_LP (8) 和 IDR_W_RADL (7) ,二者都不允许有 RASL 帧。而 IDR_N_LP 不能有前置帧,而 IDR_W_RADL 可以有前置的 RADL 帧。

  • 9 CRA 纯随机接入图像。处于比特流 GOP的第一帧,而且引领开放式GOP,在解码过程中允许参考其他GOP中的图像数据。CRA图像是和相应前置图像有关联的图像。CRA图像的前置图像被允许参考CRA前面的已解码图像。

  • 10 GDR 渐近解码刷新图像,是与IDR相关的一个概念,这是一种特别帧,每个GDR包含一个I-Slice,几个连续的GDR帧可以起到一个I帧相同的作用,但由于GDR帧的大小与P帧接近,因此可以起到平滑码率的作用,减少超大I帧对网络的冲击,减少传输延迟,同时也能起到阻隔错误的作用。GDR是通过P帧内包括I块组的方法来实现渐近刷新。
    在这里插入图片描述

  • 12 OPI 操作点信息,提供解码器的操作点信息。
    在这里插入图片描述

  • 13 DCI 解码能力信息。
    在这里插入图片描述

  • 后面的比较常规,不再赘述。

这篇关于VVC/H.266 VTM10.0 代码阅读记录 (1. nal_unit_type 与 帧类型)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

SpringBoot中四种AOP实战应用场景及代码实现

《SpringBoot中四种AOP实战应用场景及代码实现》面向切面编程(AOP)是Spring框架的核心功能之一,它通过预编译和运行期动态代理实现程序功能的统一维护,在SpringBoot应用中,AO... 目录引言场景一:日志记录与性能监控业务需求实现方案使用示例扩展:MDC实现请求跟踪场景二:权限控制与

利用Python调试串口的示例代码

《利用Python调试串口的示例代码》在嵌入式开发、物联网设备调试过程中,串口通信是最基础的调试手段本文将带你用Python+ttkbootstrap打造一款高颜值、多功能的串口调试助手,需要的可以了... 目录概述:为什么需要专业的串口调试工具项目架构设计1.1 技术栈选型1.2 关键类说明1.3 线程模

Python Transformers库(NLP处理库)案例代码讲解

《PythonTransformers库(NLP处理库)案例代码讲解》本文介绍transformers库的全面讲解,包含基础知识、高级用法、案例代码及学习路径,内容经过组织,适合不同阶段的学习者,对... 目录一、基础知识1. Transformers 库简介2. 安装与环境配置3. 快速上手示例二、核心模

Java的栈与队列实现代码解析

《Java的栈与队列实现代码解析》栈是常见的线性数据结构,栈的特点是以先进后出的形式,后进先出,先进后出,分为栈底和栈顶,栈应用于内存的分配,表达式求值,存储临时的数据和方法的调用等,本文给大家介绍J... 目录栈的概念(Stack)栈的实现代码队列(Queue)模拟实现队列(双链表实现)循环队列(循环数组

Java使用SLF4J记录不同级别日志的示例详解

《Java使用SLF4J记录不同级别日志的示例详解》SLF4J是一个简单的日志门面,它允许在运行时选择不同的日志实现,这篇文章主要为大家详细介绍了如何使用SLF4J记录不同级别日志,感兴趣的可以了解下... 目录一、SLF4J简介二、添加依赖三、配置Logback四、记录不同级别的日志五、总结一、SLF4J

使用Java将DOCX文档解析为Markdown文档的代码实现

《使用Java将DOCX文档解析为Markdown文档的代码实现》在现代文档处理中,Markdown(MD)因其简洁的语法和良好的可读性,逐渐成为开发者、技术写作者和内容创作者的首选格式,然而,许多文... 目录引言1. 工具和库介绍2. 安装依赖库3. 使用Apache POI解析DOCX文档4. 将解析

C++使用printf语句实现进制转换的示例代码

《C++使用printf语句实现进制转换的示例代码》在C语言中,printf函数可以直接实现部分进制转换功能,通过格式说明符(formatspecifier)快速输出不同进制的数值,下面给大家分享C+... 目录一、printf 原生支持的进制转换1. 十进制、八进制、十六进制转换2. 显示进制前缀3. 指

在Spring Boot中浅尝内存泄漏的实战记录

《在SpringBoot中浅尝内存泄漏的实战记录》本文给大家分享在SpringBoot中浅尝内存泄漏的实战记录,结合实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录使用静态集合持有对象引用,阻止GC回收关键点:可执行代码:验证:1,运行程序(启动时添加JVM参数限制堆大小):2,访问 htt

MySQL 中查询 VARCHAR 类型 JSON 数据的问题记录

《MySQL中查询VARCHAR类型JSON数据的问题记录》在数据库设计中,有时我们会将JSON数据存储在VARCHAR或TEXT类型字段中,本文将详细介绍如何在MySQL中有效查询存储为V... 目录一、问题背景二、mysql jsON 函数2.1 常用 JSON 函数三、查询示例3.1 基本查询3.2

使用Python实现全能手机虚拟键盘的示例代码

《使用Python实现全能手机虚拟键盘的示例代码》在数字化办公时代,你是否遇到过这样的场景:会议室投影电脑突然键盘失灵、躺在沙发上想远程控制书房电脑、或者需要给长辈远程协助操作?今天我要分享的Pyth... 目录一、项目概述:不止于键盘的远程控制方案1.1 创新价值1.2 技术栈全景二、需求实现步骤一、需求