| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 1 | //===- LinkerScript.h -------------------------------------------*- C++ -*-===// | 
|  | 2 | // | 
|  | 3 | //                             The LLVM Linker | 
|  | 4 | // | 
|  | 5 | // This file is distributed under the University of Illinois Open Source | 
|  | 6 | // License. See LICENSE.TXT for details. | 
|  | 7 | // | 
|  | 8 | //===----------------------------------------------------------------------===// | 
|  | 9 |  | 
|  | 10 | #ifndef LLD_ELF_LINKER_SCRIPT_H | 
|  | 11 | #define LLD_ELF_LINKER_SCRIPT_H | 
|  | 12 |  | 
| Eugene Leviant | bbe3860 | 2016-07-19 09:25:43 +0000 | [diff] [blame] | 13 | #include "Writer.h" | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 14 | #include "lld/Core/LLVM.h" | 
|  | 15 | #include "llvm/ADT/DenseMap.h" | 
|  | 16 | #include "llvm/ADT/MapVector.h" | 
| Rui Ueyama | f9de0d6 | 2016-02-11 21:38:55 +0000 | [diff] [blame] | 17 | #include "llvm/Support/Allocator.h" | 
|  | 18 | #include "llvm/Support/MemoryBuffer.h" | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 19 |  | 
|  | 20 | namespace lld { | 
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 21 | namespace elf { | 
| Eugene Leviant | e63d81b | 2016-07-20 14:43:20 +0000 | [diff] [blame] | 22 | template <class ELFT> class InputSectionBase; | 
|  | 23 | template <class ELFT> class OutputSectionBase; | 
|  | 24 | template <class ELFT> class OutputSectionFactory; | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 25 |  | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 26 | // Parses a linker script. Calling this function updates | 
|  | 27 | // Config and ScriptConfig. | 
|  | 28 | void readLinkerScript(MemoryBufferRef MB); | 
|  | 29 |  | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 30 | class ScriptParser; | 
| Rui Ueyama | 1ebc8ed | 2016-02-12 21:47:28 +0000 | [diff] [blame] | 31 | template <class ELFT> class InputSectionBase; | 
| George Rimar | 652852c | 2016-04-16 10:10:32 +0000 | [diff] [blame] | 32 | template <class ELFT> class OutputSectionBase; | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 33 |  | 
| Rui Ueyama | 1ebc8ed | 2016-02-12 21:47:28 +0000 | [diff] [blame] | 34 | // This class represents each rule in SECTIONS command. | 
| Rui Ueyama | c9f402e | 2016-04-22 00:23:52 +0000 | [diff] [blame] | 35 | struct SectionRule { | 
| Rui Ueyama | 8ec77e6 | 2016-04-21 22:00:51 +0000 | [diff] [blame] | 36 | SectionRule(StringRef D, StringRef S) | 
|  | 37 | : Dest(D), SectionPattern(S) {} | 
| Rui Ueyama | 1ebc8ed | 2016-02-12 21:47:28 +0000 | [diff] [blame] | 38 |  | 
| Rui Ueyama | 1ebc8ed | 2016-02-12 21:47:28 +0000 | [diff] [blame] | 39 | StringRef Dest; | 
|  | 40 |  | 
| Rui Ueyama | 1ebc8ed | 2016-02-12 21:47:28 +0000 | [diff] [blame] | 41 | StringRef SectionPattern; | 
|  | 42 | }; | 
|  | 43 |  | 
| George Rimar | 076fe15 | 2016-07-21 06:43:01 +0000 | [diff] [blame] | 44 | // This enum represents what we can observe in SECTIONS tag of script. | 
|  | 45 | // Each sections-command may of be one of the following: | 
|  | 46 | // (https://sourceware.org/binutils/docs/ld/SECTIONS.html#SECTIONS) | 
|  | 47 | // * An ENTRY command. | 
|  | 48 | // * A symbol assignment. | 
|  | 49 | // * An output section description. | 
|  | 50 | // * An overlay description. | 
|  | 51 | // We support only AssignmentKind and OutputSectionKind for now. | 
|  | 52 | enum SectionsCommandKind { AssignmentKind, OutputSectionKind }; | 
| George Rimar | 652852c | 2016-04-16 10:10:32 +0000 | [diff] [blame] | 53 |  | 
| George Rimar | 076fe15 | 2016-07-21 06:43:01 +0000 | [diff] [blame] | 54 | struct BaseCommand { | 
|  | 55 | BaseCommand(int K) : Kind(K) {} | 
|  | 56 | virtual ~BaseCommand() {} | 
|  | 57 | int Kind; | 
|  | 58 | }; | 
|  | 59 |  | 
|  | 60 | struct SymbolAssignment : BaseCommand { | 
|  | 61 | SymbolAssignment(StringRef Name, std::vector<StringRef> &Expr) | 
|  | 62 | : BaseCommand(AssignmentKind), Name(Name), Expr(std::move(Expr)) {} | 
|  | 63 | static bool classof(const BaseCommand *C); | 
|  | 64 | StringRef Name; | 
| George Rimar | 652852c | 2016-04-16 10:10:32 +0000 | [diff] [blame] | 65 | std::vector<StringRef> Expr; | 
| George Rimar | 076fe15 | 2016-07-21 06:43:01 +0000 | [diff] [blame] | 66 | }; | 
|  | 67 |  | 
|  | 68 | struct OutputSectionCommand : BaseCommand { | 
|  | 69 | OutputSectionCommand(StringRef Name) | 
|  | 70 | : BaseCommand(OutputSectionKind), Name(Name) {} | 
|  | 71 | static bool classof(const BaseCommand *C); | 
| Eugene Leviant | eda81a1 | 2016-07-12 06:39:48 +0000 | [diff] [blame] | 72 | StringRef Name; | 
| Eugene Leviant | bbe3860 | 2016-07-19 09:25:43 +0000 | [diff] [blame] | 73 | std::vector<StringRef> Phdrs; | 
| George Rimar | 076fe15 | 2016-07-21 06:43:01 +0000 | [diff] [blame] | 74 | std::vector<uint8_t> Filler; | 
| Eugene Leviant | bbe3860 | 2016-07-19 09:25:43 +0000 | [diff] [blame] | 75 | }; | 
|  | 76 |  | 
|  | 77 | struct PhdrsCommand { | 
|  | 78 | StringRef Name; | 
|  | 79 | unsigned Type; | 
|  | 80 | bool HasFilehdr; | 
|  | 81 | bool HasPhdrs; | 
| Eugene Leviant | 865bf86 | 2016-07-21 10:43:25 +0000 | [diff] [blame] | 82 | unsigned Flags; | 
| George Rimar | 652852c | 2016-04-16 10:10:32 +0000 | [diff] [blame] | 83 | }; | 
|  | 84 |  | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 85 | // ScriptConfiguration holds linker script parse results. | 
|  | 86 | struct ScriptConfiguration { | 
| Rui Ueyama | 1ebc8ed | 2016-02-12 21:47:28 +0000 | [diff] [blame] | 87 | // SECTIONS commands. | 
|  | 88 | std::vector<SectionRule> Sections; | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 89 |  | 
| George Rimar | 652852c | 2016-04-16 10:10:32 +0000 | [diff] [blame] | 90 | // Used to assign addresses to sections. | 
| George Rimar | 076fe15 | 2016-07-21 06:43:01 +0000 | [diff] [blame] | 91 | std::vector<std::unique_ptr<BaseCommand>> Commands; | 
| George Rimar | 652852c | 2016-04-16 10:10:32 +0000 | [diff] [blame] | 92 |  | 
| Eugene Leviant | bbe3860 | 2016-07-19 09:25:43 +0000 | [diff] [blame] | 93 | // Used to assign sections to headers. | 
| George Rimar | 70ce0a9 | 2016-07-20 15:09:10 +0000 | [diff] [blame] | 94 | std::vector<PhdrsCommand> PhdrsCommands; | 
|  | 95 |  | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 96 | bool DoLayout = false; | 
|  | 97 |  | 
| Rui Ueyama | f9de0d6 | 2016-02-11 21:38:55 +0000 | [diff] [blame] | 98 | llvm::BumpPtrAllocator Alloc; | 
| Rui Ueyama | 8ec77e6 | 2016-04-21 22:00:51 +0000 | [diff] [blame] | 99 |  | 
|  | 100 | // List of section patterns specified with KEEP commands. They will | 
|  | 101 | // be kept even if they are unused and --gc-sections is specified. | 
|  | 102 | std::vector<StringRef> KeptSections; | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 103 | }; | 
|  | 104 |  | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 105 | extern ScriptConfiguration *ScriptConfig; | 
|  | 106 |  | 
|  | 107 | // This is a runner of the linker script. | 
|  | 108 | template <class ELFT> class LinkerScript { | 
| Rui Ueyama | 0b3868e | 2016-04-22 20:41:07 +0000 | [diff] [blame] | 109 | typedef typename ELFT::uint uintX_t; | 
|  | 110 |  | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 111 | public: | 
| Rafael Espindola | 74df5c7 | 2016-07-19 12:33:46 +0000 | [diff] [blame] | 112 | typedef PhdrEntry<ELFT> Phdr; | 
| Eugene Leviant | bbe3860 | 2016-07-19 09:25:43 +0000 | [diff] [blame] | 113 |  | 
| Rui Ueyama | a7f7884 | 2016-07-20 17:19:03 +0000 | [diff] [blame] | 114 | std::vector<OutputSectionBase<ELFT> *> | 
|  | 115 | createSections(OutputSectionFactory<ELFT> &Factory); | 
|  | 116 |  | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 117 | StringRef getOutputSection(InputSectionBase<ELFT> *S); | 
|  | 118 | ArrayRef<uint8_t> getFiller(StringRef Name); | 
|  | 119 | bool isDiscarded(InputSectionBase<ELFT> *S); | 
|  | 120 | bool shouldKeep(InputSectionBase<ELFT> *S); | 
| George Rimar | dbbd8b1 | 2016-04-21 11:21:48 +0000 | [diff] [blame] | 121 | void assignAddresses(ArrayRef<OutputSectionBase<ELFT> *> S); | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 122 | int compareSections(StringRef A, StringRef B); | 
| Eugene Leviant | eda81a1 | 2016-07-12 06:39:48 +0000 | [diff] [blame] | 123 | void addScriptedSymbols(); | 
| Eugene Leviant | bbe3860 | 2016-07-19 09:25:43 +0000 | [diff] [blame] | 124 | std::vector<Phdr> createPhdrs(ArrayRef<OutputSectionBase<ELFT> *> S); | 
|  | 125 | bool hasPhdrsCommands(); | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 126 |  | 
|  | 127 | private: | 
| Rui Ueyama | c998a8c | 2016-04-22 00:03:13 +0000 | [diff] [blame] | 128 | // "ScriptConfig" is a bit too long, so define a short name for it. | 
|  | 129 | ScriptConfiguration &Opt = *ScriptConfig; | 
|  | 130 |  | 
| Rui Ueyama | c3e2a4b | 2016-04-21 20:30:00 +0000 | [diff] [blame] | 131 | int getSectionIndex(StringRef Name); | 
| Eugene Leviant | bbe3860 | 2016-07-19 09:25:43 +0000 | [diff] [blame] | 132 | std::vector<size_t> getPhdrIndicesForSection(StringRef Name); | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 133 |  | 
| Rui Ueyama | 0b3868e | 2016-04-22 20:41:07 +0000 | [diff] [blame] | 134 | uintX_t Dot; | 
| Rui Ueyama | 07320e4 | 2016-04-20 20:13:41 +0000 | [diff] [blame] | 135 | }; | 
|  | 136 |  | 
|  | 137 | // Variable template is a C++14 feature, so we can't template | 
|  | 138 | // a global variable. Use a struct to workaround. | 
|  | 139 | template <class ELFT> struct Script { static LinkerScript<ELFT> *X; }; | 
|  | 140 | template <class ELFT> LinkerScript<ELFT> *Script<ELFT>::X; | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 141 |  | 
| Rafael Espindola | e0df00b | 2016-02-28 00:25:54 +0000 | [diff] [blame] | 142 | } // namespace elf | 
| Rui Ueyama | 717677a | 2016-02-11 21:17:59 +0000 | [diff] [blame] | 143 | } // namespace lld | 
|  | 144 |  | 
|  | 145 | #endif |