YUV数据格式转换

YUV学习文档


     

     


第一幅是Y分量描述黑白图像

第二幅是U(V)分量描述

第三幅是V(U)分量描述

第四幅是YUV三幅合成后得到的正常图像


1.      YUV简介

YUV是一种颜色编码方法,它和我们熟知的RGB红绿蓝颜色体系相对应,它们之间能通过公式相互转换。而YUV区别于RGB的重要一点是采用YUV色彩空间亮度信号Y和色度信号UV是分离的,这样就使得亮度Y和色差UV三个信号分别进行编码,用同一信道发送出去。

在摄像头之类编程经常是会碰到YUV格式,研究证明,人眼对亮度的敏感超过色度。利用这个原理,可以把色度信息减少一点,人眼也无法查觉这一点。这样操作达到的效果就是对于一张图片使用YUV格式保存将更少的占用存储空间。

 

 

I420: YYYYYYYY UU VV    =>YUV420P
YV12: YYYYYYYY VV UU    =>YUV420P
NV12: YYYYYYYY UVUV     =>YUV420SP
NV21: YYYYYYYY VUVU     =>YUV420SP

 

 

2.      YUV的存储格式

RGB格式中,一个像素点要占用4字节空间32比特位。并且RGB格式每个点的数据是连继保存在一起的。而YUV格式根据采样方式的不同,每个像素点所占用的空间是不同的。YUV格式的每个点不是连续保存的,可以根据UV分量的保存类型可以把YUV存储格式分为:

紧缩格式(packedformats):将YUV值储存成Macro Pixels阵列,和RGB的存放方式类似。

平面格式(planarformats):将YUV的三个分量分别存放在不同的矩阵中。

 

 

--------Planar类型存储格式

 

 

--packed格式其中A代表Alps透明度

 

 

 

 

 

 

 

3.     YUV的采样方式

 YUV采样方式,主要描述像素Y、U、V分量采样比例,即表达每个像素时,Y、U、V分量的数目,通常有三种方式:YUV4:4:4,YUV4:2:2,YUV4:2:0。

YUV4:4:4采样,每一个Y对应一组UV分量。

YUV4:2:2采样,每两个Y共用一组UV分量。

YUV4:2:0采样,每四个Y共用一组UV分量。

用图直观地表示采集的方式,以黑点表示采样该像点的Y分量,以空心圆圈表示采用该像素点的UV分量。

 

 

采样四个像素点下各种YUV采样格式的对比:

 

使用YUV4:4:4采样,一共要进行12次采样分别4Y4U4V,对每一个Y,U,V每个需要8个比特位,就需要12*8=96位,平均下每个像素点需要96/4=24位比特位表示。

 

使用YUV4:2:2采样,就需要8*8 =64位,平均每个像素64/4=16位。

 

使用YUV4:2:0采样,就需要6*8 =48位,平均每个像素48/4=12位。

 

 

 

 

 

4.      常见的YUV码流的存储方式

CbCr的含义等同于UV

YUVY 采用422采样格式,packed存储格式

 

 

 

 

 

 

 

YUVY 采用422采样格式,Planar存储格式

 

 

 

 




YV12YU12 采用420采样格式,Planar存储格式

YV12YU12 区别就是信号量VU哪一个在前。




 

NV12NV21采用420采样格式,Planar存储格式,但是区别于YV12YU12,它的UV分量是相连保存的。

NV12NV21区别就是信号量VU哪一个在前。

 


 

YUV420P YUV420SP对比图

 

 

 

 

 

 


5.    YUV数据的转换

程序在运行时,有时候需要根据需求对所处理的YUV数据进行转换。例如程序处理需要YUV420SP数据时,当前数据是YUV420P,那么就需要对数据进行处理。



 

在处理从YUV420P数据格式从YUV420SP数据格式时,对于已经对齐的分辨率如

1280*720,640*480这样的分辨率它的YUV420P数据格式是完全对齐(16位对齐)

而像176*144这样的size它的YUV420P不是16位对齐,需要补空白位,使得176*144这样的size能16位对齐。

 

YUV’420SP数据大小的计算公式是: size = width * heigh * 1.5;

 

YUV420P数据大小的计算公式是:

size = stride * height

stride = ALIGN(width /2, 16) = width +padding

 

         static int ALIGN(int x, int y) {

            return (x + y - 1) & ~(y - 1);  // y must be a power of 2.

}

其中stride的含义:

当视频图像存储在内存时,图像的每一行末尾也许【与size是否16位对齐有关】包含一些扩展的内容,这些扩展的内容只影响图像如何存储在内存中,但是不影响图像如何显示出来。

Stride 就是这些扩展内容的名称,Stride 也被称作 Pitch,如果图像的每一行像素末尾拥有扩展内容,Stride 的值一定大于图像的宽度值,就像下图所示:


Pading可能有要看size是否16位对齐,pading也可能不在末尾,视具体size确定。

Padiing = ALIGN(Width/2,16)

Stride = Padiing + Width = ALIGN(Width/2,16)+ Width

Size =Stride * Height

 

 

         static int ALIGN(int x, int y) {

            return (x + y - 1) & ~(y - 1);  // y must be a power of 2.

}

 

 

YUV420P数据格式 如果Pading的计算结果不为Width/2 那么该size不是16位对齐的

根据公式分别代入 1280*720  640*480    176*144

1280*720

Stride = 1280 + Padiing = 1280 + 640 = 1920

Padiing = ALIGN(1280/2,16) = 640

Size =Stride * Height = 1920 * 720 =1382400

 

 

640*480

Stride = 640+ Padiing = 640 + 320 = 960

Padiing = ALIGN(640/2,16) = 320

Size =Stride * Height = 960 * 480 = 460800

 

176*144

Stride = 176+ Padiing = 176+ 96 = 272

Padiing = ALIGN(176/2,16) = 96 【176/2=88 但是计算得到96 该size不是16位对齐】

Size =Stride * Height = 272* 144 = 39168

 

YUV420SP数据格式   size = width* heigh * 1.5;

根据公式分别代入 1280*720  640*480    176*144

 

1280*720

size = width * heigh * 1.5 = 1280*720*1.5 =1382400

【该size是16位对齐YUV420P 和 YUV420SP数据大小一致 都为1382400 】

 

 

640*480

size = width * heigh * 1.5 = 640*480*1.5 =460800

【该size是16位对齐YUV420P 和 YUV420SP数据大小一致 都为460800】

 

176*144

size = width * heigh * 1.5 = 176*144*1.5 = 38016

【该size不是16位对齐YUV420P的数据大小为39168,YUV420SP数据大小为38016】

 

对于16位对齐的size来说,YUV420P 和 YUV420SP的数据大小是一致的,数据码流中不存在为了补齐而填充的空位,对于这样的size来说存在一个统一的函数来进行转换。

Yv12是YUV420P格式

NV21 是YUV420SP格式


          yv12_to_nv21((char*)(node.getImgBuf()->getVirAddr()),

          (char *)(Previewnode.getImgBuf()->getVirAddr()),

          (size_t)(node.getImgBuf()->getBufSize()));

 

         bool yv12_to_nv21(char *pyv12, char *pnv21, size_t size)

{

      size_t wh_size = size * 2 / 3;

// wh_size计算的是Y分量的大小,Y分量占用总空间的二分之三

//4:2:0 中总共6份Y占用4份,UV各占用1份得出

 

     size_t i = 0;

     char *pnv = pnv21 + wh_size; //NV21格式中,V分量的第一个数据的地址

     char *pyv_v = pyv12 + wh_size;//YV12格式中V分量的第一个数据的地址

     char *pyv_u = pyv12 + wh_size*5/4; //YV12格式中U分量的第一个数据的地址

 

      if (!pyv12 || !pnv21)

               return false;

// 从YV12复制Y分量到NV21,总共wh_size个Y分量

      memcpy(pnv21, pyv12, wh_size);

 

//遍历UV分量 UV分量各自的数量是Y分量的四分之一即 wh_size/4

      for (i = 0; i < wh_size/4; i++) {

//每次循环赋值一个V分量后 接着赋值一个U分量由YUV420Sp存储格式决定

               *(pnv++) = *(pyv_v++);

               *(pnv++) = *(pyv_u++);

      }

      if (wh_size/4 == i) //最终的判断语句,//到此数据转换完成。

               return true;

      else

               return false;

}

 

 

对于16位不对齐的size来说,YUV420P 和 YUV420SP的数据大小是不一致的,这就需要根据具体的size的存储方式以及填补的空位来决定转换方法,16位不对齐的size没有统一的一个转换方法,每个16位不对齐的size的转换方法都是不相同的。

对于176*144 这个size

YUV420P的大小为 39168

YUV420Sp的大小为 38016

这就需要写一个专门的函数进行转换

首先YUV420P 数据中肯定存在空白数据,其中38016个数据是有用,可知总共有

39168 - 38016=1152个空白的数据,我们的转换的方法就是过滤掉这1152个数据点,得到有用的那38016个数据。

 

由YUV420P的计算公式

176*144

Stride = 176+ Padiing = 176+ 96 = 272

Padiing = ALIGN(176/2,16) = 96 【176/2=88 但是计算得到96 该size不是16位对齐】

Size =Stride * Height = 272* 144 = 39168


通过遍历数据点得到空白数据点的位置在UV分量中

 

 

 

 

bool yv12_to_nv21(char *pyv12, char *pnv21, size_t size){

for(int i =0 ; i<272; i++){

 for(int j=0 ; j<144; j++){

// 打印所有数据点 查看空白点

MY_LOGE("row = %d  colum =%d pyv12 = %c", i,j, *( pyv12++));

}

}

}

 

通过查看数据得到

Y分量区域不存在空白数据点

U分量区域空白数据点是在

奇数行 第一行的88 89 90 91 92 93 94 95 位上的数据 总共8位数据

第三行的88 89 90 91 92 93 94 95 位上的数据 总共8位数据

……

偶数行 第二行的40 41 42 43 44 45 46 47 位上的数据

               136 137 138 139 140 141 142 143 位上的数据为空数据

 

第四行的40 41 42 43 44 45 46 47 位上的数据

               136 137 138 139 140 141 142 143 位上的数据为空数据

U分量区域的空白数据位 =  (8+16)* 24 = 576个

 

V分量区域空白数据点排列与U分量区域的空白数据一致

奇数行 第一行的88 89 90 91 92 93 94 95 位上的数据 总共8位数据

第三行的88 89 90 91 92 93 94 95 位上的数据 总共8位数据

……

偶数行 第二行的40 41 42 43 44 45 46 47 位上的数据

               136 137 138 139 140 141 142 143 位上的数据为空数据

 

第四行的40 41 42 43 44 45 46 47 位上的数据

               136 137 138 139 140 141 142 143 位上的数据为空数据

V分量区域的空白数据位 =  (8+16)* 24 = 576个

 

总共的空白数据点位 = V分量区域的空白数据位 + U分量区域的空白数据位

= 576 +576

                   =1152

YUV420p的size = 39168

YUV420sp的size = 38016

39168 – 38016 = 1152 由此可知 空白数据已经完全被找出来,剩下的就是位这样的size写一个专门的函数,进行转换。

 

 


得到的转换函数如下:

bool yv12_to_nv21_mms(char *pyv12, char*pnv21, size_t size)

  {

         size_t i = 0;

         size_t j = 0;

         char *pyv12_mms = pyv12;

         char *pnv21_mms = pnv21;

         char *pyv12_mms_v;

         char *pyv12_mms_u;

 

                  for (i = 0; i < 176; i++){

                          for (j = 0; j <144; j++)

                                 *(pnv21_mms++) = *(pyv12_mms++);

                  }

 

                  pyv12_mms_v = pyv12+25344;

                  pyv12_mms_u =pyv12+6912+25344;

 

                  for (i = 0; i < 48; i++) {

                  if(i%2 == 0){

                        for (j=0; j < 88;j++) {

                        *(pnv21_mms++) =*(pyv12_mms_v++);

                        *(pnv21_mms++) = *(pyv12_mms_u++);

                        }

                        pyv12_mms_v =pyv12_mms_v + 8;

                        pyv12_mms_u =pyv12_mms_u + 8;

 

                        for (j=0; j < 48;j++) {

                        *(pnv21_mms++) =*(pyv12_mms_v++);

                        *(pnv21_mms++) =*(pyv12_mms_u++);

                        }

                    }

                    else {

                        for (j=0; j < 40;j++) {

                        *(pnv21_mms++) = *(pyv12_mms_v++);

                        *(pnv21_mms++) =*(pyv12_mms_u++);

                        }

                        pyv12_mms_v =pyv12_mms_v + 8;

                        pyv12_mms_u =pyv12_mms_u + 8;

 

                        for (j=0; j < 88;j++) {

                        *(pnv21_mms++) =*(pyv12_mms_v++);

                        *(pnv21_mms++) =*(pyv12_mms_u++);

                        }

                        pyv12_mms_v =pyv12_mms_v + 8;

                        pyv12_mms_u = pyv12_mms_u+ 8;

                   }

               }

         return true;

    }

 

剧终!

你可能感兴趣的:(C码,Android)