=====================================================

最简单的视频编码器系列文章列表:

最简单的视频编码器:编译

最简单的视频编码器:基于libx264(编码YUV为H.264)

最简单的视频编码器:基于libx265(编码YUV为H.265)

最简单的视频编码器:libvpx(编码YUV为VP8)

=====================================================

本文记录一个最简单的基于libx265的H.265(HEVC)视频编码器。此前记录的编码器是通过FFmpeg调用libx265完毕编码的,比如:

《最简单的基于FFmpeg的视频编码器-更新版(YUV编码为HEVC(H.265))》

相比与上文中的编码器,本文记录的编码器属于“轻量级”的编码器。由于它不再包括FFmpeg的代码。直接调用libx265完毕编码。因此项目的体积很小巧。该编码器能够将输入的YUV数据编码为H.265码流文件。

流程图

调用libx265进行视频编码的流程图例如以下所看到的。

 
从流程图中能够看出x265的API和x264的API十分类似。它们在使用方法上仅仅有微小的不同。

流程图中基本的函数例如以下所看到的。
x265_param_alloc():为參数集结构体x265_param分配内存。
x265_param_default():设置參数集结构体x265_param的缺省值。

x265_picture_alloc():为图像结构体x265_picture分配内存。
x265_picture_init():设置图像结构体x265_picture的缺省值。

x265_encoder_open():打开编码器。

x265_encoder_encode():编码一帧图像。
x265_encoder_close():关闭编码器。
x265_picture_free():释放x265_picture_alloc()申请的资源。
x265_param_free():释放x265_param_alloc()申请的资源。

存储数据的结构体例如以下所看到的。
x265_picture:存储压缩编码前的像素数据。
x265_nal:存储压缩编码后的码流数据。

此外流程图中还包括一个“flush_encoder”模块。该模块使用的函数和编码模块是一样的。唯一的不同在于不再输入视频像素数据。它的作用是输出编码器中剩余的码流数据。

源码

/**
* 最简单的基于X265的视频编码器
* Simplest X265 Encoder
*
* 雷霄骅 Lei Xiaohua
* leixiaohua1020@126.com
* 中国传媒大学/数字电视技术
* Communication University of China / Digital TV Technology
* http://blog.csdn.net/leixiaohua1020
*
* 本程序能够YUV格式的像素数据编码为H.265码流,是最简单的
* 基于libx265的视频编码器
*
* This software encode YUV data to H.265 bitstream.
* It's the simplest encoder example based on libx265.
*/
#include <stdio.h>
#include <stdlib.h> #if defined ( __cplusplus)
extern "C"
{
#include "x265.h"
};
#else
#include "x265.h"
#endif int main(int argc, char** argv){
int i,j;
FILE *fp_src=NULL;
FILE *fp_dst=NULL;
int y_size;
int buff_size;
char *buff=NULL;
int ret;
x265_nal *pNals=NULL;
uint32_t iNal=0; x265_param* pParam=NULL;
x265_encoder* pHandle=NULL;
x265_picture *pPic_in=NULL; //Encode 50 frame
//if set 0, encode all frame
int frame_num=50;
int csp=X265_CSP_I420;
int width=640,height=360; fp_src=fopen("../cuc_ieschool_640x360_yuv420p.yuv","rb");
//fp_src=fopen("../cuc_ieschool_640x360_yuv444p.yuv","rb"); fp_dst=fopen("cuc_ieschool.h265","wb");
//Check
if(fp_src==NULL||fp_dst==NULL){
return -1;
} pParam=x265_param_alloc();
x265_param_default(pParam);
pParam->bRepeatHeaders=1;//write sps,pps before keyframe
pParam->internalCsp=csp;
pParam->sourceWidth=width;
pParam->sourceHeight=height;
pParam->fpsNum=25;
pParam->fpsDenom=1;
//Init
pHandle=x265_encoder_open(pParam);
if(pHandle==NULL){
printf("x265_encoder_open err\n");
return 0;
}
y_size = pParam->sourceWidth * pParam->sourceHeight; pPic_in = x265_picture_alloc();
x265_picture_init(pParam,pPic_in);
switch(csp){
case X265_CSP_I444:{
buff=(char *)malloc(y_size*3);
pPic_in->planes[0]=buff;
pPic_in->planes[1]=buff+y_size;
pPic_in->planes[2]=buff+y_size*2;
pPic_in->stride[0]=width;
pPic_in->stride[1]=width;
pPic_in->stride[2]=width;
break;
}
case X265_CSP_I420:{
buff=(char *)malloc(y_size*3/2);
pPic_in->planes[0]=buff;
pPic_in->planes[1]=buff+y_size;
pPic_in->planes[2]=buff+y_size*5/4;
pPic_in->stride[0]=width;
pPic_in->stride[1]=width/2;
pPic_in->stride[2]=width/2;
break;
}
default:{
printf("Colorspace Not Support.\n");
return -1;
}
} //detect frame number
if(frame_num==0){
fseek(fp_src,0,SEEK_END);
switch(csp){
case X265_CSP_I444:frame_num=ftell(fp_src)/(y_size*3);break;
case X265_CSP_I420:frame_num=ftell(fp_src)/(y_size*3/2);break;
default:printf("Colorspace Not Support.\n");return -1;
}
fseek(fp_src,0,SEEK_SET);
} //Loop to Encode
for( i=0;i<frame_num;i++){
switch(csp){
case X265_CSP_I444:{
fread(pPic_in->planes[0],1,y_size,fp_src); //Y
fread(pPic_in->planes[1],1,y_size,fp_src); //U
fread(pPic_in->planes[2],1,y_size,fp_src); //V
break;}
case X265_CSP_I420:{
fread(pPic_in->planes[0],1,y_size,fp_src); //Y
fread(pPic_in->planes[1],1,y_size/4,fp_src); //U
fread(pPic_in->planes[2],1,y_size/4,fp_src); //V
break;}
default:{
printf("Colorspace Not Support.\n");
return -1;}
} ret=x265_encoder_encode(pHandle,&pNals,&iNal,pPic_in,NULL);
printf("Succeed encode %5d frames\n",i); for(j=0;j<iNal;j++){
fwrite(pNals[j].payload,1,pNals[j].sizeBytes,fp_dst);
}
}
//Flush Decoder
while(1){
ret=x265_encoder_encode(pHandle,&pNals,&iNal,NULL,NULL);
if(ret==0){
break;
}
printf("Flush 1 frame.\n"); for(j=0;j<iNal;j++){
fwrite(pNals[j].payload,1,pNals[j].sizeBytes,fp_dst);
}
} x265_encoder_close(pHandle);
x265_picture_free(pPic_in);
x265_param_free(pParam);
free(buff);
fclose(fp_src);
fclose(fp_dst); return 0;
}

执行结果

程序的输入为一个YUV文件(已经測试过YUV444P和YUV420P两种格式)。

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbGVpeGlhb2h1YTEwMjA=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="" />

输出为H.265码流文件。

watermark/2/text/aHR0cDovL2Jsb2cuY3Nkbi5uZXQvbGVpeGlhb2h1YTEwMjA=/font/5a6L5L2T/fontsize/400/fill/I0JBQkFCMA==/dissolve/70/gravity/SouthEast" alt="" />

H.265码流文件的信息例如以下所看到的。

下载

Simplest Encoder

项目主页

SourceForge:https://sourceforge.net/projects/simplestencoder/

Github:https://github.com/leixiaohua1020/simplest_encoder

开源中国:http://git.oschina.net/leixiaohua1020/simplest_encoder

CDSN下载地址:http://download.csdn.net/detail/leixiaohua1020/8284105

该解决方式包括了几个常见的编码器的使用演示样例:
simplest_vpx_encoder:最简单的基于libvpx的视频编码器
simplest_x264_encoder:最简单的基于libx264的视频编码器
simplest_x265_encoder:最简单的基于libx265的视频编码器

最简单的视频编码器:基于libx265(编码YUV为H.265)的更多相关文章

  1. 最简单的视频编码器:基于libx264(编码YUV为H.264)

    ===================================================== 最简单的视频编码器系列文章列表: 最简单的视频编码器:编译 最简单的视频编码器:基于libx ...

  2. 最简单的基于FFmpeg的视频编码器-更新版(YUV编码为HEVC(H.265))

    ===================================================== 最简单的基于FFmpeg的视频编码器文章列表: 最简单的基于FFMPEG的视频编码器(YUV ...

  3. 最简单的视频编码器:基于libvpx(编码YUV为VP8)

    ===================================================== 最简单的视频编码器系列文章列表: 最简单的视频编码器:编译 最简单的视频编码器:基于libx ...

  4. 最简单的视频编码器:编译(libx264,libx265,libvpx)

    ===================================================== 最简单的视频编码器系列文章列表: 最简单的视频编码器:编译 最简单的视频编码器:基于libx ...

  5. VP9 vs H.265——下一代视频编码标准的王道之争

    目前下一代主流的视频编码标准有 ITU-T VCEG 推出来的 H.265 和 Google 推出 VP9 . H.265 在 H.264 的基础上保留其中的部分技术,并对相关技术加以改进研发而成.新 ...

  6. 【省带宽、压成本专题】深入解析 H.265 编码模式,带你了解 Apple 全面推进 H.265 的原因

    过去几年,又拍云一直在点播.直播等视频应用方面潜心钻研,取得了不俗的成果.我们结合点播.直播.短视频等业务中的用户场景,推出了"省带宽.压成本"系列文章,从编码技术.网络架构等角度 ...

  7. 深入解析 H.265 编码模式,带你了解Apple全面推进H.265的原因

    今天我们聊聊视频编码.视频文件亘古以来存在一个矛盾:高清画质和视频体积的冲突,相同编码标准下,视频更高清,视频体积更大.因此,应用更先进的视频编码标准,降低视频体积,可以大幅降低网站的流量消耗. 目前 ...

  8. Jcompress: 一款基于huffman编码和最小堆的压缩、解压缩小程序

    前言 最近基于huffman编码和最小堆排序算法实现了一个压缩.解压缩的小程序.其源代码已经上传到github上面: Jcompress下载地址 .在本人的github上面有一个叫Utility的re ...

  9. TensorFlow上实践基于自编码的One Class Learning

    “我不知道什么是爱,但我知道什么是不爱” --One Class Learning的自白 一.单分类简介 如果将分类算法进行划分,根据类别个数的不同可以分为单分类.二分类.多分类,常见的分类算法主要解 ...

随机推荐

  1. 整理自百度知道提问的几道Java编程题

    蚂蚁爬杆 问题描述: 有一根27厘米的细木杆,在第3厘米.7厘米.11厘米.17厘米.23厘米这五个位置上各有一只蚂蚁.木杆很细,不能同时通过一只蚂蚁.开始时,蚂蚁的头朝左还是朝右是任意的,它们只会朝 ...

  2. Extjs4.2 Desktop 拖动黑色和白色的桌面图标的解决方案

    最近做了一个extjs4.2的desktop桌面demo,该desktop从原来的包中剥离出来,并实现了桌面图标休息,拖动桌面图标,但是,用户抱怨拖动桌面图标会出现黑色和白色,测试,在 extjs4. ...

  3. 关于matlab矩阵卷积conv2和傅里叶变换求卷积ifft2的关系

    先定义两个矩阵 a = [1 2 3 5 ; 4 7 9 5;1 4 6 7;5 4 3 7;8 7 5 1] %a矩阵取5*4 b = [1 5 4; 3 6 8; 1 5 7]   %b矩阵如多数 ...

  4. oschina Web应用开发

    Web应用开发 SPDY开发包(13) HTML5开发相关(105) Pjax相关项目(4) 网站API(93) REST/RESTful项目(72) 响应式 Web 框架(27) 微信相关软件(63 ...

  5. 性能测试开源小工具——http_load介绍

    淘测试 性能测试开源小工具——http_load介绍 meizhu 发表于:2009-07-02 浏览:3552次 评论:1次 所属分类: 性能测试 性能测试开源小工具——http_load介绍 ht ...

  6. [华为机试练习题]50.求M的N次方的最后三位

    题目 描写叙述: 正整数M 的N次方有可能是一个很大的数字,我们仅仅求该数字的最后三位 例1: 比方输入5和3 ,5的3次方为125.则输出为125 例2: 比方输入2和10 2的10次方为1024 ...

  7. 基于Cocos2dx开发卡牌游戏Demo_放开那三国 2.0

    PS:下载地址在最以下 1.登录 2.副本选择 3.地图 4. 选择敌人 5. 战斗 6. 战斗结算 7. 地图拓展 8. 武将拓展 9. 下载地址: 点击打开链接

  8. hdu1114(完全背包)

    题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=1114 分析:很裸的一道完全背包题,只是这里求装满背包后使得价值最少,只需初始化数组dp为inf:dp[ ...

  9. java參数传递方式问题

    java的參数传递方式到底是值传递还是引用传递,这一直是一个争论不休的问题,一直以来没有形成统一意见. 在这里,我也仅仅是说一说个人见解,不保证是对的,全当是抛砖引玉. 首先我的观点是java採用的是 ...

  10. POJ 3632 Optimal Parking(简单题)

    [题意简述]:就是选择一个停车地点.然后走遍全部的store后,再回到停车地点.就是走一圈.问要走的距离是多少. [分析]:能够直接求出距离的最大值与最小值,求出差值.乘以2就是最后的输出结果. // ...