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 <stdio.h>
#include <string.h>
#include <stdlib.h>
#include<errno.h>

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;
}

轉換之前:

轉換之後:

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章