1. 前言
1.1 工具准备:
1.2 界面:
1) UltraEdit打开ts文件(如果发现打开的是一堆乱请用ctrl +H以16进制打开):
2) EStreamAnalyzer打开TS文件如下:
2. 分析TS包:
传输流包长度必须是188字节,sync_byte必须是0x47。任何有用的数据都不会是FF。
2.1 第一个TS包:
PAT是Program
Association Table的简称,即”节目关联表”. PAT定义了当前TS流中所有的节目,其PID恒为0x00,它是PSI信息的根节点,要查找节目必须从PAT表开始.PAT表描述了当前TS流中包含什么样的PID,主要包括当前流的NIT表格的PID,当前流中有多少个不同类型PMT表,每个PMT表对应的频道号等信息.
2.2 TS包头:
TS包头只有4个字节(47 60 00 10),除掉第一个字节0x47,剩下就3个,重新分组如下:
0 1 1 0000000000000 00 01 0000
|
sync_byte |
同步字节 |
0x47: |
|
transport_error_indicator |
传输错误标识 |
0: |
|
payload_unit_start_indicator |
负载单元开始标识 |
1:该字段用来表示TS包的有效净荷带有PES包或者PSI数据的情况。 当TS包带有PES包数据时,payload_unit_start_indicator具有以下特点:置为1,表示TS包的有效净荷以PES包的第一个字节开始;置为0,表示TS包的开始不是PES包。 当TS包带有PSI数据时,payload_unit_start_indicator具有以下特点:置为1,表示TS包带有PSI部分的第一个字节,即第一个字节带有指针pointer_field;置为0,表示TS包不带有一个PSI部分的第一个字节,即在有效净荷中没有指针pointer_field。 对于空包的包,payload_unit_start_indicator应该置为0。 |
|
transport_priority |
传输优先级 |
1: |
|
pid |
PID |
0x000:即为0,表示PAT |
|
transport_scrambling_control |
传输扰乱控制 |
00: |
|
adaptation_field_control |
自适应区域控制 |
00:是保留值。 |
|
continuity_counter |
连续计数器 |
0x0: |
|
同步字节 |
传输错误标识 |
负载单元开始标识 |
传输优先级 |
PID |
传输扰乱控制 |
自适应区域控制 |
连续计数器 |
|
8 |
1 |
1 |
1 |
13 |
2 |
2 |
4 |
|
第1个字节 |
第2、3个字节 |
第4个字节 |
|||||
|
table_id |
0x00:对于PAT只能是0x00 |
|
section_syntax_indicator |
1:应设置为‘1’ |
|
0 |
0: |
|
reserved |
11: |
|
section_length |
0000 0000 1101:为0xD,13 该字段的头两个比特必为‘00’,剩余10比特指定该分段的字节数,紧随分段长度字段开始,并包括CRC。此字段中,该值应不超过1021(0x3FD) |
|
transport_stream_id |
0x00:该字节充当标签,标识网络内此传输流有别于任何其他路复用流。其值由用户规定。 |
|
reserved |
11: |
|
version_number |
00000:一旦PAT有变化,版本号加1 |
|
current_next_indicator |
1:表示传送的PAT当前可以使用,若为0表示下一个表有效 |
|
section_number |
0x00:给出了该分段的数目。在PAT中的第一个分段的section_number为0x00,PAT中每一分段将加1。 |
|
last_section_number |
0x00:该字段指出了最后一个分段号。在整个PAT中即分段的最大数目。 |
|
program_number |
0x0001:这个为PMT。该字段指出了节目对于那个program_map_PID是可以使用的。如果是0x0000,那么后面的PID是网络PID,否则其他值由用户定义。 |
|
reserved |
111: |
|
program_map_PID |
0 0000 1000 0001:0x81该字段指出TS包中的PID值。 |
|
table_id |
8 |
第1个字节 |
|
section_syntax_indicator |
1 |
第2、3个字节 |
|
0 |
1 |
|
|
reserved |
2 |
|
|
section_length |
12 |
|
|
transport_stream_id |
16 |
第4、5个字节 |
|
reserved |
2 |
第6个字节 |
|
version_number |
5 |
|
|
current_next_indicator |
1 |
|
|
section_number |
8 |
第7个字节 |
|
last_section_number |
8 |
第8个字节 |
|
循环开始 |
||
|
program_number |
16 |
2个字节 |
|
reserved |
3 |
2个字节 |
|
network_id(节目号为0时) program_map_PID(节目号为其他时) |
13 |
|
|
循环结束 |
||
|
CRC_32 |
32 |
4个字节 |
3. 实现:
3.1 PAT表格定义如下:
- typedef struct TS_PAT_Program
- {
- unsigned program_number :16; //节目号
- unsigned program_map_PID :13; //节目映射表的PID,节目号大于0时对应的PID,每个节目对应一个
- }TS_PAT_Program;
- typedef struct TS_PAT
- {
- unsigned table_id : 8; //固定为0x00 ,标志是该表是PAT
- unsigned section_syntax_indicator : 1; //段语法标志位,固定为1
- unsigned zero : 1; //0
- unsigned reserved_1 : 2; // 保留位
- unsigned section_length : 12; //表示这个字节后面有用的字节数,包括CRC32
- unsigned transport_stream_id : 16; //该传输流的ID,区别于一个网络中其它多路复用的流
- unsigned reserved_2 : 2;// 保留位
- unsigned version_number : 5; //范围0-31,表示PAT的版本号
- unsigned current_next_indicator : 1; //发送的PAT是当前有效还是下一个PAT有效
- unsigned section_number : 8; //分段的号码。PAT可能分为多段传输,第一段为00,以后每个分段加1,最多可能有256个分段
- unsigned last_section_number : 8; //最后一个分段的号码
- std::vector<TS_PAT_Program> program;
- unsigned reserved_3 : 3; // 保留位
- unsigned network_PID : 13; //网络信息表(NIT)的PID,节目号为0时对应的PID为network_PID
- unsigned CRC_32 : 32; //CRC32校验码
- } TS_PAT;
3.3 解析代码如下:
- HRESULT CTS_Stream_Parse::adjust_PAT_table( TS_PAT * packet, unsigned char * buffer)
- {
- packet->table_id = buffer[0];
- packet->section_syntax_indicator = buffer[1] >> 7;
- packet->zero = buffer[1] >> 6 & 0x1;
- packet->reserved_1 = buffer[1] >> 4 & 0x3;
- packet->section_length = (buffer[1] & 0x0F) << 8 | buffer[2];
- packet->transport_stream_id = buffer[3] << 8 | buffer[4];
- packet->reserved_2 = buffer[5] >> 6;
- packet->version_number = buffer[5] >> 1 & 0x1F;
- packet->current_next_indicator = (buffer[5] << 7) >> 7;
- packet->section_number = buffer[6];
- packet->last_section_number = buffer[7];
- int len = 0;
- len = 3 + packet->section_length;
- packet->CRC_32 = (buffer[len-4] & 0x000000FF) << 24
- | (buffer[len-3] & 0x000000FF) << 16
- | (buffer[len-2] & 0x000000FF) << 8
- | (buffer[len-1] & 0x000000FF);
- int n = 0;
- for ( n = 0; n < packet->section_length - 12; n += 4 )
- {
- unsigned program_num = buffer[8 + n ] << 8 | buffer[9 + n ];
- packet->reserved_3 = buffer[10 + n ] >> 5;
- packet->network_PID = 0x00;
- if ( program_num == 0x00)
- {
- packet->network_PID = (buffer[10 + n ] & 0x1F) << 8 | buffer[11 + n ];
- TS_network_Pid = packet->network_PID; //记录该TS流的网络PID
- TRACE(" packet->network_PID %0x /n/n", packet->network_PID );
- }
- else
- {
- TS_PAT_Program PAT_program;
- PAT_program.program_map_PID = (buffer[10 + n] & 0x1F) << 8 | buffer[11 + n];
- PAT_program.program_number = program_num;
- packet->program.push_back( PAT_program );
- TS_program.push_back( PAT_program );//向全局PAT节目数组中添加PAT节目信息
- }
- }
- return 0;
- }