TS数据流PAT和PMT分析(转载)

转自:http://www.cnblogs.com/hjj801006/p/3837435.html

TS流,是基于packet的位流格式,每个packet是188个字节或者204个字 节(一般是188字节,204字节格式是在188字节的packet后面加上16字节的CRC数据,其他格式相同),解析TS流,先解析每个packet ,然后从一个packet中,解析出PAT的PID,根据PID找到PAT包,然后从PAT包中解析出PMT的PID,根据PID找到PMT包,在从 PMT包中解析出Video和Audio(也有的包括Teletext和EPG)的PID。然后根据PID找出相应的包。

所有packet格式都是同一的,包括packet header和packet datas。

packet header格式如下
typedef struct
{
unsigned sync_byte :  8位;                         /*8bits的同步字节,固定为0x47,表示后面是一个TS分组。*/
unsigned transport_error_indicator : 1位      /*1位的错误指示信息,1表示当前packet至少有1bit的传输错误,0表示所有数据都正确一般传输错误的话就不会处理这个包了。*/
unsigned payload_unit_start_indicator: 1 位 /*负载单元开始标志,*/
unsigned transport_priority: 1位                 /*1bit的传输优先级标志,1表示高优先级,0表示低优先级,传输机制可能用到,解码好像用不到*/
unsigned PID:13 位;                                /*13 bits的packet ID号码,唯一的号码对应不同的包,指出了这个包的有效负载数据的类型,告诉我们这个包传输的是什么内容*/
unsigned transport_scrambling_control : 2位; /*2 bits 的加密标志,00表示没有加密,其他表示已被加密,表示TS分组有效负载的加密模式。TS首部(也就是前面着32个 bit)是不应被加密的*/
unsigned
adaptation_field_control : 2位       /*
2bits的附加区域控制,表示TS分组首部后面是否跟随有调整字段和有效负载。01仅有有效负载,10仅含调整字段,11含有调整字段和有效负载。00
的话解码器不进行处理。空分组没有调整字段。*/
unsigned continuity_counter: 4位              
/*4bits的包递增计数器,范围是0-15,具有相同PID的TS分组传输时每次加1,到15后清0.不过有些时候是不计数的。如下:1
、TS分组无有效负载,2、复制的TS分组和原分组这个值一样。3、后面讲到的一个标志discontinuity——indicator为1时*/
}packet_header;

PAT数据结构
program_association_section()
{
table_id     //8位  固定为0x00,表示该表是PAT表
section_syntax_indicator  //1   段语法标志位,固定为1
‘0‘    //1   固定为1 为了防止和ISO13818Video流格式中的控制字冲突而设置的
reserved      //2      保留位,一般都是0
section_length    //12    段的大小,表示这个字节后面有用的字节数,包括CRC32.假如后面的字节加上前面的字节数少于188,后面会用0xFF填充,假如这个数值比较大,则PAT会分成几部分来传输。
transport_stream_id    //16 该传输流的ID,区别于一个网络中其他多路复用的流
reserved          //2
version_number      //5     范围0-31,表示PAT的版本号,标注当前节目的版本,这是个非常有用的参数,当检测到这个字段改变时,说明TS流中的节目已经改变了,程序必须重新收索节目
current_next_indicator    //1    表示发送的PAT是当前有效还是下一个PAT有效
section_number           //8   分段的号码,PAT可能分为多个段传输,第一段为00,以后每个分段加1,最大可能有256个分段
last_section_number      //8 最后一个分段的号码
for(i=0;i<N;i++)
{
program_number      //16  节目号
reserved          //3
if(program_number == ‘0‘)
{
network_PID         //13   网络信息表(NIT)的PID,网络信息表提供了该物理网络的一些信息,和电视台相关的,节目号为0时对应的PID为network_PID
}
else
{
 program_map_PID     //13   节目映射表的PID,节目号大于0时对应的PID,每个节目对应一个
}
CRC_32        //32   CRC32校验码
}

例子  47 40 00 1c 00 00 b0 15 13 f6 e7 00 00 00 00 e0 10 00 01 e0 20 00 02 e0 21 1a 34 b4 77 ff ff ff ff ......ff

其中,前4个字节是TS分组的头部,01000111 01000000 00000000 0001 1100 ,可以解析 出PID= (buf[1]&0x0f)<<8 | buf[2] =0x00,表示为该分组的有效负载是PAT。

五个字节00称为“指针域”,表示一个偏移量,即从后面第几个字节开始时PAT部分。为00表示后面紧跟着的就是PAT:00 b0 15 13 f6
e7 00 00 00 00 e0 10 00 01 e0 20 00 02 e0 21 1a 34 b4 77
(00000000
10110000 00010101 00010011 11110110 11100111 00000000 00000000 00000000
00000000 11100000 00010000 00000000 00000001 11100000 00100000 00000000
00000010 11100000 00100001 00011010 00110100 10110100 01110111)

利用PAT数据结构解析出PAT,可得到如下信息:

table_id  : 0x00
section_syntax_indicator 0x01
section_length:   0x015  表示后面有21个字节有用
transport_stream_id: 0x13f6
version_number  :0x13
current_next_indicator  :0x01
section_number  :0x00
last_section_number :0x00  表示只有一个分段。
program_number: 0x0000
network_PID: 0010
program_number: 0001
program_map_PID: 0020
program_number: 0002
program_map_PID: 0021

CRC_32: 1a34b477
此PAT只有一段,包含了三个节目,节目号0000对应于network_PID=0010
,节目号0001对应于program_map_PID =0020,节目号0002对应于program_map_PID
=0021,从实际的角度,我们应该把这三个节目号理解为三个频道,第一个频道中的内容是网络信息,第二、三个频道包含了节目信息。在数字电视中,一个频
道上可以有多个节目,后面的PMT即是告诉了我们某个频道中所有节目对应的PID。

于是现在就搜寻PID=0x0020的TS分组,即是频道2对应的PMT信息。

PMT数据结构:

TS_program_map_section()
{
table_id                       // 8     固定为0x02,表示该表是PMT
section_syntax_indicator     //1   段语法标志位,固定为1     
‘0‘        //  1                                           
reserved              //    2                                
section_length        
//  12    
表示这个字节后面有用的字节数,包括CRC32。假如后面的字节加上前面的字节数少于188,后面会用0XFF填充。假如这个数值比较大,则PAT会分成
几部分来传输。                            
program_number           //16   节目号,表示该PMT对应的节目                             
reserved                  // 2                               
version_number            
//5   
范围0-31,表示PAT的版本号,标注当前节目的版本,这是个非常有用的参数,当检测到这个字段改变时,说明TS流中的节目已经改变了,程序必须重新收
索节目                           
current_next_indicator      //1      表示发送的PAT是当前有效还是下一个PAT有效                        
section_number             // 8   分段的号码  固定为0x00                         
last_section_number        //8      最后分段的号码 固定为0x00                        
reserved                  //3                                
PCR_PID               //13         PCR(节目时钟参考)所在TS分组的PID,根据PID可以去搜索相应的TS分组,解出PCR信息。                          
reserved 4
program_info_length         //12    该节目的信息长度,在此字段之后可能会有一些字节描述该节目的信息                         
for (i=0; i<N; i++) {
  descriptor()
}
for (i=0;i<N1;i++) {
  stream_type                    //8   指示了PID为elementary_PID的PES分组中原始流的类型,比如视频流,音频流等                      
  reserved                       //3                         
  elementary_PID                //13     该节目中包括的视频流,音频流等对应的TS分组的PID                    
  reserved                 //4                               
  ES_info_length        //12         该节目相关原始流的描述符的信息长度                        
  for (i=0; i<N2; i++) {
   descriptor()
  }
}
CRC_32                       //32                                                       
}

}

例子PMT包
47 40 20 1c 00 02 b0 1f 00 01 e7 00 00 e1 00 f0 00 02 e1 00 f0 05 02 03 b2 44 5f 04 e1 10 f0 03 03 01 67 c9 ab c8 d2

前4个字节是TS分组头部, 第五个字节00是指针域。后面的就是PMT的内容了。
table_id: 02
section_syntax_indicator: 01
section_length: 01f
program_number: 0001
version_number: 13
current_next_indicator: 01
section_number: 00
last_section_number: 00
PCR_PID: 0100
program_info_length: 000
descriptor:
steam_type: 00
elementary_PID: 0001
ES_info_length: 000
descriptor:
steam_type: 02
elementary_PID: 0001
ES_info_length: 005
descriptor: 02 03 b2 44 5f
steam_type: 04
elementary_PID: 0011
ES_info_length: 003
descriptor: 03 01 67
CRC_32: c9abc8d2

可以看出,该节目号0001包含了三个流的信息,流类型分别为00,02,04,00的流为保留值,可以不考虑,02表示原始流为视频流,其
elementary_PID为0001,04表示原始流为音频流,其elementary_PID为0011,两个流分别还带有
descriptor(描述符),说明了该原始流的一些信息。

大部分是转载的别人的,中间加了些自己的理解。非常感谢给予我帮助的大神们!谢谢你们。

时间: 2024-08-07 14:43:27

TS数据流PAT和PMT分析(转载)的相关文章

TS数据流PAT和PMT分析

TS流,是基于packet的位流格式,每个packet是188个字节或者204个字节(一般是188字节,204字节格式是在188字节的packet后面加上16字节的CRC数据,其他格式相同),解析TS流,先解析每个packet ,然后从一个packet中,解析出PAT的PID,根据PID找到PAT包,然后从PAT包中解析出PMT的PID,根据PID找到PMT包,在从PMT包中解析出Video和Audio(也有的包括Teletext和EPG)的PID.然后根据PID找出相应的包. 所有packet

修复FFMPEG 复用 PAT、PMT发送间隔小于25ms的错误

分析ffmpeg源码 分析问题 mpegtsenc.c 找到发送PAT.PMT的函数 /* send SDT, PAT and PMT tables regularly */ static void retransmit_si_info(AVFormatContext *s, int force_pat, int64_t dts) { MpegTSWrite *ts = s->priv_data; int i; if (++ts->sdt_packet_count == ts->sdt_

递归算法的时间复杂度分析 转载

在算法分析中,当一个算法中包含递归调用时,其时间复杂度的分析会转化为一个递归方程求解.实际上,这个问题是数学上求解渐近阶的问题,而递归方程的形式多种多样,其求解方法也是不一而足,比较常用的有以下四种方法: (1)代入法(Substitution Method)        代入法的基本步骤是先推测递归方程的显式解,然后用数学归纳法来验证该解是否合理.        (2)迭代法(Iteration Method)        迭代法的基本步骤是迭代地展开递归方程的右端,使之成为一个非递归的和

mpeg2 ts流PAT,PMT,SDT的定义

转自: http://blog.sina.com.cn/s/blog_5ea0192f0100vo15.html 更具体准确的信息请参考iso13818-1,都在里面定义的 PAT的定义:Table_id:为8bit字段,该字段标识节目关联分段,对于PAT,置为0x00. Section_syntax_indicator:1bit字段,对于PAT,置为0x01. Reserved:2bit保留字段,用于将来扩展,置为11. Section_length:12bit字段,指示当前section的长

门控时钟-理论分析 ---- 转载

转载自:http://www.chipsbank.com/news_detail/newsId=123.html 门控的基本要求: 1. 所需要的沿(对于正沿触发的寄存器是正沿,对于负沿触发的寄存器是负沿)不增加,不减少: 1. 不会产生毛刺: 1. 使用后功耗要能够降低: 1. 最好面积还会减小. 1. 上升沿触发的门控时钟的结构研究:应用与上升沿触发的寄存器的门控. 1. 直接与门结构: 1. 高电平使能Latch + 与门结构: 1. 低电平使能Latch + 与门结构: 1. 波形研究:

EasyUI学习总结(三)——easyloader源码分析(转载)

声明:这一篇文章是转载过来的,转载地址忘记了,原作者如果看到了,希望能够告知一声,我好加上去! easyloader模块是用来加载jquery easyui的js和css文件的,而且它可以分析模块的依赖关系,先加载依赖项.模块加载好了会调用parse模块来解析页面.把class是easyui开头的标签都转化成easyui的控件. 先看Demo1例子,再分析源代码. 1 <!DOCTYPE html> 2 <html> 3 <head> 4 <title>ea

存储分析--- 转载

2013年SSD行业发展总结[转载] (2014-06-09 23:33:07) 转载▼ 标签: flash ssd 存储 分类: SSD 1.概述 2013年是Flash存储领域非常不平凡的一年,Flash盘阵厂商Violin Memory上市之后就股票大跌,CEO被迫辞退.混合存储厂商Nimble Storage终于在年末成功上市.Cisco在自己的老本行之外于去年下半年收购全Flash盘阵厂商Whiptail,开始入足存储领域.FusionIO在去年一直收益不佳,并且引发高层动荡,创始人兼

Graphic32中TBitmap32.TextOut性能分析[转载]

转载:http://blog.csdn.net/avan_lau/article/details/6958497 最近在分析软件中画线效率问题,发现在画一些标志性符号的方法,存在瓶颈,占用较大的时间.而画这些符号的,则最终是调用TBitMap32.TextOut.大致状况如下: TextOutSignWithAngle为画特殊符号方法,中间调用了API:GetObject用于获取LogFont信息,然后填充logFont角度信息,再调用API:CreateFontIndirect,设置创建后的字

HashMap源码分析(转载)

一.HashMap概述 HashMap基于哈希表的 Map 接口的实现.此实现提供所有可选的映射操作,并允许使用 null 值和 null 键.(除了不同步和允许使用 null 之外,HashMap 类与 Hashtable 大致相同.)此类不保证映射的顺序,特别是它不保证该顺序恒久不变. 值得注意的是HashMap不是线程安全的,如果想要线程安全的HashMap,可以通过Collections类的静态方法synchronizedMap获得线程安全的HashMap. Map map = Coll