博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

第二章:从TS到PAT和PMT 【TS流分析,解析出PAT, PMT】

Posted on 2017-01-16 11:18  bw_0927  阅读(2291)  评论(0编辑  收藏  举报

https://www.onelib.biz/doc/stb/course/begin.html

https://www.onelib.biz/doc/stb/appendix/psisitable.html#pmt1  表结构对照表

 

从PAT表开始,解析TS流

说完了TS流相关的基本概念,就该开始对TS流进行更深入的研究了。 首先需要想一想:TS流的本质是什么? TS是一段码流,并且是一段由数据包(Packet)组成的码流。 那么我们要如何组织和处理如此庞大的数据包呢?

如果我们把码流看作一本书,那么我们就需要给这本书制作一个目录;有了这样的目录,我们就可以“按图索骥”地得到我们需要的信息。

是的,PSI/SI就是这样“目录”。而PAT则是“目录”的“目录”。 通过PAT及其他 PSI 信息,我们可以一点一点地把整个TS流的信息组织和管理起来,从而分离出不同的业务(Service), 以及该业务的视频、音频、图文(Teletext)、字幕(Subtitle)、其他音轨等分量; 再根据 SI信息,我们就可以得到每个业务的名称、事件(Event)等附加信息, 从而形成 EPG ,更好地管理各个业务。

需要区分的一点是,不管是PSI还是SI,都只是一种“管理信息”,它并不包含任何的视频或音频数据。

什么是PID

Introduction of PID

在上一节《预备知识》中,我们可以看到 PID(Packet Identifier) 这个被标红的字段频繁地出现。

PID 是当前TS流的Packet区别于其他Packet类型的唯一识别符,通过读取每个包的Packet Header, 我们可以知道这个Packet的数据属于何种类型。下表列出了几项固定的 PID 值,它们用于识别存储了特殊信息的Packet。 下面要谈的PAT表的 PID 值就是固定的0x0000。

TS流中PID的分配

解复用程序需要使用到的表格只有PAT、PMT、SDT;而CA解扰还需要使用CAT;EPG则还要有NIT、EIT、TDT、TOT等表。

使用工具说明

Tools Introduction

使用码流分析软件来分析一段码流,会事半功倍:泽华源码流分析软件 或 Tsparse

下图是使用 泽华源码流分析软件 的界面。在打开码流并分析完毕后,点击下方选项卡“语法分析”可以出现这个窗口。

一切从PAT开始

Begin with PAT

PAT表(Program Association Table,节目关联表) 定义了当前TS流中所有的节目,其PID为0x0000,它是PSI的根节点,要查寻找节目必须从PAT表开始查找。PAT表携带以下信息:

TS流ID transport_stream_id 该ID标志唯一的流ID
节目频道号 program_number 该号码标志TS流中的一个频道,该频道可以包含很多的节目(即可以包含多个Video PID和Audio PID)
PMT的PID program_map_PID 表示本频道使用哪个PID做为PMT的PID。
因为可以有很多的业务,因此DVB规定PMT的PID可以由用户自己定义

解析PAT表的数据包

Analyze the Package of PAT

上一幅图中,右下角粉红色的数据即为PAT表的数据包。

根据在上一章 《预备知识》中提到的包的分析方法,可以将上图的的数据这样划分:

包头(Package Header) 包数据(Package Data)
47 40 00 1C 00 B0 1D 22 01 CF 00 00 00 00 E0 10 40 13 E1 30 40 18 E1 ... FF FF

注:表格内数据均为16进制数值。前4个字节为包头,后面的184个字节为包数据。

 

PAT表的包头

Package Header of PAT

细分其表头(Package Header)数据(47 40 00 1C)如下:

位号 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 ...
Packet(二进制) 0 1 0 0 0 1 1 1 0 1 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1 0 0 ...
Packet(十六进制) 4 7 4 0 0 0 1 C ...
Packet Header信息 0*: sync_byte=0x47 1* 2* 3* 4*: PID (这里是0x0000) 5* 6* 7* ...

拖动可显示更多数据   

上表中,第一行为表头的位号(0-31,共32位), 第二行为每位的二进制数值, 第三行为每个字节的16进制数值, 最后一行的数据说明参看下面的表格。

Packet Header(包头)信息Demo
# 标识 位数 说明
0* sync_byte 8 bits 固定是0x47
1* transport_error_indicator 1 bits 值为0,表示当前包没有发生传输错误。错误指示信息(1:该包至少有1bits传输错误)
2* payload_unit_start_indicator 1 bits 值为1,含义参考ISO13818-1标准文档。负载单元开始标志(packet不满188字节时需填充)
3* transport_priority 1 bits 值为0,表示当前包是低优先级。传输优先级标志(1:优先级高)
4* PID 13 bits PID=0x0000,说明是PAT表。Packet ID号码,唯一的号码对应不同的包
5* transport_scrambling_control 2 bits 值为0x00,表示节目没有加密。加密标志(00:未加密;其他表示已加密)
6* adaptation_field_control 2 bits 值为0x01,具体含义请参考ISO13818-1。附加区域控制
7* continuity_counter 4 bits 值为0xC,表示当前传送的相同类型的包是第12个。包递增计数器

 

PAT表的包数据

Package Data of PAT

该包的数据部分(Package Data)为: 
00 B0 1D 22 01 CF 00 00 00 00 E0 10 40 13 E1 30 40 18 E1 80 40 0A E0 A0 40 0E E0 B5 10 A5 84 FF ... FF FF

注意到这里有一个绿色的 00, 这是因为在包头后需要除去一个字节才是有效数据(payload_unit_start_indicator="1")。 因此,真正要分析的数据应该是红色部分: 
B0 1D 22 01 CF 00 00 00 00 E0 10 40 13 E1 30 40 18 E1 80 40 0A E0 A0 40 0E E0 B5 10 A5 84 FF ... FF FF

 

解析前8个字节

下面对前8个字节(0-7字节,共计64 bits)进行分析: 
B0 1D 22 01 CF 00 00 00 00 E0 10 40 13 E1 30 40 18 E1 80 40 0A E0 A0 40 0E E0 B5 10 A5 84 FF ... FF FF

位号 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 ...
Binary 0 0 0 0 0 0 0 0 1 0 1 1 0 0 0 0 0 0 0 1 1 1 0 1 0 0 1 0 0 0 1 0 0 0 0 0 0 0 0 1 1 1 0 0 1 1 1 1 0 0 0 0 0 0 0 0 ...
Hex 0 0 B 0 1 D 2 2 0 1 C F 0 0 ...
结构 0*: table_id=0x00 1* 2* 3* 4*: section_length (这里是0x01D) 5* 6* 7* 8* 9* 10* ...

拖动可显示更多数据   

# 字段名 占位 具体值 次序 说明
0* table_id 8 bits 0000 0000(即0x00) 第0个字节
00
PAT的table_id只能是0x00
1* section_syntax_indicator 1 bit 1

第1、2个字节
B0 1D
段语法标志位,固定为1
2* zero 1 bit 0  
3* reserved 2 bits 11 (Binary)  
4* section_length 12 bits 00 00 00 01 11 01(即0x1D,十进制值为29) 意思是 段长度为29字节
5* transport_stream_id 16 bits ‭00 10 00 10 00 00 00 01‬(即0x2201) 第3、4个字节
22 01
TS的识别号
6* reserved 2 bits 11

第5个字节
CF
TS的识别号
7* version_number 5 bits 0 01 11 一旦PAT有变化,版本号加1
8* current_next_indicator 1 bit 1 当前传送的PAT表可以使用,若为0则要等待下一个表
9* section_number 4 bits 00 00(即0x00) 第6个字节
00
给出section号,在sub_table中,
第一个section其section_number为"0x00",
每增加一个section,section_number加一
10* last_section_number 4 bits 00 00(即0x00) 第7个字节
00
sub_table中最后一个section的section_number
循环开始(循环内的数据解析见下一节内容!)
- program_number 16 bits -- -- -
- reserved 3 bits -- -- -
- network_id

program_map_PID
13 bits -- -- program_number为0x0000时,
这里是network_id(NIT的PID);
其余情况是program_map_PID(PMT的PID)
循环结束
- CRC_32 32 bits -- 最后4个字节 -

 

解析循环部分

下面对后续几个字节(第8个字符开始)进行分析: 
B0 1D 22 01 CF 00 00 00 00 E0 10 40 13 E1 30 40 18 E1 80 40 0A E0 A0 40 0E E0 B5 10 A5 84 FF ... FF FF

在上一小节节的表格中,每一个循环都是4个字节(32 bits),我们在这里就先解析第一个循环: 
B0 1D 22 01 CF 00 00 00 00 E0 10 40 13 E1 30 40 18 E1 80 40 0A E0 A0 40 0E E0 B5 10 A5 84 FF ... FF FF

位号 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 ...
Binary 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1 0 0 0 0 0 0 0 0 1 0 0 0 0 ...
Hex 0 0 0 0 E 0 1 0 ...
结构 1*: program_number = 0x0000 2*: reserved 3*: network_id = 0x10 ...

拖动可显示更多数据   

# 字段名 占位 具体值 说明
1* program_number 16 bits 0x0000 program_number = 0x0000
2* reserved 3 bits 111 -
3* program_map_PID 13 bits 0x0000 因为program_number为0x0000,
所以这里是network_id(NIT的PID)

继续解析下一个循环: 
B0 1D 22 01 CF 00 00 00 00 E0 10 40 13 E1 30 40 18 E1 80 40 0A E0 A0 40 0E E0 B5 10 A5 84 FF ... FF FF

位号 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 ...
Binary 0 1 0 0 0 0 0 0 0 0 0 1 0 0 1 1 1 1 1 0 0 0 0 1 0 0 1 1 0 0 0 0 ...
Hex 4 0 1 3 E 1 3 0 ...
结构 1*: program_number = 0x4013(即十进制16403) 2*: reserved 3*: program_map_PID = 0x130(即PMT_PID为0x130) ...

拖动可显示更多数据   

# 字段名 占位 具体值 说明
1* program_number 16 bits 0x4013 program_number = 0x4013(即十进制16403)
2* reserved 3 bits 111 -
3* program_map_PID 13 bits 0x130 因为program_number不为0x0000,
所以这里是program_map_PID = 0x130(即PMT_PID为0x130)

通过这个循环,我们可以知道,在这个TS中,有一个节目号为0x4013(即十进制16403)的节目,其PMT的PID为0x130。那么要想获取这个节目的详细信息,就要去解析PID为0x130的PMT表。(关于PMT表的解析可参看下一节内容)

为了不占用过多空间来显示这样重复的内容,关于更多循环的分析就不显示在正文,如果你仍想看,可以 点击这里显示或隐藏

 

循环汇总

将所有循环遍历完毕后,我们就能找出4个节目。下图是码流分析软件的截图。

PMT PID列表

下面表格是通过对PAT表分析后得到的数据。在分析完PMT表后,还将对这张表进行补充。

# 节目号 PMT PID
1 16403 0x130
2 16408 0x180
3 16394 0xA0
4 16398 0xE0

 

小结

对我们来说,PAT就是一个总入口。PAT告诉了我们,这个TS流中有几个节目,以及它们的PMT PID分别是多少。有了PMT的PID,我们就可以继续下一步了。

PMT表分析

Analyze of PMT

PMT表概述

PMT表中包含的数据如下: 
    (1) 当前频道中包含的所有Video数据的PID 
    (2) 当前频道中包含的所有Audio数据的PID 
    (3) 和当前频道关联在一起的其他数据的PID(如Teletext、Subtitle、其他音轨等的PID)

只要我们处理了PMT,那么我们就可以获取频道中所有的PID信息, 如当前频道包含多少个Video、共多少个Audio和其他数据,还能知道每种数据对应的PID分别是什么。 这样如果我们要选择其中一个Video和Audio收看,那么只需要把要收看的节目的Video PID和Audio PID保存起来, 在处理Packet的时候进行过滤即可实现。

 

PMT表的数据包总览

以PMT PID=0x130来分析。我们找到PID为0x130的PMT表:

PMT表的数据包包头

表头的分析方式和PAT表头的相同,就不详细介绍了。

数据包头是红色部分: 
47 41 30 19 00 02 B0 43 40 13 C5 00 00 E1 31 F0 00 02 E1 31 
F0 03 52 01 01 04 04 E1 32 F0 09 0A 04 64 75 01 52 01 03 06
E1 37 F0 0A 52 01 0A 56 05 64 65 75 09 00 06 E1 38 F0 0C 0A
04 64 65 75 01 6A 01 00 52 01 11 28 09 C3 CB FF FF ... FF FF

分析如下:

位号 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 ...
Packet(二进制) 0 1 0 0 0 1 1 1 0 1 0 0 0 0 0 1 0 0 1 1 0 0 0 0 0 0 0 1 1 0 0 1 ...
Packet(十六进制) 4 7 4 1 3 0 1 9 ...
Packet Header信息 - - - - PMT PID (这里是0x130) - - - ...

拖动可显示更多数据   

 

PMT表的数据包数据

数据内容是红色部分: 
47 41 30 19 00

 

下面按照PMT的表结构,人为分成4段进行解析:


片段1:02 B0 43 40 13 C5
片段2:00 00 E1 31 F0 00
片段3:02 E1 31 F0 03 
片段3补充:52 01 01 
片段4:04 E1 32 F0 09
0A 04 64 75 01 52 01 03 06
E1 37 F0 0A 52 01 0A 56 05 64 65 75 09 00 06 E1 38 F0 0C 0A
04 64 65 75 01 6A 01 00 52 01 11 28 09 C3 CB FF FF ... FF FF

片段1:02 B0 43 40 13 C5

位号 0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47
Binary 0 0 0 0 0 0 1 0 1 0 1 1 0 0 0 0 0 1 0 0 0 0 1 1 0 1 0 0 0 0 0 0 0 0 0 1 0 0 1 1 1 1 0 0 0 1 0 1
Hex 0 2 B 0 4 3 4 0 1 3 C 5
结构 0*: table_id=0x02 1* 2* 3* 4* 5*: 0x4013 = 16403 6* 7*: 0x02 8*

拖动可显示更多数据   

# 字段名 占位 具体值 说明
0* table_id 8 bits 0x02  
1* Section_syntax_indicator 1 bit 1 通常设为“1”
2* zero 1 bit 0  
3* Reserved 2 bits 11  
4* Section_length 12 bits 0x43 段长度,从program_number开始,到CRC_32(含)的字节总数
5* program_number 16 bits 0x4013 节目号为16403
6* Reserved 2 bits 11  
7* Version_number 5 bits 00010 = 0x02  
8* Current_next_indicator 1 bit 1 当前未来标志符

片段2:00 00 E1 31 F0 00

位号 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 ...
Binary 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 1 1 1 0 0 0 0 1 0 0 1 1 0 0 0 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 0 0 ...
Hex 0 0 0 0 E 1 3 1 F 0 0 0 ...
结构 9* 10* 11* 12* 13* 14* ...

拖动可显示更多数据   

# 字段名 占位 具体值 说明
9* Section_number 8 bits 0x00 当前段号码
10* last_section_number 8 bits 0x00 最后段号码,含义和PAT中的对应字段相同
11* reserved 3 bits 0x00  
12* PCR_PID 13 bits 0x0131 PCR(节目参考时钟)所在TS分组的PID
13* reserved 4 bits 1111  
14* program_info_length 12 bits 0x00 节目信息长度
(之后的是N个描述符结构,一般可以忽略掉,这个字段就代表描述符总的长度,单位是Bytes)
紧接着就是频道内部包含的节目类型和对应的PID号码了

注意,上面的program_info_length之后的是N个描述符结构,但一般可以忽略掉。这个字段就代表描述符总的长度,单位是Bytes;这里是0x00,也就是意味着后面没有描述符结构。

 

片段3:02 E1 31 F0 03 
因为program_info_length=0x00,所以紧接着的是频道内部包含的节目类型和对应的PID号码

位号 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 ...
Binary 0 0 0 0 0 0 1 0 1 1 1 0 0 0 0 1 0 0 1 1 0 0 0 1 1 1 1 1 0 0 0 0 0 0 0 0 0 0 1 1 ...
Hex 0 2 E 1 3 1 F 0 0 3 ...
结构 15* 16* 17* 18* 19* ...

拖动可显示更多数据   

# 字段名 占位 具体值 说明
15* stream_type 8 bits 0x02 流类型,标志是Video还是Audio还是其他数据。这里是MPEG2视频类型
16* reserved 3 bits 111  
17* elementary_PID 13 bits 0x131 该节目的音频或视频PID。这里是视频PID=0x131
18* reserved 4 bits 1111  
19* ES_info_length 12 bits 0x003 后面紧跟着3个字节的描述子

片段3补充:52 01 01 
因为ES_info_length=0x03,所以后续的三个字节,为Stream identifier descriptor

位号 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 ...
Binary 0 1 0 1 0 0 1 0 0 0 0 0 0 0 0 1 0 0 0 0 0 0 0 1 ...
Hex 5 2 0 1 0 1 ...
结构 20* 21* 22* ...

拖动可显示更多数据   

# 字段名 占位 具体值 说明
20* descriptor_tag 8 bits 0x52  
21* descriptor_length 8 bits 0x01  
22* component_tag 8 bits 0x01  

至此,第一段单元流(Elementary Stream)的信息解析出来了【一个PMT对应一个节目,一个节目中可能会包含音频,视频,字幕等其他一些组成单元。第一个单元解析出来了(这是一个视频节目),for循环继续解析该PMT中的其他单元(音频,或者字幕)】。

这是视频ES(MPEG2),其PID为0x131

片段4:04 E1 32 F0 09 
接下来,继续第二个单元流信息解析。参照片段3的方式,先讲下一个单元流的基本信息解析出来: 
0000 0100    111    0 0001 0011 0010    1111    0000 0000 1001

# 字段名 占位 具体值 说明
23* stream_type 8 bits 0x04 流类型,标志是Video还是Audio还是其他数据。这里是MPEG2音频类型
24* reserved 3 bits 111  
25* elementary_PID 13 bits 0 0001 0011 0010 = 0x132 该节目的音频或视频PID。这里是音频PID=0x132
26* reserved 4 bits 1111  
27* ES_info_length 12 bits 0000 0000 1001 = 0x09 后面紧跟着9个字节的描述子

上图是完整的PID为0x132的ES信息,可以看到这个ES有两个描述子(descriptor)。 解析时,第一个字节代表了描述子tag,第二个代表该描述子的长度。

看第一个描述子ISO_639_language_descriptor,第一个字节值为0xa,长度为0x4 
看第二个描述子stream_identifier_descriptor,第一个字节值为0x52,长度为0x1

其余单元流和PMT PID的分析略。

PMT表的数据包数据小结

根据PAT和PMT的信息,我们可以得到下面的信息:

# 节目号 PMT PID ES PID ES说明
1 16403 0x130 0x131 MPEG2视频
0x132 MPEG2音频
0x137 MPEG2私用PES
0x138 MPEG2私用PES
2 16408 0x180 0x181 MPEG2视频
0x182 MPEG2音频
0x187 MPEG2私用PES
0x188 MPEG2私用PES
3 16394 0xA0 0xa1 MPEG2视频
0xa2 MPEG2音频
0xa7 MPEG2私用PES
4 16398 0xE0 0xe1 MPEG2视频
0xe2 MPEG2音频
0xe7 MPEG2私用PES

TS的复用与解复用

Multiplexing and Demultiplexing of Transport Stream

在数字电视系统的前端,所有的数据(视频、音频、PSI/SI信息)都会被复用器(Multiplexer)复用成TS流,从而进行传送(见下图)。

TS复用(Multiplexing)

当机顶盒等接收器收到的数据,自然就是被复用后的一个个包(Package)了。这样的数据是不能直接使用的。 这时候,我们要做的就是把被复用的TS流(音视频、字幕等混杂在一起的流数据) 解复用(Demultiplexing, 简称Demux)。

解复用的意义在于,由于TS流是一种复用的码流,里面混杂了多种类型的包;解复用TS流可以将类型相同的Packet存入相同缓存,分别处理。 这样就可以将Video、Audio或者其他业务信息(如PSI/SI信息)的数据区分开来。

上一小节制成的表格,其实就是解复用获得的信息。 通过它,我们可以组织起这个频点下的所有 节目。 以节目号为16403的 节目 为例, 过滤PID为0x131的所有包数据,就可以组成这个节目的视频; 过滤PID为0x132的所有包数据,就可以组成这个节目的音频; 这样,节目16403需要的视音频数据就集齐了;当然,要正常播放这个台,还需要时钟来同步。

DVB搜台过程

Process of DVB Scanning

DVB搜台过程

机顶盒先调整高频头到一个固定的频率(一般是主频点,如深圳天威的主频点是259MHZ),如果此频率有数字信号, 则COFDM芯片(如MT352)会自动把TS流数据传送给MPEG-2 decoder。 MPEG-2 decoder先进行数据的同步,也就是等待完整的Packet的到来.然后循环查找是否出现PID==0x0000的Packet; 如果出现了,则马上进入分析PAT的处理,获取了所有的PMT的PID。 接着循环查找是否出现PMT,如果发现了,则自动进入PMT分析,获取该频段所有的频道数据并保存。 如果没有发现PAT或者没有发现PMT,说明该频段没有信号,进入下一个频率扫描。

在解析TS流的时候,首先寻找PAT表,根据PAT获取所有PMT表的PID;再寻找PMT表,获取该频段所有节目数据并保存。 这样,只需要知道节目的PID就可以根据PacketHeade给出的PID过滤出不同的Packet,从而观看不同的节目。 这些就是PAT表和PMT表之间的关系。

 

SDT表

由于PID是一串枯燥的数字,用户不方便记忆、且容易输错,所以需要有一张表将节目名称和该节目的PID对应起来,DVB设计了 SDT表来解决这个问题。 该表格标志一个节目的名称,并且能和PMT中的PID联系起来,这样用户就可以通过直接选择节目名称来选择节目了。

根据SDT表信息(这里不再详细解析),我们可以把上一小节总结的表格进一步补充:

# 节目号 PMT PID ES PID ES说明 SDT表信息
1 16403 0x130 0x131 MPEG2视频 ProSieben
0x132 MPEG2音频
0x137 MPEG2私用PES
0x138 MPEG2私用PES
2 16408 0x180 0x181 MPEG2视频 SAT.1
0x182 MPEG2音频
0x187 MPEG2私用PES
0x188 MPEG2私用PES
3 16394 0xA0 0xa1 MPEG2视频 KABEL1
0xa2 MPEG2音频
0xa7 MPEG2私用PES
4 16398 0xE0 0xe1 MPEG2视频 N24
0xe2 MPEG2音频
0xe7 MPEG2私用PES

 

NIT表

在国内,NIT表一般会给出整个网络的频点列表。 在搜索主频点的同时解析出NIT表,就可以得到这个网络的频点列表。 机顶盒按顺序对所有频点锁频、解析,就可以将整个网络的业务搜索出来了

但在国外,一般不会有这样的情况。首先,小运营商只是租用其中一个或几个频点,不可能对整个网络“指手画脚”,自然就不能定义NIT表了。 而大运营商,它们有自己的业务群(Bouquet),更倾向于使用BAT表来组织自己的节目。

参考文档

References

 

# 文档名称 作者
1 《1.从TS流到PAT和PMT》 林晓州
2 《PSI/SI深入学习》 林晓州
3 《En300468.V1.7.1_Specification for SI in DVB Systems.pdf》 European Standard
4 《数字视频广播中文业务信息规范》 国家广播电影电视总局
5 《数字电视业务PSI-SI学习系列》 网络