Commit a8b61d16 authored by Tom Stellard's avatar Tom Stellard
Browse files

Merging r344368:

------------------------------------------------------------------------
r344368 | ruiu | 2018-10-12 10:07:32 -0700 (Fri, 12 Oct 2018) | 10 lines

[lld] Add more complete support for the INCLUDE command.

Patch by Ian Tessier.

This change adds INCLUDE support to the MEMORY and SECTION commands, and
to output sections, as per:

https://sourceware.org/binutils/docs/ld/File-Commands.html#File-Commands

Differential Revision: https://reviews.llvm.org/D52951
------------------------------------------------------------------------

llvm-svn: 344926
parent b8dce5ad
Loading
Loading
Loading
Loading
+13 −5
Original line number Diff line number Diff line
@@ -497,6 +497,9 @@ void ScriptParser::readSections() {
      for (BaseCommand *Cmd : readOverlay())
        V.push_back(Cmd);
      continue;
    } else if (Tok == "INCLUDE") {
      readInclude();
      continue;
    }

    if (BaseCommand *Cmd = readAssignment(Tok))
@@ -778,6 +781,8 @@ OutputSection *ScriptParser::readOutputSectionDescription(StringRef OutSec) {
      Cmd->Filler = readFill();
    } else if (Tok == "SORT") {
      readSort();
    } else if (Tok == "INCLUDE") {
      readInclude();
    } else if (peek() == "(") {
      Cmd->SectionCommands.push_back(readInputSectionDescription(Tok));
    } else {
@@ -1404,7 +1409,11 @@ uint64_t ScriptParser::readMemoryAssignment(StringRef S1, StringRef S2,
void ScriptParser::readMemory() {
  expect("{");
  while (!errorCount() && !consume("}")) {
    StringRef Name = next();
    StringRef Tok = next();
    if (Tok == "INCLUDE") {
      readInclude();
      continue;
    }

    uint32_t Flags = 0;
    uint32_t NegFlags = 0;
@@ -1419,10 +1428,9 @@ void ScriptParser::readMemory() {
    uint64_t Length = readMemoryAssignment("LENGTH", "len", "l");

    // Add the memory region to the region map.
    MemoryRegion *MR =
        make<MemoryRegion>(Name, Origin, Length, Flags, NegFlags);
    if (!Script->MemoryRegions.insert({Name, MR}).second)
      setError("region '" + Name + "' already defined");
    MemoryRegion *MR = make<MemoryRegion>(Tok, Origin, Length, Flags, NegFlags);
    if (!Script->MemoryRegions.insert({Tok, MR}).second)
      setError("region '" + Tok + "' already defined");
  }
}

+23 −0
Original line number Diff line number Diff line
# REQUIRES: x86

# RUN: echo '.section .text,"ax"; .global _start; nop' > %t.s
# RUN: echo '.section .data,"aw"; .quad 0' >> %t.s
# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux %t.s -o %t.o

# RUN: echo "RAM2 (rwx): ORIGIN = 0x3000, LENGTH = 0x100" > %t.inc
# RUN: ld.lld -o %t.elf --script %s %t.o -L %T
# RUN: llvm-objdump -section-headers %t.elf | FileCheck %s
# CHECK: .data         00000008 0000000000002000 DATA
# CHECK: .data2        00000008 0000000000003000 DATA

MEMORY {
  ROM (rwx): ORIGIN = 0x1000, LENGTH = 0x100
  RAM (rwx): ORIGIN = 0x2000, LENGTH = 0x100
  INCLUDE "memory-include.test.tmp.inc"
}

SECTIONS {
  .text : { *(.text*) } > ROM
  .data : { *(.data*) } > RAM
  .data2 : { QUAD(0) } > RAM2
}
+30 −0
Original line number Diff line number Diff line
# REQUIRES: x86

# RUN: echo '.section .text,"ax"; .global _start; nop' > %t.s
# RUN: echo '.section .data,"aw"; .quad 0' >> %t.s
# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux %t.s -o %t.o

## Empty include file.
# RUN: echo "" > %t.inc
# RUN: ld.lld -o %t.elf --script %s %t.o -L %T
# RUN: llvm-objdump -section-headers %t.elf | FileCheck %s --check-prefix=CHECK1
# CHECK1: .data         00000008 0000000000002000 DATA

## Non-empty include file.
# RUN: echo "QUAD(0)" > %t.inc
# RUN: ld.lld -o %t.elf --script %s %t.o -L %T
# RUN: llvm-objdump -section-headers %t.elf | FileCheck %s --check-prefix=CHECK2
# CHECK2: .data         00000010 0000000000002000 DATA

MEMORY {
  ROM (rwx): ORIGIN = 0x1000, LENGTH = 0x100
  RAM (rwx): ORIGIN = 0x2000, LENGTH = 0x100
}

SECTIONS {
  .text : { *(.text*) } > ROM
  .data : {
    *(.data*)
    INCLUDE "output-section-include.test.tmp.inc"
  } > RAM
}
+32 −0
Original line number Diff line number Diff line
# REQUIRES: x86

# RUN: echo '.section .text,"ax"; .global _start; nop' > %t.s
# RUN: echo '.section .data,"aw"; .quad 0' >> %t.s
# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux %t.s -o %t.o

## Empty include file.
# RUN: echo "" > %t.inc
# RUN: ld.lld -o %t.elf --script %s %t.o -L %T
# RUN: llvm-objdump -section-headers %t.elf | FileCheck %s --check-prefix=CHECK1
# CHECK1:      .data         00000008 0000000000002000 DATA
# CHECK1-NEXT: .data3        00000008 0000000000002008 DATA

## Non-empty include file.
# RUN: echo ".data2 : { QUAD(0) } > RAM" > %t.inc
# RUN: ld.lld -o %t.elf --script %s %t.o -L %T
# RUN: llvm-objdump -section-headers %t.elf | FileCheck %s --check-prefix=CHECK2
# CHECK2:      .data         00000008 0000000000002000 DATA
# CHECK2-NEXT: .data2        00000008 0000000000002008 DATA
# CHECK2-NEXT: .data3        00000008 0000000000002010 DATA

MEMORY {
  ROM (rwx): ORIGIN = 0x1000, LENGTH = 0x100
  RAM (rwx): ORIGIN = 0x2000, LENGTH = 0x100
}

SECTIONS {
  .text : { *(.text*) } > ROM
  .data : { *(.data*) } > RAM
  INCLUDE "section-include.test.tmp.inc"
  .data3 : { QUAD(0) } > RAM
}