左右TS分析流
字节。在TS流里能够填入非常多类型的数据。如视频、音频、自己定义信息等。他的包的结构为,包头为4个字节,负载为184个字节(这184个字节不一定都是有效数据。有一些可能为填充数据)。
由于在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 ; ?*.~..?
?
].紿

memcpy(buf, pheader, );
pheader->transport_error_indicator = buf[] >> ;
pheader->payload_unit_start_indicator = buf[] >> & ] >> & ] & | buf[];
pheader->transport_scrambling_control = buf[] >> ;
pheader->adaption_field_control = buf[] >> & ] & ;
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;

如今看看我们的TS流片断样例。看来正好是47 40 00开头的。一个TS流的头部占领了4个字节。剩下的负载部分的内容由PID来决定。样例看来就是一个PAT表。在这里有个地方须要注意一下。payload_unit_start_indicator为1时。在前4个字节之后会有一个调整字节,它的数值决定了负载内容的详细開始位置。如今看样例中的数据47 40 00 17 00第五个字节是00,说明紧跟着00之后就是详细的负载内容。

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;


;
packet->table_id = buffer[];
packet->section_syntax_indicator = buffer[] >> ;
packet->zero = buffer[] >> & ] >> & ] & | buffer[];
packet->transport_stream_id = buffer[] << | buffer[];
packet->reserved_2 = buffer[] >> ;
packet->version_number = buffer[] >> & ] << ) >> ;
packet->section_number = buffer[];
packet->last_section_number = buffer[];
+ packet->section_length;
packet->CRC_32 = (buffer[len-] &
| (buffer[len-] &
| (buffer[len-] &
| (buffer[len-] & ; n < packet->section_length - ; n ++ )
{
packet->program_number = buffer[] << | buffer[];
packet->reserved_3 = buffer[] >> ;
] << ) << | buffer[];
] << ) << | buffer[];
}
n += ;
}
}

循环的次数通过section_length元素的确定。在这个样例中program_map_PID为20,所以以下来PMT分析时。就是查找47 40 20的开头的TS包。

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;


;
packet->table_id = buffer[];
packet->section_syntax_indicator = buffer[] >> ;
packet->zero = buffer[] >> ;
packet->reserved_1 = buffer[] >> ;
packet->section_length = (buffer[] & | buffer[];
packet->program_number = buffer[] << | buffer[];
packet->reserved_2 = buffer[] >> ;
packet->version_number = buffer[] >> & ] << ) >> ;
packet->section_number = buffer[];
packet->last_section_number = buffer[];
packet->reserved_3 = buffer[] >> ;
packet->PCR_PID = ((buffer[] << ) | buffer[]) & ] >> ;
packet->program_info_length = (buffer[] & | buffer[];
;
packet->CRC_32 = (buffer[len-] &
| (buffer[len-] &
| (buffer[len-] &
| (buffer[len-] & )
pos += packet->program_info_length;
) - ; )
{
packet->stream_type = buffer[pos];
packet->reserved_5 = buffer[pos+] >> ;
packet->elementary_PID = ((buffer[pos+] << ) | buffer[pos+]) & ] >> ;
packet->ES_info_length = (buffer[pos+] & | buffer[pos+];
)
{
pos = pos+;
pos += packet->ES_info_length;
}
;
}
i++;
}
}

然后就能够把数据是用调整函数填入结构中。然后得到详细节目的PID为视频0x21, 音频0x22。
版权声明:本文博主原创文章。博客,未经同意不得转载。
左右TS分析流的更多相关文章
- 关于ES、PES、PS/TS 码流
一.基本概念 )ES ES--Elementary Streams (原始流)是直接从编码器出来的数据流,可以是编码过的视频数据流(H.264,MJPEG等),音频数据流(AAC),或其他编码 ...
- ffmpeg利用libav库把yuv视频流转换为TS串流
今天到月末了,才发我这个月的第一篇文章,因为这个月前三周一直在看ffmpeg的libavcodec和libavformat两个库源码.实验室要做一个“小传大”的软件,就是android手机或平板电脑的 ...
- 流媒体ts/ps流封装/分析
1.TS 1) 感谢星辰同学,还热乎着,
- go http 下载视频(TS码流文件)(推荐一个网站学习 go example)
视频 http下载代码 dn.go(注意:代码很ugly,没怎么花时间) 总体感觉特别简单,网上看了下 net/http ,io这2个库的使用, 几分钟就写完了,感觉cpp 在做工具这块 开发效率的 ...
- TS流分析
http://blog.csdn.net/zxh821112/article/details/17587215 一 从TS流开始 数字电视机顶盒接收到的是一段段的码流,我们称之为TS(Transpor ...
- TS流格式(转)
一 从TS流开始 数字电视机顶盒接收到的是一段段的码流,我们称之为TS(Transport Stream,传输流),每个TS流都携带一些信息,如Video.Audio以及我们需要学习的PAT.PMT等 ...
- ts文件分析(纯c解析代码)
参考链接: 1. MPEG-2 TS码流分析 https://blog.csdn.net/zhubin215130/article/details/8958567 TS Header PAT PMT ...
- TS流解析 一
一 从TS流开始 数字电视机顶盒接收到的是一段段的码流,我们称之为TS(Transport Stream,传输流),每个TS流都携带一些信息,如Video.Audio以及我们需要学习的PAT.PMT等 ...
- ffmpeg解析TS流
介绍: MPEG的系统层编码为不同的应用场景设计了两种格式: TS(Transport Stream) 和PS(Program Stream), 它们两者之间不具有层级关系, 在逻辑上,它们两者都 ...
随机推荐
- HDU 2451 Simple Addition Expression(组合数学)
主题链接:http://acm.hdu.edu.cn/showproblem.php?pid=2451 Problem Description A luxury yacht with 100 pass ...
- Conversion to Dalvik format failed with error 1
主要和添�的第三方的包有关系. ======================================= 出现,Conversion to Dalvik format failed with e ...
- SharedPreferences共享优先存储的详细解析和原理
共享优先存储: publicvoid onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setCont ...
- windows phone (23) ScrollViewer元素
原文:windows phone (23) ScrollViewer元素 ScrollViewer类表示可包含其他可见元素的可滚动区域,一般会用在屏幕的宽度和高度不够用时,作为一种延伸使用,参考书上称 ...
- 初识google多语言通信框架gRPC系列(四)C++中使用gRPC
我的这几篇文章都是使用gRPC的example,不是直接编译example,而是新建一个项目,从添加依赖,编译example代码,执行example.这样做可以为我们创建自己的项目提供借鉴.如果对gR ...
- 猫学习IOS(三)UI纯代码UI——图片浏览器
猫分享.必须精品 看看效果 主要实现相似看新闻的一个界面,不用拖拽,纯代码手工写. 首先分析app能够非常easy知道他这里有两个UILabel一个UIImageView还有两个UIButton 定义 ...
- ibatis实战之OR映射
相对Hibernate等ORM实现而言,ibatis的映射配置更为简洁直接,以下是一个典型的配置文件. <?xml version="1.0" encoding=" ...
- Linux注意到Makefile
规则: 目标 : 依靠 命令 make是怎样工作的: (1)make在当前文件夹下寻找makefile或Makefile. (2)假设找到,他会寻找文件里的第一个目标文件(target).并把这个文件 ...
- HDU 3681 BFS&像缩进DP&二分法
N*M矩阵.从F出发点.走完全部Y点.每个人格开支1电源点,去G点,电池充满,D无法访问.最小的开始问什么时候满负荷可以去完全部Y.Y和G总共高达15一 第一BFS所有的F.Y.G之间的最短距离. 然 ...
- C++ 版本的split_string
vector<string> split_string(const string &in, char del, bool skip_empty) { vector<strin ...