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

Joe*_*ett 13 c++ linker clang thread-local-storage c++11

我在链接期间发生了一个奇怪的问题.

我有一个包含以下定义的头文件foo.hpp:

struct Foo { static __thread int x; }
Run Code Online (Sandbox Code Playgroud)

以及引用该变量的源文件plugin.cpp:

#include "foo.hpp"
void bar() { int y = Foo::x; }
Run Code Online (Sandbox Code Playgroud)

编译很好:

$CXX -stdlib=libc++ -std=c++11 -fvisibility=hidden -fPIC -o plugin.cpp.o -c plugin.cpp
Run Code Online (Sandbox Code Playgroud)

但是当我尝试链接为动态库时:

$CXX -stdlib=libc++ -std=c++11 -fvisibility=hidden -dynamiclib -Wl,-undefined,dynamic_lookup -o libext.dylib ext.cpp.o
Run Code Online (Sandbox Code Playgroud)

我明白了:

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
}
Run Code Online (Sandbox Code Playgroud)

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

笔记:

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

编辑 引用全局(而不是静态类)变量时存在同样的问题.

当我使用thread_local代替时__thread,这可以正常工作,但thread_local不适用于随Xcode一起提供的LLVM版本.

Car*_*Now 1

苹果的可执行文件格式(我相信是MACH-O)不允许线程本地存储。真是让人头疼。您必须在线程库的内存分配中创建空间并在其中隐藏线程局部变量。它非常非常晦涩难懂。