Index: ELF/InputSection.cpp =================================================================== --- ELF/InputSection.cpp +++ ELF/InputSection.cpp @@ -738,17 +738,18 @@ case R_RELAX_TLS_IE_TO_LE: case R_RELAX_TLS_LD_TO_LE: case R_TLS: - // A weak undefined TLS symbol resolves to the base of the TLS - // block, i.e. gets a value of zero. If we pass --gc-sections to - // lld and .tbss is not referenced, it gets reclaimed and we don't - // create a TLS program header. Therefore, we resolve this - // statically to zero. - if (Sym.isTls() && Sym.isUndefWeak()) - return 0; + // A weak undefined TLS reference resolves to 0. With --noinhibit-exec, a + // global undefined reference may also reach here. It is not very clear what + // we should return, just resolve the symbol to zero and return A. This + // matches the formula of the non-TLS R_ABS: Sym.getVA(A). + if (Sym.isUndefined()) + return A; return Sym.getVA(A) + getTlsTpOffset(); case R_RELAX_TLS_GD_TO_LE_NEG: case R_NEG_TLS: - return -(Sym.getVA(A) + getTlsTpOffset()); + if (Sym.isUndefined()) + return A; + return -(Sym.getVA(0) + getTlsTpOffset()) + A; case R_SIZE: return Sym.getSize() + A; case R_TLSDESC: Index: test/ELF/i386-tls-le-undef.s =================================================================== --- /dev/null +++ test/ELF/i386-tls-le-undef.s @@ -0,0 +1,16 @@ +# REQUIRES: x86 +# RUN: llvm-mc -filetype=obj -triple=i386 %s -o %t.o +# RUN: ld.lld --noinhibit-exec %t.o -o %t 2>&1 +# RUN: llvm-objdump -d %t | FileCheck %s + +## Undefined TLS symbols resolve to 0. +## In --noinhibit-exec mode, a non-weak undefined symbol is not an error. + +# CHECK: subl $0, %eax +# CHECK: subl $0, %eax + +.weak weak_undef +movl %gs:0, %eax +subl $weak_undef@tpoff,%eax +movl %gs:0, %eax +subl $undef@tpoff,%eax Index: test/ELF/x86-64-tls-le-undef.s =================================================================== --- /dev/null +++ test/ELF/x86-64-tls-le-undef.s @@ -0,0 +1,15 @@ +# REQUIRES: x86 +# RUN: llvm-mc -filetype=obj -triple=x86_64 %s -o %t.o +# RUN: ld.lld --noinhibit-exec %t.o -o %t 2>&1 +# RUN: llvm-objdump -d %t | FileCheck %s + +## Undefined TLS symbols resolve to 0. +## In --noinhibit-exec mode, a non-weak undefined symbol is not an error. + +# CHECK: leaq 16(%rax), %rdx +# CHECK-NEXT: leaq 32(%rax), %rdx + +.weak weak +movq %fs:0, %rax +leaq weak@tpoff+16(%rax), %rdx +leaq global@tpoff+32(%rax), %rdx