毕设系列之Libx264实时视频流(YUV 420P转H264视频编码篇)
PS:要转载请注明出处,本人版权所有。
PS: 这个只是基于《我自己》的理解,
如果和你的原则及想法相冲突,请谅解,勿喷。
前置说明
本文作为本人csdn blog的主站的备份。(BlogID=035)
本文发布于 2017-08-07 16:30:54,现用MarkDown+图床做备份更新。blog原图已丢失,使用csdn所存的图进行更新。(BlogID=035)
环境说明
Ubuntu 16.04 LTS
前言
无
Libx264实时视频流
本文的技术实现部分参考雷博士的这篇文章。http://blog.csdn.net/leixiaohua1020/article/details/42078645
- 现在网上关于H264的文章有很多,但是我个人认为最好的就是雷霄骅博士的x264部分的文章最详细。所以许多的细节部分,我推荐大家去雷博士的blog去看。本文只提及我们使用Libx264时候,我们要注意的问题。
- 使用Libx264时候,我们需要关注的东西(下面用我的代码来说明假如我们要使用Libx264,那么我们需要注意的几个事情)。
//encoder
x264_t * pX264Handle;//结构体是一个编码器实例句柄,要使用这个编码库,我们必须有一个这种变量,没有为啥。
//param
x264_param_t * pX264Param;//这个结构体就比较重要了,他是我们设置编码器参数的载体,我们必须具体的了解各种参数的意义。具体参数在下一节进行分析。
//input,output pic
x264_picture_t *pPic_In;//这就是YUV输入图像和输出图像的载体,这里面有一个pts参数需要注意,下面小节进行说明。
x264_picture_t *pPic_Out;
//output h264 stream
x264_nal_t * pNals;//这个也是比较重要的一个东西,他的作用是用来保存编码后,网络抽象层所保存的数据(NAL HEADER,NAL BODY),想具体了解,可以去看H264编码原理。
//user config callback
//UESER_CONF_CALLBACK yX264_UserConfig;
int (*yX264_UserConfig)(struct ymx264 * mvl);//私有,忽略
//pi_nal is the number of NAL units
int pi_nal;//网络抽象单元个数
- 编码器参数分析
//* cpuFlags
mvl->pX264Param->i_threads = X264_SYNC_LOOKAHEAD_AUTO;//* 取空缓冲区继续使用不死锁的保证.
//* 视频选项
mvl->pX264Param->i_width = FRAME_WIDTH; //* 要编码的图像宽度.
mvl->pX264Param->i_height = FRAME_HEIGHT; //* 要编码的图像高度
mvl->pX264Param->i_frame_total = 0; //* 编码总帧数.不知道用0.
/* Force an IDR keyframe at this interval */
mvl->pX264Param->i_keyint_max = 10; //这个参数很重要,控制i帧的频率
mvl->pX264Param->b_repeat_headers = 1; // 重复SPS/PPS 放到关键帧前面//做实时流播放,此参数必须ENABLE
//* 流参数
//* how many b-frame between 2 references pictures */
mvl->pX264Param->i_bframe = 5;
//
mvl->pX264Param->b_open_gop = 0;
//* Keep some B-frames as references: 0=off, 1=strict hierarchical, 2=normal */
mvl->pX264Param->i_bframe_pyramid = 0;
//
mvl->pX264Param->i_bframe_adaptive = X264_B_ADAPT_TRELLIS;
//* Log参数,不需要打印编码信息时直接注释掉就行
//mvl->pX264Param->i_log_level = X264_LOG_DEBUG;
//* 速率控制参数
//pX264Param->rc.i_bitrate = 1024 * 10;//* 码率(比特率,单位Kbps) ,重要
//* muxing parameters 帧率控制,重要。
mvl->pX264Param->i_fps_den = 1; //* 帧率分母
mvl->pX264Param->i_fps_num = Y_STREAM_FPS;//* 帧率分子
mvl->pX264Param->i_timebase_den = mvl->pX264Param->i_fps_num;
mvl->pX264Param->i_timebase_num = mvl->pX264Param->i_fps_den;
最后,我们需要注意一点:关于我们设置的帧率的问题,不一定是设置多少,播放的时候就是多少,只是一个参考值,编码器会尽量的把视频编码为这个帧率。
- x264_picture_t * pPic_In->i_pts += 1; 此参数非常重要。如果不进行设置,视频流将不会正常播放。
/*
PTS:Presentation Time Stamp。PTS主要用于度量解码后的视频帧什么时候被显示出来
DTS:Decode Time Stamp。DTS主要是标识读入内存中的bit流在什么时候开始送入解码器中进行解码。
*/
- 关于颜色空间的问题,大家可以去百度YUV 420 ,YUV 422,YUV 444等这些原始图像的存储问题。具体来说,他们分为两类,一种是分组存储(例如:YYYUUUVVV*),一种是交叉存储(例如:YUYV)
- 此模块我的源代码
ym_x264.h
/*
FileName:ym_x264.h
Version:1.0
Description:
Created On: 2017-3-19
Modified date:
Author:Sky
*/
#ifndef _YM_X264_H
#define _YM_X264_H
#ifdef __cplusplus
#if __cplusplus
extern "C"{
#endif
#endif /* __cplusplus */
#include <stdint.h>
#include <stdio.h>
#include <ym_x264_config.h>
#include <x264.h>
#include <stdlib.h>
#define CLEAR_MEM(x) memset(&(x),0,sizeof(x))
enum yX264Cmd{
DO_DEFAULT_PRESET = 0,
DO_DEFAULT_USERCONF = 1,
DO_PARAM_APPLY_PROFILE = 2,
OPEN_ENCODER = 3,
ENCODER_ENCODE = 4,
};
enum yX264ColorSpace{
Y_CSP_I444 = 0,
Y_CSP_I422 = 1,
Y_CSP_I420 = 2,
Y_CSP_YUYV = 3,
};
//typedef struct ymx264 yMX264;
typedef struct ymx264{
//encoder
x264_t * pX264Handle;
//param
x264_param_t * pX264Param;
//input,output pic
x264_picture_t *pPic_In;
x264_picture_t *pPic_Out;
//output h264 stream
x264_nal_t * pNals;
//user config callback
//UESER_CONF_CALLBACK yX264_UserConfig;
int (*yX264_UserConfig)(struct ymx264 * mvl);
//pi_nal is the number of NAL units
int pi_nal;
long cur_pts;
}yMX264;
typedef int (*UESER_CONF_CALLBACK)(yMX264 * mvl);
int yInitMX264(yMX264 * mvl);
int yDestroyMX264(yMX264 * mvl);
int yIoctlX264(enum yX264Cmd cmd,...);
//CSC = ColorSpaceCovert,FIP = Fill In_Pic
int yDoCSC_And_FIP(yMX264 * mvl,enum yX264ColorSpace t_csp, int cache_id);
int yDo_Default_UserConf(yMX264 * mvl);
#ifdef __cplusplus
#if __cplusplus
}
#endif
#endif /* __cplusplus */
#endif
ym_x264.c
/*
FileName:ym_x264.c
Version:1.0
Description:
Created On: 2017-3-19
Modified date:
Author:Sky
*/
/*
x264_param_default():设置参数集结构体x264_param_t的缺省值。
x264_picture_alloc():为图像结构体x264_picture_t分配内存。
x264_encoder_open():打开编码器。
x264_encoder_encode():编码一帧图像。
x264_encoder_close():关闭编码器。
x264_picture_clean():释放x264_picture_alloc()申请的资源。
存储数据的结构体如下所示。
x264_picture_t:存储压缩编码前的像素数据。
x264_nal_t:存储压缩编码后的码流数据。
*/
#include <ym_x264.h>
uint8_t ImgCache[ImageCacheNum][FRAME_SIZE];
int yInitMX264(yMX264 * mvl){
mvl->pX264Param = (x264_param_t *)malloc(sizeof(x264_param_t));
// for (int i = 0; i < ImageCacheNum; i++){
mvl->pPic_In = (x264_picture_t *)malloc(sizeof(x264_picture_t));
mvl->pPic_Out = (x264_picture_t *)malloc(sizeof(x264_picture_t));
mvl->pNals = NULL;
mvl->pi_nal = 0;
x264_picture_init(mvl->pPic_Out);
x264_picture_alloc(mvl->pPic_In, FRAME_COLORSPACE, FRAME_WIDTH, FRAME_HEIGHT);
// PTS FROM 0,AND AUTO INCRESE 1
mvl->pPic_In->i_pts = 0;
// }
mvl->cur_pts = 0;
return 0;
}
int yDestroyMX264(yMX264 * mvl){
// 清除图像区域
//for (int i = 0; i < ImageCacheNum; i++)
x264_picture_clean(mvl->pPic_In);
x264_encoder_close(mvl->pX264Handle);
free(mvl->pX264Param);
//for (int i = 0; i < ImageCacheNum; i++){
free(mvl->pPic_In);
free(mvl->pPic_Out);
//}
return 0;
}
int yIoctlX264(enum yX264Cmd cmd,...){
va_list arg;
va_start(arg,cmd);
yMX264 *mx264;
mx264 = va_arg(arg,yMX264 *);
va_end(arg);
switch(cmd){
case DO_DEFAULT_PRESET:
{
/* Get default params for preset/tuning */
//x264_param_default(pParam); //this do default set for x264,but can not config some info
if( x264_param_default_preset( mx264->pX264Param, "veryfast", "zerolatency" ) < 0 ){
printf("x264_param_default_preset failed!\n");
return -1;
}
break;
}
case DO_DEFAULT_USERCONF:
{
//va_list arg;
//va_start(arg,cmd);
//mx264->yX264_UserConfig = va_arg(arg,UESER_CONF_CALLBACK);
//mx264->yX264_UserConfig = NULL;
//va_end(arg);
if ( (*mx264->yX264_UserConfig)(mx264) < 0){
printf("Do user conf callback failed.\n");
return -1;
}
break;
}
case DO_PARAM_APPLY_PROFILE:
{
//x264_profile_names[0] = baseline , to set stream-quality
if (x264_param_apply_profile(mx264->pX264Param, x264_profile_names[0]) < 0 ){
printf("x264_param_apply_profile failed.\n");
return -1;
}
break;
}
case OPEN_ENCODER:
{
//open encoder
if( (mx264->pX264Handle = x264_encoder_open(mx264->pX264Param)) == NULL){
printf("x264_encoder_open failed.\n");
return -1;
}
break;
}
case ENCODER_ENCODE:
{
/*
x264_encoder_encode:
* encode one picture.
* *pi_nal is the number of NAL units outputted in pp_nal.
* returns the number of bytes in the returned NALs.
* returns negative on error and zero if no NAL units returned.
* the payloads of all output NALs are guaranteed to be sequential in memory.
int x264_encoder_encode( x264_t *, x264_nal_t **pp_nal, int *pi_nal, x264_picture_t *pic_in, x264_picture_t *pic_out );
*/
if ( x264_encoder_encode(mx264->pX264Handle, &mx264->pNals, &mx264->pi_nal, mx264->pPic_In, mx264->pPic_Out) < 0){
printf("x264_encoder_encode failed.\n");
return -1;
}
/*
PTS:Presentation Time Stamp。PTS主要用于度量解码后的视频帧什么时候被显示出来
DTS:Decode Time Stamp。DTS主要是标识读入内存中的bit流在什么时候开始送入解码器中进行解码。
*/
//Presentation Time Stamp。PTS主要用于度量解码后的视频帧什么时候被显示出来
//MUST DO THIS ,IT DECIDE ,主要用于度量解码后的视频帧什么时候被显示出来
//mx264->pPic_In->i_pts += 1;
mx264->pPic_In->i_pts += 1;
printf("pts = %d\n",mx264->pPic_In->i_pts);
break;
}
default :
{
printf("No this cmd to analyse\n");
return -1;
break;
}
}
return 0;
}
int yDoCSC_And_FIP(yMX264 * mvl,enum yX264ColorSpace t_csp, int cache_id){
switch(t_csp){
case Y_CSP_I444:
{
/*
read(fd_in,pPic_In->img.plane[0],FRAME_SIZE); //Y
read(fd_in,pPic_In->img.plane[1],FRAME_SIZE); //U
read(fd_in,pPic_In->img.plane[2],FRAME_SIZE); //V
*/
break;
}
case Y_CSP_I420:
{
/*
#ifndef ENABLE_YUYVTOI420
read(fd_in,pPic_In->img.plane[0],FRAME_SIZE); //Y
read(fd_in,pPic_In->img.plane[1],FRAME_SIZE/4); //U
read(fd_in,pPic_In->img.plane[2],FRAME_SIZE/4); //V
#else
//YUYV to I420
read(fd_in,Cache,FRAME_SIZE*2); //read one frame to cache
//must set to 0
int id_u = 0, id_v = 0 , id_y = 0;
for (int i = 0; i < FRAME_SIZE*2 ;i+=4){
pPic_In->img.plane[0][id_y] = Cache[i];//get Y
id_y++;
pPic_In->img.plane[0][id_y] = Cache[i+2];//get Y
id_y++;
if ( ((int)((i)/1280)%2) == 0 ){
pPic_In->img.plane[1][id_u] = Cache[i+1];//get U
pPic_In->img.plane[2][id_v] = Cache[i+3];//get V
id_u++;
id_v++;
}
}
#endif
*/
break;
}
case Y_CSP_YUYV:{
// firstly,Do YUYV to I420 ,then, Fill in_pic
int id_u = 0, id_v = 0 , id_y = 0;
for (int i = 0; i < FRAME_SIZE ;i+=4){
mvl->pPic_In->img.plane[0][id_y] = ImgCache[cache_id][i];//get Y
id_y++;
mvl->pPic_In->img.plane[0][id_y] = ImgCache[cache_id][i+2];//get Y
id_y++;
if ( ((int)((i)/1280)%2) == 0 ){
mvl->pPic_In->img.plane[1][id_u] = ImgCache[cache_id][i+1];//get U
mvl->pPic_In->img.plane[2][id_v] = ImgCache[cache_id][i+3];//get V
id_u++;
id_v++;
}
}
break;
}
case Y_CSP_I422:
{
/*
read(fd_in,pPic_In->img.plane[0],FRAME_SIZE); //Y
read(fd_in,pPic_In->img.plane[1],FRAME_SIZE/2); //U
read(fd_in,pPic_In->img.plane[2],FRAME_SIZE/2); //V
*/
break;
}
default:
{
printf("Colorspace Not Support.\n");
return -1;
}
}
}
int yDo_Default_UserConf(yMX264 * mvl){
//* cpuFlags
mvl->pX264Param->i_threads = X264_SYNC_LOOKAHEAD_AUTO;//* 取空缓冲区继续使用不死锁的保证.
//* 视频选项
mvl->pX264Param->i_width = FRAME_WIDTH; //* 要编码的图像宽度.
mvl->pX264Param->i_height = FRAME_HEIGHT; //* 要编码的图像高度
mvl->pX264Param->i_frame_total = 0; //* 编码总帧数.不知道用0.
/* Force an IDR keyframe at this interval */
mvl->pX264Param->i_keyint_max = 10;
mvl->pX264Param->b_repeat_headers = 1; // 重复SPS/PPS 放到关键帧前面
//* 流参数
//* how many b-frame between 2 references pictures */
mvl->pX264Param->i_bframe = 5;
//
mvl->pX264Param->b_open_gop = 0;
//* Keep some B-frames as references: 0=off, 1=strict hierarchical, 2=normal */
mvl->pX264Param->i_bframe_pyramid = 0;
//
mvl->pX264Param->i_bframe_adaptive = X264_B_ADAPT_TRELLIS;
//* Log参数,不需要打印编码信息时直接注释掉就行
//mvl->pX264Param->i_log_level = X264_LOG_DEBUG;
//* 速率控制参数
//pX264Param->rc.i_bitrate = 1024 * 10;//* 码率(比特率,单位Kbps)
//* muxing parameters
mvl->pX264Param->i_fps_den = 1; //* 帧率分母
mvl->pX264Param->i_fps_num = Y_STREAM_FPS;//* 帧率分子
mvl->pX264Param->i_timebase_den = mvl->pX264Param->i_fps_num;
mvl->pX264Param->i_timebase_num = mvl->pX264Param->i_fps_den;
return 0;
}
后记
无
参考文献
打赏、订阅、收藏、丢香蕉、硬币,请关注公众号(攻城狮的搬砖之路)
PS: 请尊重原创,不喜勿喷。
PS: 要转载请注明出处,本人版权所有。
PS: 有问题请留言,看到后我会第一时间回复。
毕设系列之Libx264实时视频流(YUV 420P转H264视频编码篇)的更多相关文章
- 人脸检测及识别python实现系列(6)——终篇:从实时视频流识别出“我”
人脸检测及识别python实现系列(6)——终篇:从实时视频流识别出“我” 终于到了最后一步,激动时刻就要来临了,先平复一下心情,把剩下的代码加上,首先是为Model类增加一个预测函数: #识别人脸 ...
- 人脸检测及识别python实现系列(1)——配置、获取实时视频流
人脸检测及识别python实现系列(1)——配置.获取实时视频流 1. 前言 今天用多半天的时间把QQ空间里的几篇年前的旧文搬到了这里,算是完成了博客搬家.QQ空间里还剩下一些记录自己数学学习路线的学 ...
- opencv获取IP摄像头(IP-camera)实时视频流
之前这篇文章讲了如何通过网络摄像头(web camera)获取实时视频流,但是这种方法的缺陷就是摄像头和主机必须连在一起,那这种在室外部署的时候就会非常麻烦并且不安全,所以后来找了下用海康威视或者大华 ...
- 纠删码在实时视频流中的应用丨Dev for Dev 专栏
本文为「Dev for Dev 专栏」系列内容,作者为声网网络体验团队王瑞. 01 背景 在实时音视频通话中,音视频质量受网络丢包影响较大,特别是对于视频. 为什么视频对丢包更敏感呢?通常来说,音频的 ...
- 利用flask将opencv实时视频流输出到浏览器
opencv通过webcam可以获取本地实时视频流,但是如果需要将视频流共享给其他机器调用,就可以将利用flask框架构建一个实时视频流服务器,然后其他机器可以通过向这个服务器发送请求来获取这台机器上 ...
- 在WEB显示实时视频流
转载自:https://www.jianshu.com/p/7ef5490fbef7 安装摄像头 这里使用的是树莓派的官方摄像头,使用普通的 USB 摄像头也可以,但前提是你能够搞的定它的驱动. 大概 ...
- 视频系列:RTX实时射线追踪(下)
视频系列:RTX实时射线追踪(下) Key things from part 4 光线有效载荷是从一个着色器传递到另一个着色器的结构. 这一切都发生在RTX的引擎下. 更小的有效载荷要好得多! 新的D ...
- 视频系列:RTX实时射线追踪(上)
视频系列:RTX实时射线追踪(上) Video Series: Practical Real-Time Ray Tracing With RTX RTX在游戏和应用程序中引入了一个令人兴奋的和根本性的 ...
- [常用工具] OpenCV获取网络摄像头实时视频流
所需要硬件及软件环境: python 3/OpenCV3.4 or C++11/OpenCV3.4 1 RTSP协议 RTSP (Real Time Streaming Protocol),是一种语法 ...
- JS组件系列——表格组件神器:bootstrap table(三:终结篇,最后的干货福利)
前言:前面介绍了两篇关于bootstrap table的基础用法,这章我们继续来看看它比较常用的一些功能,来个终结篇吧,毛爷爷告诉我们做事要有始有终~~bootstrap table这东西要想所有功能 ...
随机推荐
- Arduino语言基础(萌新)
Arduino语言基础(萌新) Arduino语言注解Arduino语言是建立在C/C++基础上的,其实也就是基础的C语言,Arduino语言只不过把AVR单片机(微控制器)相关的一些参数设置都函数化 ...
- 文心一言 VS 讯飞星火 VS chatgpt (194)-- 算法导论14.3 2题
二.用go语言,改写 INTERVAL-SEARCH 的代码,使得当所有区间都是开区间时,它也能正确地工作. 文心一言,代码正常运行: 在Go语言中,处理开区间(open intervals)时,我们 ...
- NC16670 [NOIP2006]能量项链
题目链接 题目 题目描述 在Mars星球上,每个Mars人都随身佩带着一串能量项链.在项链上有N颗能量珠.能量珠是一颗有头标记与尾标记的珠子,这些标记对应着某个正整数.并且,对于相邻的两颗珠子,前一颗 ...
- Android 自动化测试项目
1 前言 在 Android自动化测试框架uiautomator2详解 中,介绍了 uiautomator2 框架的环境配置.元素定位工具以及常用接口. 本文对 uiautomator2 框架 ...
- 两个数组的交集II
两个数组的交集II 给定两个数组,编写一个函数来计算它们的交集. 示例 输入:nums1 = [1,2,2,1], nums2 = [2,2] 输出:[2,2] 输入:nums1 = [4,9,5], ...
- 微信小程序生态13-微信公众号自定义菜单配置
序 微信公众号分为订阅号和服务号两种,虽然二者很大的不同,但是这两种公众号的底部却是差不多的:都有菜单栏,而且这些底部菜单也都是自定义配置的. 如CSDN的官方公众号的底部就有精彩栏目.新程序员.CS ...
- XXL-Job框架入门介绍
框架概述 框架主页: https://www.xuxueli.com/xxl-job/ 包含组件: 1.调度中心 2.任务执行器 特点: 1.调度中心,任务执行器独立部署,互不影响. 2.调度中心和任 ...
- 案例分享:Qt出版社书籍配套U盘资源播放器软件定制(脚本关联播放器与资源文件,播放器,兼容win7,win10和mac)
红胖子(红模仿)的博文大全:开发技术集合(包含Qt实用技术.树莓派.三维.OpenCV.OpenGL.ffmpeg.OSG.单片机.软硬结合等等)持续更新中-(点击传送门) 合作案例专栏:案例分享(体 ...
- 无依赖单机尝鲜 Nebula Exchange 的 SST 导入
本文尝试分享下以最小方式(单机.容器化 Spark.Hadoop.Nebula Graph),快速趟一下 Nebula Exchange 中 SST 写入方式的步骤.本文适用于 v2.5 以上版本的 ...
- JAVA微服务分布式事务的几种实现方式
基础理论 CAP理论 一致性(Consistency) :在分布式系统中所有的数据备份,在同一时刻都保持一致状态,如无法保证状态一致,直接返回错误: 可用性(Availability):在集群中一部分 ...