diff options
author | Fangrui Song <maskray@google.com> | 2018-06-11 19:42:57 +0000 |
---|---|---|
committer | Fangrui Song <maskray@google.com> | 2018-06-11 19:42:57 +0000 |
commit | 2389a240d02bd8440995e2ad630958e1971ebec7 (patch) | |
tree | c21b97209c15ebe4a72dd65802ea8ebca5287d3a | |
parent | d1df7b7ee751206f6e9cee31ac003801376c938a (diff) | |
download | bcm5719-llvm-2389a240d02bd8440995e2ad630958e1971ebec7.tar.gz bcm5719-llvm-2389a240d02bd8440995e2ad630958e1971ebec7.zip |
[ELF] Fix copy relocation when two symbols share the same Symbol instance.
In glibc libc.so.6, the multiple versions of sys_errlist share the same Symbol instance. When sys_errlist is copy relocated, we would replace SharedSymbol with Defined in the first iteration of the following loop:
for (SharedSymbol *Sym : getSymbolsAt<ELFT>(SS))
Then in the second iteration, we think the symbol (which has been changed to Defined) is still SharedSymbol and screw up (the address ends up in the `Size` field).
llvm-svn: 334432
-rw-r--r-- | lld/ELF/Relocations.cpp | 9 | ||||
-rw-r--r-- | lld/test/ELF/Inputs/copy-rel-version.s | 11 | ||||
-rw-r--r-- | lld/test/ELF/copy-rel-version.s | 15 |
3 files changed, 31 insertions, 4 deletions
diff --git a/lld/ELF/Relocations.cpp b/lld/ELF/Relocations.cpp index 64b23ab87de..936371303b6 100644 --- a/lld/ELF/Relocations.cpp +++ b/lld/ELF/Relocations.cpp @@ -52,6 +52,7 @@ #include "Thunks.h" #include "lld/Common/Memory.h" #include "lld/Common/Strings.h" +#include "llvm/ADT/SmallSet.h" #include "llvm/Support/Endian.h" #include "llvm/Support/raw_ostream.h" #include <algorithm> @@ -432,14 +433,14 @@ template <class ELFT> static bool isReadOnly(SharedSymbol &SS) { // // If two or more symbols are at the same offset, and at least one of // them are copied by a copy relocation, all of them need to be copied. -// Otherwise, they would refer different places at runtime. +// Otherwise, they would refer to different places at runtime. template <class ELFT> -static std::vector<SharedSymbol *> getSymbolsAt(SharedSymbol &SS) { +static SmallSet<SharedSymbol *, 4> getSymbolsAt(SharedSymbol &SS) { typedef typename ELFT::Sym Elf_Sym; SharedFile<ELFT> &File = SS.getFile<ELFT>(); - std::vector<SharedSymbol *> Ret; + SmallSet<SharedSymbol *, 4> Ret; for (const Elf_Sym &S : File.getGlobalELFSyms()) { if (S.st_shndx == SHN_UNDEF || S.st_shndx == SHN_ABS || S.st_value != SS.Value) @@ -447,7 +448,7 @@ static std::vector<SharedSymbol *> getSymbolsAt(SharedSymbol &SS) { StringRef Name = check(S.getName(File.getStringTable())); Symbol *Sym = Symtab->find(Name); if (auto *Alias = dyn_cast_or_null<SharedSymbol>(Sym)) - Ret.push_back(Alias); + Ret.insert(Alias); } return Ret; } diff --git a/lld/test/ELF/Inputs/copy-rel-version.s b/lld/test/ELF/Inputs/copy-rel-version.s new file mode 100644 index 00000000000..36bb1ba54c9 --- /dev/null +++ b/lld/test/ELF/Inputs/copy-rel-version.s @@ -0,0 +1,11 @@ +.data +.global foo@v1 +.type foo@v1, @object +.size foo@v1, 4 +.global foo@@v2 +.type foo@@v2, @object +.size foo@@v2, 8 +foo@v1: +foo@@v2: +.int 0 +.int 0 diff --git a/lld/test/ELF/copy-rel-version.s b/lld/test/ELF/copy-rel-version.s new file mode 100644 index 00000000000..29fae8ecbb0 --- /dev/null +++ b/lld/test/ELF/copy-rel-version.s @@ -0,0 +1,15 @@ +// REQUIRES: x86 +// RUN: llvm-mc -filetype=obj -triple=x86_64-pc-linux %s -o %t.o +// RUN: llvm-mc -filetype=obj -triple=x86_64-pc-linux %p/Inputs/copy-rel-version.s -o %t1.o +// RUN: echo "v1 {}; v2 {};" > %t.ver +// RUN: ld.lld %t1.o -shared -soname t1.so --version-script=%t.ver -o %t1.so +// RUN: ld.lld %t.o %t1.so -o %t +// RUN: llvm-readobj -t %t | FileCheck %s + +.global _start +_start: + leaq foo, %rax + +// CHECK: Name: foo ( +// CHECK-NEXT: Value: +// CHECK-NEXT: Size: 8 |