[PATCH] D101116: [ELF] Support .rela.eh_frame with unordered r_offset values

Fangrui Song via Phabricator via llvm-commits llvm-commits at lists.llvm.org
Thu Apr 22 15:29:24 PDT 2021


MaskRay created this revision.
MaskRay added reviewers: grimar, jhenderson, peter.smith.
Herald added subscribers: arichardson, emaste.
MaskRay requested review of this revision.
Herald added a project: LLVM.
Herald added a subscriber: llvm-commits.

In rare cases, .rela.eh_frame can have unordered r_offset values.
This is currently unsupported and will trigger
`assert(pieces[i].inputOff <= off ...` in `OffsetGetter::get`
(the content is corrupted in a -DLLVM_ENABLE_ASSERTIONS=off build).

This patch supports this case. This issue arise in an Android kernel build.


Repository:
  rG LLVM Github Monorepo

https://reviews.llvm.org/D101116

Files:
  lld/ELF/InputSection.cpp
  lld/ELF/Relocations.cpp
  lld/test/ELF/eh-frame-unordered-r_offset.s


Index: lld/test/ELF/eh-frame-unordered-r_offset.s
===================================================================
--- /dev/null
+++ lld/test/ELF/eh-frame-unordered-r_offset.s
@@ -0,0 +1,30 @@
+# REQUIRES: x86
+# RUN: split-file %s %t
+# RUN: llvm-mc -filetype=obj -triple=x86_64 %t/a.s -o %t/a.o
+# RUN: cp %t/a.o %t/b.o
+# RUN: ld.lld -r -T %t/lds %t/a.o %t/b.o -o %t/c.o
+# RUN: llvm-readelf -r %t/c.o | FileCheck %s --check-prefix=REL
+
+## If we swap two input .eh_frame, the r_offset values in relocations will be
+## unordered.
+# REL:          Offset
+# REL-NEXT: 0000000000000050
+# REL-NEXT: 0000000000000020
+
+## Test we can handle the rare case.
+# RUN: ld.lld %t/c.o -o %t/c
+# RUN: llvm-dwarfdump --eh-frame %t/c | FileCheck %s
+
+# CHECK: 00000000 00000014 00000000 CIE
+# CHECK: 00000018 00000014 0000001c FDE cie=00000000
+# CHECK: 00000030 00000014 00000034 FDE cie=00000000
+
+#--- a.s
+.cfi_startproc
+nop
+.cfi_endproc
+
+#--- lds
+SECTIONS {
+  .eh_frame : { *b.o(.eh_frame) *a.o(.eh_frame) }
+}
Index: lld/ELF/Relocations.cpp
===================================================================
--- lld/ELF/Relocations.cpp
+++ lld/ELF/Relocations.cpp
@@ -1576,6 +1576,22 @@
   if (config->emachine == EM_PPC64)
     checkPPC64TLSRelax<RelTy>(sec, rels);
 
+  // For EhInputSection, OffsetGetter expects the relocations to be sorted by
+  // r_offset. In rare cases (.eh_frame pieces are reordered by a linker
+  // script), the relocations may be unordered. Allocate a temporary vector and
+  // sort the relocations.
+  SmallVector<RelTy, 0> sorted;
+  if (isa<EhInputSection>(sec)) {
+    auto cmp = [](const RelTy &a, const RelTy &b) {
+      return a.r_offset < b.r_offset;
+    };
+    if (!llvm::is_sorted(rels, cmp)) {
+      sorted.assign(rels.begin(), rels.end());
+      llvm::stable_sort(sorted, cmp);
+      rels = sorted;
+    }
+  }
+
   for (auto i = rels.begin(), end = rels.end(); i != end;)
     scanReloc<ELFT>(sec, getOffset, i, rels.begin(), end);
 
Index: lld/ELF/InputSection.cpp
===================================================================
--- lld/ELF/InputSection.cpp
+++ lld/ELF/InputSection.cpp
@@ -1326,6 +1326,18 @@
 
 template <class ELFT, class RelTy>
 void EhInputSection::split(ArrayRef<RelTy> rels) {
+  // getReloc expects the relocations to be sorted by r_offset. See the comment
+  // in scanRelocs.
+  SmallVector<RelTy, 0> sorted;
+  auto cmp = [](const RelTy &a, const RelTy &b) {
+    return a.r_offset < b.r_offset;
+  };
+  if (!llvm::is_sorted(rels, cmp)) {
+    sorted.assign(rels.begin(), rels.end());
+    llvm::stable_sort(sorted, cmp);
+    rels = sorted;
+  }
+
   unsigned relI = 0;
   for (size_t off = 0, end = data().size(); off != end;) {
     size_t size = readEhRecordSize(this, off);


-------------- next part --------------
A non-text attachment was scrubbed...
Name: D101116.339800.patch
Type: text/x-patch
Size: 2775 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20210422/29204f6e/attachment.bin>


More information about the llvm-commits mailing list