TS流的解析
因为在TS流里可以填入很多种东西,所以有必要有一种机制来确定怎么来标识这些数据。制定TS流标准的机构就规定了一些数据结构来定义。比如: PSI(Program Specific Information)表,所以解析起来就像这样: 先接收一个负载里为PAT的数据包,在整个数据包里找到一个PMT包的ID。然后再接收一个含有PMT的数据包,在这个数据包里找到有关填入数据类型的ID。之后就在接收到的TS包里找含有这个ID的负载内容,这个内容就是填入的信息。根据填入的数据类型的ID的不同,在TS流复合多种信息是可行的。关键就是找到标识的ID号。
0000f32ch: 47 40 00 17 00 00 B0 0D 00 01 C1 00 00 00 01 E0 ; G@....?..?...?
0000f33ch: 20 A2 C3 29 41 FF FF FF FF FF FF FF FF FF FF FF ; ⒚)A
0000f34ch: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f35ch: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f36ch: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f37ch: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f38ch: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f39ch: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f3ach: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f3bch: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f3cch: FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF FF ;
0000f3dch: FF FF FF FF FF FF FF FF FF FF FF FF 47 40 20 17 ; G@ .
0000f3ech: 00 02 B0 1B 00 01 C1 00 00 E0 21 F0 00 1B E0 21 ; ..?..?.??.?
0000f3fch: F0 04 2A 02 7E 1F 03 E0 22 F0 00 5D 16 BD 48 ; ?*.~..??].紿
// Adjust TS packet header
void adjust_TS_packet_header(TS_packet_header* pheader)
{
unsigned char buf[];
memcpy(buf, pheader, );
pheader->transport_error_indicator = buf[] >> ;
pheader->payload_unit_start_indicator = buf[] >> & 0x01;
pheader->transport_priority = buf[] >> & 0x01;
pheader->PID = (buf[] & 0x1F) << | buf[];
pheader->transport_scrambling_control = buf[] >> ;
pheader->adaption_field_control = buf[] >> & 0x03;
pheader->continuity_counter = buf[] & 0x03;
}
// Transport packet header
typedef struct TS_packet_header
{
unsigned sync_byte : ;
unsigned transport_error_indicator : ;
unsigned payload_unit_start_indicator : ;
unsigned transport_priority : ;
unsigned PID : ;
unsigned transport_scrambling_control : ;
unsigned adaption_field_control : ;
unsigned continuity_counter : ;
} TS_packet_header;
// PAT table
// Programm Association Table
typedef struct TS_PAT
{
unsigned table_id : ;
unsigned section_syntax_indicator : ;
unsigned zero : ;
unsigned reserved_1 : ;
unsigned section_length : ;
unsigned transport_stream_id : ;
unsigned reserved_2 : ;
unsigned version_number : ;
unsigned current_next_indicator : ;
unsigned section_number : ;
unsigned last_section_number : ;
unsigned program_number : ;
unsigned reserved_3 : ;
unsigned network_PID : ;
unsigned program_map_PID : ;
unsigned CRC_32 : ;
} TS_PAT;
// Adjust PAT table
void adjust_PAT_table ( TS_PAT * packet, char * buffer )
{
int n = , i = ;
int len = ;
packet->table_id = buffer[];
packet->section_syntax_indicator = buffer[] >> ;
packet->zero = buffer[] >> & 0x1;
packet->reserved_1 = buffer[] >> & 0x3;
packet->section_length = (buffer[] & 0x0F) << | buffer[];
packet->transport_stream_id = buffer[] << | buffer[];
packet->reserved_2 = buffer[] >> ;
packet->version_number = buffer[] >> & 0x1F;
packet->current_next_indicator = (buffer[] << ) >> ;
packet->section_number = buffer[];
packet->last_section_number = buffer[];
// Get CRC_32
len = + packet->section_length;
packet->CRC_32 = (buffer[len-] & 0x000000FF) <<
| (buffer[len-] & 0x000000FF) <<
| (buffer[len-] & 0x000000FF) <<
| (buffer[len-] & 0x000000FF);
// Parse network_PID or program_map_PID
for ( n = ; n < packet->section_length - ; n ++ )
{
packet->program_number = buffer[] << | buffer[];
packet->reserved_3 = buffer[] >> ;
if ( packet->program_number == 0x0 )
packet->network_PID = (buffer[] << ) << | buffer[];
else
{
packet->program_map_PID = (buffer[] << ) << | buffer[];
}
n += ;
}
}
// PMT table
// Program Map Table
typedef struct TS_PMT
{
unsigned table_id : ;
unsigned section_syntax_indicator : ;
unsigned zero : ;
unsigned reserved_1 : ;
unsigned section_length : ;
unsigned program_number : ;
unsigned reserved_2 : ;
unsigned version_number : ;
unsigned current_next_indicator : ;
unsigned section_number : ;
unsigned last_section_number : ;
unsigned reserved_3 : ;
unsigned PCR_PID : ;
unsigned reserved_4 : ;
unsigned program_info_length : ;
unsigned stream_type : ;
unsigned reserved_5 : ;
unsigned elementary_PID : ;
unsigned reserved_6 : ;
unsigned ES_info_length : ;
unsigned CRC_32 : ;
} TS_PMT;
// Adjust PMT table
void adjust_PMT_table ( TS_PMT * packet, char * buffer )
{
int pos = , len = ;
int i = ;
packet->table_id = buffer[];
packet->section_syntax_indicator = buffer[] >> ;
packet->zero = buffer[] >> ;
packet->reserved_1 = buffer[] >> ;
packet->section_length = (buffer[] & 0x0F) << | buffer[];
packet->program_number = buffer[] << | buffer[];
packet->reserved_2 = buffer[] >> ;
packet->version_number = buffer[] >> & 0x1F;
packet->current_next_indicator = (buffer[] << ) >> ;
packet->section_number = buffer[];
packet->last_section_number = buffer[];
packet->reserved_3 = buffer[] >> ;
packet->PCR_PID = ((buffer[] << ) | buffer[]) & 0x1FFF;
packet->reserved_4 = buffer[] >> ;
packet->program_info_length = (buffer[] & 0x0F) << | buffer[];
// Get CRC_32
len = packet->section_length + ;
packet->CRC_32 = (buffer[len-] & 0x000000FF) <<
| (buffer[len-] & 0x000000FF) <<
| (buffer[len-] & 0x000000FF) <<
| (buffer[len-] & 0x000000FF);
// program info descriptor
if ( packet->program_info_length != )
pos += packet->program_info_length;
// Get stream type and PID
for ( ; pos <= (packet->section_length + ) - ; )
{
packet->stream_type = buffer[pos];
packet->reserved_5 = buffer[pos+] >> ;
packet->elementary_PID = ((buffer[pos+] << ) | buffer[pos+]) & 0x1FFF;
packet->reserved_6 = buffer[pos+] >> ;
packet->ES_info_length = (buffer[pos+] & 0x0F) << | buffer[pos+];
// Store in es
es[i].type = packet->stream_type;
es[i].pid = packet->elementary_PID;
if ( packet->ES_info_length != )
{
pos = pos+;
pos += packet->ES_info_length;
}
else
{
pos += ;
}
i++;
}
}
TS流的解析的更多相关文章
- 关于TS流的解析
字节.在TS流里可以填入很多类型的数据,如视频.音频.自定义信息等.他的包的结构为,包头为4个字节,负载为184个字节(这184个字节不一定都是有效数据,有一些可能为填充数据). 工作形式: 因为在T ...
- TS流解析 一
一 从TS流开始 数字电视机顶盒接收到的是一段段的码流,我们称之为TS(Transport Stream,传输流),每个TS流都携带一些信息,如Video.Audio以及我们需要学习的PAT.PMT等 ...
- TS流解析 四
一 从TS流开始 数字电视机顶盒接收到的是一段段的码流,我们称之为TS(Transport Stream,传输流),每个TS流都携带一些信息,如Video.Audio以及我们需要学习的PAT.PMT等 ...
- TS流PAT/PMT详解
一 从TS流开始 从MPEG-2到DVB,看着看着突然就出现了一大堆表格,什么PAT.PMT.CAT……如此多的表该怎样深入了解呢? 我们知道,数字电视机顶盒接收到的是一段段的码流,我们称之为TS(T ...
- TS流分析
http://blog.csdn.net/zxh821112/article/details/17587215 一 从TS流开始 数字电视机顶盒接收到的是一段段的码流,我们称之为TS(Transpor ...
- 从TS流到PAT和PMT
转自:https://blog.csdn.net/rongdeguoqian/article/details/18214627 一 从TS流开始 最近开始学习数字电视机顶盒的开发,从MPEG-2到DV ...
- 分析ffmpeg解析ts流信息的源码
花费一些时间,然后全部扔了.为了不忘记和抛砖引玉,特发此贴. ffmpeg解析ts流 1.目的 打算软件方式解析出pat,pmt等码流信息 2.源代码所在位置 下载ffmpeg ...
- TS流解析 二 *****
1.TS格式介绍 TS:全称为MPEG2-TS.TS即"Transport Stream"的缩写.它是分包发送的,每一个包长为188字节(还有192和204个字节的包).包的结构为 ...
- ffmpeg解析TS流
介绍: MPEG的系统层编码为不同的应用场景设计了两种格式: TS(Transport Stream) 和PS(Program Stream), 它们两者之间不具有层级关系, 在逻辑上,它们两者都 ...
随机推荐
- HDU 1936 区间贪心
/* *区间贪心.前几天刚做了POJ 1328 ...思路完全相同... *最多有100个表情,100行文字.遍历寻找每个表情的所在区间.时间复杂度大约在10^5 ~ 10^6 可以接受. *然后对每 ...
- HDU 4635 Strongly connected (Tarjan+一点数学分析)
Strongly connected Time Limit : 2000/1000ms (Java/Other) Memory Limit : 32768/32768K (Java/Other) ...
- Spring学习笔记之Container overview
The Spring IoC container
- C中预编译详解
预处理过程扫描源代码,对其进行初步的转换,产生新的源代码提供给编译器.可见预处理过程先于编译器对源代码进行处理.在C 语言中,并没有任何内在的机制来完成如下一些功能:在编译时包含其他源文件.定义宏.根 ...
- C++设计模式之访问者模式
简述 访问者模式(Visitor Pattern)表示一个作用于某对象结构中的各元素的操作,它使你可以在不改变各元素类的前提下定义作用于这些元素的新操作. 代码实现: // Visitor.cpp : ...
- 内核事件KEVENT(同步)
转载请您注明出处:http://www.cnblogs.com/lsh123/p/7358702.html 一.驱动程序与驱动程序的事件交互 IoCreateNotificationEvent ...
- ZooKeeper客户端原生API的使用以及ZkClient第三方API的使用
这两部分内容的介绍主要讲的是节点及节点内容和子节点的操作,并且讲解的节点的事件监听以及ACL授权 ZooKeeper客户端原生API的使用 百度网盘地址: http://pan.baidu.com/s ...
- juery 安全加固 集合
来源 jquery升级坑 2 3 4 5 版本 相关源码分享 新建document jquery ajax使用说明 最近在iteye的新闻中看到jQuery已经更新到了1.6.1. 和 ...
- 手机连不上eclipse
在进行android开发时,有时候会很奇怪,手机连不上eclipse了,打开eclipse的ddms也没有,重启adb也不行,这时候我们应该怎么办呢. 首先打开资源管理器,找到 adb.exe 结束掉 ...
- (MyEclipse) MyEclipse完美破解方法(图)
http://photo.blog.sina.com.cn/list/blogpic.php?pid=53358777td408badc4071&bid=533587770101dd03&am ...