用实例分析H264 RTP payload

From: http://blog.csdn.net/zblue78/article/details/5948538

H264的RTP中有三种不同的基本负载(Single NAL,Non-interleaved,Interleaved)

应用程序可以使用第一个字节来识别。

 

在SDP中也说明了本次会话的属性

SDP 参数 
下面描述了如何在 SDP 中表示一个 H.264 流:
. "m=" 行中的媒体名必须是 "video"
. "a=rtpmap" 行中的编码名称必须是 "H264".
. "a=rtpmap" 行中的时钟频率必须是 90000.
. 其他参数都包括在 "a=fmtp" 行中.
如:
m=video 49170 RTP/AVP 98
a=rtpmap:98 H264/90000
a=fmtp:98 profile-level-id=42A01E; packetization-mode=1; sprop-parameter-sets=Z0IACpZTBYmI,aMljiA==

下面介绍一些常用的参数.
3.1 packetization-mode: 
表示支持的封包模式. 
当 packetization-mode 的值为 0 时或不存在时, 必须使用单一 NALU 单元模式.
当 packetization-mode 的值为 1 时必须使用非交错(non-interleaved)封包模式.

当 packetization-mode 的值为 2 时必须使用交错(interleaved)封包模式.

 

 

每个打包方式允许的NAL单元类型总结(yes = 允许, no = 不允许, ig = 忽略)
      Type   Packet    Single NAL    Non-Interleaved    Interleaved
                       Unit Mode           Mode             Mode
      -------------------------------------------------------------

      0      undefined     ig               ig               ig
      1-23   NAL unit     yes              yes               no
      24     STAP-A        no              yes               no
      25     STAP-B        no               no              yes
      26     MTAP16        no               no              yes
      27     MTAP24        no               no              yes
      28     FU-A          no              yes              yes
      29     FU-B          no               no              yes
      30-31  undefined     ig               ig               ig


这个参数不可以取其他的值.

3.2 sprop-parameter-sets: SPS,PPS
这个参数可以用于传输 H.264 的序列参数集和图像参数 NAL 单元. 这个参数的值采用 Base64 进行编码. 不同的参数集间用","号隔开.


3.3 profile-level-id:
这个参数用于指示 H.264 流的 profile 类型和级别. 由 Base16(十六进制) 表示的 3 个字节. 第一个字节表示 H.264 的 Profile 类型, 第三个字节表示 H.264 的 Profile 级别:

3.4 max-mbps: 
这个参数的值是一个整型, 指出了每一秒最大的宏块处理速度.

 

Rtp payload的第一个字节和264的NALU类似

 

+---------------+
|0|1|2|3|4|5|6|7|
+-+-+-+-+-+-+-+-+
|F|NRI| Type    |
+---------------+

 

F: 1 个比特.

forbidden_zero_bit. 在 H.264 规范中规定了这一位必须为 0.

NRI: 2 个比特.

nal_ref_idc. 取 00 ~ 11, 似乎指示这个 NALU 的重要性, 如 00 的 NALU 解码器可以丢弃它而不影响图像的回放. 不过一般情况下不太关心这个属性.

Type: 5 个比特.

nal_unit_type. 这个 NALU 单元的类型. 简述如下:
0     没有定义
1-23 NAL单元 单个 NAL 单元包.
24    STAP-A   单一时间的组合包
24    STAP-B   单一时间的组合包
26    MTAP16   多个时间的组合包
27    MTAP24   多个时间的组合包
28    FU-A     分片的单元
29    FU-B     分片的单元
30-31 没有定义

例子:

0x5C=01011100 (F:0  NRI:10  Type:28) FU-A

0x41=01000001 (F:0  NRI:10  Type:01)Single NAL

0x68=01000100 (F:0  NRI:10  Type:08)Single NAL

 

Single NAL Unit Mode :Type[1-23] packetization-mode=0


对于 NALU 的长度小于 MTU 大小的包, 一般采用单一 NAL 单元模式.
对于一个原始的 H.264 NALU 单元常由 [Start Code] [NALU Header] [NALU Payload] 三部分组成, 其中 Start Code 用于标示这是一个 NALU 单元的开始, 必须是 "00 00 00 01" 或 "00 00 01", NALU 头仅一个字节, 其后都是 NALU 单元内容.
打包时去除 "00 00 01" 或 "00 00 00 01" 的开始码, 把其他数据封包的 RTP 包即可.

 

 

 

 

Non-interleaved Mode:Type[1-23,24,28] packetization-mode=1

       Type=[1-23]的情况 参照 packetization-mode=0

Type=28 FU-A

+---------------+---------------+
|0|1|2|3|4|5|6|7|0|1|2|3|4|5|6|7|
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|F|NRI| Type:28 |S|E|R| Type    |
+---------------+---------------+

 

S:开始标志

E:结束标志 (与 Mark相同)

R:必须为0

 

Type:h264的NALU Type

 

例:

 

0x7C85=01111100 10000101 (开始包)

0x7C05=01111100 00000101 (中间包)

0x7C45=01111100 01000101 (结束包)


Type=23  STAP-A

0               1             2                 3
|0 1 2 3 4 5 6 7|8 9 0 1 2 3 4|5 6 7 8 9 0 1 2 3|4 5 6 7 8 9 0 1
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                          RTP Header                           |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|STAP-A NAL HDR |         NALU 1 Size           | NALU 1 HDR    |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                         NALU 1 Data                           |
:                                                               :
+               +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+

|               | NALU 2 Size                   | NALU 2 HDR    |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                         NALU 2 Data                           |
:                                                               :
|                               +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|                               :...OPTIONAL RTP padding        |
+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+

 

[cpp] view plaincopy
  1. class H264NALUParser    
  2. {  
  3. public:  
  4.     H264NALUParser(int width , int height);  
  5.     H264NALUParser();  
  6.     virtual ~H264NALUParser();  
  7.     void SetBuffer(unsigned char * buffer,int len,int f,int nri,int type);  
  8.     BOOL readOnePacket(unsigned char * buffer,int &len);  
  9.     BOOL isPacketOutstanding();  
  10. private:  
  11.     unsigned char * m_pNaluBuffer;  // NALU数据指向的缓冲区的指针  
  12.     unsigned int m_nNaluSize;       // NALU数据缓冲区的大小  
  13.     unsigned char * m_pCurNaluPos;  //指向下一个数据包要读取的缓冲区指针  
  14.     int m_nFrameWidth;  
  15.     int m_nFrameHeight;  
  16.     int m_nPacketCounts;  
  17.     int m_nPacketSeqNum;  
  18.     int m_nF;  
  19.     int m_nNRI;  
  20.     int m_nType;  
  21.     enum {  
  22.         STAP_A = 24,  
  23.         STAP_B = 25,  
  24.         MTAP16 = 26,  
  25.         MTAP24 = 27,  
  26.         FU_A   = 28,  
  27.         FU_B   = 29  
  28.     };  
  29. };    
  30.   
  31. // class H264NALUParser /  
  32. H264NALUParser::H264NALUParser(int width , int height)  
  33. {  
  34.     m_nFrameWidth   = width;  
  35.     m_nFrameHeight  = height;  
  36.     m_pNaluBuffer   = NULL;  
  37.     m_nNaluSize     = 0;  
  38.     m_nPacketCounts = 0;  
  39.     m_nPacketSeqNum = 0;  
  40.     m_nF            = 0;  
  41.     m_nNRI          = 0;  
  42.     m_nType         = 0;  
  43. }  
  44. H264NALUParser::H264NALUParser()  
  45. {  
  46.     m_pNaluBuffer   = NULL;  
  47.     m_nNaluSize     = 0;  
  48.     m_nPacketCounts = 0;  
  49.     m_nPacketSeqNum = 0;  
  50.     m_nF            = 0;  
  51.     m_nNRI          = 0;  
  52.     m_nType         = 0;  
  53. }  
  54. H264NALUParser::~H264NALUParser()  
  55. {  
  56. }  
  57. void H264NALUParser::SetBuffer(unsigned char * buffer,int len,int f,int nri,int type)  
  58. {  
  59.     m_pNaluBuffer   = buffer;  
  60.     m_nNaluSize     = len;  
  61.     m_nF            = f;  
  62.     m_nNRI          = nri;  
  63.     m_nType         = type;  
  64.     m_pCurNaluPos   = m_pNaluBuffer;  
  65.     m_nPacketCounts = (m_nNaluSize + H264_MTU - 1) / H264_MTU;  
  66.     m_nPacketSeqNum = 0;  
  67. }  
  68. BOOL H264NALUParser::readOnePacket(unsigned char * buffer,int &len)  
  69. {  
  70.     if(m_pCurNaluPos >= m_pNaluBuffer + m_nNaluSize)  
  71.     {  
  72.         return FALSE;  
  73.     }  
  74.     struct h264_rtp_hdr header;  
  75.     int headersize;  
  76.     unsigned char * pCurBuf = buffer;  
  77.     if(m_nNaluSize <= H264_MTU)// Single NALU  
  78.     {  
  79.         header.SingleNALU.f     = m_nF;  
  80.         header.SingleNALU.nri   = m_nNRI;  
  81.         header.SingleNALU.type  = m_nType;  
  82.         headersize = sizeof(header.SingleNALU);  
  83.         memcpy(pCurBuf,&(header.SingleNALU),headersize);  
  84.         pCurBuf += headersize;  
  85.     }  
  86.     else// FU-A  
  87.     {  
  88.         header.FU_A.f           = m_nF;  
  89.         header.FU_A.nri         = m_nNRI;  
  90.         header.FU_A.type_indicator  = FU_A;  
  91.         if(0 == m_nPacketSeqNum)  
  92.         {  
  93.             header.FU_A.s       = 1;  
  94.         }  
  95.         else  
  96.         {  
  97.             header.FU_A.s       = 0;  
  98.         }  
  99.         if(m_nPacketSeqNum == m_nPacketCounts - 1)  
  100.         {  
  101.             header.FU_A.e       = 1;  
  102.         }  
  103.         else  
  104.         {  
  105.             header.FU_A.e       = 0;  
  106.         }  
  107.         header.FU_A.r           = 0;  
  108.         header.FU_A.type_header = m_nType;  
  109.         //  
  110.         headersize = sizeof(header.FU_A);  
  111.         memcpy(pCurBuf,&(header.FU_A),headersize);  
  112.         pCurBuf += headersize;  
  113.     }  
  114.     if(m_nPacketSeqNum < m_nPacketCounts - 1)  
  115.     {  
  116.         memcpy(pCurBuf,m_pCurNaluPos,H264_MTU);  
  117.         m_pCurNaluPos += H264_MTU;  
  118.         len = headersize + H264_MTU;  
  119.     }  
  120.     else  
  121.     {  
  122.         int remainLen = m_nNaluSize % H264_MTU;  
  123.         if(0 == remainLen)  
  124.         {  
  125.             remainLen = H264_MTU;  
  126.         }  
  127.         memcpy(pCurBuf,m_pCurNaluPos,remainLen);  
  128.         m_pCurNaluPos += remainLen;  
  129.         len = headersize + remainLen;  
  130.     }  
  131.     m_nPacketSeqNum ++;  
  132.     return TRUE;  
  133. }  
  134. BOOL H264NALUParser::isPacketOutstanding()  
  135. {  
  136.     return (m_nPacketSeqNum < m_nPacketCounts);  
  137. }  

Interleaved Mode:Type[26-29] packetization-mode=2

 

待续

 

STAP-B

MTAP16

MTAP24

FU-B


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/404067.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[ECMAScript] module、export、import分别有什么作用?

[ECMAScript] module、export、import分别有什么作用&#xff1f; import导入模块功能。import导入的只是一个只读引用&#xff0c;等到脚本执行时根据引用才到模块里去取值。这也就是import可以按需加载的原因。 export 指定输出的代码 module 要在js中使用ES6的import。<…

实验4 颜色、字符串资源的使用

课程名称 基于Android平台移动互联网开发 实验日期 3月25 实验项目名称 颜色、字符串资源的使用 实验地点 S3010 实验类型 □验证型 √设计型 □综合型 学 时 1学时 一、实验目的及要求&#xff08;本实验所涉及并要求掌握的知识点&#xff09; 掌握Androi…

LAMP平台下构建Postfix邮件服务器

前言&#xff1a; 本人之前已经做过这个实验&#xff0c;只是版本不同&#xff0c;今天本来想用全新版本的源码包做实验&#xff0c;但是以httpd-2.4.2.tar.gz的源码包在配置虚拟主机这一块我不知道如何配置&#xff0c;按照以前的配置总是报错&#xff1a;AH00548: NameVirtua…

[NodeJs] 如何使用nodejs对base64进行编解码?

[NodeJs] 如何使用nodejs对base64进行编解码&#xff1f; const str hi//编码 const buff Buffer.from(str, utf-8) console.log(buff)const base64 buff.toString(base64) console.log(base64)// 解码 const buff2 Buffer.from(base64, base64) console.log(---, buff2)c…

FU-A分包方式,以及从RTP包里面得到H.264数据和AAC数据的方法

From: http://www.cnweblog.com/fly2700/archive/2012/02/23/319718.html RFC3984是H.264的baseline码流在RTP方式下传输的规范&#xff0c;这里只讨论FU-A分包方式&#xff0c;以及从RTP包里面得到H.264数据和AAC数据的方法。 1、单个NAL包单元 12字节的RTP头后面的就是音视频…

Maven 手动添加 JAR 包到本地仓库

Maven 手动添加 JAR 包到本地仓库Maven 确确实实是个好东西&#xff0c;用来管理项目显得很方便&#xff0c;但是如果是通过 Maven 来远程下载 JAR 包的话&#xff0c;我宿舍的带宽是4兆的&#xff0c;4个人共用&#xff0c;有时候用 Maven 来远程下载 JAR 包会显得很慢&#x…

[NodeJs] 如果发现node_modules中有个模块代码有bug,你该怎么办?

[NodeJs] 如果发现node_modules中有个模块代码有bug&#xff0c;你该怎么办&#xff1f; 删除node_modules文件夹 重新npm install 如果本身代码有问题&#xff0c;去github对应的模块库提issues 个人简介 我是歌谣&#xff0c;欢迎和大家一起交流前后端知识。放弃很容易&am…

好久不更新这个博客了。

今天又翻到了这里&#xff0c;呵呵&#xff0c;以前都是在csdn博客发表些博文的。不过有了著名的XX事件&#xff0c;也没有多大兴趣了。今天就把这里作为主阵地吧。转载于:https://www.cnblogs.com/yanpeng/archive/2012/05/30/2526094.html

IOS开发之格式化日期时间的使用 编程中常见问题

今天在做一个有关时间的一些开发的时候&#xff0c;遇见了一写问题&#xff0c;反正来说既是很简单的问题&#xff0c;但毕竟用了我一些时间去调错误&#xff0c;遂记录之。 本来是想用 NSDateFormat 来转换一下服务器返回的时间&#xff0c;然后在客户端显示一下。但是最后发现…

[NodeJs] 你有使用过npx吗?它主要解决什么问题?

[NodeJs] 你有使用过npx吗&#xff1f;它主要解决什么问题&#xff1f; npm从5.2开始增加了npx命令 node自带npm模块,npx命令可以直接使用,如果不能使用,需自行安装.$ npm install -g npx npx 的原理很简单&#xff0c;就是运行的时候&#xff0c;会到node_modules/.bin路径和…

如何在eclipse中装myeclipse的插件

本文介绍Eclipse插件的安装方法。Eclipse插件的安装方法大体有三种&#xff1a;直接复制、使用link文件&#xff0c;以及使用eclipse自带的图形界面的插件安装方法。AD&#xff1a;做为当下最流行的开源IDE之一&#xff0c;Eclipse的一大优势就在于其无数优秀的插件。一个好的插…

RTSP协议举例

From: http://www.cppblog.com/elva/archive/2010/08/13/123313.html 因为项目需要&#xff0c;学习了一下RTSP协议&#xff0c;为了防止以后忘记&#xff0c;就把学习过程和成果记载下来。期间参考了一些网上的资料&#xff0c;并分析了VLC的RTSP报文。 RTSP&#xff08;…

thread安全性(写的不错)

http://blog.csdn.net/aaa1117a8w5s6d/article/details/8295527转载于:https://www.cnblogs.com/freezone/p/5359171.html

[NodeJs] 如何获取项目的根路径?

[NodeJs] 如何获取项目的根路径&#xff1f; __dirname, process.cwd() 个人简介 我是歌谣&#xff0c;欢迎和大家一起交流前后端知识。放弃很容易&#xff0c; 但坚持一定很酷。欢迎大家一起讨论 主目录 与歌谣一起通关前端面试题

python中的help()的用法

help是一个内置函数&#xff0c;在Python中被自动加载的函数&#xff0c;参数分两种&#xff1a; 如果传一个字符串做参数的话&#xff0c;它会自动搜索以这个字符串命名的模块&#xff0c;方法&#xff0c;等。如果传入的是一个对象&#xff0c;就会显示这个对象的类型的帮助例…

H264编码 封装成MP4格式 视频流 RTP封包

From:http://www.cnblogs.com/ghw-NO1/archive/2012/08/28/2660848.html 一、概述 本文讲述的是对H264编码且封装成MP4格式的视频流进行RTP打包过程时需要了解的一些基本知识。 二、H264的基础知识 1.H264的编码格式 H.263 定义的码流结构是分级结构&#xff0c;共四层。自上而…

hadoop 分布式缓存

Hadoop 分布式缓存实现目的是在所有的MapReduce调用一个统一的配置文件&#xff0c;首先将缓存文件放置在HDFS中&#xff0c;然后程序在执行的过程中会可以通过设定将文件下载到本地具体设定如下&#xff1a; public static void main(String[] arge) throws IOException, Clas…

在Eclipse中查看JDK类库的源代码

转自&#xff1a;http://www.cnblogs.com/wuhenke/archive/2011/06/22/2087134.html 核心提示&#xff1a;在Eclipse中查看JDK类库的源代码&#xff01;&#xff01;&#xff01; 设置&#xff1a; 1.点 window- Preferences - Java - Installed JRES 2.此时Installed JRES右边…

[单选题]PHP函数,mail($param1, $param2, $param3),其中的$param2参数包含什么?

信息的内容 信息的发送地址 信息的回复地址 信息的主题正确答案&#xff1a;转载于:https://www.cnblogs.com/pizishui/p/5361848.html

不使用加减乘除实现加法

思路&#xff1a; 例如: a5&#xff0c;b9&#xff0c;ab14 a转换为二进制形式为101&#xff0c;b转换为二进制形式为1001&#xff0c;其和转换为二进制形式为1110。 对于二进制形式的相加&#xff0c;可分两步进行操作&#xff1a; 1&#xff09;先不考虑进位&#xff0c;则01…