两个共享库liba.so和libb.so。liba.so使用libb.so。所有c文件都使用-fPIC编译。链接使用-
shared。当我们在liba.so上调用dlopen时,它无法在libb.so中找到符号…我们得到“未定义符号”错误。我们可以dlopen
libb.so没有错误。我们知道liba正在找到libb,因为我们没有得到文件未找到错误。删除libb.so时,出现文件未找到错误。我们尝试了-
lutil,但没有运气。
有任何想法吗????
哦耶。gcc 4.1.2
更新:链接liba时,我们使用rpath,以便它可以找到libb。
ldd liba.so返回:
linux-gate.so.1 => (0xffffe000)
libb.so => ./libb.so (0xf6ef9000) <-------- LIBB
libutil.so.1 => /lib/libutil.so.1 (0xf6ef5000)
libdl.so.2 => /lib/libdl.so.2 (0xf6ef1000)
libm.so.6 => /lib/libm.so.6 (0xf6ec9000)
libpthread.so.0 => /lib/libpthread.so.0 (0xf6eb1000)
librt.so.1 => /lib/librt.so.1 (0xf6ea8000)
libc.so.6 => /lib/libc.so.6 (0xf6d62000)
/lib/ld-linux.so.2 (0x007d0000)
在libb末尾没有。#有意义吗???
您可以libb.so
使用以下ldd
命令轻松检查期望的位置:
$ ldd liba.so
linux-gate.so.1 => (0xb77b0000)
libb.so.1 => not found
libstdc++.so.6 => /usr/lib/libstdc++.so.6 (0xb75b6000)
libgcc_s.so.1 => /lib/libgcc_s.so.1 (0xb7572000)
libc.so.6 => /lib/i686/cmov/libc.so.6 (0xb742b000)
/lib/ld-linux.so.2 (0xb77b1000)
如果是not found
,libb.so
则应将路径添加到/etc/ld.so.conf
或shell变量中LD_LIBRARY_PATH
。
另一种方式是设置rpath
在liba.so
本身-它基本上硬编码它的路径,所以当二进制启动动态连接器会知道在哪里搜索共享库。
如果rpath
未设置,它将首先在中搜索LD_LIBRARY_PATH
,然后在/etc/ld.so.conf
(或/etc/ld.so.conf.d/)中提到的路径。添加后ls.so.conf
不要忘记执行/sbin/ldconfig
动态链接器通过它们的依赖共享库soname
(如果已设置)搜索它们-如果soname
未设置(例如,使用-
Wl,-soname,libb.so.1),则将按库的名称进行搜索。
示例:libb.so.1.0
是您的实际库,具有soname
- libb.so.1
。通常,您将具有以下文件结构:
libb.so -> libb.so.1
libb.so.1 -> libb.so.1.0
libb.so.1.0
其中libb.so
和libb.so.1
是符号链接。
libb.so
在构建某些应用程序或其他库时,通常取决于链接到libb.so
。
gcc -shared -Wl,-soname,liba.so.1 -o liba.so.1.2 -L/libb/path -lb
启动应用程序(或执行dlopen-您的情况)时,动态链接程序将搜索名称为libb.so.1
-
soname
依赖库(如果soname
已设置)的文件,而不是libb.so
。
这就是为什么您需要那个libb.so.1
指向实际库的symlink的原因。
如果使用ld.so.conf
和ldconfig
,则soname
如果缺少该符号链接,它将使用的名称创建指向该库文件的符号链接。
您可以查看ld-linux手册页以获取更多有用的信息。
如果找到了库,但是缺少一些符号,请尝试libb.so
使用-Wl,--no-undefined
option 构建
gcc -shared -Wl,-soname,libb.so.1 -Wl,--no-undefined -o libb.so.1.2
如果您错过定义某些符号的操作,应该会给您一个错误。
问题内容: 最近,我们被要求提供其中一个库的Linux版本,之前我们是在Linux下开发的,并已针对Windows发行,而在Windows上部署库通常要容易得多。我们遇到的问题是将导出的符号剥离为仅暴露界面中的符号。想要这样做的三个很好的理由 为了保护我们技术的专有方面,以免通过导出的符号被暴露。 防止用户遇到符号名称冲突的问题。 为了加快库的加载速度(至少有人告诉我)。 然后举一个简单的例子:
我正在尝试建立一个共享库,可以用来访问本机蓝牙API函数。我目前正试图编译一个c文件:https://android.googlesource.com/platform/external/bluetooth/bluedroid/ 每当我尝试将这些代码中的任何一个编译为共享库时,我都会收到许多未定义的引用错误...它看起来像是头文件引用的每个函数。在我最近的尝试中,我将整个Bluedroid/bta
问题内容: 我正在制作一个C ++共享库,当我编译使用该库的主exe时,编译器会给出以下信息: 库代码: fooclass.h fooclass.cpp 编译: 主要:main.cpp 编译: 该机器是Ubuntu Linux 12 谢谢! 问题答案: 是问题。最新版本的GCC要求您按照相互依赖的顺序放置目标文件和库-因此,根据经验法则,必须将库标志作为链接器的最后一个开关;即写 代替。
库用于将相似函数打包在一个单元中。然后这些单元就可为其他开发人员所共享,并因此有了模块化编程这种说法— 即,从模块中构建程序。Linux支持两种类型的库,每一种库都有各自的优缺点。静态库包含在编译时静态绑定到一个程序的函数。动态库则不同,它是在加载应用程序时被加载的,而且它与应用程序是在运行时绑定的。 使用共享库的方法有两种:您既可以在运行时动态链接库,也可以动态加载库并在程序控制之下使用它们。本
问题内容: 这有什么问题吗?我有以下简单的课程: 在运行一个简单的compile()之后,出现以下错误: 在共享库上运行nm表示实际上已找到它: 这是在Ubuntu 12.04上发生的。该 libmnl-dev的 和 libmnl0 包安装。的输出表明正使用该* .so文件: 问题答案: 库必须在使用它们的对象之后列出(更确切地说,仅当库包含满足遇到未定义引用的符号时,才使用库)。将移至命令末尾。
问题内容: 我正在尝试建立一个共享库。让我们说libabc.so。它使用另一个.so文件,例如lib123.so(/ usr / local / lib中的一个lib)。现在我在我的应用程序中使用共享的liblibabc.so。说我的应用程序。我想知道我应该如何链接这些二进制文件?我不想直接将我的应用程序与lib123.so链接。my- app应该仅与libabc.so链接。我怎样才能做到这一点?