TLS变量上的“非常线程局部引用常规符号”错误

时间:2015-12-10 21:38:22

标签: c++ c++11 linker clang thread-local-storage

我在链接过程中遇到了一个奇怪的问题。

我有一个包含以下定义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
}

可能导致此链接器问题的原因是什么?是否有解决方法?我似乎无法找到任何与此相关的错误报告。

注意:

  • 这纯粹是使用Apple LLVM 7.0.0
  • 我在OS X
  • 上使用gcc 5或gcc 4.9.3链接时没有问题

修改 引用全局(而不是静态类)变量时存在同样的问题。

当我使用thread_local代替__thread时,这样可以正常工作,但thread_local不适用于Xcode附带的LLVM版本。

1 个答案:

答案 0 :(得分:1)

苹果可执行文件格式(我相信MACH-O)不允许Thread本地存储。这是一个痛苦的屁股。您必须在线程库的内存分配中创建空间并在其中隐藏线程局部变量。它非常模糊。