[lld] 7c74ce3 - [ELF] --wrap: don't clear sym->isUsedInRegularObj if real->isUsedInRegularObj; set wrap's initial binding to sym's
Fangrui Song via llvm-commits
llvm-commits at lists.llvm.org
Sat Apr 17 00:29:56 PDT 2021
Author: Fangrui Song
Date: 2021-04-17T00:29:51-07:00
New Revision: 7c74ce3c686938e95a08a05ea1e2a714eac43167
URL: https://github.com/llvm/llvm-project/commit/7c74ce3c686938e95a08a05ea1e2a714eac43167
DIFF: https://github.com/llvm/llvm-project/commit/7c74ce3c686938e95a08a05ea1e2a714eac43167.diff
LOG: [ELF] --wrap: don't clear sym->isUsedInRegularObj if real->isUsedInRegularObj; set wrap's initial binding to sym's
Fix PR49897: if `__real_foo` has the isUsedInRegularObj bit set, we need to
retain `foo` in .symtab, even if `foo` is undefined. The new behavior will match
GNU ld.
Before the patch, we produced an R_X86_64_JUMP_SLOT relocation referencing the
index 0 undefined symbol, which would be erroed by glibc
(see f96ff3c0f8ebd941b3f6b345164c3d858b781484).
While here, fix another bug: if `__wrap_foo` does not exist, its initial binding
should be `foo`'s.
Added:
Modified:
lld/ELF/Driver.cpp
lld/ELF/SymbolTable.cpp
lld/test/ELF/wrap-dynamic-undef.s
Removed:
lld/test/ELF/Inputs/wrap-dynamic-undef.s
################################################################################
diff --git a/lld/ELF/Driver.cpp b/lld/ELF/Driver.cpp
index 2daee212cc005..1b840c166e098 100644
--- a/lld/ELF/Driver.cpp
+++ b/lld/ELF/Driver.cpp
@@ -1888,8 +1888,9 @@ static Symbol *addUndefined(StringRef name) {
Undefined{nullptr, name, STB_GLOBAL, STV_DEFAULT, 0});
}
-static Symbol *addUnusedUndefined(StringRef name) {
- Undefined sym{nullptr, name, STB_GLOBAL, STV_DEFAULT, 0};
+static Symbol *addUnusedUndefined(StringRef name,
+ uint8_t binding = STB_GLOBAL) {
+ Undefined sym{nullptr, name, binding, STV_DEFAULT, 0};
sym.isUsedInRegularObj = false;
return symtab->addSymbol(sym);
}
@@ -1953,7 +1954,8 @@ static std::vector<WrappedSymbol> addWrappedSymbols(opt::InputArgList &args) {
continue;
Symbol *real = addUnusedUndefined(saver.save("__real_" + name));
- Symbol *wrap = addUnusedUndefined(saver.save("__wrap_" + name));
+ Symbol *wrap =
+ addUnusedUndefined(saver.save("__wrap_" + name), sym->binding);
v.push_back({sym, real, wrap});
// We want to tell LTO not to inline symbols to be overwritten
diff --git a/lld/ELF/SymbolTable.cpp b/lld/ELF/SymbolTable.cpp
index 6283d943984af..70aea288c53f6 100644
--- a/lld/ELF/SymbolTable.cpp
+++ b/lld/ELF/SymbolTable.cpp
@@ -42,7 +42,7 @@ void SymbolTable::wrap(Symbol *sym, Symbol *real, Symbol *wrap) {
if (real->exportDynamic)
sym->exportDynamic = true;
- if (sym->isUndefined())
+ if (!real->isUsedInRegularObj && sym->isUndefined())
sym->isUsedInRegularObj = false;
// Now renaming is complete, and no one refers to real. We drop real from
diff --git a/lld/test/ELF/Inputs/wrap-dynamic-undef.s b/lld/test/ELF/Inputs/wrap-dynamic-undef.s
deleted file mode 100644
index ade79556db7b2..0000000000000
--- a/lld/test/ELF/Inputs/wrap-dynamic-undef.s
+++ /dev/null
@@ -1,2 +0,0 @@
-.global foo
-foo:
diff --git a/lld/test/ELF/wrap-dynamic-undef.s b/lld/test/ELF/wrap-dynamic-undef.s
index af2871cfe6eae..ca1698173ef59 100644
--- a/lld/test/ELF/wrap-dynamic-undef.s
+++ b/lld/test/ELF/wrap-dynamic-undef.s
@@ -1,9 +1,10 @@
# REQUIRES: x86
-# RUN: llvm-mc -filetype=obj -triple=x86_64-pc-linux %s -o %t1.o
-# RUN: llvm-mc -filetype=obj -triple=x86_64-pc-linux %p/Inputs/wrap-dynamic-undef.s -o %t2.o
-# RUN: ld.lld %t2.o -o %t2.so -shared
-# RUN: ld.lld %t1.o %t2.so -o %t --wrap foo
-# RUN: llvm-readelf --dyn-syms %t | FileCheck %s
+# RUN: split-file %s %t
+# RUN: llvm-mc -filetype=obj -triple=x86_64 %t/a.s -o %t/a.o
+# RUN: llvm-mc -filetype=obj -triple=x86_64 %t/def.s -o %t/def.o
+# RUN: ld.lld %t/def.o -o %t/def.so -shared --soname=def
+# RUN: ld.lld %t/a.o %t/def.so -o %t1 --wrap foo
+# RUN: llvm-readelf --dyn-syms %t1 | FileCheck %s
# Test that the dynamic relocation uses foo. We used to produce a
# relocation with __real_foo.
@@ -12,6 +13,29 @@
# CHECK: NOTYPE LOCAL DEFAULT UND
# CHECK-NEXT: NOTYPE GLOBAL DEFAULT UND foo
+# RUN: llvm-mc -filetype=obj -triple=x86_64 %t/b.s -o %t/b.o
+# RUN: ld.lld -shared --wrap foo %t/b.o -o %t2.so
+# RUN: llvm-readelf --dyn-syms %t2.so | FileCheck %s --check-prefix=SYM2
+
+# SYM2: Symbol table '.dynsym' contains 4 entries:
+# SYM2: NOTYPE LOCAL DEFAULT UND
+# SYM2-NEXT: NOTYPE WEAK DEFAULT UND foo
+# SYM2-NEXT: NOTYPE WEAK DEFAULT UND __wrap_foo
+# SYM2-NEXT: NOTYPE GLOBAL DEFAULT [[#]] _start
+
+#--- a.s
.global _start
_start:
callq __real_foo at plt
+
+#--- def.s
+.globl foo
+foo:
+
+#--- b.s
+.weak foo
+.weak __real_foo
+.global _start
+_start:
+ call __real_foo at plt
+ call foo at plt
More information about the llvm-commits
mailing list