我在链接过程中遇到了一个奇怪的问题。
我有一个包含以下定义foo.hpp
的头文件:
struct Foo { static __thread int x; }
引用该变量plugin.cpp
的源文件:
#include "foo.hpp"
void bar() { int y = Foo::x; }
编译好:
$CXX -stdlib=libc++ -std=c++11 -fvisibility=hidden -fPIC -o plugin.cpp.o -c plugin.cpp
但是当我尝试链接为动态库时:
$CXX -stdlib=libc++ -std=c++11 -fvisibility=hidden -dynamiclib -Wl,-undefined,dynamic_lookup -o libext.dylib ext.cpp.o
我明白了:
ld:非法线程局部变量引用常规符号 __ZN3Foo1xE for architecture x86_64
但llvm字节码意味着编译器正确地将Foo::x
视为TLS变量。
$CXX -stdlib=libc++ -std=c++11 -fvisibility=hidden -fPIC -S -emit-llvm -o -
... omitted
@_ZN3Foo1xE = external thread_local global i32
... omitted
; Function Attrs: nounwind ssp uwtable
define hidden void @_Z3barv() #0 {
%y = alloca i32, align 4
%1 = load i32* @_ZN3Foo1xE, align 4
store i32 %1, i32* %y, align 4
ret void
}
可能导致此链接器问题的原因是什么?是否有解决方法?我似乎无法找到任何与此相关的错误报告。
注意:
修改 引用全局(而不是静态类)变量时存在同样的问题。
当我使用thread_local
代替__thread
时,这样可以正常工作,但thread_local
不适用于Xcode附带的LLVM版本。
答案 0 :(得分:1)
苹果可执行文件格式(我相信MACH-O)不允许Thread本地存储。这是一个痛苦的屁股。您必须在线程库的内存分配中创建空间并在其中隐藏线程局部变量。它非常模糊。