1*437bfbebSnyanmisaka# MPP 开发参考 2*437bfbebSnyanmisaka 3*437bfbebSnyanmisaka| Project: | MPP | 4*437bfbebSnyanmisaka| --------- | ---------- | 5*437bfbebSnyanmisaka| Version: | 0.7 | 6*437bfbebSnyanmisaka| Author: | 陈恒明 | 7*437bfbebSnyanmisaka| Date: | 10/17/2023 | 8*437bfbebSnyanmisaka 9*437bfbebSnyanmisaka| Revision | Date | Description | Author | 10*437bfbebSnyanmisaka| -------- | ---------- | ------------------------------------------------------- | ----------------------- | 11*437bfbebSnyanmisaka| 0.1 | 04/18/2018 | 初始版本 | 陈恒明 | 12*437bfbebSnyanmisaka| 0.2 | 05/07/2018 | 增加解码器control命令说明,编码器部分说明和demo部分说明 | 陈恒明 | 13*437bfbebSnyanmisaka| 0.3 | 05/22/2018 | 修正一些笔误和说明错误,重新编排页码 | 陈恒明 谢雄斌(精英智通) | 14*437bfbebSnyanmisaka| 0.4 | 11/28/2018 | 更新编码器输入图像的内存排布说明。 修正编码器流程图笔误 | 陈恒明 研果(科通) | 15*437bfbebSnyanmisaka| 0.5 | 06/08/2020 | 更新编码器新配置接口 不再支持RK3188 | 陈恒明 | 16*437bfbebSnyanmisaka| 0.6 | 08/31/2022 | 更新MPP demo说明 更新编码器控制接口说明 | 陈恒明 阮学满 | 17*437bfbebSnyanmisaka| 0.7 | 10/17/2023 | 添加markdown文档 |阮学满 林言东| 18*437bfbebSnyanmisaka| | | | | 19*437bfbebSnyanmisaka| | | | | 20*437bfbebSnyanmisaka 21*437bfbebSnyanmisaka# 第一章 MPP 介绍 22*437bfbebSnyanmisaka 23*437bfbebSnyanmisaka## 1.1 概述 24*437bfbebSnyanmisaka 25*437bfbebSnyanmisaka瑞芯微提供的媒体处理软件平台(Media Process Platform,简称MPP)是适用于瑞芯微芯片系列的通用媒体处理软件平台。该平台对应用软件屏蔽了芯片相关的复杂底层处理,其目的是为了屏蔽不同芯片的差异,为使用者提供统一的视频媒体处理接口(Media Process Interface,缩写MPI)。MPP提供的功能包括: 26*437bfbebSnyanmisaka 27*437bfbebSnyanmisaka- 视频解码 28*437bfbebSnyanmisaka - H.265 / H.264 / H.263 / VP9 / VP8 / MPEG-4 / MPEG-2 / MPEG-1 / VC1 / MJPEG / AV1 29*437bfbebSnyanmisaka- 视频编码 30*437bfbebSnyanmisaka - H.265 / H.264 / VP8 / MJPEG 31*437bfbebSnyanmisaka- 视频处理 32*437bfbebSnyanmisaka - 视频拷贝,缩放,色彩空间转换,场视频解交织(Deinterlace) 33*437bfbebSnyanmisaka 34*437bfbebSnyanmisaka本文档描述了MPP框架以及组成模块,以及供用户使用的MPI接口。本文档适合于上层应用开发人员以及技术支持人员阅读。 35*437bfbebSnyanmisaka 36*437bfbebSnyanmisaka## 1.2 系统架构 37*437bfbebSnyanmisaka 38*437bfbebSnyanmisakaMPP平台在系统架构的层次图如下图: 39*437bfbebSnyanmisaka 40*437bfbebSnyanmisaka 41*437bfbebSnyanmisaka<center>图表 1 MPP系统框架</center> 42*437bfbebSnyanmisaka 43*437bfbebSnyanmisaka- 硬件层Hardware 44*437bfbebSnyanmisaka 45*437bfbebSnyanmisaka硬件层是瑞芯微系列芯片平台的视频编解码硬件加速模块,包括vdpu、vepu、rkvdec和rkvenc等不同类型、不同功能的硬件加速器。 46*437bfbebSnyanmisaka 47*437bfbebSnyanmisaka- 内核驱动层Kernel driver 48*437bfbebSnyanmisaka 49*437bfbebSnyanmisakaLinux内核的编码器硬件设备驱动,以及相关的mmu,内存,时钟,电源管理模块等。支持的平台版本主要是Linux kernel 3.10、4.4、4.19、5.10和6.1。MPP库对于内核驱动有依赖。 50*437bfbebSnyanmisaka 51*437bfbebSnyanmisaka- MPP层 52*437bfbebSnyanmisaka 53*437bfbebSnyanmisaka用户态的MPP层屏蔽了不同操作系统和不同芯片平台的差异,为上层使用者提供统一的MPI接口。 54*437bfbebSnyanmisaka 55*437bfbebSnyanmisakaMPP层包括MPI模块、OSAL模块、HAL模块以及视频编解码器(Video Decoder / Video Encoder)和视频处理功能模块(Video Process)。 56*437bfbebSnyanmisaka 57*437bfbebSnyanmisaka- 操作系统层 58*437bfbebSnyanmisaka 59*437bfbebSnyanmisakaMPP用户态的运行平台,如Android以及Debian等Linux发行版 60*437bfbebSnyanmisaka 61*437bfbebSnyanmisaka- 应用层 62*437bfbebSnyanmisaka 63*437bfbebSnyanmisakaMPP层通过MPI对接各种中间件软件,如OpenMax和gstreamer,或者直接对接客户的上层应用。 64*437bfbebSnyanmisaka 65*437bfbebSnyanmisaka## 1.3 平台支持 66*437bfbebSnyanmisaka 67*437bfbebSnyanmisaka### 1.3.1 软件平台支持 68*437bfbebSnyanmisaka 69*437bfbebSnyanmisakaMPP支持在各种版本的Android平台和纯Linux平台上运行。 70*437bfbebSnyanmisaka 71*437bfbebSnyanmisaka支持瑞芯微Linux内核3.10、4.4、4.19和5.10版本,需要有vcodec_service设备驱动支持以及相应的DTS配置支持。 72*437bfbebSnyanmisaka 73*437bfbebSnyanmisaka### 1.3.2 硬件平台支持 74*437bfbebSnyanmisaka 75*437bfbebSnyanmisaka支持瑞芯微主流的各种系列芯片平台: 76*437bfbebSnyanmisaka 77*437bfbebSnyanmisakaRK3288系列,RK3368系列,RK3399系列,RK3588系列 78*437bfbebSnyanmisaka 79*437bfbebSnyanmisakaRK30xx系列,RK312x系列芯片,RK322x系列芯片,RK332x系列 80*437bfbebSnyanmisaka 81*437bfbebSnyanmisakaRV1109 / RV1126系列(注:RV1107/RV1108会逐步退出支持) 82*437bfbebSnyanmisaka 83*437bfbebSnyanmisaka## 1.4 功能支持 84*437bfbebSnyanmisaka 85*437bfbebSnyanmisakaMPP支持的编解码功能随运行的芯片平台规格不同区别很大,请查询对应芯片的《Multimedia Benchmark》。 86*437bfbebSnyanmisaka 87*437bfbebSnyanmisaka## 1.5 注意事项 88*437bfbebSnyanmisaka 89*437bfbebSnyanmisaka如果想快速了解MPP的使用和demo,请直接转至——第四章 MPP demo说明。 90*437bfbebSnyanmisaka 91*437bfbebSnyanmisaka如果想快速编译和使用MPP代码,请直接转至——第五章 MPP库编译与使用。 92*437bfbebSnyanmisaka 93*437bfbebSnyanmisaka如果想了解MPP设计细节与设计思路,请参考MPP代码根目录的readme.txt,doc目录下的txt文档以及头文件的注释说明。 94*437bfbebSnyanmisaka 95*437bfbebSnyanmisaka# 第二章 接口设计说明 96*437bfbebSnyanmisaka 97*437bfbebSnyanmisaka本章节描述了用户在使用MPP过程中会直接接触到的数据结构,以及这些数据结构的使用说明。 98*437bfbebSnyanmisaka 99*437bfbebSnyanmisaka由于视频编解码与视频处理过程需要处理大量的数据交互,包括码流数据、图像数据以及内存数据,同时还要处理与上层应用以及内核驱动的交叉关系,所以MPP设计了MPI接口,用于与上层交互。 100*437bfbebSnyanmisaka 101*437bfbebSnyanmisaka本章节说明了MPI接口使用的数据结构,以及设计思路。 102*437bfbebSnyanmisaka 103*437bfbebSnyanmisaka## 2.1 接口结构概述 104*437bfbebSnyanmisaka 105*437bfbebSnyanmisaka下图为MPI接口使用的主要数据结构: 106*437bfbebSnyanmisaka 107*437bfbebSnyanmisaka 108*437bfbebSnyanmisaka<center>图表 2 MPI接口使用的数据结构</center> 109*437bfbebSnyanmisaka 110*437bfbebSnyanmisakaMppMem为C库malloc内存的封装。 111*437bfbebSnyanmisaka 112*437bfbebSnyanmisakaMppBuffer为硬件用的dmabuf内存的封装。 113*437bfbebSnyanmisaka 114*437bfbebSnyanmisakaMppPacket为一维缓存封装,可以从MppMem和MppBuffer生成,主要用于表示码流数据。 115*437bfbebSnyanmisaka 116*437bfbebSnyanmisakaMppFrame为二维帧数据封装,可以从MppMem和MppBuffer生成,主要用于表示图像数据。 117*437bfbebSnyanmisaka 118*437bfbebSnyanmisaka使用MppPacket和MppFrame就可以简单有效的完成一般的视频编解码工作。 119*437bfbebSnyanmisaka 120*437bfbebSnyanmisaka以视频解码为例,码流输入端把地址和大小赋值给MppPacket,通过put_packet接口输入,在输出端通过get_frame接口得到输入图像MppFrame,即可完成最简单的视频解码过程。 121*437bfbebSnyanmisaka 122*437bfbebSnyanmisaka 123*437bfbebSnyanmisaka<center>图表 3 使用简单接口实现视频解码</center> 124*437bfbebSnyanmisaka 125*437bfbebSnyanmisakaMppMeta和MppTask为输入输出用任务的高级组合接口,可以支持指定输入输出方式等复杂使用方式,支持异步数据流处理。 126*437bfbebSnyanmisaka 127*437bfbebSnyanmisaka**注意**:以上这些接口数据结构都是使用void\*句柄来引用使用,其目的是为了方便扩展和前向兼容。 128*437bfbebSnyanmisaka 129*437bfbebSnyanmisaka本段落中的提到的成员都是通过形如mpp_xxx_set/get_xxx的接口来访问。 130*437bfbebSnyanmisaka 131*437bfbebSnyanmisaka## 2.2 内存封装MppBuffer 132*437bfbebSnyanmisaka 133*437bfbebSnyanmisakaMppBuffer主要用于描述供硬件使用的内存块(即缓存),提供了内存块的分配、释放和加减引用等功能,目前支持的分配器有:ion、drm和dma_heap。MppBuffer几个重要的参数成员如下: 134*437bfbebSnyanmisaka 135*437bfbebSnyanmisaka| 成员名称 | 成员类型 | 描述说明 | 136*437bfbebSnyanmisaka|----------|----------|----------------------------------| 137*437bfbebSnyanmisaka| ptr | void \* | 表示内存块的起始虚拟地址。 | 138*437bfbebSnyanmisaka| size | size_t | 表示内存块的大小。 | 139*437bfbebSnyanmisaka| fd | int | 表示内存块的用户态空间文件句柄。 | 140*437bfbebSnyanmisaka 141*437bfbebSnyanmisaka在解码过程中,解码图像的缓存通常需要在固定的缓存池里进行轮转,为了实现这一点,MPP在MppBuffer基础之上又定义了MppBufferGroup。MppBuffer的使用方式有两种:常规使用方式和外部导入方式,由于二者的内存分配方式不同,也称为internal模式和external模式。 142*437bfbebSnyanmisaka 143*437bfbebSnyanmisaka在MppBuffer常规使用方式下,MppBufferGroup由MPP内部生成和维护。通过mpp_buffer_get和mpp_buffer_put对内存块进行申请和释放,如下图所示: 144*437bfbebSnyanmisaka 145*437bfbebSnyanmisaka 146*437bfbebSnyanmisaka<center>图表 4 MppBuffer的常规使用方式</center> 147*437bfbebSnyanmisaka 148*437bfbebSnyanmisaka其流程伪代码如下图: 149*437bfbebSnyanmisaka 150*437bfbebSnyanmisaka 151*437bfbebSnyanmisaka 152*437bfbebSnyanmisaka这种方式可以实现解码器在解码过程中**零拷贝输出**(解码器的输出帧与解码器内部使用的参考帧用是同一帧),但不容易实现**零拷贝显示**(解码器的输出帧不一定在显示端可以直接显示),同时要求用户知道需要给解码器开多大的空间进行解码。 153*437bfbebSnyanmisaka 154*437bfbebSnyanmisaka另一种使用方式是把MppBufferGroup完全做为一个缓存的管理器,用于管理外部导入的缓存。其使用方式如下图: 155*437bfbebSnyanmisaka 156*437bfbebSnyanmisaka 157*437bfbebSnyanmisaka<center>图表 5 MppBuffer外部导入使用方式</center> 158*437bfbebSnyanmisaka 159*437bfbebSnyanmisaka其流程伪代码如下图: 160*437bfbebSnyanmisaka 161*437bfbebSnyanmisaka 162*437bfbebSnyanmisaka 163*437bfbebSnyanmisaka这种方式可以使得解码器使用外部的缓存,可以对接OpenMax/gstreamer这样的中间件,也方便对接用户己有的上层代码,便于实现零拷贝显示。 164*437bfbebSnyanmisaka 165*437bfbebSnyanmisaka## 2.3 码流封装MppPacket 166*437bfbebSnyanmisaka 167*437bfbebSnyanmisakaMppPacket主要用于描述一维码流的相关信息,特别是有效数据的位置与长度。MppPacket几个重要的参数成员如下: 168*437bfbebSnyanmisaka 169*437bfbebSnyanmisaka| 成员名称 | 成员类型 | 描述说明 | 170*437bfbebSnyanmisaka|----------|----------|----------------------------------------------------------------------------------------------------| 171*437bfbebSnyanmisaka| data | void \* | 表示缓存空间的起始地址。 | 172*437bfbebSnyanmisaka| size | size_t | 表示缓存空间的大小。 | 173*437bfbebSnyanmisaka| pos | void \* | 表示缓存空间内有效数据的起始地址。 | 174*437bfbebSnyanmisaka| length | size_t | 表示缓存空间内有效数据的长度。如果在decode_put_packet调用之后length变为0,说明此包码流己消耗完成。 | 175*437bfbebSnyanmisaka 176*437bfbebSnyanmisaka其关系如下图所示: 177*437bfbebSnyanmisaka 178*437bfbebSnyanmisaka 179*437bfbebSnyanmisaka 180*437bfbebSnyanmisaka<center>图表 6 MppPacket重要参数说明</center> 181*437bfbebSnyanmisaka 182*437bfbebSnyanmisakaMppPacket的其他配置参数成员如下: 183*437bfbebSnyanmisaka 184*437bfbebSnyanmisaka| 成员名称 | 成员类型 | 描述说明 | 185*437bfbebSnyanmisaka| -------- | --------- | ------------------------------------------------------------ | 186*437bfbebSnyanmisaka| pts | RK_U64 | 表示显示时间戳(Present Time Stamp)。 | 187*437bfbebSnyanmisaka| dts | RK_U64 | 表示解码时间戳(Decoding Time Stamp)。 | 188*437bfbebSnyanmisaka| eos | RK_U32 | 表示码流结束标志(End Of Stream)。 | 189*437bfbebSnyanmisaka| buffer | MppBuffer | 表示MppPacket对应的MppBuffer。 | 190*437bfbebSnyanmisaka| flag | RK_U32 | 表示MPP内部使用的标志位,包含如下标志: <br/>\#define MPP_PACKET_FLAG_EOS (0x00000001) <br/>\#define MPP_PACKET_FLAG_EXTRA_DATA (0x00000002) <br/>\#define MPP_PACKET_FLAG_INTERNAL (0x00000004) <br/>\#define MPP_PACKET_FLAG_INTRA (0x00000008) | 191*437bfbebSnyanmisaka 192*437bfbebSnyanmisakaMppPacket做为描述一维内存的结构体,在使用时需要使用malloc出来的内存或者使用MppBuffer的内存进行初始化。在释放MppPacket时有几种情况: 193*437bfbebSnyanmisaka 194*437bfbebSnyanmisaka如果是外部malloc地址配置到MppPacket,不会做free释放处理,如下示例; 195*437bfbebSnyanmisaka 196*437bfbebSnyanmisaka 197*437bfbebSnyanmisaka 198*437bfbebSnyanmisaka如果是拷贝产生的MppPacket,会做free释放内存,如下示例; 199*437bfbebSnyanmisaka 200*437bfbebSnyanmisaka 201*437bfbebSnyanmisaka 202*437bfbebSnyanmisaka如果是MppBuffer产生的MppPacket,会在生成时对MppBuffer加引用,在释放时对MppPacket减引用。 203*437bfbebSnyanmisaka 204*437bfbebSnyanmisaka 205*437bfbebSnyanmisaka 206*437bfbebSnyanmisaka## 2.4 图像封装MppFrame 207*437bfbebSnyanmisaka 208*437bfbebSnyanmisakaMppFrame主要用于定义二维图像缓存的相关信息,有效数据的位置与长度。MppFrame几个重要的参数成员如下: 209*437bfbebSnyanmisaka 210*437bfbebSnyanmisaka| 成员名称 | 成员类型 | 描述说明 | 211*437bfbebSnyanmisaka|------------|----------|------------------------------------------------| 212*437bfbebSnyanmisaka| width | RK_U32 | 表示水平方向像素数,单位为像素个数。 | 213*437bfbebSnyanmisaka| height | RK_U32 | 表示垂直方向像素数,单位为像素个数。 | 214*437bfbebSnyanmisaka| hor_stride | RK_U32 | 表示垂直方向相邻两行之间的距离,单位为byte数。 | 215*437bfbebSnyanmisaka| ver_stride | RK_U32 | 表示图像分量之间的以行数间隔数,单位为1。 | 216*437bfbebSnyanmisaka 217*437bfbebSnyanmisaka 218*437bfbebSnyanmisaka<center>图表 7 MppFrame重要参数说明</center> 219*437bfbebSnyanmisaka 220*437bfbebSnyanmisakaMppFrame的其他配置参数成员如下: 221*437bfbebSnyanmisaka 222*437bfbebSnyanmisaka| 成员名称 | 成员类型 | 描述说明 | 223*437bfbebSnyanmisaka|-------------|--------------------|-----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| 224*437bfbebSnyanmisaka| mode | RK_U32 | 表示图像数据帧场属性:  | 225*437bfbebSnyanmisaka| pts | RK_U64 | 表示图像的显示时间戳(Present Time Stamp)。 | 226*437bfbebSnyanmisaka| dts | RK_U64 | 表示图像的解码时间戳(Decoding Time Stamp)。 | 227*437bfbebSnyanmisaka| eos | RK_U32 | 表示图像的结束标志(End Of Stream)。 | 228*437bfbebSnyanmisaka| errinfo | RK_U32 | 表示图像的错误标志,是否图像内有解码错误。 | 229*437bfbebSnyanmisaka| discard | RK_U32 | 表示图像的丢弃标志,如果图像解码时的参考关系不满足要求,则这帧图像会被标记为需要丢弃,不被显示。 | 230*437bfbebSnyanmisaka| buf_size | size_t | 表示图像需要分配的缓存大小,与图像的格式相关,也与解码数据的格式相关。 | 231*437bfbebSnyanmisaka| info_change | RK_U32 | 如果为真,表示当前MppFrame是一个用于标记码流信息变化的描述结构,说明了新的宽高,stride,以及图像格式。 可能的info_change原因有: 图像序列宽高变化。 图像序列格式变化,如8bit变为10bit。 一旦info_change产生,需要重新分析解码器使用的内存池。 | 232*437bfbebSnyanmisaka| fmt | MppFrameFormat | 表示图像色彩空间格式以及内存排布方式:  | 233*437bfbebSnyanmisaka| color_range | MppFrameColorRange | 表示图像数据彩色空间范围: YUV full range:0 \~ 255(8bit) YUV limit range:16 \~ 235(8bit)  | 234*437bfbebSnyanmisaka| buffer | MppBuffer | 表示MppFrame对应的MppBuffer。 | 235*437bfbebSnyanmisaka 236*437bfbebSnyanmisaka对于解码器来说,MppFrame是其输出的信息结构体,码流解码后的信息(包括像素数据与pts,错误信息等相关信息)都需要带在MppFrame结构体给调用者。MppFrame中的pts/dts,以及eos标志,就是继承自对应的输入MppPacket。 237*437bfbebSnyanmisaka 238*437bfbebSnyanmisaka同时,一旦发现码流分辨率改变,MppFrame中的info_change标志就会对应置位,向用户通知发生了info_change事件,需要用户进行缓存池修改处理。 239*437bfbebSnyanmisaka 240*437bfbebSnyanmisaka## 2.5 高级任务封装MppTask 241*437bfbebSnyanmisaka 242*437bfbebSnyanmisaka当MppPacket与MppFrame组成的接口无法满足需求时,需要使用MppTask做为一个数据容器,来满足复杂的输入输出需求。MppTask需要与poll/dequeuer/enqueue接口来配合使用,对比put_packet/get_frame等简单流程接口,MppTask的使用流程复杂,效率低,是为了满足复杂需求的代价。 243*437bfbebSnyanmisaka 244*437bfbebSnyanmisaka 245*437bfbebSnyanmisaka<center>图表 8 使用MppTask来进行输入输出</center> 246*437bfbebSnyanmisaka 247*437bfbebSnyanmisakaMppTask是一个通过关键字key值(MppMetaKey)来进行扩展的结构,可以通过扩展支持的数据类型来支持复杂的高级需求。可以使用通过mpp_task_meta_set/get_xxx系列接口来对MppTask里的不同关键字数据进行访问。 248*437bfbebSnyanmisaka 249*437bfbebSnyanmisaka 250*437bfbebSnyanmisaka<center>图表 9 MppTask支持的数据类型与关键字类型</center> 251*437bfbebSnyanmisaka 252*437bfbebSnyanmisaka在实际使用中,需要从MPP的输入端口通过dequeue接口获取MppTask,把需要处理的数据通过mpp_task_meta_set_xxx系列接口配置到MppTask里,之后enqueue输出到MPP实例进行处理。MPP的输出端口流程类似,需要把mpp_task_meta_set_xxx系列接口换成mpp_task_meta_get_xxx系列接口来从MppTask里获取数据。 253*437bfbebSnyanmisaka 254*437bfbebSnyanmisaka目前实用的编码器接口,以及MJPEG解码接口有使用MppTask进行实现。 255*437bfbebSnyanmisaka 256*437bfbebSnyanmisaka## 2.6 实例上下文封装MppCtx 257*437bfbebSnyanmisaka 258*437bfbebSnyanmisakaMppCtx是提供给用户使用的MPP实例上下文句柄,用于指代解码器或编码器实例。 259*437bfbebSnyanmisaka 260*437bfbebSnyanmisaka用户可以通过mpp_create接口获取MppCtx实例以及MppApi结构体,再通过mpp_init来初始化MppCtx的编解码类型与格式,之后通过decode_xxx/encode_xxx以及poll/dequeuer/enqueue接口来进行访问,使用结束时通过mpp_destroy接口进行销毁。 261*437bfbebSnyanmisaka 262*437bfbebSnyanmisaka 263*437bfbebSnyanmisaka<center>图表 10 MppCtx使用过程</center> 264*437bfbebSnyanmisaka 265*437bfbebSnyanmisaka## 2.7 API封装MppApi(MPI) 266*437bfbebSnyanmisaka 267*437bfbebSnyanmisakaMppApi结构体封装了MPP的对外API接口,用户通过使用MppApi结构中提供的函数指针实现视频编解码功能,其结构如下: 268*437bfbebSnyanmisaka 269*437bfbebSnyanmisaka| 成员名称 | 成员类型 | 描述 | 270*437bfbebSnyanmisaka| ----------------- | -------- | ------------------------------------------------------------ | 271*437bfbebSnyanmisaka| size | RK_U32 | MppApi结构体大小。 | 272*437bfbebSnyanmisaka| version | RK_U32 | MppApi结构体版本。 | 273*437bfbebSnyanmisaka| decode | 函数指针 | MPP_RET (\*decode)(MppCtx ctx, MppPacket packet, MppFrame \*frame) <br/>视频解码接口,同时进行输入与输出,单独使用。<br>ctx :MPP实例上下文。 <br/>packet :输入码流。<br/>frame :输出图像。 <br/>返回值 :0为正常,非零为错误码。 | 274*437bfbebSnyanmisaka| decode_put_packet | 函数指针 | MPP_RET (\*decode_put_packet)(MppCtx ctx, MppPacket packet) <br/>视频解码输入接口,与decode_get_frame配合使用。 <br/>ctx :MPP实例上下文。 <br/>packet :输入码流。 <br/>返回值 :0为正常,表示码流己被MPP处理;非零为出现错误,码流未被处理,需要把码流重新输入。 | 275*437bfbebSnyanmisaka| decode_get_frame | 函数指针 | MPP_RET (\*decode_get_frame)(MppCtx ctx, MppFrame \*frame) <br/>视频解码输出接口,与decode_put_packet配合使用。 <br/>ctx :MPP实例上下文。 <br/>frame :输出图像。 <br/>返回值 :0为正常,表示获取输出过程正常,需要判断是否有得到有值的frame指针;非零为出现错误。 | 276*437bfbebSnyanmisaka| encode | 函数指针 | MPP_RET (\*encode)(MppCtx ctx, MppFrame frame, MppPacket \*packet) <br/>视频编码接口,同时进行输入与输出,单独使用。 <br/>ctx :MPP实例上下文。 <br/>frame :输入图像。 <br/>packet :输出码流。 <br/>返回值 :0为正常,非零为错误码。 | 277*437bfbebSnyanmisaka| encode_put_frame | 函数指针 | MPP_RET (\*encode_put_frame)(MppCtx ctx, MppFrame frame) <br/>视频编码输入接口,与encode_get_packet配合使用。 <br/>ctx :MPP实例上下文。 <br/>frame :输入图像。 <br/>返回值 :0为正常,非零为错误码。 | 278*437bfbebSnyanmisaka| encode_get_packet | 函数指针 | MPP_RET (\*encode_get_packet)(MppCtx ctx, MppPacket \*packet) <br/>视频编码输出接口,与encode_put_frame配合使用。 <br/>ctx :MPP实例上下文。 <br/>packet :输出码流。 <br/>返回值 :0为正常,非零为错误码。 | 279*437bfbebSnyanmisaka| poll | 函数指针 | MPP_RET (\*poll)(MppCtx ctx, MppPortType type, MppPollType timeout) <br/>端口查询接口,用于查询端口是否有数据可供dequeue。 <br/>ctx :MPP实例上下文。 <br/>type :端口类型,分为输入端口与输出端口。 <br/>timeout :查询超时参数,-1为阻塞查询,0为非阻塞查询,正值为超时毫秒数。 <br/>返回值 :0为正常,有数据可取出,非零为错误码。 | 280*437bfbebSnyanmisaka| dequeue | 函数指针 | MPP_RET (\*dequeue)(MppCtx ctx, MppPortType type, MppTask \*task) <br/>端口出队列接口,用于从端口中取出MppTask结构。 <br/>ctx :MPP实例上下文。 <br/>type :端口类型,分为输入端口与输出端口。 <br/>task :MppTask。 <br/>返回值 :0为正常,有数据可取出,非零为错误码。 | 281*437bfbebSnyanmisaka| enqueue | 函数指针 | MPP_RET (\*enqueue)(MppCtx ctx, MppPortType type, MppTask task) <br/>端口入队列接口,用于往端口送入MppTask结构。 <br/>ctx :MPP实例上下文。 <br/>type :端口类型,分为输入端口与输出端口。 <br/>task :MppTask。 <br/>返回值 :0为正常,有数据可取出,非零为错误码。 | 282*437bfbebSnyanmisaka| reset | 函数指针 | MPP_RET (\*reset)(MppCtx ctx) <br/>复位接口,用于对MppCtx的内部状态进行复位,回到可用的初始化状态。需要注意的是,reset接口是阻塞的同步接口。 <br/>ctx :MPP实例上下文。 <br/>返回值 :0为正常,有数据可取出,非零为错误码。 | 283*437bfbebSnyanmisaka| control | 函数指针 | MPP_RET (\*control)(MppCtx ctx, MpiCmd cmd, MppParam param) <br/>控制接口,用于向MPP实例进行额外控制操作的接口。 <br/>ctx :MPP实例上下文。 <br/>cmd :Mpi命令类型,表示控制命令的不同类型的。 <br/>param :Mpi命令参数,表示控制控制命令的附加参数。 <br/>返回值 :0为正常,有数据可取出,非零为错误码。 | 284*437bfbebSnyanmisaka 285*437bfbebSnyanmisaka# 第三章 MPI接口使用说明 286*437bfbebSnyanmisaka 287*437bfbebSnyanmisaka本章节描述了用户使用MPI接口的具体过程,以及过程是的一些注意事项。 288*437bfbebSnyanmisaka 289*437bfbebSnyanmisakaMPI(Media Process Interface)是MPP提供给用户的接口,用于提供硬件编解码功能,以及一些必要的相关功能。MPI是通过C结构里的函数指针方式提供给用户,用户可以通过MPP上下文结构MppCtx与MPI接口结构MppApi组合使用来实现解码器与编码器的功能。 290*437bfbebSnyanmisaka 291*437bfbebSnyanmisaka 292*437bfbebSnyanmisaka<center>图表 11 MPI接口范围</center> 293*437bfbebSnyanmisaka 294*437bfbebSnyanmisaka如上图所示,mpp_create,mpp_init,mpp_destroy是操作MppCtx接口的过程,其中mpp_create接口也获取到了MPI接口结构体MppApi,真正的编码与解码过程是通过调用MppApi结构体里内的函数指针来实现,也就是上图中红框内的部分。红框内的函数调用分为编解码流程接口put/get_packet/frame和相关的control和reset接口。下文先描述编解码器接口,再对编解码器工作中的一些要点进行说明。 295*437bfbebSnyanmisaka 296*437bfbebSnyanmisaka## 3.1 解码器数据流接口 297*437bfbebSnyanmisaka 298*437bfbebSnyanmisaka解码器接口为用户提供了输入码流,输出图像的功能,接口函数为MppApi结构体里的decode_put_packet函数,decode_get_frame函数和decode函数。这组函数提供了最简洁的解码功能支持。 299*437bfbebSnyanmisaka 300*437bfbebSnyanmisaka### 3.1.1 decode_put_packet 301*437bfbebSnyanmisaka 302*437bfbebSnyanmisaka| 接口定义 | MPP_RET decode_put_packet(MppCtx ctx, MppPacket packet) | 303*437bfbebSnyanmisaka|----------|---------------------------------------------------------| 304*437bfbebSnyanmisaka| 输入参数 | ctx :MPP解码器实例 packet :待输入的码流数据 | 305*437bfbebSnyanmisaka| 返回参数 | 运行错误码 | 306*437bfbebSnyanmisaka| 功能 | 输入packet码流包数据给ctx指定的MPP解码器实例 | 307*437bfbebSnyanmisaka 308*437bfbebSnyanmisaka#### 输入码流的形式 309*437bfbebSnyanmisaka 310*437bfbebSnyanmisakaMPP的输入都是没有封装信息的裸码流,裸码流输入有两种形式: 311*437bfbebSnyanmisaka 312*437bfbebSnyanmisaka1. 外部分帧 313*437bfbebSnyanmisaka 314*437bfbebSnyanmisaka一种是外部已经按帧分段的数据,即每一包输入给decode_put_packet函数的MppPacket数据都已经包含完整的一帧,不多也不少。在这种情况下,MPP可以直接按包处理码流,是MPP的默认运行情况。 315*437bfbebSnyanmisaka 316*437bfbebSnyanmisaka2. 内部分帧 317*437bfbebSnyanmisaka 318*437bfbebSnyanmisaka另一种是按长度读取的数据,这样的数据无法判断一包MppPacket数据是否是完整的一帧,需要MPP内部进行分帧处理。MPP也可以支持这种形式的输入,但需要在mpp_init之前,通过control接口的MPP_DEC_SET_PARSER_SPLIT_MODE命令,MPP内的need_split标志打开。 319*437bfbebSnyanmisaka 320*437bfbebSnyanmisaka 321*437bfbebSnyanmisaka 322*437bfbebSnyanmisaka这样,调用decode_put_packet输入的MppPacket就会被MPP重新分帧,进入到情况一的处理。 323*437bfbebSnyanmisaka 324*437bfbebSnyanmisaka如果这两种情况出现了混用,会出现码流解码出错的问题。 325*437bfbebSnyanmisaka 326*437bfbebSnyanmisaka分帧方式处理效率高,但需要输入码流之前先进行解析与分帧; 327*437bfbebSnyanmisaka 328*437bfbebSnyanmisaka不分帧方式使用简单,但效率会受影响。 329*437bfbebSnyanmisaka 330*437bfbebSnyanmisaka在mpi_dec_test的测试用例中,使用的是方式内部分帧的方式。在瑞芯微的Android SDK中,使用的是外部分帧的方式。用户可以根据自己的应用场景和平台条件进行选择。 331*437bfbebSnyanmisaka 332*437bfbebSnyanmisaka#### 输入码流的消耗 333*437bfbebSnyanmisaka 334*437bfbebSnyanmisaka输入MppPacket的有效数据长度为length,在送入decode_put_packet之后,如果输入码流被成功地消耗,函数返回值为零(MPP_OK),同时MppPacket的length被清为0。如果输入码流还没有被处理,会返回非零错误码,MppPacket的length保持不变。 335*437bfbebSnyanmisaka 336*437bfbebSnyanmisaka#### 函数的工作模式 337*437bfbebSnyanmisaka 338*437bfbebSnyanmisakadecode_put_packet函数的功能是输入待解码码流给MPP实例,但在一些情况下,MPP实例无法接收更多的数据,这时,工作于非阻塞模式的decode_put_packet会报出错误信息并直接返回。用户得到decode_put_packet返回的错误码之后,需要进行一定时间的等待,再重新送入码流数据,避免额外的频繁cpu开销。 339*437bfbebSnyanmisaka 340*437bfbebSnyanmisaka#### 最大缓冲数据包数量 341*437bfbebSnyanmisaka 342*437bfbebSnyanmisakaMPP实例默认可以接收4个输入码流包在处理队列中,如果码流送得太快,就会报出错误码要求用户等待后再送。 343*437bfbebSnyanmisaka 344*437bfbebSnyanmisaka### 3.1.2 decode_get_frame 345*437bfbebSnyanmisaka 346*437bfbebSnyanmisaka| 接口定义 | MPP_RET decode_get_frame(MppCtx ctx, MppFrame \*frame) | 347*437bfbebSnyanmisaka|----------|------------------------------------------------------------| 348*437bfbebSnyanmisaka| 输入参数 | ctx :MPP解码器实例。 frame :用于获取MppFrame实例的指针。 | 349*437bfbebSnyanmisaka| 返回参数 | 运行错误码 | 350*437bfbebSnyanmisaka| 功能 | 从ctx指定的MPP解码器实例里获取完成解码的frame描述信息。 | 351*437bfbebSnyanmisaka 352*437bfbebSnyanmisakaMPP解码输出的图像是通过MppFrame结构来描述的,同时MppFrame结构也是MPP实例输出信息的管道,图像的错误信息,以及变宽高信息(info change)也是带在MppFrame结构进行输出的。 353*437bfbebSnyanmisaka 354*437bfbebSnyanmisaka- 输出图像的错误信息 355*437bfbebSnyanmisaka 356*437bfbebSnyanmisaka图像的错误信息为errinfo,表示图像内容是否有错误,errinfo不为零则表示码流解码时发生了错误,图像内容是有问题的,可以做丢弃处理。 357*437bfbebSnyanmisaka 358*437bfbebSnyanmisaka- 解码器图像空间需求的确认 359*437bfbebSnyanmisaka 360*437bfbebSnyanmisaka解码器在解码时,需要为输出图像获取保存像素数据的内存空间,用户需要给解码器提供足够大小,这个空间大小的需求,会在MPP解码器内部根据不同的芯片平台以及不同的视频格式需求进行计算,计算后的内存空间需求会通过MppFrame的成员变量buf_size提供给用户。用户需要按buf_size的大小进行内存分配,即可满足解码器的要求。 361*437bfbebSnyanmisaka 362*437bfbebSnyanmisaka- 输出图像的变宽高信息(Info change) 363*437bfbebSnyanmisaka 364*437bfbebSnyanmisaka当码流的宽高,格式,像素位深等信息发生变化时,需要反馈给用户,用户需要更新解码器使用的内存池,把新的内存更新给解码器。这里涉及到解码内存分配与使用模式,会在3.3.2 图像内存分配以及交互模式进行说明。 365*437bfbebSnyanmisaka 366*437bfbebSnyanmisaka### 3.1.3 decode 367*437bfbebSnyanmisaka 368*437bfbebSnyanmisakadecode函数是decode_put_packet与decode_get_frame数据的结合,为用户提供了两个函数的复合调用。其内部逻辑为: 369*437bfbebSnyanmisaka 370*437bfbebSnyanmisaka1. 获取输出图像; 371*437bfbebSnyanmisaka2. 如果输出图像获取成功即返回; 372*437bfbebSnyanmisaka3. 判断码流己送入成功则返回; 373*437bfbebSnyanmisaka4. 送入输入码流; 374*437bfbebSnyanmisaka5. 标记码流送入是否成功并循环第一步; 375*437bfbebSnyanmisaka 376*437bfbebSnyanmisaka在用户看来,decode函数首先是获取解码图像,有解码图像优先返回解码图像,没有可输出的解码图像的情况下送入码流,最后再尝试一次获取解码图像并退出。 377*437bfbebSnyanmisaka 378*437bfbebSnyanmisaka## 3.2 解码器控制接口 379*437bfbebSnyanmisaka 380*437bfbebSnyanmisaka### 3.2.1 control 381*437bfbebSnyanmisaka 382*437bfbebSnyanmisaka在定义于rk_mpi_cmd.h文件的MpiCmd枚举类型定义了control接口命令字,其中与解码器和解码过程相关的命令如下: 383*437bfbebSnyanmisaka 384*437bfbebSnyanmisaka 385*437bfbebSnyanmisaka 386*437bfbebSnyanmisaka从MPP_DEC_CMD_BASE到MPP_DEC_CMD_END之间的命令为解码器的control接口命令,分别介绍这些命令的功能如下: 387*437bfbebSnyanmisaka 388*437bfbebSnyanmisaka- MPP_DEC_SET_FRAME_INFO 389*437bfbebSnyanmisaka 390*437bfbebSnyanmisaka命令参数为MppFrame,用于配置解码器的默认宽高信息,返回的MppFrame结构会从解码器中带出需要分配的图像缓存大小。命令调用时机一般在mpp_init之后,mpi-\>decode_put_packet之前。 391*437bfbebSnyanmisaka 392*437bfbebSnyanmisaka- MPP_DEC_SET_EXT_BUF_GROUP 393*437bfbebSnyanmisaka 394*437bfbebSnyanmisaka命令参数为MppBufferGroup,用于把解码器图像解码所需的MppBufferGroup配置给解码器。命令调用时机视图像内存分配模式有不同。 395*437bfbebSnyanmisaka 396*437bfbebSnyanmisaka- MPP_DEC_SET_INFO_CHANGE_READY 397*437bfbebSnyanmisaka 398*437bfbebSnyanmisaka无命令参数,用于标记解码器使用的MppBufferGroup已经完成Info Change操作的reset处理,可以继续解码。命令调用时机视图像内存分配模式有不同。 399*437bfbebSnyanmisaka 400*437bfbebSnyanmisaka- MPP_DEC_SET_PRESENT_TIME_ORDER 401*437bfbebSnyanmisaka 402*437bfbebSnyanmisaka命令参数为RK_U32\*,用于处理异常的码流时间戳。 403*437bfbebSnyanmisaka 404*437bfbebSnyanmisaka- MPP_DEC_SET_PARSER_SPLIT_MODE 405*437bfbebSnyanmisaka 406*437bfbebSnyanmisaka命令参数为RK_U32\*,用于使能MPP内的协议解析器使用内部分帧处理,默认为码流按帧输入,不开启。命令调用时机是在- MPP_init之前。 407*437bfbebSnyanmisaka 408*437bfbebSnyanmisaka- MPP_DEC_SET_PARSER_FAST_MODE 409*437bfbebSnyanmisaka 410*437bfbebSnyanmisaka命令参数为RK_U32\*,用于使能MPP内的快速帧解析,提升解码的软硬件并行度,但副作用是会对错误码流的标志有影响,默认关闭。命令调用时机是在- MPP_init之前。 411*437bfbebSnyanmisaka 412*437bfbebSnyanmisaka- MPP_DEC_GET_STREAM_COUT 413*437bfbebSnyanmisaka 414*437bfbebSnyanmisaka命令参数为RK_U32\*,用于外部应用获取还未处理的码流包数量,历史遗留接口。 415*437bfbebSnyanmisaka 416*437bfbebSnyanmisaka- MPP_DEC_GET_VPUMEM_USED_COUT 417*437bfbebSnyanmisaka 418*437bfbebSnyanmisaka命令参数为RK_U32\*,用于外部应用获取MPP使用的MppBuffer数量,历史遗留接口。 419*437bfbebSnyanmisaka 420*437bfbebSnyanmisaka- MPP_DEC_SET_VC1_EXTRA_DATA 421*437bfbebSnyanmisaka 422*437bfbebSnyanmisaka暂未实现,历史遗留接口。 423*437bfbebSnyanmisaka 424*437bfbebSnyanmisaka- MPP_DEC_SET_OUTPUT_FORMAT 425*437bfbebSnyanmisaka 426*437bfbebSnyanmisaka命令参数为MppFrameFormat,用于外部应用配置JPEG解码器的输出格式,默认不使用。 427*437bfbebSnyanmisaka 428*437bfbebSnyanmisaka- MPP_DEC_SET_DISABLE_ERROR 429*437bfbebSnyanmisaka 430*437bfbebSnyanmisaka命令参数为RK_U32\*,用于关闭MPP解码器的错误处理。一旦使能,MPP解码会无视码流的错误情况,输出全部的可解码图像,同时不对输出的MppFrame结构里的errinfo进行标记。命令调用时机在decode_put_packet之前。 431*437bfbebSnyanmisaka 432*437bfbebSnyanmisaka- MPP_DEC_SET_IMMEDIATE_OUT 433*437bfbebSnyanmisaka 434*437bfbebSnyanmisaka命令参数为RK_U32\*,用于使能H.264解码器的立即输出模式。一旦使能,H.264解码器会忽略丢帧导致的帧序不连续情况,立即输出解码的图像。命令调用时机在decode_put_packet之前。 435*437bfbebSnyanmisaka 436*437bfbebSnyanmisaka- MPP_DEC_SET_ENABLE_FAST_PLAY 437*437bfbebSnyanmisaka 438*437bfbebSnyanmisaka命令参数为FastPlayMode\*,用于使能H.264解码器的快速起播,不必等到dpb full之后出帧。如果当前帧在I帧之后解码,但在I帧之前显示,可能会出现丢帧的情况。命令调用时机在decode_put_packet之前。 439*437bfbebSnyanmisaka 440*437bfbebSnyanmisaka### 3.2.2 reset 441*437bfbebSnyanmisaka 442*437bfbebSnyanmisakareset接口用于把解码器恢复为正常初始化后的状态。 443*437bfbebSnyanmisaka 444*437bfbebSnyanmisaka当用户发送最后一包MppPacket码流,并置上EOS标记送入解码器,解码器在处理完这最后一包数据之后会进入EOS的状态,不再接收和处理码流,需要reset之后再才能再继续接收新的码流。 445*437bfbebSnyanmisaka 446*437bfbebSnyanmisaka## 3.3 解码器使用要点 447*437bfbebSnyanmisaka 448*437bfbebSnyanmisaka解码器在使用过程中,需要注意的一些重要事项: 449*437bfbebSnyanmisaka 450*437bfbebSnyanmisaka### 3.3.1 解码器单/多线程使用方式 451*437bfbebSnyanmisaka 452*437bfbebSnyanmisakaMPP解码器的MPI接口是线程安全的,可以在多线环境下使用。单线程工作模式如mpi_dec_test的demo所示,多线程工作模式如mpi_dec_mt_test的demo所示。 453*437bfbebSnyanmisaka 454*437bfbebSnyanmisaka 455*437bfbebSnyanmisaka 456*437bfbebSnyanmisaka 457*437bfbebSnyanmisaka 458*437bfbebSnyanmisaka<center>图表 12 解码器单线程与多线程使用方式</center> 459*437bfbebSnyanmisaka 460*437bfbebSnyanmisaka### 3.3.2 图像内存分配以及交互模式 461*437bfbebSnyanmisaka 462*437bfbebSnyanmisaka解码器在解码图像时,需要获取内存空间以写入数据,在解码完成之后,这块内存空间需要交给用户使用,在用户使用完成之后要释放给解码器,在关闭解码器时要释放全部内存空间。在这种工作方式下,解码器与用户之间才可以形成零拷贝的数据交互。MPP解码器支持三种内存分配以及与用户交互图像数据的模式: 463*437bfbebSnyanmisaka 464*437bfbebSnyanmisaka#### 模式一:纯内部分配模式 465*437bfbebSnyanmisaka 466*437bfbebSnyanmisaka图像内存直接从MPP解码器内部分配,内存由解码器直接分配,用户得到解码器输出图像,在使用完成之后直接释放。 467*437bfbebSnyanmisaka 468*437bfbebSnyanmisaka 469*437bfbebSnyanmisaka<center>图表 13 解码器图像内存纯内部分配模式示意图</center> 470*437bfbebSnyanmisaka 471*437bfbebSnyanmisaka在这种方式下,用户不需要调用解码器control接口的MPP_DEC_SET_EXT_BUF_GROUP命令,只需要在解码器上报info change时直接调用control接口的MPP_DEC_SET_INFO_CHANGE_READY命令即可。解码器会自动在内部进行内存分配,用户需要把获取到的每帧数据直接释放。 472*437bfbebSnyanmisaka 473*437bfbebSnyanmisaka 474*437bfbebSnyanmisaka<center>图表 14 解码器图像内存纯内部分配模式代码流程</center> 475*437bfbebSnyanmisaka 476*437bfbebSnyanmisaka**优点:** 477*437bfbebSnyanmisaka 478*437bfbebSnyanmisaka过程简单,可以快速完成一个可用的demo,评估解码器的性能。 479*437bfbebSnyanmisaka 480*437bfbebSnyanmisaka**缺点:** 481*437bfbebSnyanmisaka 482*437bfbebSnyanmisaka1. 内存都是从解码器内部分配的,如果内存在解码器被销毁时还没有被释放,有可能出现内存泄漏或崩溃问题。 483*437bfbebSnyanmisaka2. 无法控制解码器的内存使用量。解码器可以不受限制地使用内存,如果码流输入的速度很快,用户又没有及时释放内存,解码器会很快消耗掉全部的可用内存。 484*437bfbebSnyanmisaka3. 实现零拷贝的显示比较困难,因为内存是从解码器内部分配的,不一定和用户的显示系统兼容。 485*437bfbebSnyanmisaka 486*437bfbebSnyanmisaka#### 模式二:半内部分配模式 487*437bfbebSnyanmisaka 488*437bfbebSnyanmisaka这种模式是mpi_dec_test demo使用的默认模式。用户需要根据get_frame返回的MppFrame的buf_size来创建MppBufferGroup,并通过control接口的MPP_DEC_SET_EXT_BUF_GROUP配置给解码器。用户可以通过mpp_buffer_group_limit_config接口来限制解码器的内存使用量。 489*437bfbebSnyanmisaka 490*437bfbebSnyanmisaka 491*437bfbebSnyanmisaka<center>图表 15 解码器图像内存半内部分配模式代码流程</center> 492*437bfbebSnyanmisaka 493*437bfbebSnyanmisaka**优点:** 494*437bfbebSnyanmisaka 495*437bfbebSnyanmisaka过程简单,易上手,可以一定程度限制内存的使用。 496*437bfbebSnyanmisaka 497*437bfbebSnyanmisaka**缺点:** 498*437bfbebSnyanmisaka 499*437bfbebSnyanmisaka1. 内存空间的限制并不准确,内存的使用量不是100%固定的,会有波动。 500*437bfbebSnyanmisaka2. 同样难于实现零拷贝的显示。 501*437bfbebSnyanmisaka 502*437bfbebSnyanmisaka#### 模式三:纯外部分配模式 503*437bfbebSnyanmisaka 504*437bfbebSnyanmisaka这种模式通过创建空的external模式的MppBufferGroup,从用户那里导入外部分配器分析的内存块文件句柄(一般是dmabuf/ion/drm)。在Android平台上,Mediaserver通过gralloc从SurfaceFlinger获取显示用内存,把gralloc得到的文件句柄提交(commit)到MppBufferGroup里,再把MppBufferGroup通过control接口MPP_DEC_SET_EXT_BUF_GROUP命令配置给解码器,然后MPP解码器将循环使用gralloc得到的内存空间。 505*437bfbebSnyanmisaka 506*437bfbebSnyanmisaka 507*437bfbebSnyanmisaka<center>图表 16 解码器图像内存纯外部分配模式示意图</center> 508*437bfbebSnyanmisaka 509*437bfbebSnyanmisaka 510*437bfbebSnyanmisaka<center>图表 17 解码器图像内存纯外部分配模式代码流程</center> 511*437bfbebSnyanmisaka 512*437bfbebSnyanmisaka**优点:** 513*437bfbebSnyanmisaka 514*437bfbebSnyanmisaka直接使用外部显示用的内存,容易实现零拷贝。 515*437bfbebSnyanmisaka 516*437bfbebSnyanmisaka**缺点:** 517*437bfbebSnyanmisaka 518*437bfbebSnyanmisaka1. 理解与使用较困难。 519*437bfbebSnyanmisaka2. 需要修改用户程序,一些用户程序的调用方式限制了纯外部分配方式的使用。 520*437bfbebSnyanmisaka 521*437bfbebSnyanmisaka**纯外部分配模式使用时的注意事项:** 522*437bfbebSnyanmisaka 523*437bfbebSnyanmisaka1. 如果图像内存分配是在解码器创建之前,需要有额外的方式来得到图像内存的大小。 524*437bfbebSnyanmisaka 525*437bfbebSnyanmisaka一般的YUV420图像内存空间计算方法: 526*437bfbebSnyanmisaka 527*437bfbebSnyanmisaka图像像素数据:hor_stride \* ver_stride \* 3 / 2 528*437bfbebSnyanmisaka额外附加信息:hor_stride \* ver_stride / 2 529*437bfbebSnyanmisaka 530*437bfbebSnyanmisaka2. 内存块数需要考虑解码和显示的需求,如果内存块数分配得太少,可能会卡住解码器。 531*437bfbebSnyanmisaka 532*437bfbebSnyanmisakaH.264/H.265这类参考帧较多的协议需要20+内存块能保证一定能解码。其它协议需要10+内存块能保证一定能解码。 533*437bfbebSnyanmisaka 534*437bfbebSnyanmisaka3. 如果在码流解码过程中发生了info change,需要把已有的MppBufferGroup进行reset,再commit进新的图像缓存,同时外部的显示也需要相应调整。 535*437bfbebSnyanmisaka 536*437bfbebSnyanmisaka 537*437bfbebSnyanmisaka## 3.4 编码器数据流接口 538*437bfbebSnyanmisaka 539*437bfbebSnyanmisaka编码器接口为用户提供了输入图像,输出码流的功能,接口函数为MppApi结构体里的encode_put_frame函数, encode_get_packet函数和encode函数。这组函数提供了简洁的编码功能支持。 540*437bfbebSnyanmisaka 541*437bfbebSnyanmisaka### 3.4.1 encode_put_frame 542*437bfbebSnyanmisaka 543*437bfbebSnyanmisaka| 接口定义 | MPP_RET encode_put_frame(MppCtx ctx, MppFrame frame) | 544*437bfbebSnyanmisaka|----------|------------------------------------------------------| 545*437bfbebSnyanmisaka| 输入参数 | ctx :MPP解码器实例 frame :待输入的图像数据 | 546*437bfbebSnyanmisaka| 返回参数 | 运行错误码 | 547*437bfbebSnyanmisaka| 功能 | 输入frame图像数据给ctx指定的MPP编码器实例 | 548*437bfbebSnyanmisaka 549*437bfbebSnyanmisaka#### 函数的工作模式 550*437bfbebSnyanmisaka 551*437bfbebSnyanmisaka由于编码器的输入图像一般都比较大,如果进行拷贝,效率会大幅度下降,所以编码器的输入函数需要等待编码器硬件完成对输入图像内存的使用,才能把输入函数返回,把使用后的图像归还给调用者。基于以上的考虑,encode_put_frame是阻塞式函数,会把调用者阻塞住,直到输入图像使用完成,会一定程度上导致软硬件运行无法并行,效率下降。 552*437bfbebSnyanmisaka 553*437bfbebSnyanmisaka#### 拷贝与零拷贝输入 554*437bfbebSnyanmisaka 555*437bfbebSnyanmisaka编码器的输入不支持CPU分配的空间,如果需要支持编码CPU分配的地址,需要分配MppBuffer并把数据拷贝进去,这样做会很大程度影响效率。编码器更喜欢dmabuf/ion/drm内存形式的输入,这样可以实现零拷贝的编码,额外的系统开销最小。 556*437bfbebSnyanmisaka 557*437bfbebSnyanmisaka### 3.4.2 encode_get_packet 558*437bfbebSnyanmisaka 559*437bfbebSnyanmisaka| 接口定义 | MPP_RET encode_get_packet(MppCtx ctx, MppPacket \*packet) | 560*437bfbebSnyanmisaka|----------|------------------------------------------------------------| 561*437bfbebSnyanmisaka| 输入参数 | ctx :MPP解码器实例 packet :用于获取MppPacket实例的指针。 | 562*437bfbebSnyanmisaka| 返回参数 | 运行错误码 | 563*437bfbebSnyanmisaka| 功能 | 从ctx指定的MPP编码器实例里获取完成编码的packet描述信息。 | 564*437bfbebSnyanmisaka 565*437bfbebSnyanmisaka#### 取头信息与图像数据 566*437bfbebSnyanmisaka 567*437bfbebSnyanmisaka以H.264编码器为例,编码器的输出数据分为头信息(sps/pps)和图像数据(I/P slice)两部分,头信息需要通过control接口的MPP_ENC_GET_EXTRA_INFO命令获取,图像数据则是通过encode_get_packet接口来获取。获取的时机是在control接口的SET_RC_CFG/SET_PREP_CFG/SET_CODEC_CFG参数配置命令完成之后。在参数配置命令调用时,编码器会进行各个参数的更新,在更新全部完成之后,调用MPP_ENC_GET_EXTRA_INFO获取到的头信息才是最新的。 568*437bfbebSnyanmisaka 569*437bfbebSnyanmisaka#### H.264编码器输出码流的格式 570*437bfbebSnyanmisaka 571*437bfbebSnyanmisaka目前硬件固定输出带00 00 00 01起始码的码流,所以encode_get_packet函数获取到码流都是带有00 00 00 01起始码。如果需要去掉起始码,可以从起始码之后的地址进行拷贝。 572*437bfbebSnyanmisaka 573*437bfbebSnyanmisaka#### 码流数据的零拷贝 574*437bfbebSnyanmisaka 575*437bfbebSnyanmisaka由于使用encode_put_frame和encode_get_packet接口时没有提供配置输出缓存的方法,所以使用encode_get_packet时一定会进行一次拷贝。一般来说,编码器的输出码流相对于输入图像不算大,数据拷贝可以接受。如果需要使用零拷贝的接口,需要使用enqueue/dequeue接口以及MppTask结构。 576*437bfbebSnyanmisaka 577*437bfbebSnyanmisaka### 3.4.3 encode 578*437bfbebSnyanmisaka 579*437bfbebSnyanmisaka暂未实现 580*437bfbebSnyanmisaka 581*437bfbebSnyanmisaka## 3.5 编码器控制接口 582*437bfbebSnyanmisaka 583*437bfbebSnyanmisaka编码器与解码器不同,需要用户进行一定的参数配置。编码器需要用户通过control接口配置编码器信息之后才可以进行编码工作。 584*437bfbebSnyanmisaka 585*437bfbebSnyanmisaka### 3.5.1 control与MppEncCfg 586*437bfbebSnyanmisaka 587*437bfbebSnyanmisakaMPP推荐使用封装后的MppEncCfg结构通过control接口的MPP_ENC_SET_CFG/MPP_ENC_GET_CFG命令来进行编码器信息配置。 588*437bfbebSnyanmisaka 589*437bfbebSnyanmisaka由于编码器可配置的选项与参数繁多,使用固定结构体容易出现接口结构体频繁变化,导致接口二进制兼容性无法得到保证,版本管理复杂,极大增加维护量。 590*437bfbebSnyanmisaka 591*437bfbebSnyanmisaka为了缓解这个问题MppEncCfg使用(void \*)作为类型,使用<字符串-值>进行key map式的配置,函数接口分为s32/u32/s64/u64/ptr,对应的接口函数分为set与get两组,如下: 592*437bfbebSnyanmisaka 593*437bfbebSnyanmisaka```c 594*437bfbebSnyanmisaka配置编码器信息: 595*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_set_s32(MppEncCfg cfg, const char *name, RK_S32 val); 596*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_set_u32(MppEncCfg cfg, const char *name, RK_U32 val); 597*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_set_s64(MppEncCfg cfg, const char *name, RK_S64 val); 598*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_set_u64(MppEncCfg cfg, const char *name, RK_U64 val); 599*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_set_ptr(MppEncCfg cfg, const char *name, void *val); 600*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_set_st(MppEncCfg cfg, const char *name, void *val); 601*437bfbebSnyanmisaka获取编码器信息: 602*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_get_s32(MppEncCfg cfg, const char *name, RK_S32 *val); 603*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_get_u32(MppEncCfg cfg, const char *name, RK_U32 *val); 604*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_get_s64(MppEncCfg cfg, const char *name, RK_S64 *val); 605*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_get_u64(MppEncCfg cfg, const char *name, RK_U64 *val); 606*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_get_ptr(MppEncCfg cfg, const char *name, void **val); 607*437bfbebSnyanmisakaMPP_RET mpp_enc_cfg_get_st(MppEncCfg cfg, const char *name, void *val); 608*437bfbebSnyanmisaka``` 609*437bfbebSnyanmisaka 610*437bfbebSnyanmisaka 611*437bfbebSnyanmisaka字符串一般用\[类型:参数\]的方式进行定义,可支持的字符串与参数类型如下: 612*437bfbebSnyanmisaka 613*437bfbebSnyanmisaka| 参数字串 | 接口 | 实际类型 | 描述说明 | 614*437bfbebSnyanmisaka| ----------------------- | ---- | ----------------------------------- | ------------------------------------------------------------ | 615*437bfbebSnyanmisaka| base:low_delay | S32 | RK_S32 | 表示低延时输出模式。 0 – 表示关闭;1 – 表示开启。 | 616*437bfbebSnyanmisaka| rc:mode | S32 | MppEncRcMode | 表示码率控制模式,目前支持CBR、VBR和AVBR三种: CBR为Constant Bit Rate,固定码率模式。 在固定码率模式下,目标码率起决定性作用。 VBR为Variable Bit Rate,可变码率模式。 在可变码率模式下,最大最小码率起决定性作用。 AVBR为Adaptive Variable Bit Rate,自适应码率模式。 在自适应码率模式下,静止场景中最小码率起决定性作用,运动场景中最大码率起决定性作用。最终平均码率将接近目标码率。 FIX_QP为固定QP模式,用于调试和性能评估。  | 617*437bfbebSnyanmisaka| rc:bps_target | S32 | RK_S32 | 表示CBR模式下的目标码率。 | 618*437bfbebSnyanmisaka| rc:bps_max | S32 | RK_S32 | 表示VBR/AVBR模式下的最高码率。 | 619*437bfbebSnyanmisaka| rc:bps_min | S32 | RK_S32 | 表示VBR/AVBR模式下的最低码率。 | 620*437bfbebSnyanmisaka| rc:fps_in_flex | S32 | RK_S32 | 表示输入帧率是否可变的标志位,默认为0。 为0表示输入帧率固定,帧率计算方式为: fps_in_num / fps_in_denom,可以表示分数帧率。 为1表示输入帧率可变。可变帧率的情况下,帧率不固定,对应的码率计算与分配的规则变为按实际时间进行计算。 | 621*437bfbebSnyanmisaka| rc:fps_in_num | S32 | RK_S32 | 表示输入帧率分数值的分子部分,默认值为30。 | 622*437bfbebSnyanmisaka| rc:fps_in_denom | S32 | RK_S32 | 表示输入帧率分数值的分母部分,默认值为1。 | 623*437bfbebSnyanmisaka| rc:fps_out_flex | S32 | RK_S32 | 表示输出帧率是否可变的标志位,默认为0。 为0表示输出帧率固定,帧率计算方式为: fps_out_num / fps_out_denom,可以表示分数帧率。 为1表示输出帧率可变。可变帧率的情况下,帧率不固定,对应的码流输出时间按实际时间进行计算。 | 624*437bfbebSnyanmisaka| rc:fps_out_num | S32 | RK_S32 | 表示输出帧率分数值的分子部分,默认值为30。 | 625*437bfbebSnyanmisaka| rc:fps_out_denom | S32 | RK_S32 | 表示输出帧率分数值的分母部分,默认值为1。 | 626*437bfbebSnyanmisaka| rc:gop | S32 | RK_S32 | 表示Group Of Picture,即两个I帧之间的间隔,含义如下: 0 – 表示只有一个I帧,其他帧均为P帧。 1 – 表示全为I帧。 2 – 表示序列为I P I P I P… 3 – 表示序列为I P P I P P I P P… 一般情况下,gop应配置为输出帧率的整数倍,默认值为两倍输出帧率。 | 627*437bfbebSnyanmisaka| rc:max_reenc_times | U32 | RK_U32 | 表示一帧图像最大重编码次数,默认值为1。在低延时输出模式下,max_reenc_times只能配置为0。 | 628*437bfbebSnyanmisaka| rc:priority | U32 | MppEncRcPriority | 表示超大帧重编优先级。 0 – 表示目标码率优先。 1 – 表示超大帧阈值优先。 此优先级只在超大帧重编时有效。  | 629*437bfbebSnyanmisaka| rc:drop_mode | U32 | MppEncRcDropFrmMode | 表示丢帧模式。 0 – 表示丢帧模式未使能。 1 – 表示正常丢帧模式。当瞬时码率超过丢帧阈值时,正常丢帧。 2 – 表示pskip构造模式。当瞬时码率超过丢帧阈值时,编码pskip帧替代当前帧。  | 630*437bfbebSnyanmisaka| rc:drop_thd | U32 | RK_U32 | 表示丢帧阈值控制变量,默认值为20。丢帧阈值计算公式为: bps_max \* (1 + drop_thd / 100)。 | 631*437bfbebSnyanmisaka| rc:drop_gap | U32 | RK_U32 | 表示最大允许连续丢帧数。当输出码流的poc_type为2时,drop_gap只能配置为1。 | 632*437bfbebSnyanmisaka| rc:max_i_prop | S32 | RK_S32 | 表示最大 IP 比例,用于钳位IP比例的范围,默认值为30。当max_i_proportion被调整较小时,会导致I帧模糊,P 帧清晰。 | 633*437bfbebSnyanmisaka| rc:min_i_prop | S32 | RK_S32 | 表示最小 IP 比例,用于钳位IP比例的范围,默认值为10。当min_i_proportion被调整较大时,会导致I帧清晰,P 帧模糊。 | 634*437bfbebSnyanmisaka| rc:init_ip_ratio | S32 | RK_S32 | 表示初始IP比例,默认值为160。IP比例表示I帧和P帧的bits数的比例,有效范围为\[160, 640\]。 | 635*437bfbebSnyanmisaka| rc:super_mode | U32 | MppEncRcSuperFrameMode | 表示超大帧处理模式。 0 – 表示无特殊策略。 1 – 表示丢弃超大帧。 2 – 表示重编超大帧。<br>  | 636*437bfbebSnyanmisaka| rc:super_i_thd | U32 | RK_U32 | 表示超大I帧阈值。 | 637*437bfbebSnyanmisaka| rc:super_p_thd | U32 | RK_U32 | 表示超大P帧阈值。 | 638*437bfbebSnyanmisaka| rc:debreath_en | U32 | RK_U32 | 表示去除呼吸效应使能标志。 0 – 表示关闭;1 – 表示开启。 | 639*437bfbebSnyanmisaka| rc:debreath_strength | U32 | RK_U32 | 表示去除呼吸效应强度调节参数,有效范围为\[0, 35\]。值越大,呼吸效应改善会越弱;值越小,呼吸效应改善越明显。 | 640*437bfbebSnyanmisaka| rc:qp_init | S32 | RK_S32 | 表示初始QP值。 | 641*437bfbebSnyanmisaka| rc:qp_min | S32 | RK_S32 | 表示P、B帧的最小QP值。 | 642*437bfbebSnyanmisaka| rc:qp_max | S32 | RK_S32 | 表示P、B帧的最大QP值。 | 643*437bfbebSnyanmisaka| rc:qp_min_i | S32 | RK_S32 | 表示I帧的最小QP值。 | 644*437bfbebSnyanmisaka| rc:qp_max_i | S32 | RK_S32 | 表示I帧的最大QP值。 | 645*437bfbebSnyanmisaka| rc:qp_step | S32 | RK_S32 | 表示相临两帧之间的帧级QP变化幅度。目前仅配备于RV1109/RV1126系统芯片,默认值为4。 | 646*437bfbebSnyanmisaka| rc:qp_ip | S32 | RK_S32 | 表示I帧和P帧的QP差值,有效范围为\[0, 8\]。 | 647*437bfbebSnyanmisaka| rc:qp_vi | S32 | RK_S32 | 表示虚拟I帧和P帧的QP差值,有效范围为\[0, 6\]。 | 648*437bfbebSnyanmisaka| rc:hier_qp_en | S32 | RK_S32 | 表示QP分层使能标识。 0 – 表示关闭;1 – 表示开启。 | 649*437bfbebSnyanmisaka| rc:hier_qp_delta | St | RK_S32 \* | 表示各层帧相对于第0层P帧的QP差值,层数为4,用数组存储。 | 650*437bfbebSnyanmisaka| rc:hier_frame_num | St | RK_S32 \* | 表示各层帧数,层数为4,用数组存储。 | 651*437bfbebSnyanmisaka| rc:stats_time | S32 | RK_S32 | 表示瞬时码率统计时间,单位为秒,有效范围为\[1, 60\]。默认值为3。 | 652*437bfbebSnyanmisaka| prep:width | S32 | RK_S32 | 表示图像水平方向像素数,单位为像素个数。 | 653*437bfbebSnyanmisaka| prep:height | S32 | RK_S32 | 表示图像垂直方向像素数,单位为像素个数。 | 654*437bfbebSnyanmisaka| prep:format | S32 | MppFrameFormat | 表示图像色彩空间格式以及内存排布方式。  | 655*437bfbebSnyanmisaka| prep:hor_stride | S32 | RK_S32 | 表示图像垂直方向相邻两行之间的距离,单位为byte数。 | 656*437bfbebSnyanmisaka| prep:ver_stride | S32 | RK_S32 | 表示图像分量之间的以行数间隔数,单位为1。 | 657*437bfbebSnyanmisaka| prep:colorspace | S32 | MppFrameColorSpace | 表示VUI信息中色域空间类型,用于colour_primaries和transfer_characteristics的备注。 | 658*437bfbebSnyanmisaka| prep:colorprim | S32 | MppFrameColorPrimaries | 表示VUI信息中colour_primaries参数,具体含义请参考H.264/H.265协议。 | 659*437bfbebSnyanmisaka| prep:colortrc | S32 | MppFrameColorTransferCharacteristic | 表示VUI信息中transfer_characteristics参数,具体含义请参考H.264/H.265协议。 | 660*437bfbebSnyanmisaka| prep:colorrange | S32 | MppFrameColorRange | 表示YUV转RGB的色彩范围。 0 – 表示未指定,由MPP配置。 1 – 表示码流格式为MPEG,限制色彩范围。 2 – 表示码流格式为JPEG,不限制色彩范围。  | 661*437bfbebSnyanmisaka| prep:range | S32 | MppFrameColorRange | 同prep:colorrange,用于前向兼容MPP版本。 | 662*437bfbebSnyanmisaka| prep:rotation | S32 | MppEncRotationCfg | 表示图像旋转属性,默认值为0。除了RK3588芯片,其他芯片均不支持FBC数据结构的旋转操作。 0 – 表示图像不旋转。 1 – 表示图像逆时针旋转90度。 2 – 表示图像逆时针旋转180度。 3 – 表示图像逆时针旋转270度。 | 663*437bfbebSnyanmisaka| prep:mirroring | S32 | RK_S32 | 表示图像镜像属性,默认值为0。除了RK3588芯片,其他芯片均不支持FBC数据结构的镜像操作。 0 – 表示图像不做镜像。 1 – 表示图像做水平镜像。 2 – 表示图像做垂直镜像。 | 664*437bfbebSnyanmisaka| codec:type | S32 | MppCodingType | 表示MppEncCodecCfg对应的协议类型,需要与MppCtx初始化函数mpp_init的参数一致。  | 665*437bfbebSnyanmisaka| h264:stream_type | S32 | RK_S32 | 表示H.264码流格式类型。 0 – 表示Annex-B格式,即用00 00 00 01的起始码对齐、分割码流数据。 1 – 表示AVCC格式,仅支持extradata中码流的解析。 目前MPP内部固定为带00 00 00 01起始码的格式。 | 666*437bfbebSnyanmisaka| h264:profile | S32 | RK_S32 | 表示SPS中的profile_idc参数。  | 667*437bfbebSnyanmisaka| h264:level | S32 | RK_S32 | 表示SPS中的level_idc参数,其中10表示level 1.0: 10/11/12/13 – qcif@15fps / cif@7.5fps / cif@15fps / cif@30fps 20/21/22 – cif@30fps / half-D1@25fps / D1@12.5fps 30/31/32 – D1@25fps / 720p@30fps / 720p@60fps 40/41/42 – 1080p@30fps / 1080p@30fps / 1080p@60fps 50/51/52 – 4K@30fps / 4K@30fps / 4K@60fps 一般配置为level 4.1即可满足要求。 | 668*437bfbebSnyanmisaka| h264:poc_type | U32 | RK_U32 | 表示SPS中的pic_order_cnt_type参数。 | 669*437bfbebSnyanmisaka| h264:log2_max_poc_lsb | U32 | RK_U32 | 表示SPS中的log2_max_pic_order_cnt_lsb_minus4参数,只在pic_order_cnt_type为0时使用。 | 670*437bfbebSnyanmisaka| h264:log2_max_frm_num | U32 | RK_U32 | 表示SPS中的log2_max_frame_num_minus4参数。 | 671*437bfbebSnyanmisaka| h264:gaps_not_allowed | U32 | RK_U32 | 表示SPS中的gaps_in_frame_num_value_allowed_flag参数的非。 | 672*437bfbebSnyanmisaka| h264:cabac_en | S32 | RK_S32 | 表示编码器使用的熵编码格式。 0 – 表示CAVLC,自适应变长编码。 1 – 表示CABAC,自适应算术编码。 | 673*437bfbebSnyanmisaka| h264:cabac_idc | S32 | RK_S32 | 表示H.264协议中的cabac_init_idc参数,只在cabac_en为1时使用,有效范围为\[0, 2\]。 | 674*437bfbebSnyanmisaka| h264:trans8x8 | S32 | RK_S32 | 表示H.264协议中transform_8x8_mode_flag参数,即8x8变换的使能标志。 0 – 表示关闭,在Baseline/Main profile时固定关闭。 1 – 表示开启,在High profile时可选开启。 | 675*437bfbebSnyanmisaka| h264:const_intra | S32 | RK_S32 | 表示H.264协议中constrained_intra_pred_flag参数,即 constrained_intra_pred_mode模式使能标志。 0 – 表示关闭;1 – 表示开启。 | 676*437bfbebSnyanmisaka| h264:scaling_list | S32 | RK_S32 | 表示H.264协议中scaling_list_matrix模式。 0 – 表示flat matrix;1 – 表示默认matrix。 | 677*437bfbebSnyanmisaka| h264:cb_qp_offset | S32 | RK_S32 | 表示H.264协议中chroma_qp_index_offset参数,即应添加到QPY和QSY的偏移量,用于寻址Cb色度分量的QPC表,有效范围为\[-12, 12\]。 | 678*437bfbebSnyanmisaka| h264:cr_qp_offset | S32 | RK_S32 | 表示H.264协议中second_chroma_qp_index_offset参数,即应添加到QPY和QSY的偏移量,用于寻址Cr色度分量的QPC表,有效范围为\[-12, 12\]。 | 679*437bfbebSnyanmisaka| h264:dblk_disable | S32 | RK_S32 | 表示H.264协议中deblocking_filter_control_present_flag参数,即deblock禁用标志,有效范围为\[0, 2\]。 <br/>0 – deblocking开启。 <br/>1 – deblocking关闭。<br/>2 – 在slice边界关闭deblocking。 | 680*437bfbebSnyanmisaka| h264:dblk_alpha | S32 | RK_S32 | 表示H.264协议中slice_alpha_c0_offset_div2参数,有效范围为\[-6, 6\]。 | 681*437bfbebSnyanmisaka| h264:dblk_beta | S32 | RK_S32 | 表示H.264协议中slice_beta_offset_div2参数,有效范围为\[-6, 6\]。 | 682*437bfbebSnyanmisaka| h264:qp_init | S32 | RK_S32 | 表示初始QP值,同rc:qp_init,用于前向兼容MPP版本。 | 683*437bfbebSnyanmisaka| h264:qp_max | S32 | RK_S32 | 表示P、B帧的最大QP值,同rc:qp_max,用于前向兼容MPP版本。 | 684*437bfbebSnyanmisaka| h264:qp_min | S32 | RK_S32 | 表示P、B帧的最小QP值,同rc:qp_min,用于前向兼容MPP版本。 | 685*437bfbebSnyanmisaka| h264:qp_max_i | S32 | RK_S32 | 表示I帧的最大QP值,同rc:qp_max_i,用于前向兼容MPP版本。 | 686*437bfbebSnyanmisaka| h264:qp_min_i | S32 | RK_S32 | 表示I帧的最小QP值,同rc:qp_min_i,用于前向兼容MPP版本。 | 687*437bfbebSnyanmisaka| h264:qp_step | S32 | RK_S32 | 表示相临两帧之间的帧级QP变化幅度,同rc:qp_step,用于前向兼容MPP版本。 | 688*437bfbebSnyanmisaka| h264:qp_delta_ip | S32 | RK_S32 | 表示I帧和P帧的QP差值,同rc:qp_ip,用于前向兼容MPP版本。 | 689*437bfbebSnyanmisaka| h264:max_tid | S32 | RK_S32 | 表示最大时序层ID。 | 690*437bfbebSnyanmisaka| h264:max_ltr | S32 | RK_S32 | 表示最大长期参考帧数。 | 691*437bfbebSnyanmisaka| h264:prefix_mode | S32 | RK_S32 | 表示添加prefix nal的使能标志。 0 – 表示关闭。 1 – 表示开启,在SEI信息和硬件编码的码流数据之间添加prefix nal。 | 692*437bfbebSnyanmisaka| h264:base_layer_pid | S32 | RK_S32 | 表示基准层优先级ID。 | 693*437bfbebSnyanmisaka| h264:constraint_set | U32 | RK_U32 | 表示SPS中的constraint_set0_flag至constraint_set5_flag参数。 | 694*437bfbebSnyanmisaka| h265:profile | S32 | RK_S32 | 表示VPS中的profile_idc参数。目前MPP内部固定为1,Main profile。 | 695*437bfbebSnyanmisaka| h265:level | S32 | RK_S32 | 表示VPS中的level_idc参数。 | 696*437bfbebSnyanmisaka| h265:scaling_list | S32 | RK_S32 | 表示H.265协议中scaling_list_matrix模式。 0 – 表示flat matrix,1 – 表示默认matrix。 | 697*437bfbebSnyanmisaka| h265:cb_qp_offset | S32 | RK_S32 | 表示H.265协议中chroma_qp_index_offset参数,即应添加到QPY和QSY的偏移量,用于寻址Cb色度分量的QPC表,有效范围为\[-12, 12\]。 | 698*437bfbebSnyanmisaka| h265:cr_qp_offset | S32 | RK_S32 | 表示H.265协议中second_chroma_qp_index_offset参数,即应添加到QPY和QSY的偏移量,用于寻址Cr色度分量的QPC表,有效范围为\[-12, 12\]。 | 699*437bfbebSnyanmisaka| h265:dblk_disable | S32 | RK_S32 | 表示H.265协议中deblocking_filter_control_present_flag参数,即deblock禁用标志,有效范围为\[0, 2\]。 <br/>0 – deblocking开启。 <br/>1 – deblocking关闭。 <br/>2 – 在slice边界关闭deblocking。 | 700*437bfbebSnyanmisaka| h265:dblk_alpha | S32 | RK_S32 | 表示H.265协议中slice_alpha_c0_offset_div2参数,有效范围为\[-6, 6\]。 | 701*437bfbebSnyanmisaka| h265:dblk_beta | S32 | RK_S32 | 表示H.265协议中slice_beta_offset_div2参数,有效范围为\[-6, 6\]。 | 702*437bfbebSnyanmisaka| h265:qp_init | S32 | RK_S32 | 表示初始QP值,同rc:qp_init,用于前向兼容MPP版本。 | 703*437bfbebSnyanmisaka| h265:qp_max | S32 | RK_S32 | 表示P、B帧的最大QP值,同rc:qp_max,用于前向兼容MPP版本。 | 704*437bfbebSnyanmisaka| h265:qp_min | S32 | RK_S32 | 表示P、B帧的最小QP值,同rc:qp_min,用于前向兼容MPP版本。 | 705*437bfbebSnyanmisaka| h265:qp_max_i | S32 | RK_S32 | 表示I帧的最大QP值,同rc:qp_max_i,用于前向兼容MPP版本。 | 706*437bfbebSnyanmisaka| h265:qp_min_i | S32 | RK_S32 | 表示I帧的最小QP值,同rc:qp_min_i,用于前向兼容MPP版本。 | 707*437bfbebSnyanmisaka| h265:qp_step | S32 | RK_S32 | 表示相临两帧之间的帧级QP变化幅度,同rc:qp_step,用于前向兼容MPP版本。 | 708*437bfbebSnyanmisaka| h265:qp_delta_ip | S32 | RK_S32 | 表示I帧和P帧的QP差值,同rc:qp_ip,用于前向兼容MPP版本。 | 709*437bfbebSnyanmisaka| h265:sao_luma_disable | S32 | RK_S32 | 表示H.265协议中slice_sao_luma_flag参数的非,即当前slice亮度分量的采样点自适应偏移的禁用标志。 <br/>0 – 亮度分量的SAO开启。 <br/>1 – 亮度分量的SAO关闭。 | 710*437bfbebSnyanmisaka| h265:sao_chroma_disable | S32 | RK_S32 | 表示H.265协议中slice_sao_chroma_flag参数的非,即当前slice色度分量的采样点自适应偏移的禁用标志。 <br/>0 – 色度分量的SAO开启。 <br/>1 – 色度分量的SAO关闭。 | 711*437bfbebSnyanmisaka| vp8:qp_init | S32 | RK_S32 | 表示初始QP值,同rc:qp_init,用于前向兼容MPP版本。 | 712*437bfbebSnyanmisaka| vp8:qp_max | S32 | RK_S32 | 表示P、B帧的最大QP值,同rc:qp_max,用于前向兼容MPP版本。 | 713*437bfbebSnyanmisaka| vp8:qp_min | S32 | RK_S32 | 表示P、B帧的最小QP值,同rc:qp_min,用于前向兼容MPP版本。 | 714*437bfbebSnyanmisaka| vp8:qp_max_i | S32 | RK_S32 | 表示I帧的最大QP值,同rc:qp_max_i,用于前向兼容MPP版本。 | 715*437bfbebSnyanmisaka| vp8:qp_min_i | S32 | RK_S32 | 表示I帧的最小QP值,同rc:qp_min_i,用于前向兼容MPP版本。 | 716*437bfbebSnyanmisaka| vp8:qp_step | S32 | RK_S32 | 表示相临两帧之间的帧级QP变化幅度,同rc:qp_step,用于前向兼容MPP版本。 | 717*437bfbebSnyanmisaka| vp8:qp_delta_ip | S32 | RK_S32 | 表示I帧和P帧的QP差值,同rc:qp_ip,用于前向兼容MPP版本。 | 718*437bfbebSnyanmisaka| vp8:disable_ivf | S32 | RK_S32 | 表示ivf封装的禁用标志,禁用后硬件编码的码流数据不封装成ivf格式。 <br/>0 – 表示开启;<br/>1 – 表示关闭。 | 719*437bfbebSnyanmisaka| jpeg:quant | S32 | RK_S32 | 表示JPEG编码器使用的量化参数等级,编码器一共内置了11级量化系数表格,从0到10,图像质量从差到好。 | 720*437bfbebSnyanmisaka| jpeg:qtable_y | Ptr | RK_U8 \* | 表示样本亮度分量量化表,大小为64,用数组存储。 | 721*437bfbebSnyanmisaka| jpeg:qtable_u | Ptr | RK_U8 \* | 表示样本色度分量u量化表,大小为64,用数组存储。 | 722*437bfbebSnyanmisaka| jpeg:qtable_v | Ptr | RK_U8 \* | 表示样本色度分量v量化表,大小为64,用数组存储。 | 723*437bfbebSnyanmisaka| jpeg:q_factor | S32 | RK_S32 | 表示量化表因子,有效范围为\[1, 99\]。qfactor 越大,量化表中的量化系数越小,得到的图像质量更好,但编码压缩率更低。同理,qfactor 越小,量化表中的量化系数越大,编码压缩率更高,但得到的图像质量更差。默认值为80。 | 724*437bfbebSnyanmisaka| jpeg:qf_max | S32 | RK_S32 | 表示量化表因子最大值,默认值为99。 | 725*437bfbebSnyanmisaka| jpeg:qf_min | S32 | RK_S32 | 表示量化表因子最小值,默认值为1。 | 726*437bfbebSnyanmisaka| split:mode | U32 | MppEncSplitMode | 表示H.264/H.265协议的slice切分模式  <br>0– 不切分。<br>1– BY_BYTE 切分 slice 根据 slice 大小。<br>2– BY_CTU 切分 slice 根据宏块或 CTU 个数。 | 727*437bfbebSnyanmisaka| split:arg | U32 | RK_U32 | Slice切分参数: 在BY_BYTE模式下,参数表示每个slice的最大大小。 在BY_CTU模式下,参数表示每个slice包含的宏块或CTU个数。 | 728*437bfbebSnyanmisaka 729*437bfbebSnyanmisaka其他的字符串与参数会进行后续扩展。 730*437bfbebSnyanmisaka 731*437bfbebSnyanmisaka### 3.5.2 control其他命令 732*437bfbebSnyanmisaka 733*437bfbebSnyanmisaka在定义于rk_mpi_cmd.h文件的MpiCmd枚举类型定义了control接口命令字,其中与编码器和编码过程相关的命令如下: 734*437bfbebSnyanmisaka 735*437bfbebSnyanmisaka 736*437bfbebSnyanmisaka 737*437bfbebSnyanmisaka从MPP_ENC_CMD_BASE到MPP_ENC_CMD_END之间的命令为编码器的control接口命令,其中配置命令的MPP_ENC_SET/GET\_ CFG已经做为基本的配置命令在3.5.1进行了介绍。剩下的命令在下面进行简要的介绍,其中的命令与编码器硬件相关,只有部分硬件支持。 738*437bfbebSnyanmisaka 739*437bfbebSnyanmisaka目前MPP支持的编码器硬件分为vepu系列和rkvenc系列,vepu系列支持H.264编码,vp8编码和jpeg编码,配备于绝大多数RK芯片中。rkvenc系列只支持H.264编码,目前只配备于RV1109/RV1126系统芯片,其支持的编码功能相对于vepu系统会更多更强。 740*437bfbebSnyanmisaka 741*437bfbebSnyanmisaka部分CMD命令简要说明: 742*437bfbebSnyanmisaka 743*437bfbebSnyanmisaka- ~~MPP_ENC_SET_PREP_CFG/ MPP_ENC_GET_PREP_CFG~~ 744*437bfbebSnyanmisaka 745*437bfbebSnyanmisaka- ~~MPP_ENC_SET_RC_CFG/ MPP_ENC_GET_RC_CFG~~ 746*437bfbebSnyanmisaka 747*437bfbebSnyanmisaka- ~~MPP_ENC_SET_CODEC_CFG/ MPP_ENC_GET_CODEC_CFG~~ 748*437bfbebSnyanmisaka 749*437bfbebSnyanmisaka废弃命令,为了前向兼容保留,不要使用 750*437bfbebSnyanmisaka 751*437bfbebSnyanmisaka- MPP_ENC_SET_IDR_FRAME 752*437bfbebSnyanmisaka 753*437bfbebSnyanmisaka无命令参数,用于向编码器请求I帖,编码器收到请求之后,会将待编码的下一帧编码为IDR帧。 754*437bfbebSnyanmisaka 755*437bfbebSnyanmisaka全部硬件都支持。 756*437bfbebSnyanmisaka 757*437bfbebSnyanmisaka- ~~MPP_ENC_SET_OSD_LEGACY_0~~ 758*437bfbebSnyanmisaka 759*437bfbebSnyanmisaka- ~~MPP_ENC_SET_OSD_LEGACY_1~~ 760*437bfbebSnyanmisaka 761*437bfbebSnyanmisaka- ~~MPP_ENC_SET_OSD_LEGACY_2~~ 762*437bfbebSnyanmisaka 763*437bfbebSnyanmisaka废弃命令,前向兼容用保留,不要使用 764*437bfbebSnyanmisaka 765*437bfbebSnyanmisaka- MPP_ENC_GET_HDR_SYNC/ ~~MPP_ENC_GET_EXTRA_INFO~~ 766*437bfbebSnyanmisaka 767*437bfbebSnyanmisaka用于单独获取码流头数据的命令,其中MPP_ENC_GET_EXTRA_INFO为旧命令,不推荐使用。 768*437bfbebSnyanmisaka 769*437bfbebSnyanmisakaMPP_ENC_GET_HDR_SYNC输入参数为MppPacket,需要外部用户分配好空间并封装为MppPacket再control到编码器,control接口调用返回时就完成了数据拷贝,线程安全。调用时机在编码器基本配置完成之后。需要用户手动释放之前分配的MppPacket。 770*437bfbebSnyanmisaka 771*437bfbebSnyanmisakaMPP_ENC_GET_EXTRA_INFO输入参数为MppPacket\*,会获取编码器的内部MppPacket来进行访问。调用时机在编码器基本配置完成之后。需要注意的是,这里得到的MppPacket是MPP的内部空间,不需要用户释放。 772*437bfbebSnyanmisaka 773*437bfbebSnyanmisaka由于在多线程情况下,MPP_ENC_GET_EXTRA_INFO命令获取的MppPacket有可能在读取时被其他control修改,所以这个命令并不是线程安全的,仅做为旧vpu_api的兼容用,不要再使用。 774*437bfbebSnyanmisaka 775*437bfbebSnyanmisaka- ~~MPP_ENC_SET_SEI_CFG/MPP_ENC_GET_SEI_DATA~~ 776*437bfbebSnyanmisaka 777*437bfbebSnyanmisaka废弃命令,前向兼容用保留,不要使用 778*437bfbebSnyanmisaka 779*437bfbebSnyanmisaka- ~~MPP_ENC_PRE_ALLOC_BUFF/ MPP_ENC_SET_QP_RANGE/ MPP_ENC_SET_ROI_CFG/ MPP_ENC_SET_CTU_QP~~ 780*437bfbebSnyanmisaka 781*437bfbebSnyanmisaka废弃命令,前向兼容用保留,不要使用 782*437bfbebSnyanmisaka 783*437bfbebSnyanmisaka- MPP_ENC_GET_RC_API_ALL 784*437bfbebSnyanmisaka 785*437bfbebSnyanmisaka获取MPP目前支持的码率控制策略API信息的接口,输入 RcApiQueryAll\*指针,在返回时填充好结构体内容 786*437bfbebSnyanmisaka 787*437bfbebSnyanmisaka- MPP_ENC_GET_RC_API_BY_TYPE 788*437bfbebSnyanmisaka 789*437bfbebSnyanmisaka获取指定MppCodingType类型的所有码率控制策略API信息,输入RcApiQueryType\*指针并指定MppCodingType,在返回时会填充好结构体内容。 790*437bfbebSnyanmisaka 791*437bfbebSnyanmisaka- MPP_ENC_SET_RC_API_CFG 792*437bfbebSnyanmisaka 793*437bfbebSnyanmisaka注册外部码率控制策略API,输入RcImplApi\*指针,该结构中的函数指针定义了码率控制策略插件的行为,注册之后的码率控制策略才可以被查询和激活使用。 794*437bfbebSnyanmisaka 795*437bfbebSnyanmisaka- MPP_ENC_GET_RC_API_CURRENT 796*437bfbebSnyanmisaka 797*437bfbebSnyanmisaka返回当前使用的码率控制策略API信息,输入RcApiBrief\*指针,在返回时会填充好结构体内容。 798*437bfbebSnyanmisaka 799*437bfbebSnyanmisaka- MPP_ENC_SET_RC_API_CURRENT 800*437bfbebSnyanmisaka 801*437bfbebSnyanmisaka激活指定名字的码率控制策略API,输入RcApiBrief\*指针,编码器会搜索到RcApiBrief中指定字符串名字的码率控制策略API并激活为当前码率控制策略。 802*437bfbebSnyanmisaka 803*437bfbebSnyanmisaka- MPP_ENC_SET_HEADER_MODE/MPP_ENC_GET_HEADER_MODE 804*437bfbebSnyanmisaka 805*437bfbebSnyanmisaka配置和获取H.264/H.265编码器的SEI调试信息输出方式,调试用开关,以后会被环境变量取代,不要使用。 806*437bfbebSnyanmisaka 807*437bfbebSnyanmisaka- MPP_ENC_SET_SPLIT/MPP_ENC_GET_SPLIT 808*437bfbebSnyanmisaka 809*437bfbebSnyanmisaka配置和获取H.264/H265编码器的slice切分配置信息,己被MppEncCfg中的split:mode和split:arg取代,不要使用 810*437bfbebSnyanmisaka 811*437bfbebSnyanmisaka- MPP_ENC_SET_REF_CFG 812*437bfbebSnyanmisaka 813*437bfbebSnyanmisaka配置编码器高级参考帧模式,默认不需要配置,在需要配置长期参考帧,短期参考帧参考关系模式时使用,用于配置特殊的参考关系模式。高级接口,文档待完善。 814*437bfbebSnyanmisaka 815*437bfbebSnyanmisaka- MPP_ENC_SET_OSD_PLT_CFG 816*437bfbebSnyanmisaka 817*437bfbebSnyanmisaka用于配置rkvenc系列硬件的OSD调色板,命令参数为MppEncOSDPlt。一般只在编码开始时配置一次,全编码过程使用统一的调色板。仅RV1109/RV1126系列支持。 818*437bfbebSnyanmisaka 819*437bfbebSnyanmisaka- MPP_ENC_GET_OSD_PLT_CFG 820*437bfbebSnyanmisaka 821*437bfbebSnyanmisaka用于获取rkvenc系列硬件的OSD调色板,命令参数为MppEncOSDPlt\*。一般不使用 822*437bfbebSnyanmisaka 823*437bfbebSnyanmisaka- MPP_ENC_SET_OSD_DATA_CFG 824*437bfbebSnyanmisaka 825*437bfbebSnyanmisaka用于配置rkvenc系列硬件的OSD数据,命令参数为MppEncOSDData。需要每帧进行配置,每编码一帧之后需要重新配置。本命令被MppFrame带的MppMeta中的KEY_OSD_DATA进行替代,不再使用。 826*437bfbebSnyanmisaka 827*437bfbebSnyanmisaka## 3.6 编码器使用要点 828*437bfbebSnyanmisaka 829*437bfbebSnyanmisaka### 3.6.1 输入图像的宽高与stride 830*437bfbebSnyanmisaka 831*437bfbebSnyanmisaka编码器的输入图像宽高配置需要与图像数据在内存中的排布一致。以1920x1080大小的YUV420图像编码为例,参考`图表 7 `MppFrame重要参数说明的内容,假设有两种情况如下: 832*437bfbebSnyanmisaka 833*437bfbebSnyanmisaka 834*437bfbebSnyanmisaka 835*437bfbebSnyanmisaka 836*437bfbebSnyanmisaka<center>图表 18 编码器输入帧内存排布</center> 837*437bfbebSnyanmisaka 838*437bfbebSnyanmisaka上图图情况:亮度分量的宽度为1920,高度为1080,亮度数据与色度数据不直接相接,中间有8行的空行。 839*437bfbebSnyanmisaka 840*437bfbebSnyanmisaka这种情况下,水平stride为1920,垂直stride为1088,应用需要以1920\*1088\*3/2的大小分配空间并写入数据,使用宽1920,高1080,水平stride 1920,垂直stride 1088的配置即可以正常进行编码。 841*437bfbebSnyanmisaka 842*437bfbebSnyanmisaka下图情况:亮度分量的宽度为1920,高度为1080,亮度数据与色度数据直接相接,中间没有空行。 843*437bfbebSnyanmisaka 844*437bfbebSnyanmisaka这种情况下,水平stride为1920,垂直stride为1080,但由于编码器对数据的访问是16对齐的,在读取亮度下边缘数据时会读取到色度部分,读取色度下边缘数据时会读取到色度数据之外的部分,需要用户开出额外的空间,这里的空间为1920\*1080\*3/2+1920\*4的填充,才能保证编码器不出现访问未分配空间的情况。 845*437bfbebSnyanmisaka 846*437bfbebSnyanmisaka### 3.6.2 编码器控制信息输入方式以及扩展 847*437bfbebSnyanmisaka 848*437bfbebSnyanmisaka编码器控制信息的输入方式分为两种: 849*437bfbebSnyanmisaka 850*437bfbebSnyanmisaka一种是全局性控制信息,如码率配置,宽高配置等,作用于整个编码器和编码过程;另一种是临时性控制信息,如每帧的OSD配置信息,用户数据信息等,只作用于单帧编码过程。 851*437bfbebSnyanmisaka 852*437bfbebSnyanmisaka第一类控制信息主要通过control接口来进行配置,第二类控制信息主要是通过MppFrame所带的MppMeta接口来进行配置。 853*437bfbebSnyanmisaka 854*437bfbebSnyanmisaka今后对控制信息的扩展也会遵循这两种规则来进行扩展。 855*437bfbebSnyanmisaka 856*437bfbebSnyanmisaka### 3.6.3 编码器输入输出流程 857*437bfbebSnyanmisaka 858*437bfbebSnyanmisaka目前编码器默认输入接口仅支持阻塞式调用,输出接口支持非阻塞式和阻塞式调用,默认为非阻塞式调用,有可能出现获取数据失败的情况,需要在使用中注意。 859*437bfbebSnyanmisaka 860*437bfbebSnyanmisaka### 3.6.4 插件式自定义码率控制策略机制 861*437bfbebSnyanmisaka 862*437bfbebSnyanmisakaMPP支持用户自己定义码率控制策略,码率控制策略接口RcImplApi定义了几个编码处理流程上的钩子函数,用于在指定环节插入用户自定义的处理方法。具体使用方法可以参考默认的H.264/H.265码控策略实现(default_h264e/default_h265e结构)。 863*437bfbebSnyanmisaka 864*437bfbebSnyanmisaka码控插件机制在MPP中有预留,接口与流程都不算稳定,可以预见将来还会有不少调整,只建议有能力阅读理解代码,以及持续维护更新的用户使用这个机制,一般用户不建议使用。 865*437bfbebSnyanmisaka 866*437bfbebSnyanmisaka# 第四章 MPP demo说明 867*437bfbebSnyanmisaka 868*437bfbebSnyanmisakaMPP的demo程序变化比较快,以下说明仅供参考,具体情况以实际运行结果为准。Demo的运行环境均以Android 32bit平台为准。 869*437bfbebSnyanmisaka 870*437bfbebSnyanmisaka## 4.1 解码器demo 871*437bfbebSnyanmisaka 872*437bfbebSnyanmisaka解码器demo为mpi_dec_test系列程序,包括使用decode_put_packet和decode_get_frame接口的单线程mpi_dec_test、多线程的mpi_dec_mt_test以及多实例的mpi_dec_multi_test。 873*437bfbebSnyanmisaka 874*437bfbebSnyanmisaka以Android平台上的mpi_dec_test为例进行使用说明。直接运行测试用例mpi_dec_test,可以在日志中打印帮助文档,如下图所示: 875*437bfbebSnyanmisaka 876*437bfbebSnyanmisaka 877*437bfbebSnyanmisaka 878*437bfbebSnyanmisaka帮助文档可以分为两部分:一是mpi_dec_test的命令参数说明;二是码流文件的协议类型说明。 879*437bfbebSnyanmisaka 880*437bfbebSnyanmisaka命令参数的描述说明如下: 881*437bfbebSnyanmisaka 882*437bfbebSnyanmisaka| 命令参数 | 描述说明 | 883*437bfbebSnyanmisaka|----------|-------------------------------------------------| 884*437bfbebSnyanmisaka| -i | 输入的码流文件。 | 885*437bfbebSnyanmisaka| -o | 输出的图像文件。 | 886*437bfbebSnyanmisaka| -w | 图像宽度,单位为像素。 | 887*437bfbebSnyanmisaka| -h | 图像高度,单位为像素。 | 888*437bfbebSnyanmisaka| -t | 码流文件的协议类型。 | 889*437bfbebSnyanmisaka| -f | 图像色彩空间格式以及内存排布方式,默认为NV12。 | 890*437bfbebSnyanmisaka| -n | 最大解码帧数。测试时若码流较长,可仅输出前n帧。 | 891*437bfbebSnyanmisaka| -s | MPP实例数,默认为1。 | 892*437bfbebSnyanmisaka| -v | 日志选项:q为静默标志;f为fps显示标志。 | 893*437bfbebSnyanmisaka| -slt | 输出帧对应的校验文件。 | 894*437bfbebSnyanmisaka| -help | 打开帮助文档。 | 895*437bfbebSnyanmisaka 896*437bfbebSnyanmisakampi_dec_test的命令参数中,输入文件(i)和码流类型(t)为强制要求配置的参数,其他参数如输出文件(o)、图像宽度(w)、图像高度(h)和解码帧数(n)等为可选参数,可以根据不同的测试需求进行配置。 897*437bfbebSnyanmisaka 898*437bfbebSnyanmisakampi_dec_test的命令参数中,输出帧对应的校验文件(slt)将输出帧数据转换为对应的循环冗余校验码(具体逻辑见utils/utils.c)。校验文件的大小往往只有几kB,在芯片的slt测试中,将输出帧文件的对比转换成校验文件的对比,可以显著缩短测试周期。 899*437bfbebSnyanmisaka 900*437bfbebSnyanmisakaMPP库支持的输入文件的编码格式(t)为MPEG2/4、H.263/4/5、VP8/9和JPEG等,id后的数字为不同编码格式对应的参数值。参数值来源于OMX的定义,值得注意的是,HEVC和AVS格式的参数值与其他格式的有显著区别。 901*437bfbebSnyanmisaka 902*437bfbebSnyanmisaka以目录/data/下的ocrean.h264解码30帧为例,对demo和输出进行说明。运行的命令为: 903*437bfbebSnyanmisaka 904*437bfbebSnyanmisaka```bash 905*437bfbebSnyanmisaka mpi_dec_test -t 7 -i /data/ocrean.h264 -n 30 906*437bfbebSnyanmisaka``` 907*437bfbebSnyanmisaka 908*437bfbebSnyanmisaka其中,-t 7表示输入码流文件的协议类型是H.264,-i表示输入文件,-n 30表示解码30帧。测试用例正常运行结果如下: 909*437bfbebSnyanmisaka 910*437bfbebSnyanmisaka 911*437bfbebSnyanmisaka 912*437bfbebSnyanmisaka其中,打印的信息里包含了部分命令参数和部分解码日志。 913*437bfbebSnyanmisaka 914*437bfbebSnyanmisakaMPP库的版本信息: 915*437bfbebSnyanmisaka 916*437bfbebSnyanmisaka``` 917*437bfbebSnyanmisakaI mpp_info: mpp version: 6cc173d1 author: Ding Wei 2022-08-29 [hal_avsd]: Fix crash on avsd ref err path 918*437bfbebSnyanmisaka``` 919*437bfbebSnyanmisaka 920*437bfbebSnyanmisaka`I mpi_dec_test: 0xeebc01c0 decode_get_frame get info changed found` 921*437bfbebSnyanmisaka 922*437bfbebSnyanmisaka为mpi_dec_test本身的打印,表示发现解码器上报了info change事件。 923*437bfbebSnyanmisaka 924*437bfbebSnyanmisaka`I mpi_dec_test: 0xeebc01c0 decoder require buffer w:h [1920:1080] stride [1920:1088] buf_size 4177920` 925*437bfbebSnyanmisaka 926*437bfbebSnyanmisaka为mpi_dec_test本身的打印,表示解码器请求的图像内存情况。 927*437bfbebSnyanmisaka 928*437bfbebSnyanmisaka`I mpi_dec_test: 0xf1c40730 decode get frame 0` 929*437bfbebSnyanmisaka 930*437bfbebSnyanmisaka为mpi_dec_test本身的打印,表示解码器在正常解码和输出图像。 931*437bfbebSnyanmisaka 932*437bfbebSnyanmisaka`I mpi_dec_test: decode 10 frames time 263ms delay 69ms fps 113.99` 933*437bfbebSnyanmisaka 934*437bfbebSnyanmisaka为mpi_dec_test本身的打印,表示解码器解码30帧所用的时间为263ms,解码第一包数据的延迟时间为69ms,帧率为113.99。 935*437bfbebSnyanmisaka 936*437bfbebSnyanmisaka`I mpi_dec_test: test success max memory 19.92 MB` 937*437bfbebSnyanmisaka 938*437bfbebSnyanmisaka为mpi_dec_test本身的打印,表示解码器完成了解码30帧的功能,最大的内存开销为19.92 MB。 939*437bfbebSnyanmisaka 940*437bfbebSnyanmisaka解码器的demo具体代码参见test/mpi_dec_test.c。 941*437bfbebSnyanmisaka 942*437bfbebSnyanmisaka## 4.2 编码器demo 943*437bfbebSnyanmisaka 944*437bfbebSnyanmisaka编码器demo为mpi_enc_test系列程序,包括单线程的mpi_enc_test及多实例的mpi_enc_multi_test。 945*437bfbebSnyanmisaka 946*437bfbebSnyanmisaka以下以Android平台上的mpi_enc_test为例进行使用说明。直接运行测试用例mpi_enc_test,可以在日志中打印帮助文档,如下图所示: 947*437bfbebSnyanmisaka 948*437bfbebSnyanmisaka 949*437bfbebSnyanmisaka 950*437bfbebSnyanmisaka帮助文档可以分为三部分:一是mpi_enc_test的命令参数说明;二是码流文件的协议类型说明;三是图像色彩空间格式以及内存排布方式说明。 951*437bfbebSnyanmisaka 952*437bfbebSnyanmisaka命令参数的描述说明如下: 953*437bfbebSnyanmisaka 954*437bfbebSnyanmisaka| 命令参数 | 描述说明 | 955*437bfbebSnyanmisaka|------------------------------|-------------------------------------------------------------------------------------------------| 956*437bfbebSnyanmisaka| -i | 输入的图像文件。 | 957*437bfbebSnyanmisaka| -o | 输出的码流文件。 | 958*437bfbebSnyanmisaka| -w | 图像宽度,单位为像素。 | 959*437bfbebSnyanmisaka| -h | 图像高度,单位为像素。 | 960*437bfbebSnyanmisaka| -hstride | 垂直方向相邻两行之间的距离,单位为byte。 | 961*437bfbebSnyanmisaka| -vstride | 图像分量之间的以行数间隔数,单位为1。 | 962*437bfbebSnyanmisaka| -f | 图像色彩空间格式以及内存排布方式,默认为NV12。 | 963*437bfbebSnyanmisaka| -t | 码流文件的协议类型。 | 964*437bfbebSnyanmisaka| -tsrc | 源码流格式,仅在测试整体编解码性能时使用。 | 965*437bfbebSnyanmisaka| -n | 最大解码帧数。测试时若码流较长,可仅输出前n帧。 | 966*437bfbebSnyanmisaka| -g | gop参考模式,对应不同的TSVC码流。 | 967*437bfbebSnyanmisaka| -rc | 码率控制模式。0:VBR; 1:CBR; 2:FIXQP; 3:AVBR。 | 968*437bfbebSnyanmisaka| -bps | 码率约束参数。命令格式:bps_target:bps_min:bps_max。 | 969*437bfbebSnyanmisaka| -fps | 输入/输出帧率控制,默认为30。该命令参数仅说明输入帧率和输出帧率之间的比例关系,与实际帧率无关。 | 970*437bfbebSnyanmisaka| -qc | 质量控制。 | 971*437bfbebSnyanmisaka| -s | MPP实例数,默认为1。 | 972*437bfbebSnyanmisaka| -v | 日志选项:q为静默标志;f为fps显示标志。 | 973*437bfbebSnyanmisaka| -ini | 额外的编码配置文件ini(暂未生效)。 | 974*437bfbebSnyanmisaka| -slt | 输出码流对应的校验文件。 | 975*437bfbebSnyanmisaka 976*437bfbebSnyanmisakampi_enc_test的命令参数中,图像宽度(w)、图像高度(h)和码流类型(t)为强制要求配置的参数,其他参数如输入文件(i)、输出文件(o)、编码帧数(n)和色彩空间格式及内存排布方式(f)等为可选参数。如果没有指定输入文件,mpi_enc_test会生成默认的彩条图像进行编码。 977*437bfbebSnyanmisaka 978*437bfbebSnyanmisakampi_enc_test的命令参数提供了多样化的码率控制方案,用户可以通过码率控制模式(rc)和码率约束参数(bps)对输出码流的码率进行控制。码率控制模式(rc)分为可变码率模式(VBR)、固定码率模式(CBR)、qp修正的码率模式(FIXQP)和自适应码率模式(AVBR),默认模式为VBR;码率约束参数(bps)则是为MPP内部配置码率边界提供参考。 979*437bfbebSnyanmisaka 980*437bfbebSnyanmisakampi_enc_test的命令参数中,输入/输出帧率控制(fps)的格式为: 981*437bfbebSnyanmisaka 982*437bfbebSnyanmisaka```bash 983*437bfbebSnyanmisaka-fps fps_in_num:fps_in_den:fps_in_flex/fps_out_num:fps_out_den:fps_out_flex 984*437bfbebSnyanmisaka``` 985*437bfbebSnyanmisaka 986*437bfbebSnyanmisaka其中,in/out分别表示输入/输出;num表示分子;den表示分母;flex为0表示帧率固定,为1表示帧率可变。输入和输出默认的num和den分别为30和1,即默认的输入/输出帧率为30。该命令参数仅说明输入帧率和输出帧率之间的比例关系,与实际帧率无关。 987*437bfbebSnyanmisaka 988*437bfbebSnyanmisakampi_enc_test的命令参数中,质量控制(qc)仅在输出码流格式为H.264、H.265、VP8和JPEG时生效,命令格式为: 989*437bfbebSnyanmisaka 990*437bfbebSnyanmisaka```bash 991*437bfbebSnyanmisaka-qc qp_init/min/max/min_i/max_i 992*437bfbebSnyanmisaka``` 993*437bfbebSnyanmisaka 994*437bfbebSnyanmisaka其中,qp表示质量参数;init表示初值;min表示最小值;max表示最大值;后缀i表示I帧最值,未标注则表示B、P帧最值。 995*437bfbebSnyanmisaka 996*437bfbebSnyanmisakampi_enc_test的命令参数中,日志选项(v)为q时,MPP日常日志关闭;日志选项(v)为f时,每秒会打印一次平均帧率和当前帧率。 997*437bfbebSnyanmisaka 998*437bfbebSnyanmisaka图像的色彩空间格式分为YUV和RGB两类。MPP支持多种内存排布方式(f),id后的数字为不同内存排布方式对应的参数值,值得注意的是,YUV和RGB格式的参数值有显著区别。 999*437bfbebSnyanmisaka 1000*437bfbebSnyanmisaka以目录/data/下的ocrean.yuv编码30帧为例,对demo和输出进行说明。运行的命令为: 1001*437bfbebSnyanmisaka 1002*437bfbebSnyanmisaka```bash 1003*437bfbebSnyanmisakampi_enc_test -w 1920 -h 1080 -t 7 -i /data/ocrean.yuv -o /data/out.h264 -n 30 1004*437bfbebSnyanmisaka``` 1005*437bfbebSnyanmisaka 1006*437bfbebSnyanmisaka测试用例正常运行结果如下: 1007*437bfbebSnyanmisaka 1008*437bfbebSnyanmisaka 1009*437bfbebSnyanmisaka 1010*437bfbebSnyanmisaka在解码器demo中已经介绍的日志略。 1011*437bfbebSnyanmisaka 1012*437bfbebSnyanmisaka`I mpp_enc : MPP_ENC_SET_RC_CFG bps 7776000 [486000 : 8262000] fps [30:30] gop 60` 1013*437bfbebSnyanmisaka 1014*437bfbebSnyanmisaka默认的编码器的码率控制参数,目标码率为7.8Mbps,码率参考下界为0.5Mbps,码率参考上界为8.3Mbps;默认的输入和输出帧率为30;默认gop数为60。 1015*437bfbebSnyanmisaka 1016*437bfbebSnyanmisaka`I mpi_enc_test: chn 0 encoded frame 0 size 218616 qp 11` 1017*437bfbebSnyanmisaka 1018*437bfbebSnyanmisaka为mpi_enc_test本身的打印,表示编码器在正常编码,输出单帧码流大小为0.2M,质量参数为11。 1019*437bfbebSnyanmisaka 1020*437bfbebSnyanmisaka`I mpi_enc_test: chn 0 encode 30 frames time 628 ms delay 4 ms fps 47.72 bps 10265048` 1021*437bfbebSnyanmisaka 1022*437bfbebSnyanmisaka为mpi_enc_test本身的打印,表示编码器编码30帧所用的时间为628 ms,编码第一帧数据的延迟时间为4ms,帧率为47.72,码率为10.2Mbps。 1023*437bfbebSnyanmisaka 1024*437bfbebSnyanmisaka`I mpi_enc_test: mpi_enc_test average frame rate 47.72` 1025*437bfbebSnyanmisaka 1026*437bfbebSnyanmisaka为mpi_enc_test本身的打印,表示编码器编码平均帧率为47.72。 1027*437bfbebSnyanmisaka 1028*437bfbebSnyanmisaka编码器的控制参数还可以通过环境变量配置。在android环境下,环境变量配置命令为: 1029*437bfbebSnyanmisaka 1030*437bfbebSnyanmisaka```bash 1031*437bfbebSnyanmisakasetprop <控制参数> value 1032*437bfbebSnyanmisaka``` 1033*437bfbebSnyanmisaka 1034*437bfbebSnyanmisaka在linux环境下,环境变量配置命令为: 1035*437bfbebSnyanmisaka 1036*437bfbebSnyanmisaka```bash 1037*437bfbebSnyanmisakaexport <控制参数>=value 1038*437bfbebSnyanmisaka``` 1039*437bfbebSnyanmisaka 1040*437bfbebSnyanmisaka相应的描述说明如下: 1041*437bfbebSnyanmisaka 1042*437bfbebSnyanmisaka| 控制参数 | 类型 | 描述说明 | 1043*437bfbebSnyanmisaka|------------------|--------|--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------| 1044*437bfbebSnyanmisaka| constraint_set | RK_U32 | 仅对H.264格式的码流生效,对应语法中的constraint_set0_flag至constraint_set5_flag。其中,强制标志force_flag和强制约束参数constraint_force存储格式为:\| 00 \| force_flag \| 00 \| constraint_force \|。 只有低6位生效,对应constraint_set5_flag至constraint_set0_flag,当force_flag为1时,对应的constraint_set将被配置入编码器。 | 1045*437bfbebSnyanmisaka| split_mode | RK_U32 | 编码器分片模式。0:不分片; 1:按字节分片; 2:按CTU分片。 | 1046*437bfbebSnyanmisaka| split_arg | RK_U32 | 分片大小参数,编码器分片模式开启后生效。按字节分片时,该参数为各片字节上限;按CTU分片时,该参数为各片CTU数上限。 | 1047*437bfbebSnyanmisaka| split_out | RK_U32 | 分片输出模式,编码器分片模式开启后生效。1:低延时输出模式; 2:段信息输出模式。低延时输出模式下,编码器输出包内每个片的延时较低;段信息输出模式下,编码器会为每个片封装段信息。 | 1048*437bfbebSnyanmisaka| sei_mode | RK_U32 | SEI写入模式。0:不写入SEI; 1:序列写入模式; 2:帧写入模式。序列写入模式下,每个gop只有一个SEI;帧写入模式下,若SEI信息发生改变,则还会在每帧数据前添加SEI。 | 1049*437bfbebSnyanmisaka| gop_mode | RK_U32 | gop参考模式。若环境变量gop_mode未配置,则按命令参数配置;否则,按环境变量配置。 | 1050*437bfbebSnyanmisaka| osd_enable | RK_U32 | 使能OSD调色板。 | 1051*437bfbebSnyanmisaka| osd_mode | RK_U32 | OSD调色板模式,使能OSD调色板后生效。0:默认配置; 1:用户定义配置。 | 1052*437bfbebSnyanmisaka| roi_enable | RK_U32 | 使能ROI测试,并根据平台配置默认的roi_type。 | 1053*437bfbebSnyanmisaka| roi_type | RK_U32 | 强制配置roi_type。 | 1054*437bfbebSnyanmisaka| user_data_enable | RK_U32 | 使能用户数据。 | 1055*437bfbebSnyanmisaka 1056*437bfbebSnyanmisaka编码器的demo具体代码参见test/mpi_enc_test.c。 1057*437bfbebSnyanmisaka 1058*437bfbebSnyanmisaka## 4.3 实用工具 1059*437bfbebSnyanmisaka 1060*437bfbebSnyanmisakaMPP提供了一些单元测试用的工具程序,这种程序可以对软硬件平台以及MPP库本身进行测试 1061*437bfbebSnyanmisaka 1062*437bfbebSnyanmisaka- mpp_info_test 1063*437bfbebSnyanmisaka 1064*437bfbebSnyanmisaka用于读取和打印MPP库的版本信息,在反馈问题时,可以把打印出来信息附上。 1065*437bfbebSnyanmisaka 1066*437bfbebSnyanmisaka- mpp_buffer_test 1067*437bfbebSnyanmisaka 1068*437bfbebSnyanmisaka用于测试内核的内存分配器是否正常。 1069*437bfbebSnyanmisaka 1070*437bfbebSnyanmisaka- mpp_mem_test 1071*437bfbebSnyanmisaka 1072*437bfbebSnyanmisaka用于测试C库的内存分配器是否正常。 1073*437bfbebSnyanmisaka 1074*437bfbebSnyanmisaka- mpp_runtime_test 1075*437bfbebSnyanmisaka 1076*437bfbebSnyanmisaka用于测试一些软硬件运行时环境是否正常。 1077*437bfbebSnyanmisaka 1078*437bfbebSnyanmisaka- mpp_platform_test 1079*437bfbebSnyanmisaka 1080*437bfbebSnyanmisaka用于读取和测试芯片平台信息是否正常。 1081*437bfbebSnyanmisaka 1082*437bfbebSnyanmisaka# 第五章 MPP库编译与使用 1083*437bfbebSnyanmisaka 1084*437bfbebSnyanmisaka## 5.1 下载源代码 1085*437bfbebSnyanmisaka 1086*437bfbebSnyanmisakaMPP源代码发布官方地址:<https://github.com/rockchip-linux/mpp> 1087*437bfbebSnyanmisaka 1088*437bfbebSnyanmisaka发布分支为release分支,开发分支为develop分支,默认为开发分支。 1089*437bfbebSnyanmisaka 1090*437bfbebSnyanmisaka下载命令:git clone https://github.com/rockchip-linux/mpp.git 1091*437bfbebSnyanmisaka 1092*437bfbebSnyanmisaka## 5.2 编译 1093*437bfbebSnyanmisaka 1094*437bfbebSnyanmisakaMPP源代码编译脚本为cmake,需要依赖2.8.12以上的版本,建议使用3.x版。cmake-3.28版本验证通过。使用高版本的cmake工具可能会产生较多的warning。 1095*437bfbebSnyanmisaka 1096*437bfbebSnyanmisaka### 5.2.1 Android平台交叉编译 1097*437bfbebSnyanmisaka 1098*437bfbebSnyanmisaka编译Android库需要使用ndk环境,默认脚本使用android-ndk-r16b进行编译。android-ndk-r16b和android-ndk-r25c均验证通过,前者适用于android 14之前的sdk,后者适用于android 14及其之后的sdk。 1099*437bfbebSnyanmisaka 1100*437bfbebSnyanmisaka以r16b ndk为例进行说明: 1101*437bfbebSnyanmisaka 1102*437bfbebSnyanmisakar16b ndk的下载路径可以在源代码目录下的build/android/ndk_links.md文件里查找。 1103*437bfbebSnyanmisaka 1104*437bfbebSnyanmisaka把下载好的ndk解压到/home/pub/ndk/android-ndk-r16b,或者手动修改build/android/目录下env_setup.sh脚本的ANDROID_NDK变量路径。 1105*437bfbebSnyanmisaka 1106*437bfbebSnyanmisaka进入build/android/arm/目录,运行make-Android.bash脚本生成编译用Makefile,运行make –j16进行编译。 1107*437bfbebSnyanmisaka 1108*437bfbebSnyanmisaka### 5.2.2 Unix/Linux平台编译 1109*437bfbebSnyanmisaka 1110*437bfbebSnyanmisaka先配置build/linux/arm/目录下arm.linux.cross.cmake文件里的工具链,再运行make-Makefiles.bash脚本通过cmake生成Makefile,最后运行make –j16进行编译。 1111*437bfbebSnyanmisaka 1112*437bfbebSnyanmisakaMPP也支持直接在开发板运行的Debian上编译。 1113*437bfbebSnyanmisaka 1114*437bfbebSnyanmisaka# 第六章 常见问题FAQ 1115*437bfbebSnyanmisaka 1116*437bfbebSnyanmisakaQ:aarch64编译出错,报错为undefined reference to \`__system_property_get'。 1117*437bfbebSnyanmisaka 1118*437bfbebSnyanmisakaA:这是google 64bit ndk的问题,其libc.so中缺少一些符号定义,问题情况参见: 1119*437bfbebSnyanmisaka 1120*437bfbebSnyanmisaka<http://stackoverflow.com/questions/28413530/api-to-get-android-system-properties-is-removed-in-arm64-platforms> 1121*437bfbebSnyanmisaka 1122*437bfbebSnyanmisaka解决方法:MPP中已经把对应的libc.so放到build/android/aarch64/fix/目录下,把库拷贝到path_to_ndk/platforms/android-21/arch-arm64/usr/lib/路径下,重新编译即正常。 1123*437bfbebSnyanmisaka 1124*437bfbebSnyanmisakaQ:运行时会打印如下这样的内核log,是不是有问题? 1125*437bfbebSnyanmisaka 1126*437bfbebSnyanmisakavpu_service_ioctl:1844: error: unknow vpu service ioctl cmd 40086c01 1127*437bfbebSnyanmisaka 1128*437bfbebSnyanmisakaA:没有问题,mpp对内核驱动有一些依赖,内核驱动有不同版本的接口,mpp会进行多次尝试。如果遇到尝试失败会进行另外一种接口的尝试。这个打印是尝试失败时打印出来的,只会打印一次,可以忽略这个打印。 1129*437bfbebSnyanmisaka 1130*437bfbebSnyanmisakaQ:MPP运行异常如何分析? 1131*437bfbebSnyanmisaka 1132*437bfbebSnyanmisakaA:首先先分析错误日志,如果出现打开内核设备失败的log,需要先分析内核平台的视频编解码器硬件设备配置文件是否有使能,然后提交问题到redmine上,分析运行环境问题之后,再来分析MPP运行的内部问题。 1133