Index: ELF/InputFiles.h =================================================================== --- ELF/InputFiles.h +++ ELF/InputFiles.h @@ -71,6 +71,9 @@ // Returns "(internal)", "foo.a(bar.o)" or "baz.o". std::string getFilename(const InputFile *F); +// Compares filenames. Used in linkerscript for SORT. +bool compareFileNames(const InputFile *A, const InputFile *B); + template class ELFFileBase : public InputFile { public: typedef typename ELFT::Shdr Elf_Shdr; Index: ELF/InputFiles.cpp =================================================================== --- ELF/InputFiles.cpp +++ ELF/InputFiles.cpp @@ -37,6 +37,14 @@ return F->getName(); } +bool elf::compareFileNames(const InputFile *A, const InputFile *B) { + if (!A || !B) + return A; + StringRef NameA = A->ArchiveName.empty() ? A->getName() : A->ArchiveName; + StringRef NameB = B->ArchiveName.empty() ? B->getName() : B->ArchiveName; + return NameA < NameB; +} + template static ELFFile createELFObj(MemoryBufferRef MB) { std::error_code EC; Index: ELF/LinkerScript.h =================================================================== --- ELF/LinkerScript.h +++ ELF/LinkerScript.h @@ -82,6 +82,7 @@ struct InputSectionDescription : BaseCommand { InputSectionDescription() : BaseCommand(InputSectionKind) {} static bool classof(const BaseCommand *C); + bool Sort = false; std::vector Patterns; }; Index: ELF/LinkerScript.cpp =================================================================== --- ELF/LinkerScript.cpp +++ ELF/LinkerScript.cpp @@ -115,6 +115,7 @@ if (!InCmd) continue; + std::vector *> Candidates; for (ObjectFile &F : Symtab::X->getObjectFiles()) { for (InputSectionBase *S : F->getSections()) { if (isDiscarded(S) || S->OutSec) @@ -124,10 +125,21 @@ if (OutCmd->Name == "/DISCARD/") S->Live = false; else - AddInputSec(S, OutCmd->Name, OutCmd->Constraint); + Candidates.push_back(S); } } } + + if (InCmd->Sort) + std::stable_sort( + Candidates.begin(), Candidates.end(), + [](InputSectionBase *A, InputSectionBase *B) { + if (A->getSectionName() != B->getSectionName()) + return A->getSectionName() < B->getSectionName(); + return compareFileNames(A->getFile(), B->getFile()); + }); + for (InputSectionBase *I : Candidates) + AddInputSec(I, OutCmd->Name, OutCmd->Constraint); } } @@ -420,6 +432,7 @@ SymbolAssignment *readAssignment(StringRef Name); void readOutputSectionDescription(StringRef OutSec); + void readInputSectionDescription(InputSectionDescription *InCmd, bool Keep); std::vector readOutputSectionPhdrs(); unsigned readPhdrType(); void readProvide(bool Hidden); @@ -648,6 +661,25 @@ .Default(-1); } +void ScriptParser::readInputSectionDescription(InputSectionDescription *InCmd, + bool Keep) { + if (peek() == "SORT") { + expect("SORT"); + expect("("); + InCmd->Sort = true; + readInputSectionDescription(InCmd, Keep); + expect(")"); + return; + } + + + while (!Error && !skip(")")) { + if (Keep) + Opt.KeptSections.push_back(peek()); + InCmd->Patterns.push_back(next()); + } +} + void ScriptParser::readOutputSectionDescription(StringRef OutSec) { OutputSectionCommand *Cmd = new OutputSectionCommand(OutSec); Opt.Commands.emplace_back(Cmd); @@ -667,25 +699,30 @@ expect("{"); while (!Error && !skip("}")) { - StringRef Tok = next(); - if (Tok == "*") { + StringRef Peek = peek(); + if (Peek == "*" || Peek == "KEEP") { auto *InCmd = new InputSectionDescription(); Cmd->Commands.emplace_back(InCmd); - expect("("); - while (!Error && !skip(")")) - InCmd->Patterns.push_back(next()); - } else if (Tok == "KEEP") { - expect("("); - expect("*"); - expect("("); - auto *InCmd = new InputSectionDescription(); - Cmd->Commands.emplace_back(InCmd); - while (!Error && !skip(")")) { - Opt.KeptSections.push_back(peek()); - InCmd->Patterns.push_back(next()); + + // Input section wildcard can be surrounded by KEEP. + // https://sourceware.org/binutils/docs/ld/Input-Section-Keep.html#Input-Section-Keep + if (Peek == "KEEP") { + expect("KEEP"); + expect("("); + expect("*"); + expect("("); + readInputSectionDescription(InCmd, true); + expect(")"); + } else { + expect("*"); + expect("("); + readInputSectionDescription(InCmd, false); } - expect(")"); - } else if (Tok == "PROVIDE") { + continue; + } + + StringRef Tok = next(); + if (Tok == "PROVIDE") { readProvide(false); } else if (Tok == "PROVIDE_HIDDEN") { readProvide(true); Index: test/ELF/linkerscript/Inputs/linkerscript-sort.s =================================================================== --- test/ELF/linkerscript/Inputs/linkerscript-sort.s +++ test/ELF/linkerscript/Inputs/linkerscript-sort.s @@ -0,0 +1,14 @@ +.section .aaa.5, "a" +.quad 0x55 + +.section .aaa.1, "a" +.quad 0x11 + +.section .aaa.3, "a" +.quad 0x33 + +.section .aaa.2, "a" +.quad 0x22 + +.section .aaa.4, "a" +.quad 0x44 Index: test/ELF/linkerscript/linkerscript-sort.s =================================================================== --- test/ELF/linkerscript/linkerscript-sort.s +++ test/ELF/linkerscript/linkerscript-sort.s @@ -0,0 +1,50 @@ +# REQUIRES: x86 +# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux %s -o %t1.o +# RUN: llvm-mc -filetype=obj -triple=x86_64-unknown-linux \ +# RUN: %p/Inputs/linkerscript-sort.s -o %t2.o + +# RUN: echo "SECTIONS { .aaa : { *(.aaa.*) } }" > %t1.script +# RUN: ld.lld -o %t1 --script %t1.script %t2.o %t1.o +# RUN: llvm-objdump -s %t1 | FileCheck -check-prefix=UNSORTED %s +# UNSORTED: Contents of section .aaa: +# UNSORTED-NEXT: 0120 55000000 00000000 11000000 00000000 +# UNSORTED-NEXT: 0130 33000000 00000000 22000000 00000000 +# UNSORTED-NEXT: 0140 44000000 00000000 05000000 00000000 +# UNSORTED-NEXT: 0150 01000000 00000000 03000000 00000000 +# UNSORTED-NEXT: 0160 02000000 00000000 04000000 00000000 + +## Check that SORT works (sorted by name of section and by filename if +## section names are equal). +# RUN: echo "SECTIONS { .aaa : { *(SORT(.aaa.*)) } }" > %t2.script +# RUN: ld.lld -o %t2 --script %t2.script %t2.o %t1.o +# RUN: llvm-objdump -s %t2 | FileCheck -check-prefix=SORTED %s +# SORTED: Contents of section .aaa: +# SORTED-NEXT: 0120 01000000 00000000 11000000 00000000 +# SORTED-NEXT: 0130 02000000 00000000 22000000 00000000 +# SORTED-NEXT: 0140 03000000 00000000 33000000 00000000 +# SORTED-NEXT: 0150 04000000 00000000 44000000 00000000 +# SORTED-NEXT: 0160 05000000 00000000 55000000 00000000 + +## Check that SORT surrounded with KEEP also works. +# RUN: echo "SECTIONS { .aaa : { KEEP (*(SORT(.aaa.*))) } }" > %t3.script +# RUN: ld.lld -o %t3 --script %t3.script %t2.o %t1.o +# RUN: llvm-objdump -s %t3 | FileCheck -check-prefix=SORTED %s + +.global _start +_start: + nop + +.section .aaa.5, "a" +.quad 5 + +.section .aaa.1, "a" +.quad 1 + +.section .aaa.3, "a" +.quad 3 + +.section .aaa.2, "a" +.quad 2 + +.section .aaa.4, "a" +.quad 4