[lld] [LLD][ELF] Support OVERLAY NOCROSSREFS (PR #133807)

Daniel Thornburgh via llvm-commits llvm-commits at lists.llvm.org
Mon Mar 31 14:52:19 PDT 2025


https://github.com/mysterymath created https://github.com/llvm/llvm-project/pull/133807

This allows NOCROSSREFS to be specified in OVERLAY linker script descriptions. This is a particularly useful part of the OVERLAY syntax, since it's very rarely possible for one overlay section to sensibly reference another.

Closes #128790

>From b067b9027db2f5a1f364fda1e60888d312775ce9 Mon Sep 17 00:00:00 2001
From: Daniel Thornburgh <mysterymath at gmail.com>
Date: Mon, 31 Mar 2025 10:47:43 -0700
Subject: [PATCH] [LLD][ELF] Support OVERLAY NOCROSSREFS

This allows NOCROSSREFS to be specified in OVERLAY linker script
descriptions. This is a particularly useful part of the OVERLAY syntax,
since it's very rarely possible for one overlay section to sensibly
reference another.

Closes #128790
---
 lld/ELF/ScriptParser.cpp                   | 19 +++++++++++++++++--
 lld/docs/ReleaseNotes.rst                  |  3 +++
 lld/test/ELF/linkerscript/nocrossrefs.test | 15 +++++++++++++++
 lld/test/ELF/linkerscript/overlay.test     | 11 +++++++++++
 4 files changed, 46 insertions(+), 2 deletions(-)

diff --git a/lld/ELF/ScriptParser.cpp b/lld/ELF/ScriptParser.cpp
index 4345b7bac1173..95e1d24042644 100644
--- a/lld/ELF/ScriptParser.cpp
+++ b/lld/ELF/ScriptParser.cpp
@@ -565,8 +565,17 @@ SmallVector<SectionCommand *, 0> ScriptParser::readOverlay() {
     addrExpr = readExpr();
     expect(":");
   }
-  Expr lmaExpr = consume("AT") ? readParenExpr() : Expr{};
-  expect("{");
+
+  Expr lmaExpr;
+  bool noCrossRefs = false;
+  while (auto tok = till("{")) {
+    if (tok == "AT")
+      lmaExpr = readParenExpr();
+    else if (tok == "NOCROSSREFS")
+      noCrossRefs = true;
+    else
+      setError("{ expected, but got " + tok);
+  }
 
   SmallVector<SectionCommand *, 0> v;
   OutputSection *prev = nullptr;
@@ -595,6 +604,12 @@ SmallVector<SectionCommand *, 0> ScriptParser::readOverlay() {
       static_cast<OutputDesc *>(od)->osec.memoryRegionName =
           std::string(regionName);
   }
+  if (noCrossRefs) {
+    NoCrossRefCommand cmd;
+    for (SectionCommand *od : v)
+      cmd.outputSections.push_back(static_cast<OutputDesc *>(od)->osec.name);
+    ctx.script->noCrossRefs.push_back(std::move(cmd));
+  }
 
   // According to the specification, at the end of the overlay, the location
   // counter should be equal to the overlay base address plus size of the
diff --git a/lld/docs/ReleaseNotes.rst b/lld/docs/ReleaseNotes.rst
index 43ed6fa73589b..2b7b7fe52ea12 100644
--- a/lld/docs/ReleaseNotes.rst
+++ b/lld/docs/ReleaseNotes.rst
@@ -39,6 +39,9 @@ ELF Improvements
   items that kept it live during garbage collection. This is inspired by the
   Mach-O LLD feature of the same name.
 
+* Linker script ``OVERLAY`` descriptions now support virtual memory regions
+  (e.g. ``>region``) and ``NOCROSSREFS``.
+
 Breaking changes
 ----------------
 
diff --git a/lld/test/ELF/linkerscript/nocrossrefs.test b/lld/test/ELF/linkerscript/nocrossrefs.test
index 5eb56190fe63b..de0f1d0300b00 100644
--- a/lld/test/ELF/linkerscript/nocrossrefs.test
+++ b/lld/test/ELF/linkerscript/nocrossrefs.test
@@ -28,6 +28,10 @@
 # RUN: not ld.lld a.o data.o -T 3.t 2>&1 | FileCheck %s --check-prefix=CHECK3 --implicit-check-not=error:
 # CHECK3:      error: a.o:(.nonalloc+0x0): prohibited cross reference from '.nonalloc' to '.text' in '.text'
 
+# RUN: not ld.lld a.o data.o -T overlay.t 2>&1 | FileCheck %s --check-prefix=OVERLAY --implicit-check-not=error:
+# OVERLAY: error: a.o:(.text.start+0x11): prohibited cross reference from '.text' to 'data' in '.data'
+# OVERLAY-NEXT: error: a.o:(.text.start+0x17): prohibited cross reference from '.text' to 'str1' in '.rodata'
+
 #--- 0.t
 ## Some cases that do not cause errors.
 abs = 42;
@@ -50,6 +54,17 @@ NOCROSSREFS(.text .text1 .text2 .data .rodata .data .nonalloc)
 abs = 42;
 NOCROSSREFS_TO(.text .text .text1 .text2 .data .nonalloc)
 
+#--- overlay.t
+abs = 42;
+_edata = 43;
+SECTIONS {
+  OVERLAY : NOCROSSREFS {
+    .text { *(.text*) }
+    .data { *(.data*) }
+    .rodata { *(.rodata*) }
+  }
+}
+
 #--- err1.t
 NOCROSSREFS )
 
diff --git a/lld/test/ELF/linkerscript/overlay.test b/lld/test/ELF/linkerscript/overlay.test
index e230134ad5541..7bc98b5206a04 100644
--- a/lld/test/ELF/linkerscript/overlay.test
+++ b/lld/test/ELF/linkerscript/overlay.test
@@ -133,3 +133,14 @@ SECTIONS {
 # RUN: not ld.lld a.o -T unclosed.lds 2>&1 | FileCheck %s --check-prefix=UNCLOSED
 # UNCLOSED:     error: unclosed.lds:2: unexpected EOF
 # UNCLOSED-NOT: {{.}}
+
+#--- invalid-before-brace.lds
+SECTIONS {
+  OVERLAY 0x1000 : INVALID {
+  }
+}
+
+# RUN: not ld.lld a.o -T invalid-before-brace.lds 2>&1 | FileCheck %s --check-prefix=INVALID-BEFORE-BRACE --strict-whitespace
+# INVALID-BEFORE-BRACE:{{.*}}error: invalid-before-brace.lds:2: { expected, but got INVALID
+# INVALID-BEFORE-BRACE-NEXT:>>>   OVERLAY 0x1000 : INVALID {
+# INVALID-BEFORE-BRACE-NEXT:>>>                    ^



More information about the llvm-commits mailing list