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

From: http://www.cnweblog.com/fly2700/archive/2012/02/23/319718.html

RFC3984是H.264的baseline码流在RTP方式下传输的规范,这里只讨论FU-A分包方式,以及从RTP包里面得到H.264数据和AAC数据的方法。

1、单个NAL包单元

12字节的RTP头后面的就是音视频数据,比较简单。一个封装单个NAL单元包到RTP的NAL单元流的RTP序号必须符合NAL单元的解码顺序。

2、FU-A的分片格式
数据比较大的H264视频包,被RTP分片发送。12字节的RTP头后面跟随的就是FU-A分片:
FU indicator有以下格式:
      +---------------+
      |0|1|2|3|4|5|6|7|
      +-+-+-+-+-+-+-+-+
      |F|NRI|  Type   |
      +---------------+
   FU指示字节的类型域 Type=28表示FU-A。。NRI域的值必须根据分片NAL单元的NRI域的值设置。
 
   FU header的格式如下:
      +---------------+
      |0|1|2|3|4|5|6|7|
      +-+-+-+-+-+-+-+-+
      |S|E|R|  Type   |
      +---------------+
   S: 1 bit
   当设置成1,开始位指示分片NAL单元的开始。当跟随的FU荷载不是分片NAL单元荷载的开始,开始位设为0。
   E: 1 bit
   当设置成1, 结束位指示分片NAL单元的结束,即, 荷载的最后字节也是分片NAL单元的最后一个字节。当跟随的FU荷载不是分片NAL单元的最后分片,结束位设置为0。
   R: 1 bit
   保留位必须设置为0,接收者必须忽略该位。
   Type: 5 bits
   NAL单元荷载类型定义见下表


表1.  单元类型以及荷载结构总结
      Type   Packet      Type name                       
      ---------------------------------------------------------
      0      undefined                                    -
      1-23   NAL unit    Single NAL unit packet per H.264  
      24     STAP-A     Single-time aggregation packet    
      25     STAP-B     Single-time aggregation packet    
      26     MTAP16    Multi-time aggregation packet     
      27     MTAP24    Multi-time aggregation packet     
      28     FU-A      Fragmentation unit                
      29     FU-B      Fragmentation unit                 
      30-31  undefined                                    -
3、拆包和解包

拆包:当编码器在编码时需要将原有一个NAL按照FU-A进行分片,原有的NAL的单元头与分片后的FU-A的单元头有如下关系:
原始的NAL头的前三位为FU indicator的前三位,原始的NAL头的后五位为FU header的后五位,FU indicator与FU header的剩余位数根据实际情况决定。
 
解包:当接收端收到FU-A的分片数据,需要将所有的分片包组合还原成原始的NAl包时,FU-A的单元头与还原后的NAL的关系如下:
还原后的NAL头的八位是由FU indicator的前三位加FU header的后五位组成,即:
nal_unit_type = (fu_indicator & 0xe0) | (fu_header & 0x1f)

4、代码实现

从RTP包里面得到H264视频数据的方法:

 
 // 功能:解码RTP H.264视频
 // 参数:1.RTP包缓冲地址 2.RTP包数据大小 3.H264输出地址 4.输出数据大小
 // 返回:true:表示一帧结束  false:FU-A分片未结束或帧未结束 
 
#define  RTP_HEADLEN 12 
 bool  UnpackRTPH264( void   *  bufIn,  int  len,   void **  pBufOut,   int   *  pOutLen)
 {
     * pOutLen  =   0 ;
     if  (len  <  RTP_HEADLEN)
     {
         return   false ;
    } 

 
    unsigned  char *  src  =  (unsigned  char * )bufIn  +  RTP_HEADLEN;
    unsigned  char  head1  =   * src; // 获取第一个字节 
 
    unsigned  char  head2  =   * (src + 1 ); // 获取第二个字节 
 
    unsigned  char  nal  =  head1  &   0x1f ; // 获取FU indicator的类型域, 
 
    unsigned  char  flag  =  head2  &   0xe0 ; // 获取FU header的前三位,判断当前是分包的开始、中间或结束 
 
    unsigned  char  nal_fua  =  (head1  &   0xe0 )  |  (head2  &   0x1f ); // FU_A nal 
 
     bool  bFinishFrame  =   false ;
     if  (nal == 0x1c ) // 判断NAL的类型为0x1c=28,说明是FU-A分片 
 
     // fu-a 
 
         if  (flag == 0x80 ) // 开始 
 
         {
             * pBufOut  =  src - 3 ;
             * (( int * )( * pBufOut))  =   0x01000000  ; // zyf:大模式会有问题 
 
             * (( char * )( * pBufOut) + 4 )  =  nal_fua;
             *  pOutLen  =  len  -  RTP_HEADLEN  +   3 ;
        } 

         else   if (flag == 0x40 ) // 结束 
 
         {
             * pBufOut  =  src + 2 ;
             *  pOutLen  =  len  -  RTP_HEADLEN  -   2 ;
        } 

         else // 中间 
 
         {
             * pBufOut  =  src + 2 ;
             *  pOutLen  =  len  -  RTP_HEADLEN  -   2 ;
        } 

    } 

     else // 单包数据 
 
     {
         * pBufOut  =  src - 4 ;
         * (( int * )( * pBufOut))  =   0x01000000 ; // zyf:大模式会有问题 
 
         *  pOutLen  =  len  -  RTP_HEADLEN  +   4 ;
    } 

 
    unsigned  char *  bufTmp  =  (unsigned  char * )bufIn;
     if  (bufTmp[ 1 ]  &   0x80 )
     {
        bFinishFrame  =   true ; // rtp mark 
 
    } 

     else 
      {
        bFinishFrame  =   false ;
    } 

     return  bFinishFrame;
  


从RTP包里面得到AAC音频数据的方法:

//功能:解RTP AAC音频包,声道和采样频率必须知道。
//参数:1.RTP包缓冲地址 2.RTP包数据大小 3.H264输出地址 4.输出数据大小
//返回:true:表示一帧结束  false:帧未结束 一般AAC音频包比较小,没有分片。
bool UnpackRTPAAC(void * bufIn, int recvLen, void** pBufOut,  int* pOutLen)
{
    unsigned char*  bufRecv = (unsigned char*)bufIn;
    //char strFileName[20];
    
    unsigned char ADTS[] = {0xFF, 0xF1, 0x00, 0x00, 0x00, 0x00, 0xFC}
    int audioSamprate = 32000;//音频采样率
    int audioChannel = 2;//音频声道 1或2
    int audioBit = 16;//16位 固定
    switch(audioSamprate)
    {
    case  16000:
        ADTS[2] = 0x60;
        break;
    case  32000:
        ADTS[2] = 0x54;
        break;
    case  44100:
        ADTS[2] = 0x50;
        break;
    case  48000:
        ADTS[2] = 0x4C;
        break;
    case  96000:
        ADTS[2] = 0x40;
        break;
    default:
        break;
    }

    ADTS[3] = (audioChannel==2)?0x80:0x40;

    int len = recvLen - 16 + 7;
    len <<= 5;//8bit * 2 - 11 = 5(headerSize 11bit)
    len |= 0x1F;//5 bit    1            
    ADTS[4] = len>>8;
    ADTS[5] = len & 0xFF;
    *pBufOut = (char*)bufIn+16-7;
    memcpy(*pBufOut, ADTS, sizeof(ADTS));
    *pOutLen = recvLen - 16 + 7;

    unsigned char* bufTmp = (unsigned char*)bufIn;
    bool bFinishFrame = false;
    if (bufTmp[1] & 0x80)
    {
        //DebugTrace::D("Marker");
        bFinishFrame = true;
    }

    else
    {
        bFinishFrame = false;
    }

    return true;
}

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

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

相关文章

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…

VLC 源代码结构

From&#xff1a; http://blog.csdn.net/jack_incredible/article/details/7301155 译者&#xff1a;捞刀客 VLC源代码目录树 本节介绍VLC源代码目录树结构&#xff0c;以期使开发者对源代码的分布结构有一个整体的认识。下面的目录按照字母顺序列出&#xff0c;右边为该目录包…

让开!!!谁也别拦着我封装React组件!

1简介 我是歌谣 放弃很容易 但是坚持一定很酷 喜欢我就一键三连哈 2前言 在我们的工作生活中 每次学习一个框架我们就不免要封装组件 而具备封装一个完美组件的能力 我称之为"优秀" 3准备工作 父组件 <Geyao/> 子组件 import React, { Component } from re…

UI-UIButton、UILable、UITextField总结

UIButton按钮 第一、UIButton的定义 UIButton *button[[UIButton buttonWithType:(UIButtonType); 能够定义的button类型有以下6种&#xff0c; typedef enum { UIButtonTypeCustom 0, 自定义风格 UIButtonTypeRoundedRect, 圆角矩形 UIButtonTypeDetailDisclosure, 蓝色小…

JAVA中线程同步的方法

1、wait方法。 2、notify方法和notifyAll方法。3、synchronized关键字。

ffmpeg教程

From&#xff1a; http://blog.csdn.net/cffishappy/article/details/7352898 概要 电影文件有很多基本的组成部分。首先&#xff0c;文件本身被称为容器Container&#xff0c;容器的类型决定了信息被存放在文件中的位置。AVI和Quicktime就是容器的例子。接着&#xff0c;你有一…