转自http://blog.sina.com.cn/s/blog_af9acfc60101alxx.html

一、YUV422转换规律 

  做视频采集与处理,自然少不了要学会分析YUV数据。因为从采集的角度来说,一般的视频采集芯片输出的码流一般都是YUV数据流的形式,而从视频处理(例如H.264、MPEG视频编解码)的角度来说,也是在原始YUV码流进行编码和解析,所以,了解如何分析YUV数据流对于做视频领域的人而言,至关重要。本文就是根据我的学习和了解,简单地介绍如何分析YUV数据流。

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

与我们熟知的RGB类似,YUV也是一种颜色编码方法,主要用于电视系统以及模拟视频领域,它将亮度信息(Y)与色彩信息(UV)分离,没有UV信息一样可以显示完整的图像,只不过是黑白的,这样的设计很好地解决了彩色电视机与黑白电视的兼容问题。并且,YUV不像RGB那样要求三个独立的视频信号同时传输,所以用YUV方式传送占用极少的频宽。

好了,言归正传,谈谈如何分析YUV码流吧。YUV码流有多种不同的格式,要分析YUV码流,就必须搞清楚你面对的到底是哪一种格式,并且必须搞清楚这种格式的YUV采样和分布情况。下面我将介绍几种常用的YUV码流格式,供大家参考。

YUV码流的存储格式其实与其采样的方式密切相关,主流的采样方式有三种,YUV4:4:4,YUV4:2:2,YUV4:2:0,关于其详细原理,可以通过网上其它文章了解,这里我想强调的是如何根据其采样格式来从码流中还原每个像素点的YUV值,因为只有正确地还原了每个像素点的YUV值,才能通过YUV与RGB的转换公式提取出每个像素点的RGB值,然后显示出来。

1.1 YUV格式

为了方便后面叙述,图片的大小定 义为:w * h,宽高分别为w和h

YUV420格式

先Y,后V,中间是U。其中的Y是w * h,U和V是w/2 * (h/2)

如果w = 4,h = 2,则:

yyyy

yyyy

uu

vv

内存则是:yyyyyyyyuuvv

需要占用的内存:w * h * 3 / 2

采样规律是:每个像素点都采样Y,寄数行采样1/2个U,不采样V,偶数行采样1/2个V,不采样U

YUV422格式

本格式使用较为广泛

每两个点为一组,共占用4个字节

YUYVYUYV…

对于每一组YUYV,前面一个Y和本组中的UV组成第一个点,第二个Y和本组中的UV组成第二个点

所以,在内存中,宽高分别为w * 2、h。

如果w = 4,h = 2,则:

YUYVYUYV

YUYVYUYV

需要占用的内存:w * h * 2

UYUY422格式

本格式和YUYV422一样,只是YUV的位置不一样罢了

每组中YUV的排列顺序为:UYUV

需要占用的内存:w * h * 2

YUV的采样格式及每种格式中单像素所占内 存大小

YUV主要的采样格式有YCbCr 4:2:0、YCbCr 4:2:2、YCbCr 4:1:1和 YCbCr 4:4:4。

采样格式       单像素所占内存大小        存放的码流

YCbCr 4:4:4    3byte       Y0 U0 V0 Y1 U1 V1 Y2 U2 V2 Y3 U3 V3(4像素为例)

YCbCr 4:2:2    2byte              Y0 U0 Y1 V1 Y2 U2 Y3 V3(4像素为例)

YCbCr 4:2:0    1.5byte     Y0 U0 Y1 Y2 U2 Y3 Y5 V5 Y6 Y7 V7 Y8(8像素为例)

YCbCr 4:1:1    1.5byte              Y0 U0 Y1 Y2 V2 Y3(4像素为例)

1.2  存储方式

下面我用图的形式给出常见的YUV码流的存储方式,并在存储方式后面附有取样每个像素点的YUV数据的方法,其中,Cb、Cr的含义等同于U、V。

(1) YUVY 格式 (属于YUV422)

YUYV为YUV422采样的存储格式中的一种,相邻的两个Y共用其相邻的两个Cb、Cr,分析,对于像素点Y'00、Y'01
而言,其Cb、Cr的值均为 Cb00、Cr00,其他的像素点的YUV取值依次类推。

(2) UYVY 格式
(属于YUV422)

UYVY格式也是YUV422采样的存储格式中的一种,只不过与YUYV不同的是UV的排列顺序不一样而已,还原其每个像素点的YUV值的方法与上面一样。

(3)
YUV422P(属于YUV422)

YUV422P也属于YUV422的一种,它是一种Plane模式,即打包模式,并不是将YUV数据交错存储,而是先存放所有的Y分量,然后存储所有的U(Cb)分量,最后存储所有的V(Cr)分量,如上图所示。其每一个像素点的YUV值提取方法也是遵循YUV422格式的最基本提取方法,即两个Y共用一个UV。比如,对于像素点Y'00、Y'01
而言,其Cb、Cr的值均为 Cb00、Cr00。

二、移植x264

首先下载对应的源文件http://ftp.videolan.org/pub/videolan/x264/snapshots/

我选择了x264-snapshot-20181021-2245-stable.tar.bz2

tar jxvf x264-snapshot---stable.tar.bz

./configure --host=arm-linux --prefix=/opt/wecam/ffmpeg --enable-shared --disable-asm

host:是要使用的平台
prefix:是make install的目录
enable-shared:是使能动态链接库
disable-asm:是关闭汇编命令

然后需要修改config.mak文件

然后运行指令:

make
make install

然后在/opt/wecam/ffmpeg目录下就有了对应的文件:

然后把libx264.so.155和pkgconfig目录放到4412开发板的/lib目录,再用指令创建软链接

ln -s libx264.so. libx264.so

而头文件x264.h是编译程序时需要使用的头文件

三、使用库编写YUV422转x264应用

#include <stdio.h>
#include <stdlib.h> #include "stdint.h" #include "include/x264.h" int main(int argc, char *argv[])
{
int ret;
int y_size;
int i, j; if(argc != ) {
printf("usage: %s [source file] [dest file] \n", argv[]);
return -;
} //source file
FILE *fp_src = fopen(argv[], "rb");
FILE *fp_dst = fopen(argv[], "wb"); //Encode 0 frame
int frame_num = ;
int csp = X264_CSP_I422; //YUYV
int width=,height=; //640*480 int iNal = ;
x264_nal_t *pNals = NULL;
x264_t *pHandle = NULL;
x264_picture_t *pPic_in = (x264_picture_t *)malloc(sizeof(x264_picture_t));
x264_picture_t *pPic_out = (x264_picture_t *)malloc(sizeof(x264_picture_t));
x264_param_t *pParam = (x264_param_t *)malloc(sizeof(x264_param_t)); if(fp_src == NULL || fp_dst == NULL) {
printf("Error open files.\n");
return -;
} x264_param_default(pParam);
pParam->i_width = width;
pParam->i_height = height;
pParam->i_csp = csp;
x264_param_apply_profile(pParam, x264_profile_names[]);
pHandle = x264_encoder_open(pParam); x264_picture_init(pPic_out);
x264_picture_alloc(pPic_in, csp, pParam->i_width, pParam->i_height);
y_size = pParam->i_width * pParam->i_height;
printf("w:%d h:%d\r\n",pParam->i_width,pParam->i_height); //detect frame number
if(frame_num == ) {
fseek(fp_src, , SEEK_END);
switch(csp) {
case X264_CSP_I444:
frame_num = ftell(fp_src)/(y_size*);
break;
case X264_CSP_I420:
frame_num = ftell(fp_src)/(y_size*/);
break;
case X264_CSP_I422:
frame_num = ftell(fp_src)/(y_size*);
break;
default:
printf("Colorspace Not Support.\n");
return -;
}
fseek(fp_src, , SEEK_SET);
} printf("frame_num:%d y_size:%d\r\n",frame_num,y_size);
//Loop to Encode
for(i=;i<frame_num;i++) {
switch(csp) {
case X264_CSP_I444:
fread(pPic_in->img.plane[], y_size, , fp_src);
fread(pPic_in->img.plane[], y_size, , fp_src);
fread(pPic_in->img.plane[], y_size, , fp_src);
break;
case X264_CSP_I420:
fread(pPic_in->img.plane[], y_size, , fp_src);
fread(pPic_in->img.plane[], y_size/, , fp_src);
fread(pPic_in->img.plane[], y_size/, , fp_src);
break;
case X264_CSP_I422:
{
int index = ;
int y_i = , u_i = , v_i = ;
for(index = ; index < y_size*; ) {
fread(&pPic_in->img.plane[][y_i++], , , fp_src); //Y
index++;
fread(&pPic_in->img.plane[][u_i++], , , fp_src); //U
index++;
fread(&pPic_in->img.plane[][y_i++], , , fp_src); //Y
index++;
fread(&pPic_in->img.plane[][v_i++], , , fp_src); //V
index++;
}
break;
}
default:
printf("Colorspace Not Support.\n");
return -;
} pPic_in->i_pts = i;
ret = x264_encoder_encode(pHandle, &pNals, &iNal, pPic_in, pPic_out);
if(ret < ) {
printf("Error.\n");
return -;
} printf("Succeed encode frame: %5d\n", i); for(j=;j<iNal;j++) {
fwrite(pNals[j].p_payload, , pNals[j].i_payload, fp_dst);
}
} //flush encoder
while() {
ret = x264_encoder_encode(pHandle, &pNals, &iNal, NULL, pPic_out);
if(ret == )
break;
printf("Flush 1 frame.\n");
for(j=;j<iNal;j++)
fwrite(pNals[j].p_payload, , pNals[j].i_payload, fp_dst);
}
x264_picture_clean(pPic_in);
x264_encoder_close(pHandle);
pHandle = NULL; free(pPic_in);
free(pPic_out);
free(pParam); fclose(fp_src);
fclose(fp_dst); return ;
}

然后是Makefile文件

encode_h264:
arm-none-linux-gnueabi-gcc encode_h264.c -o encode_h264 -L./lib/ -lx264
cp encode_h264 /home/topeet/linux/camera/x264_code
clean:
rm -f encode_h264

打完收工!

4412 移植x264并且YUV422转x264的更多相关文章

  1. x264 编码器选项分析 (x264 Codec Strong and Weak Points) 1

    文章文件夹: x264 编码器选项分析 (x264 Codec Strong and Weak Points) 1 x264 编码器选项分析 (x264 Codec Strong and Weak P ...

  2. x264 编码器选项分析 (x264 Codec Strong and Weak Points) 2

    文章目录: x264 编码器选项分析 (x264 Codec Strong and Weak Points) 1 x264 编码器选项分析 (x264 Codec Strong and Weak Po ...

  3. 4412 移植mpu9250尝试

    4412的板子IO都是1.8v的.只有I2C6是用了电平转换到了3.3v.所以我准备使用I2C6来驱动mpu9250 一.首先去掉占用的模块 menuconfig中去掉触摸的驱动 Device Dri ...

  4. X264库直接压缩BITMAP格式数据

    最近帮朋友看了下X264压缩视频,主要参考了雷霄骅(leixiaohua1020)的专栏的开源代码: http://blog.csdn.net/leixiaohua1020/article/detai ...

  5. x264 - 高品质 H.264 编码器

    转自:http://www.5i01.cn/topicdetail.php?f=510&t=3735840&r=18&last=48592660 H.264 / MPEG-4 ...

  6. 2016-06-06:X264码率控制

    H.264与x264 H264是一个视频压缩编码标准.https://zh.wikipedia.org/wiki/H.264/MPEG-4_AVC X264实现H264视频压缩标准的开源项目.http ...

  7. 转:视频压缩的基本概念(x264解压包)

    第1页:前言——视频压缩无处不在H.264 或者说 MPEG-4 AVC 是目前使用最广泛的高清视频编码标准,和上一代 MPEG-2.h.263/MPEG-4 Part4 相比,它的压缩率大为提高,例 ...

  8. Android camera采集视频 X264编码

    参考 http://blog.csdn.net/zblue78/article/details/6058147 感谢 ExperiencesOfCode 硬件平台:CPU Intel G630 @2. ...

  9. X264的版本号

    0 X264官方地扯 http://www.videolan.org/developers/x264.html 1 X264官方编译的二进制程序命名格式 官方编译出了LINUX,Win32,Win64 ...

随机推荐

  1. mysql的小练习

    建立如下表: 建表语句: class表创建语句 create table class(cid int not null auto_increment primary key, caption varc ...

  2. 应用安全 - SuiteCRM - 漏洞汇总

    CVE-2019-12598.CVE-2019-12601 SuiteCRM SQL注入与远程代码执行漏洞 SalesAgility SuiteCRM .x版本..x版本和7..5之前的7..x版本中 ...

  3. python批量下载验证码,用来做验证码处理

    刚学到爬虫识别验证码,所以自己建一个获取验证码的类,感兴趣的道友,可以看看,代码如下: import requests import time import os import re class Pi ...

  4. display:table

    display:table的CSS声明能够让一个HTML元素和它的子节点像table元素一样.使用基于表格的CSS布局,使我们能够轻松定义一个单元格的边界.背景等样式,而不会产生因为使用了table那 ...

  5. Mybatis-学习笔记(2)Mybatis配置文件

      3>typeAliases:类型别名.2种指定方式. 1>给某个类起个别名 <typeAliases> <typeAlias type="com.lfy.b ...

  6. Airbnb开源 快速搭建企业级BI数据平台

    最近在公司做一个数据可视化相关的项目,使用了Airbnb开源维护的企业级BI数据平台superset,相较于tableau这种收费的商业软件,Superset是开源维护的,同时图表的种类和颜值普遍偏高 ...

  7. 2018年牛客多校寒假 第四场 F (call to your teacher) (图的连通性)

    题目链接 传送门:https://ac.nowcoder.com/acm/contest/76/F 思路: 题目的意思就是判断图的连通性可以用可达性矩阵来求,至于图的存储可以用邻接矩阵来储存,求出来可 ...

  8. [2019杭电多校第六场][hdu6635]Nonsense Time

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=6635 题意是说一开始所有数都冻结,第i秒会解冻第ki个数,求每秒状态下的最长上上升子序列长度. 这种题 ...

  9. [BZOJ 2301] [HAOI 2011] Problem b (莫比乌斯反演)(有证明)

    [BZOJ 2301] [HAOI 2011] Problem b (莫比乌斯反演)(有证明) 题面 T组询问,每次给出a,b,c,d,k,求\(\sum _{i=a}^b\sum _{j=c}^d[ ...

  10. 在学习linux磁盘管理期间学习的逻辑卷管理笔记

    LVM(逻辑分区)的创建顺序:物理分区-物理卷-卷组-逻辑卷-挂载. 物理卷(Physical Volume,PV):就是指硬盘分区,也可以是整个硬盘或已创建的软RAID,是LVM的基本存储设备. 卷 ...