所以...
我正在尝试获取 H264 Annex B 字节流视频并将其编码为纯 Java 中的 MPEG-TS。我的目标是创建一个最小的 MPEG-TS、单一节目、有效流并且不包含任何时间信息信息(PCR、PTS、DTS)。
我目前可以将生成的文件传递给 ffmpeg (ffmpeg -i myVideo.ts) 和 ffmpeg 报告...
[NULL @ 0x7f8103022600] start time is not set in estimate_timings_from_pts
Input #0, mpegts, from 'video.ts':
Duration: N/A, bitrate: N/A
Program 1
Stream #0:0[0x100]: Video: h264 (Main) ([27][0][0][0] / 0x001B), yuv420p(tv, bt709), 1280x720 [SAR 1:1 DAR 16:9], 29.97 fps, 29.97 tbr, 90k tbn, 59.94 tbc
...似乎这个开始时间的警告没什么大不了的...而且 ffmpeg 无法确定视频的长度。如果我从我的视频文件(ffmpeg -i myVideo.ts -vcodec copy validVideo.ts)创建另一个 mpeg-ts 文件并运行 ffmpeg -i validVideo.ts 我得到......
Input #0, mpegts, from 'video2.ts':
Duration: 00:00:11.61, start: 1.400000, bitrate: 3325 kb/s
Program 1
Metadata:
service_name : Service01
service_provider: FFmpeg
Stream #0:0[0x100]: Video: h264 (Main) ([27][0][0][0] / 0x001B), yuv420p(tv, bt709), 1280x720 [SAR 1:1 DAR 16:9], 29.97 fps, 29.97 tbr, 90k tbn, 59.94 tbc
...所以你可以看到时间信息和比特率,元数据也是如此。
我的 H264 视频仅包含 I 和 P 帧(当然,SPS 和 PPS 在 I 帧之前),我创建 MPEG-TS 流的方式是......
- 在文件开头写一个 PAT
- 在文件开头写一个 PMT
- 从 SPS、PPS 和 I 帧创建 TS 和 PES 数据包(AUD NAL,如果需要的话?)
- 从 P 帧创建 TS 和 PES 数据包(同样,如果需要,也可以使用 AUD NAL)
- 对于 I 帧或 P 帧的最后一个有效载荷,将填充字节添加到适配字段以确保它适合完整的 TS 数据包
- 对整个文件重复 3-5
...我的 PAT 看起来像这样...
4740 0010 0000 b00d 0001 c100 0000 01f0
002a b104 b2ff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff
...我的 PMT 看起来像这样...
4750 0010
0002 b012 0001 c100 00ff fff0 001b e100
f000 c15b 41e0 ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff
...注意在 c100 00 之后,“ff ff”,f0... 表示我们没有使用 PCR... 另请注意,我已更新我的 CRC 以反映 PMT 的此更改。我的第一个 I Frame 数据包看起来像......
4741 0010 0000 01e0
0000 8000 0000 0000 0109 f000 0000 0127
4d40 288d 8d60 2802 dd80 b501 0101 4000
00fa 4000 3a98 3a18 00b7 2000 3380 2ef2
e343 0016 e400 0670 05de 5c16 345d c000
0000 0128 ee3c 8000 0000 0165 8880 0020
0000 4fe5 63b5 4e90 b11c 9f8f f891 10f3
13b1 666b 9fc6 03e9 e321 36bf 1788 347b
eb23 fc89 5772 6e2e 1714 96df ed16 9b30
252d ceb7 07e9 a0c7 c6e7 9515 be87 2df1
81f3 b9d2 ba5f 243e 2d5c cba2 8ca5 b798
6bec 8c43 0b5d bbda bc5b 6e7c e15c 84e8
2f13 be84
...您会注意到 01e0 0000, 8000 00 是 PES 标头扩展名,我没有指定 PTS / DTS 并且剩余长度为零。我的第一个 P 帧数据包看起来像......
4741 001d
0000 01e0 0000 8000 0000 0000 0109 f000
0000 0141 9a00 0200 0593 ff45 a7ae 1acd
f2d7 f9ec 557f cdb6 ba38 60d6 a626 5edb
4bb9 9783 89e2 d7e1 102e 4625 2fbf ce16
f952 d8c9 f027 e55a 6b2a 81c3 48d4 6a45
050a f355 fbec db01 6562 6405 04aa e011
50ec 0b45 45e5 0df7 2fed a3f8 ac13 2e69
6739 6d81 f13d 2455 e6ca 1c6b dc96 65d5
3bad f250 7dab 42e4 7ba9 f564 ee61 29fb
1b2c 974c 6924 1a1f 99ef 063c b99a c507
8c22 b0f8 b14c 3e4d 01d0 6120 4e19 8725
2fda 6550 f907 3f87
...并且每当 I 帧或 P 帧结束时,我都有一个带有适应字段的 TS 数据包...
4701 003c b000 ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff ffff ffff
ffff ffff ffff ffff ffff ffff
...其中第一个 b0 字节是适配字段填充字节,其余字节是 I 或 P 帧的最后字节。所以你可以告诉我,我可以使用 ffmpeg 并将我的文件传递给它,以创建任何格式的有效电影。但是,我需要我创建的文件格式正确,我无法弄清楚我缺少的最后一块是什么。有任何想法吗?