如何将与索引并行创建的多个图像传输到 ffmpeg,以便它可以匹配图像创建的速度?

sy_*_*001 5 python encoding png ffmpeg webm

png我们有一个逐帧输出 4 通道图像的系统(我们也控制这些图像的输出格式,因此只要它支持透明度,我们就可以使用其他东西)。现在,我们正在等待所有图像,然后使用(编码器)将它们编码ffmpegwebm视频文件。但我们现在希望将这些图像传输到 FFmpeg,以便在图像喷出时同时编码到 WebM 视频中,这样我们就不必等待之后对所有图像进行编码。vp8libvpxffmpeg

这是当前命令,采用 python 语法:

['/usr/bin/ffmpeg', '-hide_banner', '-y', '-loglevel', 'info', '-f', 'rawvideo', '-pix_fmt', 'bgra', '-s', '1573x900', '-framerate', '30', '-i', '-', '-i', 'audio.wav', '-c:v', 'libvpx', '-b:v', '0', '-crf', '30', '-tile-columns', '2', '-quality', 'good', '-speed', '4', '-threads', '16', '-auto-alt-ref', '0', '-g', '300000', '-map', '0:v:0', '-map', '1:a:0', '-shortest', 'video.webm']
# for ease of read:
# /usr/bin/ffmpeg -hide_banner -y -loglevel info -f rawvideo -pix_fmt bgra -s 1573x900 -framerate 30 -i - -i audio.wav -c:v libvpx -b:v 0 -crf 30 -tile-columns 2 -quality good -speed 4 -threads 16 -auto-alt-ref 0 -g 300000 -map 0:v:0 -map 1:a:0 -shortest video.webm

proc = subprocess.Popen(args, stdin=subprocess.PIPE)
Run Code Online (Sandbox Code Playgroud)

以下是将图像传递到 FFMPEG proc stdin 的示例:

# wait for the next frame to get ready
for frame_path in frame_path_list:
    while not os.path.exists(frame_path):
        time.sleep(0.25)
    frame = cv2.imread(frame_path, cv2.IMREAD_UNCHANGED)
    
    # put the frame in stdin so that it gets ready
    proc.stdin.write(frame.astype(np.uint8).tobytes())
Run Code Online (Sandbox Code Playgroud)

目前该过程的速度约为 0.135 倍,这对我们来说是一个巨大的瓶颈。早些时候,当我们接受输入时,-pattern_type glob -i images/*.png我们在单核上获得了大约 1x-1.2x 的性能。因此,我们的结论是,我们正受到标准输入的瓶颈,因此正在寻找通过多个源传递输入的方法,或者以某种方式帮助ffmpeg并行化这项工作 - 我们正在考虑的一些选项:

  • 以某种方式将其提供给不同的管道并让 ffmpeg 从中读取。
  • 将新图像附加到 ffmpeg,无需重新编码整个视频,但我们没有找到直接提供输入图像的方法。

但我们还无法让这两种解决方案发挥作用,也无法接受任何其他解决方案。非常感谢对此的帮助。谢谢!

Tav*_*nes 1

您可以使用 ffmpeg 的选项,正如我在此处-f image2pipe发现的那样。例子:

cat frames/{0000..0512}.png | ffmpeg -f image2pipe -r 60 -i - -c:v libx265 video.mkv
Run Code Online (Sandbox Code Playgroud)