首页 > 解决方案 > 为什么 ffmpeg 链接的库太多?

问题描述

我正在尝试构建用于在 Linux 系统上解码视频的ffmpeg共享库。构建是在 Arch Linux 上完成的,但生成的共享库需要尽可能可移植。

幸运的是,ffmpeg 内置的功能就足够了,所以我不想与目标系统上可能存在或不存在的任何库链接。我在libvdpau这里举了一个不需要的依赖的例子,但还有更多。

这是我尝试过的:

$ git clone https://git.ffmpeg.org/ffmpeg.git --branch n4.3.1 --depth 1
$ cd ffmpeg
$ mkdir build
$ cd build
$ ../configure \
    --enable-shared \
    --disable-programs --disable-doc \
    --disable-avdevice --disable-postproc --disable-avfilter \
    --disable-autodetect
$ make -j9

注意--disable-autodetect哪个应该防止自动检测构建系统上碰巧存在哪些依赖项。

然而,当我检查生成的库时,它们都具有我没有要求的依赖项。例如:

$ ldd libavcodec/libavcodec.so
    linux-vdso.so.1 (0x00007ffcd73cd000)
    libswresample.so.3 => /usr/lib/libswresample.so.3 (0x00007fba1e45c000)
    libavutil.so.56 => /usr/lib/libavutil.so.56 (0x00007fba1e1a7000)
    libm.so.6 => /usr/lib/libm.so.6 (0x00007fba1e061000)
    libpthread.so.0 => /usr/lib/libpthread.so.0 (0x00007fba1e03f000)
    libc.so.6 => /usr/lib/libc.so.6 (0x00007fba1de76000)
    libsoxr.so.0 => /usr/lib/libsoxr.so.0 (0x00007fba1ddf4000)
    libva-drm.so.2 => /usr/lib/libva-drm.so.2 (0x00007fba1dded000)
    libva.so.2 => /usr/lib/libva.so.2 (0x00007fba1ddc0000)
    libva-x11.so.2 => /usr/lib/libva-x11.so.2 (0x00007fba1ddb8000)
    libvdpau.so.1 => /usr/lib/libvdpau.so.1 (0x00007fba1ddb3000)     <- Why is this here?
    libX11.so.6 => /usr/lib/libX11.so.6 (0x00007fba1dc72000)
    libdrm.so.2 => /usr/lib/libdrm.so.2 (0x00007fba1dc5d000)
    libmfx.so.1 => /usr/lib/libmfx.so.1 (0x00007fba1dc4d000)
    libdl.so.2 => /usr/lib/libdl.so.2 (0x00007fba1dc47000)
    /usr/lib64/ld-linux-x86-64.so.2 (0x00007fba1f971000)
    libgomp.so.1 => /usr/lib/libgomp.so.1 (0x00007fba1dc06000)
    libXext.so.6 => /usr/lib/libXext.so.6 (0x00007fba1dbf1000)
    libXfixes.so.3 => /usr/lib/libXfixes.so.3 (0x00007fba1dbe8000)
    libxcb.so.1 => /usr/lib/libxcb.so.1 (0x00007fba1dbbc000)
    libstdc++.so.6 => /usr/lib/libstdc++.so.6 (0x00007fba1d9df000)
    libgcc_s.so.1 => /usr/lib/libgcc_s.so.1 (0x00007fba1d9c5000)
    libXau.so.6 => /usr/lib/libXau.so.6 (0x00007fba1d9c0000)
    libXdmcp.so.6 => /usr/lib/libXdmcp.so.6 (0x00007fba1d9b8000)

我还尝试明确列出所有禁用标志,例如--disable-vdpau,但这并没有改变任何东西;libvdpau.so.1仍然在ldd输出中列出。

也许它们只是被链接,但没有实际使用?readelf -d似乎指向那个方向:

$ ls */*.so
libavcodec/libavcodec.so    libavutil/libavutil.so          libswscale/libswscale.so
libavformat/libavformat.so  libswresample/libswresample.so
$ readelf -d */*.so | grep 'Shared library:' | sort -u
 0x0000000000000001 (NEEDED)             Shared library: [libavcodec.so.58]
 0x0000000000000001 (NEEDED)             Shared library: [libavutil.so.56]
 0x0000000000000001 (NEEDED)             Shared library: [libc.so.6]
 0x0000000000000001 (NEEDED)             Shared library: [libm.so.6]
 0x0000000000000001 (NEEDED)             Shared library: [libpthread.so.0]
 0x0000000000000001 (NEEDED)             Shared library: [libswresample.so.3]

这是一组很好且紧密的依赖关系,也是我真正希望它们成为的方式。

所以也许构建系统很愚蠢,并且总是将所有检测到的库添加到链接器命令行?为了抑制这种情况,我尝试添加--extra-ldflags=-Wl,--as-needed,但这似乎也没有任何效果。

生成的链接器命令行如下所示(删节):

$ make -n libavcodec/libavcodec.so.58
gcc \
    -shared \
    -Wl,-soname,libavcodec.so.58 \
    -Wl,-Bsymbolic \
    -Wl,--version-script,libavcodec/libavcodec.ver \
    -Llibavcodec -Llibavdevice -Llibavfilter -Llibavformat -Llibavresample -Llibavutil -Llibpostproc -Llibswscale -Llibswresample \
    -Wl,--as-needed \
    -Wl,-z,noexecstack \
    -Wl,--warn-common \
    -Wl,-rpath-link=:libpostproc:libswresample:libswscale:libavfilter:libavdevice:libavformat:libavcodec:libavutil:libavresample \
    -Wl,--as-needed \
    -o libavcodec/libavcodec.so.58 \
    [...all .o files...] \
    -lswresample -lavutil -pthread -lm -lm -pthread -lm

反正好像-Wl,--as-needed已经加了。

有趣的-Llibavdevice是,也列出来了,尽管我传递--disable-avdevice --disable-postproc --disable-avfilter给了配置脚本。这可能是一个线索:它会获取我系统的libavdevice,这当然会带来更多的依赖关系。我仍然希望-Wl,--as-needed将这些未构建的依赖项也删掉。

这些依赖从何而来,我该如何摆脱它们?

标签: ffmpeglinkerdynamic-linking

解决方案


当然:运行时ldd,对其他 ffmpeg 库的任何依赖都会解析为我系统上安装的那些。而那些有更多的依赖关系。所以问题不在于构建系统,而在于我检查结果输出的方式。

设置LD_LIBRARY_PATH揭示了传递依赖的真实列表:

$ LD_LIBRARY_PATH=libavcodec:libavformat:libavutil:libswresample:libswscale ldd libavcodec/libavcodec.so
    linux-vdso.so.1 (0x00007fffaab0f000)
    libswresample.so.3 => libswresample/libswresample.so.3 (0x00007f9a8f213000)
    libavutil.so.56 => libavutil/libavutil.so.56 (0x00007f9a8ef6c000)
    libm.so.6 => /usr/lib/libm.so.6 (0x00007f9a8ede0000)
    libpthread.so.0 => /usr/lib/libpthread.so.0 (0x00007f9a8edbe000)
    libc.so.6 => /usr/lib/libc.so.6 (0x00007f9a8ebf5000)
    /usr/lib64/ld-linux-x86-64.so.2 (0x00007f9a906e1000)

推荐阅读