Index: ELF/InputSection.cpp =================================================================== --- ELF/InputSection.cpp +++ ELF/InputSection.cpp @@ -584,24 +584,27 @@ // A TLS symbol's virtual address is relative to the TLS segment. Add a // target-specific adjustment to produce a thread-pointer-relative offset. -static int64_t getTlsTpOffset() { +static int64_t getTlsTpOffset(const Symbol &S) { switch (Config->EMachine) { case EM_ARM: case EM_AARCH64: // Variant 1. The thread pointer points to a TCB with a fixed 2-word size, // followed by a variable amount of alignment padding, followed by the TLS // segment. - return alignTo(Config->Wordsize * 2, Out::TlsPhdr->p_align); + return S.getVA(0) + alignTo(Config->Wordsize * 2, Out::TlsPhdr->p_align); case EM_386: case EM_X86_64: // Variant 2. The TLS segment is located just before the thread pointer. - return -alignTo(Out::TlsPhdr->p_memsz, Out::TlsPhdr->p_align); + // On targets that support TLSDESC, _TLS_MODULE_BASE_@tpoff = 0. + if (&S == Symtab->TlsBase) + return 0; + return S.getVA(0) - alignTo(Out::TlsPhdr->p_memsz, Out::TlsPhdr->p_align); case EM_PPC64: // The thread pointer points to a fixed offset from the start of the // executable's TLS segment. An offset of 0x7000 allows a signed 16-bit // offset to reach 0x1000 of TCB/thread-library data and 0xf000 of the // program's TLS segment. - return -0x7000; + return S.getVA(0) - 0x7000; default: llvm_unreachable("unhandled Config->EMachine"); } @@ -745,12 +748,12 @@ // loaders. if (Sym.isUndefined()) return A; - return Sym.getVA(A) + getTlsTpOffset(); + return getTlsTpOffset(Sym) + A; case R_RELAX_TLS_GD_TO_LE_NEG: case R_NEG_TLS: if (Sym.isUndefined()) return A; - return -Sym.getVA(0) - getTlsTpOffset() + A; + return -getTlsTpOffset(Sym) + A; case R_SIZE: return Sym.getSize() + A; case R_TLSDESC: Index: ELF/SymbolTable.h =================================================================== --- ELF/SymbolTable.h +++ ELF/SymbolTable.h @@ -59,6 +59,9 @@ // is used to uniquify them. llvm::DenseMap ComdatGroups; + // The special TLS symbol _TLS_MODULE_BASE_. + Symbol *TlsBase = nullptr; + private: std::vector findByVersion(SymbolVersion Ver); std::vector findAllByVersion(SymbolVersion Ver); Index: ELF/Writer.cpp =================================================================== --- ELF/Writer.cpp +++ ELF/Writer.cpp @@ -1606,6 +1606,17 @@ if (!dyn_cast_or_null(Symtab->find("__global_pointer$"))) addOptionalRegular("__global_pointer$", findSection(".sdata"), 0x800); + // _TLS_MODULE_BASE_ is defined in such a way that _TLS_MODULE_BASE_@tpoff = + // 0 (lowest address in the TLS block). To achieve this, we define it as an + // absolute symbol of zero. This is different from GNU linkers which define + // _TLS_MODULE_BASE_ relative to the first TLS section. + Symtab->TlsBase = Symtab->find("_TLS_MODULE_BASE_"); + if (Symtab->TlsBase && Symtab->TlsBase->isUndefined()) + Symtab->TlsBase->resolve(Defined{/*File=*/nullptr, + Symtab->TlsBase->getName(), STB_GLOBAL, + STV_HIDDEN, STT_TLS, /*Value=*/0, 0, + /*Section=*/nullptr}); + // This responsible for splitting up .eh_frame section into // pieces. The relocation scan uses those pieces, so this has to be // earlier. Index: test/ELF/x86-64-tlsdesc-ld.s =================================================================== --- /dev/null +++ test/ELF/x86-64-tlsdesc-ld.s @@ -0,0 +1,40 @@ +# REQUIRES: x86 +# RUN: llvm-mc -filetype=obj -triple=x86_64 %s -o %t.o + +# RUN: ld.lld -shared %t.o -o %t.so +# RUN: llvm-readobj -r %t.so | FileCheck --check-prefix=LD-REL %s +# RUN: llvm-objdump -d --no-show-raw-insn %t.so | FileCheck --check-prefix=LD %s + +# RUN: ld.lld %t.o -o %t +# RUN: llvm-readelf -r %t | FileCheck --check-prefix=NOREL %s +# RUN: llvm-objdump -d --no-show-raw-insn %t | FileCheck --check-prefix=LE %s + +# LD-REL: .rela.dyn { +# LD-REL-NEXT: 0x20A0 R_X86_64_TLSDESC - 0x0 +# LD-REL-NEXT: } + +# 0x20a0-0x1007 = 4249 +# LD: leaq 4249(%rip), %rax +# LD-NEXT: 1007: callq *(%rax) +# LD-NEXT: movl %fs:8(%rax), %edx +# LD-NEXT: addl %fs:12(%rax), %edx + +# NOREL: no relocations + +# tpoff(a) = -8, tpoff(b) = -4 +# LE: movq $0, %rax +# LE-NEXT: nop +# LE-NEXT: movl %fs:-8(%rax), %edx +# LE-NEXT: addl %fs:-4(%rax), %edx + +leaq _TLS_MODULE_BASE_@tlsdesc(%rip), %rax +call *_TLS_MODULE_BASE_@tlscall(%rax) +movl %fs:a@dtpoff(%rax), %edx +addl %fs:b@dtpoff(%rax), %edx + +.section .tbss +.zero 8 +a: +.zero 4 +b: +.zero 4