[PATCH] D81986: [ELF] Refine LMA offset propagation rule in D76395
Fangrui Song via Phabricator via llvm-commits
llvm-commits at lists.llvm.org
Tue Jun 16 19:42:37 PDT 2020
MaskRay created this revision.
MaskRay added reviewers: grimar, psmith, ruiu.
Herald added subscribers: llvm-commits, arichardson, emaste.
Herald added a reviewer: espindola.
Herald added a project: LLVM.
If neither AT(lma) nor AT>lma_region is specified,
D76395 <https://reviews.llvm.org/D76395> keeps `lmaOffset` (LMA - VMA) if the previous section is in the
default LMA region.
This patch additionally checks that the two sections are in the same
memory region.
Add a test case derived from https://bugs.llvm.org/show_bug.cgi?id=45313
.mdata : AT(0xfb01000) { *(.data); } > TCM
// It is odd to make .bss inherit lmaOffset, because the two sections
// are in different memory regions.
.bss : { *(.bss) } > DDR
With this patch, section VMA/LMA match GNU ld. Note, GNU ld supports
out-of-order (w.r.t sh_offset) sections and places .text and .bss in the
same PT_LOAD. We don't have that behavior.
Repository:
rG LLVM Github Monorepo
https://reviews.llvm.org/D81986
Files:
lld/ELF/LinkerScript.cpp
lld/docs/ELF/linker_script.rst
lld/test/ELF/linkerscript/lma-offset2.s
Index: lld/test/ELF/linkerscript/lma-offset2.s
===================================================================
--- /dev/null
+++ lld/test/ELF/linkerscript/lma-offset2.s
@@ -0,0 +1,24 @@
+# REQUIRES: x86
+# RUN: echo '.globl _start; _start: ret; \
+# RUN: .data; .byte 0; \
+# RUN: .bss; .byte 0' | \
+# RUN: llvm-mc -filetype=obj -triple=x86_64 - -o %t.o
+# RUN: ld.lld -T %s %t.o -o %t
+# RUN: llvm-readelf -l %t | FileCheck %s
+
+## GNU ld places .text and .bss in the same RWX PT_LOAD.
+# CHECK: Type Offset VirtAddr PhysAddr FileSiz MemSiz Flg
+# CHECK-NEXT: LOAD 0x001000 0x000000000fb00000 0x000000000fb00000 0x000001 0x000001 R E
+# CHECK-NEXT: LOAD 0x002000 0x000000001f400000 0x000000000fb01000 0x000001 0x000001 RW
+# CHECK-NEXT: LOAD 0x002001 0x000000000fb00001 0x000000000fb00001 0x000000 0x000001 RW
+
+MEMORY {
+ DDR : o = 0xfb00000, l = 185M
+ TCM : o = 0x1f400000, l = 128K
+}
+
+SECTIONS {
+ .text : { *(.text) } > DDR
+ .mdata : AT(0xfb01000) { *(.data); } > TCM
+ .bss : { *(.bss) } > DDR
+}
Index: lld/docs/ELF/linker_script.rst
===================================================================
--- lld/docs/ELF/linker_script.rst
+++ lld/docs/ELF/linker_script.rst
@@ -71,7 +71,7 @@
If neither ``AT(lma)`` nor ``AT>lma_region`` is specified:
-- If the previous section is also in the default LMA region, the difference
- between the LMA and the VMA is computed to be the same as the previous
- difference.
+- If the previous section is also in the default LMA region, and the two
+ section have the same memory regions, the difference between the LMA and the
+ VMA is computed to be the same as the previous difference.
- Otherwise, the LMA is set to the VMA.
Index: lld/ELF/LinkerScript.cpp
===================================================================
--- lld/ELF/LinkerScript.cpp
+++ lld/ELF/LinkerScript.cpp
@@ -851,6 +851,7 @@
if (!(sec->flags & SHF_ALLOC))
dot = 0;
+ bool sameMemRegion = ctx->memRegion == sec->memRegion;
bool prevLMARegionIsDefault = ctx->lmaRegion == nullptr;
ctx->memRegion = sec->memRegion;
ctx->lmaRegion = sec->lmaRegion;
@@ -872,14 +873,15 @@
// ctx->lmaOffset is LMA minus VMA. If LMA is explicitly specified via AT() or
// AT>, recompute ctx->lmaOffset; otherwise, if both previous/current LMA
- // region is the default, reuse previous lmaOffset; otherwise, reset lmaOffset
- // to 0. This emulates heuristics described in
+ // region is the default, and the two sections are in the same memory region,
+ // reuse previous lmaOffset; otherwise, reset lmaOffset to 0. This emulates
+ // heuristics described in
// https://sourceware.org/binutils/docs/ld/Output-Section-LMA.html
if (sec->lmaExpr)
ctx->lmaOffset = sec->lmaExpr().getValue() - dot;
else if (MemoryRegion *mr = sec->lmaRegion)
ctx->lmaOffset = alignTo(mr->curPos, sec->alignment) - dot;
- else if (!prevLMARegionIsDefault)
+ else if (!sameMemRegion || !prevLMARegionIsDefault)
ctx->lmaOffset = 0;
// Propagate ctx->lmaOffset to the first "non-header" section.
-------------- next part --------------
A non-text attachment was scrubbed...
Name: D81986.271262.patch
Type: text/x-patch
Size: 3104 bytes
Desc: not available
URL: <http://lists.llvm.org/pipermail/llvm-commits/attachments/20200617/bbd2952d/attachment.bin>
More information about the llvm-commits
mailing list