Index: ELF/OutputSections.h =================================================================== --- ELF/OutputSections.h +++ ELF/OutputSections.h @@ -102,6 +102,7 @@ bool NonAlloc = false; bool Noload = false; bool ExpressionsUseSymbols = false; + bool InOverlay = false; template void finalize(); template void writeTo(uint8_t *Buf); Index: ELF/ScriptParser.cpp =================================================================== --- ELF/ScriptParser.cpp +++ ELF/ScriptParser.cpp @@ -79,7 +79,8 @@ uint32_t readFill(); uint32_t parseFill(StringRef Tok); void readSectionAddressType(OutputSection *Cmd); - OutputSection *readOutputSectionDescription(StringRef OutSec); + OutputSection *readOutputSectionDescription(StringRef OutSec, bool InOverlay); + std::vector readOverlay(); std::vector readOutputSectionPhdrs(); InputSectionDescription *readInputSectionDescription(StringRef Tok); StringMatcher readFilePatterns(); @@ -428,6 +429,80 @@ expect(")"); } +static std::vector +createStartStopSymbols(ArrayRef V, std::string Loc) { + // This crafts start/stop symbols. Names of such symbols are produced + // from the output section names.We exclude all non-valid C identifiers + // and combine the result with the Prefix. + auto MakeSymbol = [=](StringRef Prefix, StringRef Name, Expr E) { + std::string S; + for (char C : Name) + if (C == '_' || isAlnum(C)) + S += C; + return make(Saver.save(Prefix + S), E, Loc, + "" /*CommandString*/); + }; + + // For each section, we define the start and stop symbols which are defined + // as the start and final load address respectively. + std::vector Ret; + for (const BaseCommand *Cmd : V) { + const OutputSection *OS = cast(Cmd); + Ret.push_back( + MakeSymbol("__load_start_", OS->Name, [=] { return OS->getLMA(); })); + Ret.push_back(MakeSymbol("__load_stop_", OS->Name, + [=] { return OS->getLMA() + OS->Size; })); + } + return Ret; +} + +std::vector ScriptParser::readOverlay() { + // VA and LMA expressions are optional, though for simplicity of + // implementation we assume they are not. That is what OVERLAY was designed + // for first of all: to allow sections with overlapping VAs at different LMAs. + Expr AddrExpr = readExpr(); + expect(":"); + expect("AT"); + Expr LMAExpr = readParenExpr(); + expect("{"); + + std::vector V; + while (!errorCount() && !consume("}")) { + OutputSection *Prev = V.empty() ? nullptr : cast(V.back()); + OutputSection *OS = + readOutputSectionDescription(next(), /*InOverlay=*/true); + + // VA is the same for all sections. The LMAs are consecutive in memory + // starting from the base load address specified. + OS->AddrExpr = AddrExpr; + if (Prev) + OS->LMAExpr = [=] { return Prev->getLMA() + Prev->Size; }; + else + OS->LMAExpr = LMAExpr; + + OS->InOverlay = true; + V.push_back(OS); + } + + for (BaseCommand *Sym : createStartStopSymbols(V, getCurrentLocation())) + V.push_back(Sym); + + // 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 + // largest section seen in the overlay. + // Here we want to create the Dot assignment command to achieve that. + Expr MoveDotExpr = [=] { + uint64_t MaxSize = 0; + for (BaseCommand *Cmd : V) + if (auto *Sec = dyn_cast(Cmd)) + MaxSize = std::max(MaxSize, Sec->Size); + return AddrExpr().getValue() + MaxSize; + }; + V.push_back(make(".", MoveDotExpr, getCurrentLocation(), + "" /*CommandString*/)); + return V; +} + void ScriptParser::readSections() { Script->HasSectionsCommand = true; @@ -440,12 +515,18 @@ std::vector V; while (!errorCount() && !consume("}")) { StringRef Tok = next(); + if (Tok == "OVERLAY") { + for (BaseCommand *Cmd : readOverlay()) + V.push_back(Cmd); + continue; + } + BaseCommand *Cmd = readProvideOrAssignment(Tok); if (!Cmd) { if (Tok == "ASSERT") Cmd = readAssert(); else - Cmd = readOutputSectionDescription(Tok); + Cmd = readOutputSectionDescription(Tok, /*InOverlay=*/false); } V.push_back(Cmd); } @@ -673,29 +754,32 @@ }; } -OutputSection *ScriptParser::readOutputSectionDescription(StringRef OutSec) { +OutputSection *ScriptParser::readOutputSectionDescription(StringRef OutSec, + bool InOverlay) { OutputSection *Cmd = Script->createOutputSection(OutSec, getCurrentLocation()); size_t SymbolsReferenced = Script->ReferencedSymbols.size(); - if (peek() != ":") - readSectionAddressType(Cmd); - expect(":"); + if (!InOverlay) { + if (peek() != ":") + readSectionAddressType(Cmd); + expect(":"); - std::string Location = getCurrentLocation(); - if (consume("AT")) - Cmd->LMAExpr = readParenExpr(); - if (consume("ALIGN")) - Cmd->AlignExpr = checkAlignment(readParenExpr(), Location); - if (consume("SUBALIGN")) - Cmd->SubalignExpr = checkAlignment(readParenExpr(), Location); - - // Parse constraints. - if (consume("ONLY_IF_RO")) - Cmd->Constraint = ConstraintKind::ReadOnly; - if (consume("ONLY_IF_RW")) - Cmd->Constraint = ConstraintKind::ReadWrite; + std::string Location = getCurrentLocation(); + if (consume("AT")) + Cmd->LMAExpr = readParenExpr(); + if (consume("ALIGN")) + Cmd->AlignExpr = checkAlignment(readParenExpr(), Location); + if (consume("SUBALIGN")) + Cmd->SubalignExpr = checkAlignment(readParenExpr(), Location); + + // Parse constraints. + if (consume("ONLY_IF_RO")) + Cmd->Constraint = ConstraintKind::ReadOnly; + if (consume("ONLY_IF_RW")) + Cmd->Constraint = ConstraintKind::ReadWrite; + } expect("{"); while (!errorCount() && !consume("}")) { @@ -724,17 +808,19 @@ } } - if (consume(">")) - Cmd->MemoryRegionName = next(); + if (!InOverlay) { + if (consume(">")) + Cmd->MemoryRegionName = next(); + + if (consume("AT")) { + expect(">"); + Cmd->LMARegionName = next(); + } - if (consume("AT")) { - expect(">"); - Cmd->LMARegionName = next(); + if (Cmd->LMAExpr && !Cmd->LMARegionName.empty()) + error("section can't have both LMA and a load region"); } - if (Cmd->LMAExpr && !Cmd->LMARegionName.empty()) - error("section can't have both LMA and a load region"); - Cmd->Phdrs = readOutputSectionPhdrs(); if (consume("=")) Index: ELF/Writer.cpp =================================================================== --- ELF/Writer.cpp +++ ELF/Writer.cpp @@ -1966,13 +1966,15 @@ } } +enum class OverlapKind : uint8_t { File, VirtualAddress, LoadAddress }; + // Check whether sections overlap for a specific address range (file offsets, // load and virtual adresses). // // This is a helper function called by Writer::checkNoOverlappingSections(). template static void checkForSectionOverlap(ArrayRef AllSections, - StringRef Kind, Getter GetStart, + OverlapKind Kind, Getter GetStart, Predicate ShouldSkip) { std::vector Sections; // By removing all zero-size sections we can simplify the check for overlap to @@ -1993,6 +1995,7 @@ [=](const OutputSection *A, const OutputSection *B) { return GetStart(A) < GetStart(B); }); + StringRef KindS[] = {"file", "virtual address", "load address"}; for (size_t I = 0; I < Sections.size(); ++I) { OutputSection *Sec = Sections[I]; uint64_t Start = GetStart(Sec); @@ -2003,7 +2006,13 @@ // will also start after the end of Sec. if (Start + Sec->Size <= GetStart(Other)) break; - errorOrWarn("section " + Sec->Name + " " + Kind + + + // If both sections are in OVERLAY we allow the overlapping of virtual + // addresses, because it is what OVERLAY was designed for. + if (Kind == OverlapKind::VirtualAddress && Sec->InOverlay) + continue; + + errorOrWarn("section " + Sec->Name + " " + KindS[(uint8_t)Kind] + " range overlaps with " + Other->Name + "\n>>> " + Sec->Name + " range is " + rangeToString(Start, Sec->Size) + "\n>>> " + Other->Name + " range is " + @@ -2023,7 +2032,7 @@ // space in the file so Sec->Offset + Sec->Size can overlap with others. // If --oformat binary is specified only add SHF_ALLOC sections are added to // the output file so we skip any non-allocated sections in that case. - checkForSectionOverlap(OutputSections, "file", + checkForSectionOverlap(OutputSections, OverlapKind::File, [](const OutputSection *Sec) { return Sec->Offset; }, [](const OutputSection *Sec) { return Sec->Type == SHT_NOBITS || @@ -2045,14 +2054,14 @@ auto SkipNonAllocSections = [](const OutputSection *Sec) { return (Sec->Flags & SHF_ALLOC) == 0 || (Sec->Flags & SHF_TLS); }; - checkForSectionOverlap(OutputSections, "virtual address", + checkForSectionOverlap(OutputSections, OverlapKind::VirtualAddress, [](const OutputSection *Sec) { return Sec->Addr; }, SkipNonAllocSections); // Finally, check that the load addresses don't overlap. This will usually be // the same as the virtual addresses but can be different when using a linker // script with AT(). - checkForSectionOverlap(OutputSections, "load address", + checkForSectionOverlap(OutputSections, OverlapKind::LoadAddress, [](const OutputSection *Sec) { return Sec->getLMA(); }, SkipNonAllocSections); } Index: test/ELF/linkerscript/overlay-reject.test =================================================================== --- test/ELF/linkerscript/overlay-reject.test +++ test/ELF/linkerscript/overlay-reject.test @@ -0,0 +1,13 @@ +# REQUIRES: x86 +# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux /dev/null -o %t.o +# RUN: not ld.lld %t.o --script %s -o %t 2>&1 | FileCheck %s + +# CHECK: {{.*}}.test:{{.*}}: { expected, but got 0x3000 +# CHECK-NEXT: >>> .out.aaa 0x3000 : { *(.aaa) } +# CHECK-NEXT: >>> ^ + +SECTIONS { + OVERLAY 0x1000 : AT ( 0x2000 ) { + .out.aaa 0x3000 : { *(.aaa) } + } +} Index: test/ELF/linkerscript/overlay-reject2.test =================================================================== --- test/ELF/linkerscript/overlay-reject2.test +++ test/ELF/linkerscript/overlay-reject2.test @@ -0,0 +1,17 @@ +# REQUIRES: x86 +# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux /dev/null -o %t.o +# RUN: not ld.lld %t.o --script %s -o %t 2>&1 | FileCheck %s + +# CHECK: {{.*}}.test:{{.*}}: { expected, but got AX +# CHECK-NEXT: >>> .out.aaa { *(.aaa) } > AX AT>FLASH +# CHECK-NEXT: >>> ^ + +MEMORY { + AX (ax) : ORIGIN = 0x3000, LENGTH = 0x4000 +} + +SECTIONS { + OVERLAY 0x1000 : AT ( 0x2000 ) { + .out.aaa { *(.aaa) } > AX AT>FLASH + } +} Index: test/ELF/linkerscript/overlay.test =================================================================== --- test/ELF/linkerscript/overlay.test +++ test/ELF/linkerscript/overlay.test @@ -0,0 +1,36 @@ +# REQUIRES: x86 +# RUN: echo 'nop; .section .small, "a"; .long 0; .section .big, "a"; .quad 1;' \ +# RUN: | llvm-mc -filetype=obj -triple=x86_64-unknown-linux - -o %t.o +# RUN: ld.lld %t.o --script %s -o %t + +SECTIONS { + OVERLAY 0x1000 : AT ( 0x4000 ) { + .out.big { *(.big) } + .out.small { *(.small) } + } +} + +## Here we check that can handle OVERLAY which will produce sections +## .out.big and .out.small with the same starting VAs, but different LMAs. +## Section .big is larger than .small, we check that placing of section +## .text does not cause overlapping error and that +## .text's VA is 0x1000 + max(sizeof(.out.big), sizeof(.out.small)). + +# RUN: llvm-readobj -t -sections -program-headers -elf-output-style=GNU %t | FileCheck %s + +# CHECK: Section Headers: +# CHECK: Name Type Address Off Size +# CHECK: .out.big PROGBITS 0000000000001000 001000 000008 +# CHECK: .out.small PROGBITS 0000000000001000 002000 000004 +# CHECK: .text PROGBITS 0000000000001008 002008 000001 + +# CHECK: Value Size Type Bind Vis Ndx Name +# CHECK: 0000000000004000 0 NOTYPE GLOBAL DEFAULT ABS __load_start_outbig +# CHECK: 0000000000004008 0 NOTYPE GLOBAL DEFAULT ABS __load_stop_outbig +# CHECK: 0000000000004008 0 NOTYPE GLOBAL DEFAULT ABS __load_start_outsmall +# CHECK: 000000000000400c 0 NOTYPE GLOBAL DEFAULT ABS __load_stop_outsmall + +# CHECK: Program Headers: +# CHECK: Type Offset VirtAddr PhysAddr FileSiz MemSiz Flg Align +# CHECK-NEXT: LOAD 0x001000 0x0000000000001000 0x0000000000004000 0x000008 0x000008 R E 0x1000 +# CHECK-NEXT: LOAD 0x002000 0x0000000000001000 0x0000000000004008 0x000009 0x000009 R E 0x1000