video - 如何使用ffmpeg overlay_cuda滤镜制作SBS视频?

标签 video ffmpeg nvidia video-processing

FFMPEG 几个月前推出了带有新过滤器“overlay_cuda”的新版本 FFMPEG,此过滤器与“覆盖”相同,但使用 Nvidia 卡来应用它。

我在 FFMPEG 网站上找到了过滤器的描述,但没有关于如何使用它的示例。我发现的唯一示例来自开发人员提交的内容,但将视频或照片放在另一个视频上。

我在普通叠加过滤器之前使用宽度为两倍的 nullsrc 图像来执行此操作,但现在我不知道如何使用此过滤器。

提交说明:https://patchwork.ffmpeg.org/project/ffmpeg/patch/20200318071955.2329-1-yyyaroslav@gmail.com/ ffmpeg 文档网页:https://ffmpeg.org/ffmpeg-filters.html#overlay_005fcuda-1

我希望你能帮助我。

更新:

我制作了这个 FFmpeg 命令:

  1. 输入每个视频。
  2. 第一个视频在右侧创建填充,然后上传到卡的内存中。
  3. 通过覆盖 Cuda,另一个视频放在原始视频的右侧。
ffmpeg -y -loglevel info \
-i $video_1  \
-hwaccel cuda -hwaccel_output_format cuda -i $video_2 \
-filter_complex \
" \
[0:v]pad=w=2*iw:h=ih:x=0:y=0,hwupload_cuda[base];
[base][1:v]overlay_cuda=x=800:y=0" \
-an -c:v h264_nvenc overlay_test.mp4

但我收到此错误消息:

[overlay_cuda @ 0x55fdec4b2ec0] Can't overlay nv12 on yuv420p 
[Parsed_overlay_cuda_2 @ 0x55fdec4b2d80] Failed to configure output pad on Parsed_overlay_cuda_2
Error reinitializing filters!
Failed to inject frame into filter network: Invalid argument
Error while processing the decoded data for stream #1:0

我对像素格式有疑问,希望您能帮助我。

更新 2:

我终于解决了像素格式问题,现在用填充进行叠加(为叠加视频添加空间)

这是成功的命令:

ffmpeg -y -loglevel info \
-i $video_1  \
-hwaccel cuda -hwaccel_output_format cuda -i $video_2 \
-filter_complex \
" \
[0:v]pad=w=2*iw:h=ih:x=0:y=0,hwupload_cuda,scale_npp=format=nv12[base];
[base][1:v]overlay_cuda=x=800:y=0" \
-an -c:v h264_nvenc overlay_test.mp4

现在我想将 x=800 更改为像 x=iw+1 这样的变量,但似乎这个过滤器不支持它。有没有办法设置全局变量?

最佳答案

我明白了! 再次阅读我可以用 Cuda 过滤器做什么后,我发现 scale_npp 不仅可以调整视频大小,还可以更改像素格式。

经过几次测试,我找到了一个非常好的解决方案:

ffmpeg -y -loglevel info \
-hwaccel cuda -hwaccel_output_format cuda -i $video_1  \
-hwaccel cuda -hwaccel_output_format cuda -i $video_2 \
-filter_complex \
" \
[0:v]scale_npp=640:-2:format=yuv420p,hwdownload,pad=w=2*iw:h=ih:x=0:y=0,hwupload_cuda,scale_npp=format=nv12[base];
[1:v]scale_npp=640:-2:format=nv12[overlay_video];
[base][overlay_video]overlay_cuda=x=640:y=0" \
-an -c:v h264_nvenc overlay_test.mp4

是什么造就了这个 FFMPEG 命令:

  1. 使用 Cuda 解码输入两个视频。
  2. 第一个视频[0:v]:
    • 缩放到 640 像素宽度,保持纵横比为 YUV420P 像素格式
    • 从GPU内存下载到系统内存
    • 应用填充过滤器在视频右侧添加 640 像素宽度
    • 再次上传到GPU内存
    • 将像素格式更改为 nv12
    • 标记为[base]
  3. 第二个视频[1:v]
    • 缩放到 640 像素宽度,保持纵横比为 NV12 像素格式
    • 标记为[overlay_video]
  4. 应用 overlay_cuda 过滤器
    • [base]视频作为背景视频
    • [overlay_video] 视频作为前景视频
    • 向右插入 [overlay_video] 640 像素作为 [base] 视频
  5. 完成编码
    • -an as Audio null(这个可以去掉,实际使用中需要混合音频信号或者两个视频中选择一个,甚至添加外部音频源。)
    • -c:v h264_nvenc 使用带有 h264 编解码器的 GPU 对视频进行编码(此处您可以根据需要进行更改)。

唯一的缺点是你需要事先设置一个分辨率,没有办法设置输入分辨率(就像在普通的叠加过滤器中一样)。幸运的是,您可以将其设为脚本的变量,并使用 ffprobe 获取此变量。

关于video - 如何使用ffmpeg overlay_cuda滤镜制作SBS视频?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/63471028/

相关文章:

javascript - IE9 HTML5 视频 "play"事件在第二次播放时未触发

video - 在 Latex 文档中插入 FLV 视频

ios - 使用ffmpeg转换时视频向左旋转90度

FFmpeg 编码数据包持续时间始终为零

ffmpeg - 使用 FFmpeg 从大电影创建缩略图需要太长时间

linux - OpenGL 应用程序导致 Xorg 泄漏

c++ - CUDA C和C++的解释

opengl - 打开全屏 OpenGL 窗口

video - 如果嵌入YouTube实时流视频,是否会增加服务器负载?

c# - FFMPEG C# Winforms 输出到文本框