gpt4 book ai didi

c++ - 随应用程序一起运送 libstdc++.so.6

转载 作者:可可西里 更新时间:2023-11-01 17:54:20 28 4
gpt4 key购买 nike

我想为我的应用程序使用 gcc 4.8.1(需要 libstdc++.so.6.0.18),但是客户只有 libstdc++.so.6.0.13。我使用 -static-libgcc -static-stdlibc++ 有一段时间了,但我的应用程序包含几个动态链接库和一个主应用程序。这意味着在编译每个动态库时,他们必须静态编译标准库,这是多余和浪费的。我只想随我的产品一起发布我选择的标准库,但是每次我在像他们这样的环境中运行我的应用程序时,它总是加载错误的标准库。无论我做什么,它都更喜欢 /usr/lib64/ 版本(它似乎优先于 LD_LIBRARY_PATH)。

约束:

  1. 我不能强制他们升级到新的标准库。

  2. 我不想将动态库设为静态。 (我可以一次将所有内容静态编译到主应用程序中,但是有一些后勤障碍阻止我将一些库重新编译成静态库)。

-Wl,-rpath=$(path_to_directory) 有点危险,但它是合法的,因为客户确实提供了一些允许我设置路径变量的设置。但是,设置我的新 stdlibc++ 的 rpath 似乎并没有覆盖默认的 /usr/lib64 版本。我仍然收到 GLIBCXX 错误,因为它不会使用正确的库。

肯定有一个优雅的解决方案吗?

也许只是我的程序出错了。这是一个示例(对审查员感到抱歉,但这只是用户名的东西):

~/example$ pwd
/home/username/example
~/example$ echo $LD_LIBRARY_PATH

~/example$ ls
Makefile libstdc++.so.6.0.18 test.cpp
~/example$ make
g++ -std=c++11 -Wall -Werror test.cpp -o test
~/example$ ldd test
./test: /usr/lib64/libstdc++.so.6: version `GLIBCXX_3.4.14' not found (required by ./test)
linux-vdso.so.1 => (0x00007fffe5919000)
libstdc++.so.6 => /usr/lib64/libstdc++.so.6 (0x000000390b800000)
libm.so.6 => /lib64/libm.so.6 (0x0000003904800000)
libgcc_s.so.1 => /lib64/libgcc_s.so.1 (0x000000390b400000)
libc.so.6 => /lib64/libc.so.6 (0x0000003904400000)
/lib64/ld-linux-x86-64.so.2 (0x0000003904000000)
~/example$ setenv LD_LIBRARY_PATH /home/username/example
~/example$ echo $LD_LIBRARY_PATH
/home/username/example
~/example$ ldd test
./test: /usr/lib64/libstdc++.so.6: version `GLIBCXX_3.4.14' not found (required by ./test)
linux-vdso.so.1 => (0x00007fff2d3ff000)
libstdc++.so.6 => /usr/lib64/libstdc++.so.6 (0x000000390b800000)
libm.so.6 => /lib64/libm.so.6 (0x0000003904800000)
libgcc_s.so.1 => /lib64/libgcc_s.so.1 (0x000000390b400000)
libc.so.6 => /lib64/libc.so.6 (0x0000003904400000)
/lib64/ld-linux-x86-64.so.2 (0x0000003904000000)

对不起各位,我犯了一个相当愚蠢的错误......

~/example$ file libstdc++.so.6.0.18 
libstdc++.so.6.0.18: ELF 32-bit LSB shared object, Intel 80386, version 1 (SYSV), dynamically linked, not stripped

一些笨蛋构建了错误版本的库,另一个笨蛋(也就是我自己)尝试在 64 位机器上使用它。使用 LD_LIBRARY_PATH 一直有效...

最佳答案

您的问题是可执行文件链接到 soname libstdc++.so.6 而不是完整的库文件名 libstdc++.so.6.0.16。动态链接器将在常用位置(即 LD_LIBRARY_PATH、DT_RPATH、ldconfig 目录等)查找 libstdc++.so.6,因此要确保找到 6.0.18 版本,您需要一个名为 libstdc++.so.6 指向它。

我更喜欢使用 '-Wl,-rpath,$ORIGIN' 链接,而不是使用 LD_LIBRARY_PATH(这在您无法控制的机器上很脆弱,因为用户可能会改变他们的环境)(注意引号是停止 shell 扩展所必需的 $ORIGIN)

$ORIGIN 的 RPATH 告诉动态链接器开始在与可执行文件相同的目录中寻找共享库,因此如果您将 libstdc++.so 与您的可执行文件一起发布,它将被找到。如果您想将可执行文件放在 bin 目录中并将库放在 lib 目录中,您可以使用 '-Wl,-rpath,$ORIGIN/。 ./lib' 或其他相对于可执行文件位置的路径。

关于c++ - 随应用程序一起运送 libstdc++.so.6,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/27385995/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com