ffmpeg在多线程环境中运行时挂起

时间:2016-05-12 09:24:34

标签: java multithreading video ffmpeg hang

我有一项服务需要转换大量不同格式的视频。该服务生成五个线程,一个用于单个视频,每个线程使用以下命令运行ffmpeg:

ffmpeg -i %%FILEPATH%% -vf scale=X:Y -ab 128k -c:a aac -movflags faststart -strict -2 -ar 22050 -r 24 -c:v libx264 -crf 25 -y %%OUTPUT.MP4%%

其中X和Y是基于原始文件方向的所需尺寸,基本上是横向的640:trunc(ow*a/2)*2或纵向的trunc(oh*a/2)*2:640

这是我的ffmpeg信息:

ffmpeg version 2.4.3-1ubuntu1~trusty6 Copyright (c) 2000-2014 the FFmpeg developers
  built on Nov 22 2014 17:07:19 with gcc 4.8 (Ubuntu 4.8.2-19ubuntu1)
  configuration: --prefix=/usr --extra-version='1ubuntu1~trusty6' --build-suffix=-ffmpeg --toolchain=hardened --extra-cflags= --extra-cxxflags= --libdir=/usr/lib/x86_64-linux-gnu --shlibdir=/usr/lib/x86_64-linux-gnu --incdir=/usr/include/x86_64-linux-gnu --enable-gpl --enable-shared --disable-stripping --enable-avresample --enable-avisynth --enable-fontconfig --enable-gnutls --enable-ladspa --enable-libass --enable-libbluray --enable-libbs2b --enable-libcaca --enable-libcdio --enable-libflite --enable-libfreetype --enable-libfribidi --enable-libgme --enable-libgsm --enable-libmodplug --enable-libmp3lame --enable-libopenjpeg --enable-libopus --enable-libpulse --enable-librtmp --enable-libschroedinger --enable-libshine --enable-libspeex --enable-libssh --enable-libtheora --enable-libtwolame --enable-libvorbis --enable-libvpx --enable-libwavpack --enable-libwebp --enable-opengl --enable-x11grab --enable-libxvid --enable-libx265 --enable-libdc1394 --enable-libiec61883 --enable-libzvbi --enable-libzmq --enable-frei0r --enable-libx264 --enable-libsoxr --enable-openal --enable-libopencv
  libavutil      54.  7.100 / 54.  7.100
  libavcodec     56.  1.100 / 56.  1.100
  libavformat    56.  4.101 / 56.  4.101
  libavdevice    56.  0.100 / 56.  0.100
  libavfilter     5.  1.100 /  5.  1.100
  libavresample   2.  1.  0 /  2.  1.  0
  libswscale      3.  0.100 /  3.  0.100
  libswresample   1.  1.100 /  1.  1.100
  libpostproc    53.  0.100 / 53.  0.100
Hyper fast Audio and Video encoder
usage: ffmpeg [options] [[infile options] -i infile]... {[outfile options] outfile}...

该服务是用Java编写的,在Ubuntu Server 14.04上运行,该机器是64位八核服务器。

这是执行ffmpeg:

的代码块
try 
{
    ProcessBuilder procBuilder = null;


    String sArgs = String.format("ffmpeg -i %s -vf scale=%s:%s -ab 128k -c:a aac -movflags faststart -strict -2 -ar 22050 -r 24 -c:v libx264 -crf 25 -y %s",
                originalPath,
                outWidth,
                outHeight,
                targetPath
        );

    }

    String[] arrArgs = sArgs.split("\\s+");
    procBuilder = new ProcessBuilder(Arrays.asList(arrArgs));
    procBuilder.redirectErrorStream(true);
   procBuilder.redirectOutput();


    Process process = procBuilder.start();

    try (BufferedReader reader = new BufferedReader(new InputStreamReader(process.getInputStream()))) 
    {
        String line = null;
        while ((line = reader.readLine()) != null) 
        {
            System.out.println(line);
        }
        errorCode = process.waitFor();
    }
}
catch(Throwable ex)
{
}

我目前正在产生五个线程,每个线程运行一个针对单个视频文件的ffmpeg实例。这在大多数情况下工作正常,但每隔一段时间线程就会开始挂起。我从顶部注意到ffmpeg无限期挂起一些文件,其中一个线程使用100%的核心CPU而没有进展。它发生在不同的文件类型,因为我注意到mkv,avi,wmv和mp4文件。

我不确定导致ffmpeg挂起的原因是什么,它不会在转码过程开始时发生,ffmpeg开始转换文件很好但是在中间某处卡住了。

现在这不是文件的问题,因为当我在同一个文件上手动尝试相同的命令时它运行正常。而且只有当ffmpeg的多个实例同时运行时才会发生这种情况,因为我现在将服务更改为仅运行一个线程并且它已经运行了近一个月而没有任何问题。

我是否需要使用一个选项来允许多个ffmpeg实例同时运行?在我目前使用的命令行中是什么导致了这个?

1 个答案:

答案 0 :(得分:1)

libx264默认使用多个线程,等于1.5 *逻辑核心(用于基于帧的多线程)。将其与五个实例相乘,您可以获得大量线程,这仅用于编码视频流。

您可以尝试通过设置-x264opts threads=n选项来减少每个实例上使用的线程数。