我在链接期间发生了一个奇怪的问题.
我有一个包含以下定义的头文件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,用于体系结构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 }
可能导致此链接器问题的原因是什么,是否有解决方法?我似乎无法找到任何与此相关的错误报告.
笔记:
这纯粹是使用Apple LLVM 7.0.0
在OS X上使用gcc 5或gcc 4.9.3链接没有问题
编辑 引用全局(而不是静态类)变量时存在同样的问题.
当我使用thread_local
代替时__thread
,这可以正常工作,但thread_local
不适用于随Xcode一起提供的LLVM版本.