EasyRTMPClient:RTMP拉流客户端扩展支持HEVC(H.265)解决方案之兼容H264和H265帧数据解析详解

本文主要是介绍EasyRTMPClient:RTMP拉流客户端扩展支持HEVC(H.265)解决方案之兼容H264和H265帧数据解析详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

视频流媒体中视频数据的传输占据了绝大部分的带宽,如何提升编码效率、减小带宽使用、提升画面质量,成为音视频开发者努力的重点。随着互联网、流媒体技术的发展,兼容支持H.264、H.265编码器(可减少计算的复杂性、提高压缩率,并降低编码时间)已经成为迫在眉睫的事。

在此之前,两篇关于EasyRTMPClient扩展支持HEVC(H.265)解决方案的文章中,我们已经完成了对H265的支持,本文主要阐述将H264和H265支持兼容起来,实现不同视频编码格式的自适应兼容适配。

1. 根据CodecId判断数据编码类型

根据视频编码ID判断视频编码类型,如果视频编码ID==FlvCodeId_Hevc(12),则判断视频编码格式为H265,反之则为H264(目前我们只支持这两种编码格式的视频推送),代码如下所示:

```
parser_VideoTag *video_tag = (parser_VideoTag*)(buf+parser_offset);
FlvCodeId video_code_id = (FlvCodeId)(video_tag->code_id&0x0f);
if (video_code_id == FlvCodeId_Hevc)
{av_frame.u32AVFrameFlag = EASY_SDK_VIDEO_CODEC_H265;// HEVC;
} 
else
{av_frame.u32AVFrameFlag = EASY_SDK_VIDEO_CODEC_H264;// 默认h264, 其他类型是否需要判断?!;
}
```

2. 数据帧头部判断

根据FLV/RTMP扩展支持H265标准,支持HEVC的VideoTagHeader定义如下图所示:
tsingsee

当CodecID == 12时,AVCPacketType为HEVCPacketType:

  • 如果HEVCPacketType为0,则表示HEVCVIDEOPACKET中存放的是HEVC sequence header;

  • 如果HEVCPacketType为1,则表示HEVCVIDEOPACKET中存放的是HEVC NALU;

  • 如果HEVCPacketType为2,则表示HEVCVIDEPACKET中存放的是HEVC end of sequence,即 HEVCDecoderConfigurationRecord;

而当CodecID == 7时,AVCPacketType为AVCPacketType:

  • 如果AVCPacketType为0,则表示HEVCVIDEOPACKET中存放的是AVC sequence header;

  • 如果AVCPacketType为1,则表示HEVCVIDEOPACKET中存放的是AVC NALU;

  • 如果AVCPacketType为2,则表示HEVCVIDEPACKET中存放的是AVC end of sequence,即AVCDecoderConfigurationRecord;

EasyRTMPClient对sequence header的解析函数如下代码段所示:

```
int ParserVideoSequencePacket(FlvCodeId video_code_id, char *buf,int len)
{int parser_offset = 0;char *parser_config = buf;if (video_code_id == FlvCodeId_Hevc){if(len <= sizeof(Parser_HEVCDecoderConfigurationRecord)){return -1001;}			......’//Parser HEVCDecoderConfigurationRecord ......rtmpclient_h265_decode_sps((unsigned char *)sps_buf_, sps_len_, width_, height_);	} else{if(len <= sizeof(parser_AVCDecoderHeader)){return -1001;}......’//Parser HEVCDecoderHeader......rtmpclient_h264_decode_sps((unsigned char *)sps_buf_, sps_len_,     width_, height_);}return 0;
}
```

3. 视频数据体帧数据nalu类型判断

根据FLV/RTMP扩展支持协议标准,支持H265的VideoTagBody定义如下, 扩展后的VideoTagBody如下图所示(红色字体为HEVC新增内容):
tsingsee

当CodecID为12时,VideoTagBody中存放的就是HEVC视频帧内容。
EasyRTMPClient视频帧nalu解析如下代码所示:

```
int ParserOneVideoNalu(EasyRTMPClient_AV_Frame& av_frame,char *buf,int len,char* processbuf)
{if(processbuf == NULL || buf == NULL || len == 0){return -3001;}if(sps_len_ == 0 || pps_len_ == 0){printf("do not get sequence head yet\n");return -3002;}int parse_offset = 0;int nalu_len = 0;int nalu_type = 0;int processlen = 0;while(parse_offset < len - 4){nalu_len = ntohl(*(int*)(buf + parse_offset));  parse_offset += 4;//如果视频帧编码类型为H265if(av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H265){nalu_type = (buf[parse_offset] >> 1) & 0x3F;if(nalu_type == e_H265_NAL_UNIT_VPS){ASSERT_PARSER(nalu_len,MAX_VPS_LEN);memcpy(vps_buf_,buf + parse_offset,nalu_len);vps_len_ = nalu_len;parse_offset += nalu_len;continue;}}else{nalu_type = buf[parse_offset]&0x1F;}//H265 以及 H264的SPS头解析兼容if((av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H265&&nalu_type ==e_H265_NAL_UNIT_SPS) || av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H264&&nalu_type == e_H264_Frame_Type_Sps){ASSERT_PARSER(nalu_len,MAX_PPS_LEN);memcpy(sps_buf_,buf + parse_offset,nalu_len);sps_len_ = nalu_len;parse_offset += nalu_len;if(width_ == 0 && sps_len_ > 0){if(av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H265)rtmpclient_h265_decode_sps((unsigned char *)sps_buf_, sps_len_, width_, height_);elsertmpclient_h264_decode_sps((unsigned char *)sps_buf_, sps_len_, width_, height_);}continue;}//H265 以及 H264的PPS头解析兼容else if((av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H265&&nalu_type ==e_H265_NAL_UNIT_PPS) || av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H264&&nalu_type == e_H264_Frame_Type_Pps){memcpy(pps_buf_,buf + parse_offset,nalu_len);pps_len_ = nalu_len;parse_offset += nalu_len;continue;}//H265 以及 H264的I帧解析兼容else if((av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H265&&nalu_type >=e_H265_NAL_UNIT_SLICE_BLA&&nalu_type <=e_H265_NAL_UNIT_SLICE_CRA ) || av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H264&&nalu_type == e_H264_Frame_Type_Idr){if(av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H265){memcpy(processbuf + processlen,nalu_head_,4);processlen += 4;memcpy(processbuf + processlen,vps_buf_,vps_len_);processlen += vps_len_;}.......//拷贝SPS和PPS以及Idr nalu......av_frame.u32VFrameType = EASY_SDK_VIDEO_FRAME_I;continue ;}//H265 以及 H264的P帧解析兼容else if((av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H265&&nalu_type >=e_H265_NAL_UNIT_SLICE_TRAIL_R&&nalu_type <=e_H265_NAL_UNIT_SLICE_TFD ) || av_frame.u32AVFrameFlag == EASY_SDK_VIDEO_CODEC_H264&&nalu_type == e_H264_Frame_Type_Slice){memcpy(processbuf + processlen,nalu_head_,4);processlen += 4;memcpy(processbuf + processlen,buf + parse_offset,nalu_len);processlen += nalu_len;parse_offset += nalu_len;av_frame.u32VFrameType = EASY_SDK_VIDEO_FRAME_P;continue ;}else{parse_offset += nalu_len;continue;}}av_frame.pBuffer = (uint8_t *)processbuf;av_frame.u32AVFrameLen = processlen;return 0;
}
```

至此,EasyRTMPClient对H264和H265的兼容适配就完成了,我们可以通过EasyRTMPClient拉取任意编码格式为H264或者H265的RTMP进行拉流,均能取得完整的视频帧数据进行解码和播放。

EasyDSS方案应用

本文介绍了EasyRTMPClient对H264和H265的兼容适配,然而,要将H264、H265兼容适配应用于音视频网络直播方案中,除推流端和播放端要提供相应能力外,源站、CDN、转码服务同样都需要提供这种能力,而EasyDSS流媒体解决方案完全具备了这种能力,并节省了开发时间与成本。

EasyDSS商用流媒体服务器(下载地址)是EasyDarwin流媒体团队开发的一款支持视频点播、转码、RTMP推流直播、RTMP/HLS直播分发、服务端录像、录像检索、录像下载、时移回放的商用流媒体服务器,采用业界优秀的流媒体框架模式设计,服务运行高效、稳定、可靠、易维护,支持RTMP直播、RTMP推送、HTTP点播、HLS直播,并支持关键帧缓冲,画面秒开等多种特性,能够接入WEB、Android、iOS、微信等全平台客户端,是移动互联网时代贴近企业点播/直播需求的一款接地气的流媒体服务器,配套OBS、EasyRTMP等直播推流工具以及EasyPlayer等网络播放器,可以形成一套完整的视频直播、录播解决方案,满足用户在各种行业场景的流媒体业务需求。

tsingsee
tsingsee

这篇关于EasyRTMPClient:RTMP拉流客户端扩展支持HEVC(H.265)解决方案之兼容H264和H265帧数据解析详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/764936

相关文章

Java中Arrays类和Collections类常用方法示例详解

《Java中Arrays类和Collections类常用方法示例详解》本文总结了Java中Arrays和Collections类的常用方法,涵盖数组填充、排序、搜索、复制、列表转换等操作,帮助开发者高... 目录Arrays.fill()相关用法Arrays.toString()Arrays.sort()A

Python 字典 (Dictionary)使用详解

《Python字典(Dictionary)使用详解》字典是python中最重要,最常用的数据结构之一,它提供了高效的键值对存储和查找能力,:本文主要介绍Python字典(Dictionary)... 目录字典1.基本特性2.创建字典3.访问元素4.修改字典5.删除元素6.字典遍历7.字典的高级特性默认字典

MySQL逻辑删除与唯一索引冲突解决方案

《MySQL逻辑删除与唯一索引冲突解决方案》本文探讨MySQL逻辑删除与唯一索引冲突问题,提出四种解决方案:复合索引+时间戳、修改唯一字段、历史表、业务层校验,推荐方案1和方案3,适用于不同场景,感兴... 目录问题背景问题复现解决方案解决方案1.复合唯一索引 + 时间戳删除字段解决方案2:删除后修改唯一字

MySQL 主从复制部署及验证(示例详解)

《MySQL主从复制部署及验证(示例详解)》本文介绍MySQL主从复制部署步骤及学校管理数据库创建脚本,包含表结构设计、示例数据插入和查询语句,用于验证主从同步功能,感兴趣的朋友一起看看吧... 目录mysql 主从复制部署指南部署步骤1.环境准备2. 主服务器配置3. 创建复制用户4. 获取主服务器状态5

一文详解如何使用Java获取PDF页面信息

《一文详解如何使用Java获取PDF页面信息》了解PDF页面属性是我们在处理文档、内容提取、打印设置或页面重组等任务时不可或缺的一环,下面我们就来看看如何使用Java语言获取这些信息吧... 目录引言一、安装和引入PDF处理库引入依赖二、获取 PDF 页数三、获取页面尺寸(宽高)四、获取页面旋转角度五、判断

Spring Boot中的路径变量示例详解

《SpringBoot中的路径变量示例详解》SpringBoot中PathVariable通过@PathVariable注解实现URL参数与方法参数绑定,支持多参数接收、类型转换、可选参数、默认值及... 目录一. 基本用法与参数映射1.路径定义2.参数绑定&nhttp://www.chinasem.cnbs

MyBatis-Plus通用中等、大量数据分批查询和处理方法

《MyBatis-Plus通用中等、大量数据分批查询和处理方法》文章介绍MyBatis-Plus分页查询处理,通过函数式接口与Lambda表达式实现通用逻辑,方法抽象但功能强大,建议扩展分批处理及流式... 目录函数式接口获取分页数据接口数据处理接口通用逻辑工具类使用方法简单查询自定义查询方法总结函数式接口

MySql基本查询之表的增删查改+聚合函数案例详解

《MySql基本查询之表的增删查改+聚合函数案例详解》本文详解SQL的CURD操作INSERT用于数据插入(单行/多行及冲突处理),SELECT实现数据检索(列选择、条件过滤、排序分页),UPDATE... 目录一、Create1.1 单行数据 + 全列插入1.2 多行数据 + 指定列插入1.3 插入否则更

Redis中Stream详解及应用小结

《Redis中Stream详解及应用小结》RedisStreams是Redis5.0引入的新功能,提供了一种类似于传统消息队列的机制,但具有更高的灵活性和可扩展性,本文给大家介绍Redis中Strea... 目录1. Redis Stream 概述2. Redis Stream 的基本操作2.1. XADD

Spring StateMachine实现状态机使用示例详解

《SpringStateMachine实现状态机使用示例详解》本文介绍SpringStateMachine实现状态机的步骤,包括依赖导入、枚举定义、状态转移规则配置、上下文管理及服务调用示例,重点解... 目录什么是状态机使用示例什么是状态机状态机是计算机科学中的​​核心建模工具​​,用于描述对象在其生命