C 实现 YUV420SP (NV12)和(NV21)的相互转换

YUV,分为三个分量,“Y”表示明亮度(Luminance或Luma),也就是灰度值;而“U”和“V” 表示的则是色度(Chrominance或Chroma),作用是描述影像色彩及饱和度,用于指定像素的颜色。

YUV420数据的长度,Y = 宽*高, U = 宽*高/4, V = 宽*高/4。

NV12数据的排列顺序为:YYYYYYYYY......, UVUVUV.......

NV21数据的排列顺序为:YYYYYYYYY......, VUVUVU.......

所以NV12和NV21的转换就是:前面 宽*高 的数据不变,后面 宽*高/2 的数据调换一下位置。实现代码如下:

#include 
#include 
#include 
#include

int main(int argc, char const *argv[])
{
    FILE *fp = NULL;
    FILE *fp_w = NULL;
    int fp_size = 0;

    fp = fopen("nv12_720p.yuv","rb");
    if(NULL == fp)
    {
        printf("open is error\n");
        return 1;
    }

    fseek(fp, 0, SEEK_END);
    fp_size = ftell(fp);
    fseek(fp, 0, SEEK_SET);	
    char *nv12_1 = (char *)malloc(fp_size);
    char *nv21_1 = (char *)malloc(fp_size);
    char *nv12 = nv12_1;  
    char *nv21 = nv21_1;  
    fread(nv12, 1, fp_size, fp);
    int framesize = 1280*720;   //12和21前面一部分的大小的数据是一样的,即宽*高的部分的数据是一样的
    int i = 0,j = 0;

    memcpy(nv21, nv12, framesize);
    
    for(i = 0; i < framesize; i++){
        nv21[i] = nv12[i];      //前面宽*高的数据是一样的
    }
    
    //NV12 宽*高 后的数据是:UVUVUV...,UV总的数据是 宽*高/2 。现在把V放在 宽*高 后的第一个位置,+2奇数位置。
    for (j = 0; j < framesize/2; j+=2)
    {
        nv21[framesize + j - 1] = nv12[j+framesize];  
    }

    //NV12 宽*高 后的数据是:UVUVUV...,UV总的数据是 宽*高/2 。现在把U放在 宽*高 后的第二个位置,+2偶数数位置。
    for (j = 0; j < framesize/2; j+=2)
    {
        nv21[framesize + j] = nv12[j+framesize-1]; 
    }

    fclose(fp);
    fp_w = fopen("nv21_720p.yuv","w");
    fwrite(nv21, 1, fp_size, fp_w);
    fclose(fp_w);

    if(nv12_1 != NULL){
        free(nv12_1);
        nv12_1 = NULL;
    }
    if(nv21_1 != NULL){
        free(nv21_1);
        nv21_1 = NULL;
    }

    return 0;
}

转换之前:

转换之后:

你可能感兴趣的:(C 实现 YUV420SP (NV12)和(NV21)的相互转换)