ffmpeg使用H265 vcopy查找关键帧不准确

时间:2017-08-19 15:55:59

标签: video ffmpeg mp4 seek hevc

我有一个H265 4K MP4 29.97fps视频,GOP大小正好是30帧。 当我尝试从头开始使用:

ffmpeg -ss 1 -i INPUT.MP4 -vcodec copy OUTPUT_1SEC.MP4
ffmpeg -ss 2 -i INPUT.MP4 -vcodec copy OUTPUT_2SEC.MP4
ffmpeg -ss 3 -i INPUT.MP4 -vcodec copy OUTPUT_3SEC.MP4
ffmpeg -ss 4 -i INPUT.MP4 -vcodec copy OUTPUT_4SEC.MP4
ffmpeg -ss 5 -i INPUT.MP4 -vcodec copy OUTPUT_5SEC.MP4
ffmpeg -ss 6 -i INPUT.MP4 -vcodec copy OUTPUT_6SEC.MP4
ffmpeg -ss 7 -i INPUT.MP4 -vcodec copy OUTPUT_7SEC.MP4
ffmpeg -ss 8 -i INPUT.MP4 -vcodec copy OUTPUT_8SEC.MP4
ffmpeg -ss 9 -i INPUT.MP4 -vcodec copy OUTPUT_9SEC.MP4

输出视频从0(-ss 1~4),4(-ss 5~8)或8 sec(-ss 9)开始,如下所示:

Premiere Pro Timeline

所以似乎ffmpeg以某种方式检测到4秒而不是1秒的GOP。这是正常的吗?

另外,如何在输出视频中刻录正确的时间码? 例如,我尝试了许多组合,例如:

ffmpeg -ss 5 -i INPUT.MP4 -vcodec copy -timecode 00:00:05:00 OUTPUT_5SEC.MP4
ffmpeg -ss 5 -i INPUT.MP4 -vcodec copy -copyts OUTPUT_5SEC.MP4
ffmpeg -start_at_zero -ss 5 -i INPUT.MP4 -vcodec copy -copyts OUTPUT_5SEC.MP4

但它要么给我确切的时间码(第一行)或从零开始(最后两行)

最初,我正在考虑在确切的第二个(或之后的几帧)寻找,所以我知道我会得到一个关键帧,所以我可以猜出输出将开始的确切时间码,但似乎ffmpeg -ss并不完全基于关键帧?也许我在这里遗漏了什么?谢谢你的帮助。

Additionnal infos

我想编写剪切过程的脚本,这就是为什么我想知道这个4秒“关键帧间隔”的来源。

以下是我输入的ffprobe输出:

Input #0, mov,mp4,m4a,3gp,3g2,mj2, from 'INPUT.MP4':
  Metadata:
    major_brand     : isom
    minor_version   : 512
    compatible_brands: isomiso2mp41
    encoder         : Lavf57.72.101
    comment         : DE=None, Mode=M, DSW=0001
    location-{    : +XX.4914-0XX.5164+XX.000000/
    location        : +XX.4914-0XX.5164+XX.000000/
  Duration: 00:01:45.31, start: 0.000000, bitrate: 100065 kb/s
    Stream #0:0(eng): Video: hevc (Main) (hev1 / 0x31766568), yuv420p(tv, bt709), 4096x2160 [SAR 1:1 DAR 256:135], 100062 kb/s, 29.97 fps, 29.97 tbr, 30k tbn, 29.97 tbc (default)
    Metadata:
      handler_name    : VideoHandler
    Stream #0:1(eng): Subtitle: mov_text (tx3g / 0x67337874), 2 kb/s (default)
    Metadata:
      handler_name    : SubtitleHandler

这是我用来检查gop-size的命令('I'类型为1,31,61,...和'P'介于两者之间):

ffprobe -i INPUT.MP4 -select_streams v -show_frames -show_entries frame=pict_type -of csv > OUTPUT.CSV

ffmpeg version N-86330-gbd1179e 和ffmpeg version N-86330-gbd1179e

修改: Sample video here

1 个答案:

答案 0 :(得分:1)

虽然每秒都有一个关键帧,但在MOOV框中,只有三帧被设置为同步样本

/moov/trak/mdia/minf/stbl/stss                              @ 0x77e8515
  Box size: 0x1c    version: 0x0    flags: 0x0
  entry_count:              0x3
    sample_number:
    0x1    0x79    0xf1

(那些是第1帧,第121帧和第241帧。)

FFmpeg在寻求时依赖此信息。

解决方法是多路复用到TS,然后重新启动到MP4。

ffmpeg -i input.mp4 -c copy input.ts

然后

ffmpeg -i input.ts -c copy newinput.mp4

或在一个命令中

ffmpeg -i input.mp4 -c copy -f mpegts - | ffmpeg -f mpegts -i - -c copy newinput.mp4

MPEG-TS文件没有索引,因此如果要使用该文件进行提取,请在之前指定要剪切的关键帧的搜索点

至于为什么同步表就是这样,不知道。这取决于原始的编写应用程序和那里使用的设置/参数。