- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
背景:我有应用程序的一部分用作其他独立应用程序的库。他们在链接时链接到那个库(比方说 lib.so)。这种方法的问题是我们必须使用相同的外部库,如 boost、ace 等,否则我们将有重复的符号,最终会导致崩溃。我们想解决这个问题。
我知道两种技术 - 一种是隐藏所有符号(不确定共享库的全局/局部范围的顺序),另一种是使用动态链接。我们选择了第二个选项(动态链接),因为它让客户有机会使用 stub lib.so 进行简单测试。我们有非常简单的 API。
我在下面写了一个应用程序的小例子,它加载了示例共享库并且它崩溃了(我想了解它崩溃的原因以及它应该如何编写)。崩溃发生在 dlopen 中,恰好在分配给 std::string(Aclass 类型的构造函数)时全局变量的初始化中。从我们的测试来看,在库的持续初始化过程中对 std 库的任何访问都将导致崩溃。
我们通过向 EXECUTABLE 添加 -fPIcflags设法消除了崩溃(为什么这解决了我们的问题,我认为它应该为共享库设置,谁能更准确地解释我)?就我的理解而言,这个标志是不必要的,因为它会减慢应用程序的速度,并且在我的情况下(低延迟应用程序)这是非常有问题的。
总结:1.为什么会出现这个crash?2. 为什么 -fPIcflags足以解决此崩溃问题?3. 为什么将-fPIcflags设置为可执行文件就足够了?4. 是否有可能以其他方式解决我的问题,以便共享库和客户端应用程序可以使用不同版本的库(如 boost、ace 等,编译器、linux 版本和 std 库保证相同)?5. 删除标志 RTLD_DEEPBIND 也将修复崩溃,但从 gcc man 看来我应该使用此标志,因为它将更改共享库的符号范围顺序 - 首先它将在本地范围内搜索符号,然后在全局范围内搜索符号 - 看起来对我来说是必须的因为共享库将使用与可执行文件不同的外部库(并且动态加载将通过污染其符号范围来保护可执行文件)。为什么在这个简单的案例中删除这个标志修复崩溃?
共享库 dynLib.cpp:
#include <string>
class Aclass
{
std::string s;
s = "123";
}
Aclass a;
可执行的main.cpp:
#include <stdlib.h>
#include <dlfcn.h>
#include <string>
#include <unistd.h>
#include <iostream>
int main()
{
std::string dummyCrasher;
dlerror();
void* handle = dlopen("./libdynLib.so", RTLD_LAZY | RTLD_LOCAL | RTLD_DEEPBIND);
if(!handle)
{
std::cout << "handle is null" << dlerror();
}
usleep(1000 * 1000 * 10);
}
生成文件:生成文件
CXXFLAGS=-m32 -march=x86-64 -Wl,v -g -O3 -Wformat -Werror=format -c
CLINKFLAGS=-Wl,-Bstatic -Wl,Bdynamic -ldl -m32 -march=x86-64
all: dynLib.so dynamiclinking
dynLib.so: dynLib.o
g++44 $(CLINKFLAGS) -shared -o libdynLib.so dynLib.o
dynLib.o: dynLib.cpp
g++44 $(CXXFLAGS) dynLib.cpp
dynamiclinking: main.o
g++44 $(CLINKFLAGS) -o dynamiclinking main.o -ldl
main.o: main.cpp
g++44 (CXXFLAGS) main.cpp
.PHONY: clean
clean:
rm dynLib.o main.o dynamiclinking libdynLib.so
附言。我手写代码(可能会出现一些拼写错误)PS 2. 使用 -fPIcflags它会起作用:
main.o: main.cpp
g++44 (CXXFLAGS) main.cpp -fPIC
更新可以通过 libstdc++ 的静态链接来解决这个问题。但是我的问题仍然没有得到解答 :( 也许有人有时间看一下?
UPDATE2 GCC 4.4.6 和 4.8.1 出现同样的问题。
最佳答案
我认为您遇到的问题与 When we are supposed to use RTLD_DEEPBIND? 中的问题相同,其中可执行文件获取全局变量的拷贝:
well that's a wonderful feature of you building the main application without the
-fPIC
option. [...] This means that when the symbol is found inlibdep.so
, it gets copied into the initial data segment of the main executable at that address. Then the reference toduplicate
inlibdep.so
is looked up and it points to the copy of the symbol that's in the main executable.
由于 RTLD_DEEPBIND
,dynLib.so 在初始化 std::string
时从原始 libstdc++ 中看到错误的全局变量集,因此崩溃。
至于为什么链接器有这样的行为,this article有详细的解释(强调我的):
Recall that the program/executable is not relocatable, and thus its data addresses have to bound at link time. Therefore, the linker has to create a copy of the variable in the program's address space, and the dynamic loader will use that as the relocation address. This is similar to the discussion in the previous section - in a sense,
myglob
in the main program overrides the one in the shared library, and according to the global symbol lookup rules, it's being used instead.
最后一点:这种行为是特定于平台的,至少在 PowerPC 上,主可执行文件中没有这样的全局变量的额外拷贝。
关于c++ - 使用 RTLD_DEEPBIND 动态加载共享库,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/37296995/
Java 库和 android 库有什么区别,各自有什么优点/缺点? 最佳答案 您可以在 Android 应用程序中包含标准 Java .jar 文件库。它们在 .apk 构建时被翻译成 Dalvik
所以,我现在的代码就像从 Java 层加载库(比如 liba.so),并在内部 liba.so 加载 libb.so。因此,如果我必须将所有库打包到 APK 中并将其安装在没有 root 访问权限的设
我想在我的系统中设置 LEDA 库。 我已经从以下链接下载了 LEDA 库 http://www.algorithmic-solutions.info/free/d5.php Instruct
我想用 autoconf 创建一个共享库。但是,我希望共享库具有“.so”扩展名,而不是以“lib”开头。基本上,我想制作一个加载 dlopen 的插件。 .是否有捷径可寻? 当我尝试使用 autoc
我需要在 Apps 脚本应用程序上修改 PDF。为此,我想使用 JS 库:PDF-LIB 我的代码: eval(UrlFetchApp.fetch("https://unpkg.com/pdf-lib
我正在构建一个使用以下 Boost header 的程序(我使用的是 Microsoft Visual C++ 10), #include #include #include #include
当我通过 cygwin 在 hadoop 上运行此命令时: $bin/hadoop jar hadoop-examples-*.jar grep input output 'dfs[a-z.]+' 我
我已经通过 vcpgk 成功安装了一个 C++ 库,名为:lmdb:x64-windows 我还安装了lmdb通过 Cabal 安装的 Haskell 绑定(bind)包 在尝试测试 lmdb 包时:
我该如何解决这个问题? 我刚刚将 javacv jar 文件复制到我的项目 Lib 文件夹下,但出现了这个错误! 我可以找到这个thread来自谷歌,但不幸的是,由于我国的谷歌限制政策,该页面无法打开
我有一个 Android 库项目 FooLib。 FooLib 引用 Android Context 之类的东西,但不需要任何资源文件(res/ 中的东西)所以我目前将其打包为供我的应用使用的 JAR
我正在开发一个 Android 应用程序(使用 Android Studio),它能够通过手势识别算法了解您正在进行的 Activity 。对于我使用 nickgillian ithub 帐户上可用的
关于从 .NET Framework 项目中引用 .NET Standard 类库的问题有很多类似的问题,其中 netstandard 库中的 NuGet 包依赖项不会流向 netframework
我已经从互联网上下载了 jna-4.2.2.jar,现在想将这个 jar 导入到我的项目中。但是当我试图将这个 jar 导入我的项目时,出现以下错误。 [2016-06-20 09:35:01 - F
我正在尝试通过编译在 Mac 上安装 rsync 3.2.3。但是,我想安装所有功能。为此,它需要一些库,此处 ( https://download.samba.org/pub/rsync/INSTA
进入 Web 开发有点困难。过去 5 年我一直致力于 winforms 工作。所以我正在努力从一种切换到另一种。前段时间,我使用过 JavaScript,但现在还没有大量的 JavaScript 库
很难说出这里要问什么。这个问题模棱两可、含糊不清、不完整、过于宽泛或夸夸其谈,无法以目前的形式得到合理的回答。如需帮助澄清此问题以便重新打开,visit the help center . 关闭 1
我正在寻找一个用Python编写的与logstash(ruby + java)类似的工具/库。 我的目标是: 从 syslog 中解析所有系统日志 解析应用程序特定日志(apache、django、m
就目前情况而言,这个问题不太适合我们的问答形式。我们希望答案得到事实、引用资料或专业知识的支持,但这个问题可能会引发辩论、争论、民意调查或扩展讨论。如果您觉得这个问题可以改进并可能重新开放,visit
我花了几天时间试图寻找用于 JavaPOS 实现的 .jar 库,但我找不到任何可以工作的东西。我找到了很多像这样的文档:http://jpos.1045706.n5.nabble.com/file/
这个问题在这里已经有了答案: Merge multiple .so shared libraries (2 个答案) 关闭 9 年前。 我有我在代码中使用的第三方库的源代码和对象。该库附带有关如何使
我是一名优秀的程序员,十分优秀!