[PATCH] D114179: [ELF] Ensure output section is not discarded in addStartEndSymbols()

Andrew Ng via Phabricator via llvm-commits llvm-commits at lists.llvm.org
Thu Nov 18 10:56:53 PST 2021


andrewng created this revision.
andrewng added a reviewer: MaskRay.
Herald added subscribers: arichardson, emaste.
andrewng requested review of this revision.

Fixes https://bugs.llvm.org/show_bug.cgi?id=52534.


https://reviews.llvm.org/D114179

Files:
  lld/ELF/LinkerScript.cpp
  lld/ELF/LinkerScript.h
  lld/ELF/Writer.cpp
  lld/test/ELF/linkerscript/preinit-array-empty.test


Index: lld/test/ELF/linkerscript/preinit-array-empty.test
===================================================================
--- /dev/null
+++ lld/test/ELF/linkerscript/preinit-array-empty.test
@@ -0,0 +1,39 @@
+# REQUIRES: x86
+# RUN: split-file %s %t
+# RUN: llvm-mc -filetype=obj -triple=x86_64 %t/t.s -o %t.o
+
+## PR52534: https://bugs.llvm.org/show_bug.cgi?id=52534
+## Check case where .preinit_array is discarded.
+## Link should succeed without causing an out of range relocation error.
+# RUN: ld.lld --script %t/discarded.script %t.o -o %t1 --image-base=0x80000000
+# RUN: llvm-readelf -s %t1 | FileCheck --check-prefixes=CHECK,DISCARDED %s
+
+## Check case where .preinit_array is emitted but empty.
+# RUN: ld.lld --script %t/empty.script %t.o -o %t2
+# RUN: llvm-readelf -s %t2 | FileCheck --check-prefixes=CHECK,EMPTY %s
+
+# CHECK:          [[#%x,ADDR:]] 0 NOTYPE  LOCAL  HIDDEN      [[#]] __preinit_array_start
+# CHECK-NEXT:     [[#ADDR]]     0 NOTYPE  LOCAL  HIDDEN      [[#]] __preinit_array_end
+
+# DISCARDED-NEXT: [[#ADDR]]     0 NOTYPE  GLOBAL DEFAULT     [[#]] _start
+
+# EMPTY-NOT:      [[#ADDR]]     0 NOTYPE  GLOBAL DEFAULT     [[#]] _start
+# EMPTY:          [[#ADDR]]     0 NOTYPE  GLOBAL DEFAULT     [[#]] ADDR
+
+#--- t.s
+.global _start
+_start:
+ movq __preinit_array_start at GOTPCREL(%rip),%rax
+ movq __preinit_array_end at GOTPCREL(%rip),%rax
+
+#--- discarded.script
+SECTIONS {
+  .text : { *(.text); }
+  .preinit_array : { *(.preinit_array); }
+}
+
+#--- empty.script
+SECTIONS {
+  .text : { *(.text); }
+  .preinit_array : { ADDR = .; *(.preinit_array); }
+}
Index: lld/ELF/Writer.cpp
===================================================================
--- lld/ELF/Writer.cpp
+++ lld/ELF/Writer.cpp
@@ -2276,7 +2276,7 @@
     Default = Out::elfHeader;
 
   auto define = [=](StringRef start, StringRef end, OutputSection *os) {
-    if (os) {
+    if (os && !script->isDiscarded(os)) {
       addOptionalRegular(start, os, 0);
       addOptionalRegular(end, os, -1);
     } else {
Index: lld/ELF/LinkerScript.h
===================================================================
--- lld/ELF/LinkerScript.h
+++ lld/ELF/LinkerScript.h
@@ -318,6 +318,8 @@
   void processSymbolAssignments();
   void declareSymbols();
 
+  bool isDiscarded(const OutputSection *) const;
+
   // Used to handle INSERT AFTER statements.
   void processInsertCommands();
 
Index: lld/ELF/LinkerScript.cpp
===================================================================
--- lld/ELF/LinkerScript.cpp
+++ lld/ELF/LinkerScript.cpp
@@ -1030,7 +1030,7 @@
   }
 }
 
-static bool isDiscardable(OutputSection &sec) {
+static bool isDiscardable(const OutputSection &sec) {
   if (sec.name == "/DISCARD/")
     return true;
 
@@ -1059,6 +1059,11 @@
   return true;
 }
 
+bool LinkerScript::isDiscarded(const OutputSection *sec) const {
+  return hasSectionsCommand && (getFirstInputSection(sec) == nullptr) &&
+         isDiscardable(*sec);
+}
+
 static void maybePropagatePhdrs(OutputSection &sec,
                                 std::vector<StringRef> &phdrs) {
   if (sec.phdrs.empty()) {


-------------- next part --------------
A non-text attachment was scrubbed...
Name: D114179.388266.patch
Type: text/x-patch
Size: 3103 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20211118/cab1f123/attachment.bin>


More information about the llvm-commits mailing list