- c - 在位数组中找到第一个零
- linux - Unix 显示有关匹配两种模式之一的文件的信息
- 正则表达式替换多个文件
- linux - 隐藏来自 xtrace 的命令
我的应用程序打开了一个带有静态初始化代码的库。所有其他库都做同样的事情并且之前加载得很好,但是当从另一个库调用函数时,这个库就死了。这类似于:
0x12311 <-- bad address
_static_initialization_0 <-- function call
....
dlopen
现在,反汇编中的函数调用看起来像
call _Z6MyFuncRA37_Kc@plt
然而,此调用最终调用了无效地址 0x12311,即 PLT 条目获得了错误的地址。
问题很可能是所讨论的库是一种第三方库,即以二进制预构建形式出现,即使它依赖于其他库。上周我们做了很大的优化,改变了很多标题等等。 PLT 错误的函数 MyFunc 位于我们的(另一个)库中,该库进行了大量优化更改。
这怎么可能?确切的问题是:
此外,同一个应用程序在使用 -O2 优化编译时工作正常,这就是我所说的奇怪(两种情况下的二进制库相同)。
附言ubuntu 12.04 x86_64 但应用程序是 i386。
更新:评论(由于某种原因删除)中检查 LD_DEBUG 的建议很好,在 LD_DEBUG=bindings 中,我在“崩溃”版本的应用程序中看到了这一点:
10272: /media/EXT/work/build32/bin/libMyLib.so: error:
symbol lookup error: undefined symbol: omp_set_num_threads (fatal)
然后它停止绑定(bind) libMyLib.so 符号,而在非失败版本中它继续绑定(bind)其他符号。但我不明白为什么它会继续执行并尝试加载父库。实际上方案如下:
libA -> libB -> libMyLib
libMyLib 失败(如上面的 LD_DEBUG 输出所示)因此它跳过它和 libB 完全(!)并继续绑定(bind) libA 符号。非失败版本完全加载 libMyLib 符号,然后继续加载 libB 符号,然后加载 libA 符号。
坦率地说,它看起来像 ld bug。
至于为什么优化版本有效,我想 omp_ 方法并不是真正需要的,它被链接器优化丢弃,因此它不会在运行时找不到它。
这是在找不到 libC 的 omp_ 符号后我在 LD_DEBUG=all 日志中看到的内容:
19225: symbol=omp_set_num_threads; lookup in file=/usr/lib/i386-linux-gnu/libXdmcp.so.6 [0]
19225: /media/EXT/Work/libC.so: error: symbol lookup error: undefined symbol: omp_set_num_threads (fatal)
19225:
19225: file=/media/EXT/libA.so [0]; destroying link map
19225:
19225: file=/media/EXT/libA.so [0]; dynamically loaded by /media/EXT/libX.so [0]
19225: file=/media/EXT/libA.so [0]; generating link map
19225: dynamic: 0xf2fdb764 base: 0xf2f81000 size: 0x00064a28
19225: entry: 0xf2f8ffd0 phdr: 0xf2f81034 phnum: 7
19225:
19225: checking for version `GCC_3.0' in file /lib/i386-linux-gnu/libgcc_s.so.1 [0] required by file /media/EXT/libA.so [0]
... few more checking
19225: object=/media/EXT/libA.so [0]
19225: scope 0: bin/mainapp /lib/i386-linux-gnu/libpthread.so.0 /media/EXT/libX.so ...
19225: scope 1:...
19225:
19225:
19225: relocation processing: /media/EXT/libA.so
19225: symbol=_ZTVN10__cxxabiv117__class_type_infoE; lookup in file=bin/mainapp [0]
19225: symbol=_ZTVN10__cxxabiv117__class_type_infoE; lookup in file=/lib/i386-linux-gnu/libpthread.so.0 [0]
19225: symbol=_ZTVN10__cxxabiv117__class_type_infoE; lookup in file=/media/EXT/libX.so [0]
19225: binding file /media/EXT/libA.so [0] to /media/EXT/libX.so [0]: normal symbol `_ZTVN10__cxxabiv117__class_type_infoE'
... here it continues to bind libA symbols, and after finishing that
19225:
19225:
19225: calling init: /media/EXT/libC.so
19225:
它为未初始化的 libC.so 模块调用 init。
(仅需提及 libX.so 是调用 dlopen 的基本模块,还包含所有其他库使用的基本方法。)
销毁 libA 的链接映射后,日志显示它再次生成,我只是不明白加载程序是否继续加载 libA 或这次从头开始而不关心 libB/libC。好吧,在为 libC 调用 init 之前,它无论如何都会忽略 libB/libC。
最佳答案
omp_set_num_threads
与 OpenMP support 有关在 GCC 内部。
您可能应该通过 -fopenmp
在编译和链接时标记为 gcc(即使您只是dlopen
-ing 使用 OpenMP 的库)。
也许原始图书馆提供者忘记了这一点。
(OpenMP 正在改变编译过程的整个行为)
关于c++ - 调用@plt函数时dlopen/static init上的共享库SIGSEGV,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/13817333/
每当在同一进程中使用 Linux 上的 dlopen() 函数从任何其他库加载任何新库时,我能否在我的库中获得通知?谢谢。 最佳答案 构建一个 library interposer插入 dlopen(
如果我的可执行文件调用 dlopen 来加载一个库但忽略了调用 dlclose,该库将保持加载状态直到进程退出并且操作系统强制它卸载。 如果我加载 a.so 加载 b.so,然后在 a.so 上调用
两次不同的 dlopen 和 dlclose 几次,在 dlopen 上被阻止。 卡在dlopen上,它不输出任何内容,cpuidle降至0%,并且无法通过ctrl+c退出。 LOG_TRACE("a
package com.test.nativeapp; import android.support.v7.app.ActionBarActivity; import android.os.Bundl
我在 channelReceiver.c 中使用 md5使用 jni 文件 MD5_CTX td; MD5_Init(&td); block = malloc(176); MD5_Update(&td
假设我有一个 parent 和一个 child , child 用 dlopen 在 child 中调用函数“hello”。然后 child 可以在 parent 中调用函数“world”吗?我不断收
根据文档,dlopen与 dlsym 结合使用加载库,并获取指向符号的指针。 但这已经是动态加载器/链接器所做的。 而且,这两种方法都是基于ld.so . 使用 dlopen 时实际上似乎有两个不同之
我需要从另一个程序调用一个函数。如果另一个程序是一个库,我可以简单地使用 dlopen 和 dlsym 来获取该函数的句柄。不幸的是,另一个程序是 Unix 可执行文件,并且不能将其构建为库。在可执行
我的程序使用通过 dlopen() 动态加载的插件。这些插件的位置可以是任意的,因此它们不一定在库路径中。在某些情况下,一个插件需要依赖另一个插件。因此,如果 A 和 B 是动态库,我将首先加载 A,
尝试通过 dlopen 加载模块(共享对象)时,加载失败。 比如说,我的 testshobj.c 具有以下内容: // testobj.c int dummy() { return 5; }
我正在 linux 中使用“dlopen”制作一个插件系统。有一个在插件之间传输数据(JSON)的中央应用程序。我们可以使用 try-catch 子句处理抛出的异常。我的问题是关于终止程序的错误,比如
我有一个奇怪的错误,上面写着: java.lang.UnsatisfiedLinkError: dlopen failed: cannot find "./obj/local/armeabi-v7a/
我在网上搜索后似乎找不到答案。 当我第一次使用 dlopen 时,它似乎比之后的任何时候都花费更长的时间,包括如果我从一个程序的多个实例运行它。 dlopen 是否将 so 加载到内存中一次并让操作系
在 C++ 中使用 dlopen 时,我正在努力思考工厂模式在内部是如何工作的。很抱歉发了很长的帖子。 tl;博士;问题在下面的粗体中。 来自 http://www.tldp.org/HOWTO/C+
我正在使用 dlopen 加载动态生成的代码。程序在代码上调用编译器并生成一个 .so 文件,然后程序加载该文件以扩展自身。 问题是,如果我对生成的代码使用相同的名称,dlopen 会返回旧对象的句柄
sharedlibrary通过LD_PRELOAD加载,同库的构造函数调用dlopen("libc.so.6") 问题是dlopen一直在用,调试显示如下dlopen 调用 __dlopen 调用 c
作为这篇文章的延续 C pluginsystem: symbol lookup error ,我还在写我的插件系统,遇到新的bug。 回顾一下插件是什么,该程序由一个由外壳接口(interface)的
我有一个程序调用 dlopen(使用 RTLD_NOW)来动态加载一个库,该库的完整路径在运行时指定,但程序首次执行时不知道。指定的库动态链接到另一个 .so 文件,该文件的位置直到程序启动后才知道,
除了共享对象不存在之外,dlopen 可能出现段错误的一些原因是什么? 在我的例子中,我知道共享对象存在,但是当我的程序使用 dlopen 加载它时,它会出现段错误。我检查了我的 lib 文件夹,共享
我试图让一个共享库从它被加载到的进程中调用一个函数。该库是用 C 语言编写的,即 C++ 中的“内核”。 内核.cpp: #include #include typedef void(*func_
我是一名优秀的程序员,十分优秀!