【问题标题】:How to run an old binary on modern GNU/Linux distribution?如何在现代 GNU/Linux 发行版上运行旧的二进制文件?
【发布时间】:2021-05-11 20:49:42
【问题描述】:

我正在尝试在现代 GNU/Linux 发行版 (Gentoo Linux) 上运行在 CentOS 6 上编译的旧 C++ 程序。现在 CentOS 6 已经过时,推荐的库也已经过时了,所以有必要为二进制提供工作环境。

我已经尝试使用ldd 命令和一些脚本从 CentOS 6 复制二进制所需的所有必要共享库。现在我有一个包含所有必需库的目录。但是,当我尝试使用 LD_LIBRARY_PATH="the directory with .so-files" 运行我的程序时,终端会显示一条消息

$ LD_LIBRARY_PATH=`pwd`/lib ./my-program
Inconsistency detected by ld.so: dl-call-libc-early-init.c: 37: _dl_call_libc_early_init: Assertion `sym != NULL' failed!

所以它出于某种原因不想工作。如果没有强制链接到旧的 .so,二进制文件也不会运行(显然)。我没有时间在现代 Linux 上重新编译我的程序或制作它的静态版本,因为它需要旧库,而且程序必须在少数我不维护的 PC 上运行。

有没有一种可靠的方法可以在没有容器或虚拟机的情况下在现代 Linux 上运行旧的二进制文件?也许我走对了,但需要解决一些问题,或者我走错了路,需要尝试其他方法。

如果您读到此行,感谢您的关注!

我附上了所需库的列表(ldd my-program 的输出):

linux-vdso.so.1 =>  (0x00007ffc26b40000)
librfftw.so.2 => /usr/lib64/librfftw.so.2 (0x00007f026f631000)
libfftw.so.2 => /usr/lib64/libfftw.so.2 (0x00007f026f3f8000)
libdl.so.2 => /lib64/libdl.so.2 (0x00007f026f1f4000)
libboost_thread-mt.so.5 => /usr/lib64/libboost_thread-mt.so.5 (0x00007f026efdf000)
libboost_system-mt.so.5 => /usr/lib64/libboost_system-mt.so.5 (0x00007f026eddc000)
libpthread.so.0 => /lib64/libpthread.so.0 (0x00007f026ebbf000)
libxml++-2.6.so.2 => /usr/lib64/libxml++-2.6.so.2 (0x00007f026e99c000)
libxml2.so.2 => /usr/lib64/libxml2.so.2 (0x00007f026e649000)
libglibmm-2.4.so.1 => /usr/lib64/libglibmm-2.4.so.1 (0x00007f026e3f4000)
libgobject-2.0.so.0 => /lib64/libgobject-2.0.so.0 (0x00007f026e1a8000)
libgthread-2.0.so.0 => /lib64/libgthread-2.0.so.0 (0x00007f026dfa4000)
librt.so.1 => /lib64/librt.so.1 (0x00007f026dd9c000)
libglib-2.0.so.0 => /lib64/libglib-2.0.so.0 (0x00007f026da85000)
libsigc-2.0.so.0 => /usr/lib64/libsigc-2.0.so.0 (0x00007f026d880000)
libstdc++.so.6 => /usr/lib64/libstdc++.so.6 (0x00007f026d57a000)
libm.so.6 => /lib64/libm.so.6 (0x00007f026d2f6000)
libgcc_s.so.1 => /lib64/libgcc_s.so.1 (0x00007f026d0e0000)
libc.so.6 => /lib64/libc.so.6 (0x00007f026cd4c000)
/lib64/ld-linux-x86-64.so.2 (0x00007f026f864000)
libz.so.1 => /lib64/libz.so.1 (0x00007f026cb36000)
libgmodule-2.0.so.0 => /lib64/libgmodule-2.0.so.0 (0x00007f026c933000)

【问题讨论】:

  • 在 chroot 中安装 centos6 或使用 docker run centos6 + 运行二进制文件。无论如何,使用旧解释器运行二进制文件。 LD_LIBRARY_PATH=$PWD/lib $PWD/lib/ld-linux-x86-64.so.2 ./my-program.
  • @KamilCuk 非常感谢!我已经使用 LXD 容器,但我的同事还没有准备好使用 LXD/Docker/chroot。您使用旧链接器的建议已解决问题????????

标签: c++ linux shared-libraries ld ldd


【解决方案1】:

有没有靠谱的方法

没有。 :p

有没有办法在没有容器或虚拟机的情况下在现代 Linux 上运行旧的二进制文件?

您可以使用旧链接器和库路径运行程序。当您拥有旧系统 chroot 时,您可以:

LD_LIBRARY_PATH=$PWD/lib $PWD/lib/ld-linux-x86-64.so.2 ./my-program

但请注意,如果./my-program 将通过fork+exec 运行任何东西,则可能会出现问题。您可以将PATH 设置为chroot 的位置,但解释器仍然可能是错误的。因此,如果它是一个简单的程序,它“通常可以工作”。

无论如何,如果没有docker 或普通chroot,您可能希望看到proot 程序。

【讨论】:

    猜你喜欢
    • 2017-09-02
    • 2017-02-16
    • 1970-01-01
    • 2013-01-28
    • 1970-01-01
    • 2016-08-13
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多