annotate lld/ELF/SyntheticSections.h @ 173:0572611fdcc8 llvm10 llvm12

reorgnization done
author Shinji KONO <kono@ie.u-ryukyu.ac.jp>
date Mon, 25 May 2020 11:55:54 +0900
parents 1d019706d866
children 2e18cbf3894f
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
rev   line source
150
anatofuz
parents:
diff changeset
1 //===- SyntheticSection.h ---------------------------------------*- C++ -*-===//
anatofuz
parents:
diff changeset
2 //
anatofuz
parents:
diff changeset
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
anatofuz
parents:
diff changeset
4 // See https://llvm.org/LICENSE.txt for license information.
anatofuz
parents:
diff changeset
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
anatofuz
parents:
diff changeset
6 //
anatofuz
parents:
diff changeset
7 //===----------------------------------------------------------------------===//
anatofuz
parents:
diff changeset
8 //
anatofuz
parents:
diff changeset
9 // Synthetic sections represent chunks of linker-created data. If you
anatofuz
parents:
diff changeset
10 // need to create a chunk of data that to be included in some section
anatofuz
parents:
diff changeset
11 // in the result, you probably want to create that as a synthetic section.
anatofuz
parents:
diff changeset
12 //
anatofuz
parents:
diff changeset
13 // Synthetic sections are designed as input sections as opposed to
anatofuz
parents:
diff changeset
14 // output sections because we want to allow them to be manipulated
anatofuz
parents:
diff changeset
15 // using linker scripts just like other input sections from regular
anatofuz
parents:
diff changeset
16 // files.
anatofuz
parents:
diff changeset
17 //
anatofuz
parents:
diff changeset
18 //===----------------------------------------------------------------------===//
anatofuz
parents:
diff changeset
19
anatofuz
parents:
diff changeset
20 #ifndef LLD_ELF_SYNTHETIC_SECTIONS_H
anatofuz
parents:
diff changeset
21 #define LLD_ELF_SYNTHETIC_SECTIONS_H
anatofuz
parents:
diff changeset
22
anatofuz
parents:
diff changeset
23 #include "DWARF.h"
anatofuz
parents:
diff changeset
24 #include "EhFrame.h"
anatofuz
parents:
diff changeset
25 #include "InputSection.h"
anatofuz
parents:
diff changeset
26 #include "llvm/ADT/MapVector.h"
anatofuz
parents:
diff changeset
27 #include "llvm/MC/StringTableBuilder.h"
anatofuz
parents:
diff changeset
28 #include "llvm/Support/Endian.h"
anatofuz
parents:
diff changeset
29 #include <functional>
anatofuz
parents:
diff changeset
30
anatofuz
parents:
diff changeset
31 namespace lld {
anatofuz
parents:
diff changeset
32 namespace elf {
anatofuz
parents:
diff changeset
33 class Defined;
anatofuz
parents:
diff changeset
34 struct PhdrEntry;
anatofuz
parents:
diff changeset
35 class SymbolTableBaseSection;
anatofuz
parents:
diff changeset
36 class VersionNeedBaseSection;
anatofuz
parents:
diff changeset
37
anatofuz
parents:
diff changeset
38 class SyntheticSection : public InputSection {
anatofuz
parents:
diff changeset
39 public:
anatofuz
parents:
diff changeset
40 SyntheticSection(uint64_t flags, uint32_t type, uint32_t alignment,
anatofuz
parents:
diff changeset
41 StringRef name)
anatofuz
parents:
diff changeset
42 : InputSection(nullptr, flags, type, alignment, {}, name,
anatofuz
parents:
diff changeset
43 InputSectionBase::Synthetic) {
anatofuz
parents:
diff changeset
44 markLive();
anatofuz
parents:
diff changeset
45 }
anatofuz
parents:
diff changeset
46
anatofuz
parents:
diff changeset
47 virtual ~SyntheticSection() = default;
anatofuz
parents:
diff changeset
48 virtual void writeTo(uint8_t *buf) = 0;
anatofuz
parents:
diff changeset
49 virtual size_t getSize() const = 0;
anatofuz
parents:
diff changeset
50 virtual void finalizeContents() {}
anatofuz
parents:
diff changeset
51 // If the section has the SHF_ALLOC flag and the size may be changed if
anatofuz
parents:
diff changeset
52 // thunks are added, update the section size.
anatofuz
parents:
diff changeset
53 virtual bool updateAllocSize() { return false; }
anatofuz
parents:
diff changeset
54 virtual bool isNeeded() const { return true; }
anatofuz
parents:
diff changeset
55
anatofuz
parents:
diff changeset
56 static bool classof(const SectionBase *d) {
anatofuz
parents:
diff changeset
57 return d->kind() == InputSectionBase::Synthetic;
anatofuz
parents:
diff changeset
58 }
anatofuz
parents:
diff changeset
59 };
anatofuz
parents:
diff changeset
60
anatofuz
parents:
diff changeset
61 struct CieRecord {
anatofuz
parents:
diff changeset
62 EhSectionPiece *cie = nullptr;
anatofuz
parents:
diff changeset
63 std::vector<EhSectionPiece *> fdes;
anatofuz
parents:
diff changeset
64 };
anatofuz
parents:
diff changeset
65
anatofuz
parents:
diff changeset
66 // Section for .eh_frame.
anatofuz
parents:
diff changeset
67 class EhFrameSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
68 public:
anatofuz
parents:
diff changeset
69 EhFrameSection();
anatofuz
parents:
diff changeset
70 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
71 void finalizeContents() override;
anatofuz
parents:
diff changeset
72 bool isNeeded() const override { return !sections.empty(); }
anatofuz
parents:
diff changeset
73 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
74
anatofuz
parents:
diff changeset
75 static bool classof(const SectionBase *d) {
anatofuz
parents:
diff changeset
76 return SyntheticSection::classof(d) && d->name == ".eh_frame";
anatofuz
parents:
diff changeset
77 }
anatofuz
parents:
diff changeset
78
anatofuz
parents:
diff changeset
79 void addSection(EhInputSection *sec);
anatofuz
parents:
diff changeset
80
anatofuz
parents:
diff changeset
81 std::vector<EhInputSection *> sections;
anatofuz
parents:
diff changeset
82 size_t numFdes = 0;
anatofuz
parents:
diff changeset
83
anatofuz
parents:
diff changeset
84 struct FdeData {
anatofuz
parents:
diff changeset
85 uint32_t pcRel;
anatofuz
parents:
diff changeset
86 uint32_t fdeVARel;
anatofuz
parents:
diff changeset
87 };
anatofuz
parents:
diff changeset
88
anatofuz
parents:
diff changeset
89 std::vector<FdeData> getFdeData() const;
anatofuz
parents:
diff changeset
90 ArrayRef<CieRecord *> getCieRecords() const { return cieRecords; }
anatofuz
parents:
diff changeset
91
anatofuz
parents:
diff changeset
92 private:
anatofuz
parents:
diff changeset
93 // This is used only when parsing EhInputSection. We keep it here to avoid
anatofuz
parents:
diff changeset
94 // allocating one for each EhInputSection.
anatofuz
parents:
diff changeset
95 llvm::DenseMap<size_t, CieRecord *> offsetToCie;
anatofuz
parents:
diff changeset
96
anatofuz
parents:
diff changeset
97 uint64_t size = 0;
anatofuz
parents:
diff changeset
98
anatofuz
parents:
diff changeset
99 template <class ELFT, class RelTy>
anatofuz
parents:
diff changeset
100 void addRecords(EhInputSection *s, llvm::ArrayRef<RelTy> rels);
anatofuz
parents:
diff changeset
101 template <class ELFT>
anatofuz
parents:
diff changeset
102 void addSectionAux(EhInputSection *s);
anatofuz
parents:
diff changeset
103
anatofuz
parents:
diff changeset
104 template <class ELFT, class RelTy>
anatofuz
parents:
diff changeset
105 CieRecord *addCie(EhSectionPiece &piece, ArrayRef<RelTy> rels);
anatofuz
parents:
diff changeset
106
anatofuz
parents:
diff changeset
107 template <class ELFT, class RelTy>
anatofuz
parents:
diff changeset
108 bool isFdeLive(EhSectionPiece &piece, ArrayRef<RelTy> rels);
anatofuz
parents:
diff changeset
109
anatofuz
parents:
diff changeset
110 uint64_t getFdePc(uint8_t *buf, size_t off, uint8_t enc) const;
anatofuz
parents:
diff changeset
111
anatofuz
parents:
diff changeset
112 std::vector<CieRecord *> cieRecords;
anatofuz
parents:
diff changeset
113
anatofuz
parents:
diff changeset
114 // CIE records are uniquified by their contents and personality functions.
anatofuz
parents:
diff changeset
115 llvm::DenseMap<std::pair<ArrayRef<uint8_t>, Symbol *>, CieRecord *> cieMap;
anatofuz
parents:
diff changeset
116 };
anatofuz
parents:
diff changeset
117
anatofuz
parents:
diff changeset
118 class GotSection : public SyntheticSection {
anatofuz
parents:
diff changeset
119 public:
anatofuz
parents:
diff changeset
120 GotSection();
anatofuz
parents:
diff changeset
121 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
122 void finalizeContents() override;
anatofuz
parents:
diff changeset
123 bool isNeeded() const override;
anatofuz
parents:
diff changeset
124 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
125
anatofuz
parents:
diff changeset
126 void addEntry(Symbol &sym);
anatofuz
parents:
diff changeset
127 bool addDynTlsEntry(Symbol &sym);
anatofuz
parents:
diff changeset
128 bool addTlsIndex();
anatofuz
parents:
diff changeset
129 uint64_t getGlobalDynAddr(const Symbol &b) const;
anatofuz
parents:
diff changeset
130 uint64_t getGlobalDynOffset(const Symbol &b) const;
anatofuz
parents:
diff changeset
131
anatofuz
parents:
diff changeset
132 uint64_t getTlsIndexVA() { return this->getVA() + tlsIndexOff; }
anatofuz
parents:
diff changeset
133 uint32_t getTlsIndexOff() const { return tlsIndexOff; }
anatofuz
parents:
diff changeset
134
anatofuz
parents:
diff changeset
135 // Flag to force GOT to be in output if we have relocations
anatofuz
parents:
diff changeset
136 // that relies on its address.
anatofuz
parents:
diff changeset
137 bool hasGotOffRel = false;
anatofuz
parents:
diff changeset
138
anatofuz
parents:
diff changeset
139 protected:
anatofuz
parents:
diff changeset
140 size_t numEntries = 0;
anatofuz
parents:
diff changeset
141 uint32_t tlsIndexOff = -1;
anatofuz
parents:
diff changeset
142 uint64_t size = 0;
anatofuz
parents:
diff changeset
143 };
anatofuz
parents:
diff changeset
144
anatofuz
parents:
diff changeset
145 // .note.GNU-stack section.
anatofuz
parents:
diff changeset
146 class GnuStackSection : public SyntheticSection {
anatofuz
parents:
diff changeset
147 public:
anatofuz
parents:
diff changeset
148 GnuStackSection()
anatofuz
parents:
diff changeset
149 : SyntheticSection(0, llvm::ELF::SHT_PROGBITS, 1, ".note.GNU-stack") {}
anatofuz
parents:
diff changeset
150 void writeTo(uint8_t *buf) override {}
anatofuz
parents:
diff changeset
151 size_t getSize() const override { return 0; }
anatofuz
parents:
diff changeset
152 };
anatofuz
parents:
diff changeset
153
anatofuz
parents:
diff changeset
154 class GnuPropertySection : public SyntheticSection {
anatofuz
parents:
diff changeset
155 public:
anatofuz
parents:
diff changeset
156 GnuPropertySection();
anatofuz
parents:
diff changeset
157 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
158 size_t getSize() const override;
anatofuz
parents:
diff changeset
159 };
anatofuz
parents:
diff changeset
160
anatofuz
parents:
diff changeset
161 // .note.gnu.build-id section.
anatofuz
parents:
diff changeset
162 class BuildIdSection : public SyntheticSection {
anatofuz
parents:
diff changeset
163 // First 16 bytes are a header.
anatofuz
parents:
diff changeset
164 static const unsigned headerSize = 16;
anatofuz
parents:
diff changeset
165
anatofuz
parents:
diff changeset
166 public:
anatofuz
parents:
diff changeset
167 const size_t hashSize;
anatofuz
parents:
diff changeset
168 BuildIdSection();
anatofuz
parents:
diff changeset
169 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
170 size_t getSize() const override { return headerSize + hashSize; }
anatofuz
parents:
diff changeset
171 void writeBuildId(llvm::ArrayRef<uint8_t> buf);
anatofuz
parents:
diff changeset
172
anatofuz
parents:
diff changeset
173 private:
anatofuz
parents:
diff changeset
174 uint8_t *hashBuf;
anatofuz
parents:
diff changeset
175 };
anatofuz
parents:
diff changeset
176
anatofuz
parents:
diff changeset
177 // BssSection is used to reserve space for copy relocations and common symbols.
anatofuz
parents:
diff changeset
178 // We create three instances of this class for .bss, .bss.rel.ro and "COMMON",
anatofuz
parents:
diff changeset
179 // that are used for writable symbols, read-only symbols and common symbols,
anatofuz
parents:
diff changeset
180 // respectively.
anatofuz
parents:
diff changeset
181 class BssSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
182 public:
anatofuz
parents:
diff changeset
183 BssSection(StringRef name, uint64_t size, uint32_t alignment);
anatofuz
parents:
diff changeset
184 void writeTo(uint8_t *) override {
anatofuz
parents:
diff changeset
185 llvm_unreachable("unexpected writeTo() call for SHT_NOBITS section");
anatofuz
parents:
diff changeset
186 }
anatofuz
parents:
diff changeset
187 bool isNeeded() const override { return size != 0; }
anatofuz
parents:
diff changeset
188 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
189
anatofuz
parents:
diff changeset
190 static bool classof(const SectionBase *s) { return s->bss; }
anatofuz
parents:
diff changeset
191 uint64_t size;
anatofuz
parents:
diff changeset
192 };
anatofuz
parents:
diff changeset
193
anatofuz
parents:
diff changeset
194 class MipsGotSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
195 public:
anatofuz
parents:
diff changeset
196 MipsGotSection();
anatofuz
parents:
diff changeset
197 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
198 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
199 bool updateAllocSize() override;
anatofuz
parents:
diff changeset
200 void finalizeContents() override;
anatofuz
parents:
diff changeset
201 bool isNeeded() const override;
anatofuz
parents:
diff changeset
202
anatofuz
parents:
diff changeset
203 // Join separate GOTs built for each input file to generate
anatofuz
parents:
diff changeset
204 // primary and optional multiple secondary GOTs.
anatofuz
parents:
diff changeset
205 void build();
anatofuz
parents:
diff changeset
206
anatofuz
parents:
diff changeset
207 void addEntry(InputFile &file, Symbol &sym, int64_t addend, RelExpr expr);
anatofuz
parents:
diff changeset
208 void addDynTlsEntry(InputFile &file, Symbol &sym);
anatofuz
parents:
diff changeset
209 void addTlsIndex(InputFile &file);
anatofuz
parents:
diff changeset
210
anatofuz
parents:
diff changeset
211 uint64_t getPageEntryOffset(const InputFile *f, const Symbol &s,
anatofuz
parents:
diff changeset
212 int64_t addend) const;
anatofuz
parents:
diff changeset
213 uint64_t getSymEntryOffset(const InputFile *f, const Symbol &s,
anatofuz
parents:
diff changeset
214 int64_t addend) const;
anatofuz
parents:
diff changeset
215 uint64_t getGlobalDynOffset(const InputFile *f, const Symbol &s) const;
anatofuz
parents:
diff changeset
216 uint64_t getTlsIndexOffset(const InputFile *f) const;
anatofuz
parents:
diff changeset
217
anatofuz
parents:
diff changeset
218 // Returns the symbol which corresponds to the first entry of the global part
anatofuz
parents:
diff changeset
219 // of GOT on MIPS platform. It is required to fill up MIPS-specific dynamic
anatofuz
parents:
diff changeset
220 // table properties.
anatofuz
parents:
diff changeset
221 // Returns nullptr if the global part is empty.
anatofuz
parents:
diff changeset
222 const Symbol *getFirstGlobalEntry() const;
anatofuz
parents:
diff changeset
223
anatofuz
parents:
diff changeset
224 // Returns the number of entries in the local part of GOT including
anatofuz
parents:
diff changeset
225 // the number of reserved entries.
anatofuz
parents:
diff changeset
226 unsigned getLocalEntriesNum() const;
anatofuz
parents:
diff changeset
227
anatofuz
parents:
diff changeset
228 // Return _gp value for primary GOT (nullptr) or particular input file.
anatofuz
parents:
diff changeset
229 uint64_t getGp(const InputFile *f = nullptr) const;
anatofuz
parents:
diff changeset
230
anatofuz
parents:
diff changeset
231 private:
anatofuz
parents:
diff changeset
232 // MIPS GOT consists of three parts: local, global and tls. Each part
anatofuz
parents:
diff changeset
233 // contains different types of entries. Here is a layout of GOT:
anatofuz
parents:
diff changeset
234 // - Header entries |
anatofuz
parents:
diff changeset
235 // - Page entries | Local part
anatofuz
parents:
diff changeset
236 // - Local entries (16-bit access) |
anatofuz
parents:
diff changeset
237 // - Local entries (32-bit access) |
anatofuz
parents:
diff changeset
238 // - Normal global entries || Global part
anatofuz
parents:
diff changeset
239 // - Reloc-only global entries ||
anatofuz
parents:
diff changeset
240 // - TLS entries ||| TLS part
anatofuz
parents:
diff changeset
241 //
anatofuz
parents:
diff changeset
242 // Header:
anatofuz
parents:
diff changeset
243 // Two entries hold predefined value 0x0 and 0x80000000.
anatofuz
parents:
diff changeset
244 // Page entries:
anatofuz
parents:
diff changeset
245 // These entries created by R_MIPS_GOT_PAGE relocation and R_MIPS_GOT16
anatofuz
parents:
diff changeset
246 // relocation against local symbols. They are initialized by higher 16-bit
anatofuz
parents:
diff changeset
247 // of the corresponding symbol's value. So each 64kb of address space
anatofuz
parents:
diff changeset
248 // requires a single GOT entry.
anatofuz
parents:
diff changeset
249 // Local entries (16-bit access):
anatofuz
parents:
diff changeset
250 // These entries created by GOT relocations against global non-preemptible
anatofuz
parents:
diff changeset
251 // symbols so dynamic linker is not necessary to resolve the symbol's
anatofuz
parents:
diff changeset
252 // values. "16-bit access" means that corresponding relocations address
anatofuz
parents:
diff changeset
253 // GOT using 16-bit index. Each unique Symbol-Addend pair has its own
anatofuz
parents:
diff changeset
254 // GOT entry.
anatofuz
parents:
diff changeset
255 // Local entries (32-bit access):
anatofuz
parents:
diff changeset
256 // These entries are the same as above but created by relocations which
anatofuz
parents:
diff changeset
257 // address GOT using 32-bit index (R_MIPS_GOT_HI16/LO16 etc).
anatofuz
parents:
diff changeset
258 // Normal global entries:
anatofuz
parents:
diff changeset
259 // These entries created by GOT relocations against preemptible global
anatofuz
parents:
diff changeset
260 // symbols. They need to be initialized by dynamic linker and they ordered
anatofuz
parents:
diff changeset
261 // exactly as the corresponding entries in the dynamic symbols table.
anatofuz
parents:
diff changeset
262 // Reloc-only global entries:
anatofuz
parents:
diff changeset
263 // These entries created for symbols that are referenced by dynamic
anatofuz
parents:
diff changeset
264 // relocations R_MIPS_REL32. These entries are not accessed with gp-relative
anatofuz
parents:
diff changeset
265 // addressing, but MIPS ABI requires that these entries be present in GOT.
anatofuz
parents:
diff changeset
266 // TLS entries:
anatofuz
parents:
diff changeset
267 // Entries created by TLS relocations.
anatofuz
parents:
diff changeset
268 //
anatofuz
parents:
diff changeset
269 // If the sum of local, global and tls entries is less than 64K only single
anatofuz
parents:
diff changeset
270 // got is enough. Otherwise, multi-got is created. Series of primary and
anatofuz
parents:
diff changeset
271 // multiple secondary GOTs have the following layout:
anatofuz
parents:
diff changeset
272 // - Primary GOT
anatofuz
parents:
diff changeset
273 // Header
anatofuz
parents:
diff changeset
274 // Local entries
anatofuz
parents:
diff changeset
275 // Global entries
anatofuz
parents:
diff changeset
276 // Relocation only entries
anatofuz
parents:
diff changeset
277 // TLS entries
anatofuz
parents:
diff changeset
278 //
anatofuz
parents:
diff changeset
279 // - Secondary GOT
anatofuz
parents:
diff changeset
280 // Local entries
anatofuz
parents:
diff changeset
281 // Global entries
anatofuz
parents:
diff changeset
282 // TLS entries
anatofuz
parents:
diff changeset
283 // ...
anatofuz
parents:
diff changeset
284 //
anatofuz
parents:
diff changeset
285 // All GOT entries required by relocations from a single input file entirely
anatofuz
parents:
diff changeset
286 // belong to either primary or one of secondary GOTs. To reference GOT entries
anatofuz
parents:
diff changeset
287 // each GOT has its own _gp value points to the "middle" of the GOT.
anatofuz
parents:
diff changeset
288 // In the code this value loaded to the register which is used for GOT access.
anatofuz
parents:
diff changeset
289 //
anatofuz
parents:
diff changeset
290 // MIPS 32 function's prologue:
anatofuz
parents:
diff changeset
291 // lui v0,0x0
anatofuz
parents:
diff changeset
292 // 0: R_MIPS_HI16 _gp_disp
anatofuz
parents:
diff changeset
293 // addiu v0,v0,0
anatofuz
parents:
diff changeset
294 // 4: R_MIPS_LO16 _gp_disp
anatofuz
parents:
diff changeset
295 //
anatofuz
parents:
diff changeset
296 // MIPS 64:
anatofuz
parents:
diff changeset
297 // lui at,0x0
anatofuz
parents:
diff changeset
298 // 14: R_MIPS_GPREL16 main
anatofuz
parents:
diff changeset
299 //
anatofuz
parents:
diff changeset
300 // Dynamic linker does not know anything about secondary GOTs and cannot
anatofuz
parents:
diff changeset
301 // use a regular MIPS mechanism for GOT entries initialization. So we have
anatofuz
parents:
diff changeset
302 // to use an approach accepted by other architectures and create dynamic
anatofuz
parents:
diff changeset
303 // relocations R_MIPS_REL32 to initialize global entries (and local in case
anatofuz
parents:
diff changeset
304 // of PIC code) in secondary GOTs. But ironically MIPS dynamic linker
anatofuz
parents:
diff changeset
305 // requires GOT entries and correspondingly ordered dynamic symbol table
anatofuz
parents:
diff changeset
306 // entries to deal with dynamic relocations. To handle this problem
anatofuz
parents:
diff changeset
307 // relocation-only section in the primary GOT contains entries for all
anatofuz
parents:
diff changeset
308 // symbols referenced in global parts of secondary GOTs. Although the sum
anatofuz
parents:
diff changeset
309 // of local and normal global entries of the primary got should be less
anatofuz
parents:
diff changeset
310 // than 64K, the size of the primary got (including relocation-only entries
anatofuz
parents:
diff changeset
311 // can be greater than 64K, because parts of the primary got that overflow
anatofuz
parents:
diff changeset
312 // the 64K limit are used only by the dynamic linker at dynamic link-time
anatofuz
parents:
diff changeset
313 // and not by 16-bit gp-relative addressing at run-time.
anatofuz
parents:
diff changeset
314 //
anatofuz
parents:
diff changeset
315 // For complete multi-GOT description see the following link
anatofuz
parents:
diff changeset
316 // https://dmz-portal.mips.com/wiki/MIPS_Multi_GOT
anatofuz
parents:
diff changeset
317
anatofuz
parents:
diff changeset
318 // Number of "Header" entries.
anatofuz
parents:
diff changeset
319 static const unsigned headerEntriesNum = 2;
anatofuz
parents:
diff changeset
320
anatofuz
parents:
diff changeset
321 uint64_t size = 0;
anatofuz
parents:
diff changeset
322
anatofuz
parents:
diff changeset
323 // Symbol and addend.
anatofuz
parents:
diff changeset
324 using GotEntry = std::pair<Symbol *, int64_t>;
anatofuz
parents:
diff changeset
325
anatofuz
parents:
diff changeset
326 struct FileGot {
anatofuz
parents:
diff changeset
327 InputFile *file = nullptr;
anatofuz
parents:
diff changeset
328 size_t startIndex = 0;
anatofuz
parents:
diff changeset
329
anatofuz
parents:
diff changeset
330 struct PageBlock {
anatofuz
parents:
diff changeset
331 size_t firstIndex;
anatofuz
parents:
diff changeset
332 size_t count;
anatofuz
parents:
diff changeset
333 PageBlock() : firstIndex(0), count(0) {}
anatofuz
parents:
diff changeset
334 };
anatofuz
parents:
diff changeset
335
anatofuz
parents:
diff changeset
336 // Map output sections referenced by MIPS GOT relocations
anatofuz
parents:
diff changeset
337 // to the description (index/count) "page" entries allocated
anatofuz
parents:
diff changeset
338 // for this section.
anatofuz
parents:
diff changeset
339 llvm::SmallMapVector<const OutputSection *, PageBlock, 16> pagesMap;
anatofuz
parents:
diff changeset
340 // Maps from Symbol+Addend pair or just Symbol to the GOT entry index.
anatofuz
parents:
diff changeset
341 llvm::MapVector<GotEntry, size_t> local16;
anatofuz
parents:
diff changeset
342 llvm::MapVector<GotEntry, size_t> local32;
anatofuz
parents:
diff changeset
343 llvm::MapVector<Symbol *, size_t> global;
anatofuz
parents:
diff changeset
344 llvm::MapVector<Symbol *, size_t> relocs;
anatofuz
parents:
diff changeset
345 llvm::MapVector<Symbol *, size_t> tls;
anatofuz
parents:
diff changeset
346 // Set of symbols referenced by dynamic TLS relocations.
anatofuz
parents:
diff changeset
347 llvm::MapVector<Symbol *, size_t> dynTlsSymbols;
anatofuz
parents:
diff changeset
348
anatofuz
parents:
diff changeset
349 // Total number of all entries.
anatofuz
parents:
diff changeset
350 size_t getEntriesNum() const;
anatofuz
parents:
diff changeset
351 // Number of "page" entries.
anatofuz
parents:
diff changeset
352 size_t getPageEntriesNum() const;
anatofuz
parents:
diff changeset
353 // Number of entries require 16-bit index to access.
anatofuz
parents:
diff changeset
354 size_t getIndexedEntriesNum() const;
anatofuz
parents:
diff changeset
355 };
anatofuz
parents:
diff changeset
356
anatofuz
parents:
diff changeset
357 // Container of GOT created for each input file.
anatofuz
parents:
diff changeset
358 // After building a final series of GOTs this container
anatofuz
parents:
diff changeset
359 // holds primary and secondary GOT's.
anatofuz
parents:
diff changeset
360 std::vector<FileGot> gots;
anatofuz
parents:
diff changeset
361
anatofuz
parents:
diff changeset
362 // Return (and create if necessary) `FileGot`.
anatofuz
parents:
diff changeset
363 FileGot &getGot(InputFile &f);
anatofuz
parents:
diff changeset
364
anatofuz
parents:
diff changeset
365 // Try to merge two GOTs. In case of success the `Dst` contains
anatofuz
parents:
diff changeset
366 // result of merging and the function returns true. In case of
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
367 // overflow the `Dst` is unchanged and the function returns false.
150
anatofuz
parents:
diff changeset
368 bool tryMergeGots(FileGot & dst, FileGot & src, bool isPrimary);
anatofuz
parents:
diff changeset
369 };
anatofuz
parents:
diff changeset
370
anatofuz
parents:
diff changeset
371 class GotPltSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
372 public:
anatofuz
parents:
diff changeset
373 GotPltSection();
anatofuz
parents:
diff changeset
374 void addEntry(Symbol &sym);
anatofuz
parents:
diff changeset
375 size_t getSize() const override;
anatofuz
parents:
diff changeset
376 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
377 bool isNeeded() const override;
anatofuz
parents:
diff changeset
378
anatofuz
parents:
diff changeset
379 // Flag to force GotPlt to be in output if we have relocations
anatofuz
parents:
diff changeset
380 // that relies on its address.
anatofuz
parents:
diff changeset
381 bool hasGotPltOffRel = false;
anatofuz
parents:
diff changeset
382
anatofuz
parents:
diff changeset
383 private:
anatofuz
parents:
diff changeset
384 std::vector<const Symbol *> entries;
anatofuz
parents:
diff changeset
385 };
anatofuz
parents:
diff changeset
386
anatofuz
parents:
diff changeset
387 // The IgotPltSection is a Got associated with the PltSection for GNU Ifunc
anatofuz
parents:
diff changeset
388 // Symbols that will be relocated by Target->IRelativeRel.
anatofuz
parents:
diff changeset
389 // On most Targets the IgotPltSection will immediately follow the GotPltSection
anatofuz
parents:
diff changeset
390 // on ARM the IgotPltSection will immediately follow the GotSection.
anatofuz
parents:
diff changeset
391 class IgotPltSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
392 public:
anatofuz
parents:
diff changeset
393 IgotPltSection();
anatofuz
parents:
diff changeset
394 void addEntry(Symbol &sym);
anatofuz
parents:
diff changeset
395 size_t getSize() const override;
anatofuz
parents:
diff changeset
396 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
397 bool isNeeded() const override { return !entries.empty(); }
anatofuz
parents:
diff changeset
398
anatofuz
parents:
diff changeset
399 private:
anatofuz
parents:
diff changeset
400 std::vector<const Symbol *> entries;
anatofuz
parents:
diff changeset
401 };
anatofuz
parents:
diff changeset
402
anatofuz
parents:
diff changeset
403 class StringTableSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
404 public:
anatofuz
parents:
diff changeset
405 StringTableSection(StringRef name, bool dynamic);
anatofuz
parents:
diff changeset
406 unsigned addString(StringRef s, bool hashIt = true);
anatofuz
parents:
diff changeset
407 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
408 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
409 bool isDynamic() const { return dynamic; }
anatofuz
parents:
diff changeset
410
anatofuz
parents:
diff changeset
411 private:
anatofuz
parents:
diff changeset
412 const bool dynamic;
anatofuz
parents:
diff changeset
413
anatofuz
parents:
diff changeset
414 uint64_t size = 0;
anatofuz
parents:
diff changeset
415
anatofuz
parents:
diff changeset
416 llvm::DenseMap<StringRef, unsigned> stringMap;
anatofuz
parents:
diff changeset
417 std::vector<StringRef> strings;
anatofuz
parents:
diff changeset
418 };
anatofuz
parents:
diff changeset
419
anatofuz
parents:
diff changeset
420 class DynamicReloc {
anatofuz
parents:
diff changeset
421 public:
anatofuz
parents:
diff changeset
422 DynamicReloc(RelType type, const InputSectionBase *inputSec,
anatofuz
parents:
diff changeset
423 uint64_t offsetInSec, bool useSymVA, Symbol *sym, int64_t addend)
anatofuz
parents:
diff changeset
424 : type(type), sym(sym), inputSec(inputSec), offsetInSec(offsetInSec),
anatofuz
parents:
diff changeset
425 useSymVA(useSymVA), addend(addend), outputSec(nullptr) {}
anatofuz
parents:
diff changeset
426 // This constructor records dynamic relocation settings used by MIPS
anatofuz
parents:
diff changeset
427 // multi-GOT implementation. It's to relocate addresses of 64kb pages
anatofuz
parents:
diff changeset
428 // lie inside the output section.
anatofuz
parents:
diff changeset
429 DynamicReloc(RelType type, const InputSectionBase *inputSec,
anatofuz
parents:
diff changeset
430 uint64_t offsetInSec, const OutputSection *outputSec,
anatofuz
parents:
diff changeset
431 int64_t addend)
anatofuz
parents:
diff changeset
432 : type(type), sym(nullptr), inputSec(inputSec), offsetInSec(offsetInSec),
anatofuz
parents:
diff changeset
433 useSymVA(false), addend(addend), outputSec(outputSec) {}
anatofuz
parents:
diff changeset
434
anatofuz
parents:
diff changeset
435 uint64_t getOffset() const;
anatofuz
parents:
diff changeset
436 uint32_t getSymIndex(SymbolTableBaseSection *symTab) const;
anatofuz
parents:
diff changeset
437
anatofuz
parents:
diff changeset
438 // Computes the addend of the dynamic relocation. Note that this is not the
anatofuz
parents:
diff changeset
439 // same as the addend member variable as it also includes the symbol address
anatofuz
parents:
diff changeset
440 // if useSymVA is true.
anatofuz
parents:
diff changeset
441 int64_t computeAddend() const;
anatofuz
parents:
diff changeset
442
anatofuz
parents:
diff changeset
443 RelType type;
anatofuz
parents:
diff changeset
444
anatofuz
parents:
diff changeset
445 Symbol *sym;
anatofuz
parents:
diff changeset
446 const InputSectionBase *inputSec = nullptr;
anatofuz
parents:
diff changeset
447 uint64_t offsetInSec;
anatofuz
parents:
diff changeset
448 // If this member is true, the dynamic relocation will not be against the
anatofuz
parents:
diff changeset
449 // symbol but will instead be a relative relocation that simply adds the
anatofuz
parents:
diff changeset
450 // load address. This means we need to write the symbol virtual address
anatofuz
parents:
diff changeset
451 // plus the original addend as the final relocation addend.
anatofuz
parents:
diff changeset
452 bool useSymVA;
anatofuz
parents:
diff changeset
453 int64_t addend;
anatofuz
parents:
diff changeset
454 const OutputSection *outputSec;
anatofuz
parents:
diff changeset
455 };
anatofuz
parents:
diff changeset
456
anatofuz
parents:
diff changeset
457 template <class ELFT> class DynamicSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
458 using Elf_Dyn = typename ELFT::Dyn;
anatofuz
parents:
diff changeset
459 using Elf_Rel = typename ELFT::Rel;
anatofuz
parents:
diff changeset
460 using Elf_Rela = typename ELFT::Rela;
anatofuz
parents:
diff changeset
461 using Elf_Relr = typename ELFT::Relr;
anatofuz
parents:
diff changeset
462 using Elf_Shdr = typename ELFT::Shdr;
anatofuz
parents:
diff changeset
463 using Elf_Sym = typename ELFT::Sym;
anatofuz
parents:
diff changeset
464
anatofuz
parents:
diff changeset
465 // finalizeContents() fills this vector with the section contents.
anatofuz
parents:
diff changeset
466 std::vector<std::pair<int32_t, std::function<uint64_t()>>> entries;
anatofuz
parents:
diff changeset
467
anatofuz
parents:
diff changeset
468 public:
anatofuz
parents:
diff changeset
469 DynamicSection();
anatofuz
parents:
diff changeset
470 void finalizeContents() override;
anatofuz
parents:
diff changeset
471 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
472 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
473
anatofuz
parents:
diff changeset
474 private:
anatofuz
parents:
diff changeset
475 void add(int32_t tag, std::function<uint64_t()> fn);
anatofuz
parents:
diff changeset
476 void addInt(int32_t tag, uint64_t val);
anatofuz
parents:
diff changeset
477 void addInSec(int32_t tag, InputSection *sec);
anatofuz
parents:
diff changeset
478 void addInSecRelative(int32_t tag, InputSection *sec);
anatofuz
parents:
diff changeset
479 void addOutSec(int32_t tag, OutputSection *sec);
anatofuz
parents:
diff changeset
480 void addSize(int32_t tag, OutputSection *sec);
anatofuz
parents:
diff changeset
481 void addSym(int32_t tag, Symbol *sym);
anatofuz
parents:
diff changeset
482
anatofuz
parents:
diff changeset
483 uint64_t size = 0;
anatofuz
parents:
diff changeset
484 };
anatofuz
parents:
diff changeset
485
anatofuz
parents:
diff changeset
486 class RelocationBaseSection : public SyntheticSection {
anatofuz
parents:
diff changeset
487 public:
anatofuz
parents:
diff changeset
488 RelocationBaseSection(StringRef name, uint32_t type, int32_t dynamicTag,
anatofuz
parents:
diff changeset
489 int32_t sizeDynamicTag);
anatofuz
parents:
diff changeset
490 void addReloc(RelType dynType, InputSectionBase *isec, uint64_t offsetInSec,
anatofuz
parents:
diff changeset
491 Symbol *sym);
anatofuz
parents:
diff changeset
492 // Add a dynamic relocation that might need an addend. This takes care of
anatofuz
parents:
diff changeset
493 // writing the addend to the output section if needed.
anatofuz
parents:
diff changeset
494 void addReloc(RelType dynType, InputSectionBase *inputSec,
anatofuz
parents:
diff changeset
495 uint64_t offsetInSec, Symbol *sym, int64_t addend, RelExpr expr,
anatofuz
parents:
diff changeset
496 RelType type);
anatofuz
parents:
diff changeset
497 void addReloc(const DynamicReloc &reloc);
anatofuz
parents:
diff changeset
498 bool isNeeded() const override { return !relocs.empty(); }
anatofuz
parents:
diff changeset
499 size_t getSize() const override { return relocs.size() * this->entsize; }
anatofuz
parents:
diff changeset
500 size_t getRelativeRelocCount() const { return numRelativeRelocs; }
anatofuz
parents:
diff changeset
501 void finalizeContents() override;
anatofuz
parents:
diff changeset
502 int32_t dynamicTag, sizeDynamicTag;
anatofuz
parents:
diff changeset
503 std::vector<DynamicReloc> relocs;
anatofuz
parents:
diff changeset
504
anatofuz
parents:
diff changeset
505 protected:
anatofuz
parents:
diff changeset
506 size_t numRelativeRelocs = 0;
anatofuz
parents:
diff changeset
507 };
anatofuz
parents:
diff changeset
508
anatofuz
parents:
diff changeset
509 template <class ELFT>
anatofuz
parents:
diff changeset
510 class RelocationSection final : public RelocationBaseSection {
anatofuz
parents:
diff changeset
511 using Elf_Rel = typename ELFT::Rel;
anatofuz
parents:
diff changeset
512 using Elf_Rela = typename ELFT::Rela;
anatofuz
parents:
diff changeset
513
anatofuz
parents:
diff changeset
514 public:
anatofuz
parents:
diff changeset
515 RelocationSection(StringRef name, bool sort);
anatofuz
parents:
diff changeset
516 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
517
anatofuz
parents:
diff changeset
518 private:
anatofuz
parents:
diff changeset
519 bool sort;
anatofuz
parents:
diff changeset
520 };
anatofuz
parents:
diff changeset
521
anatofuz
parents:
diff changeset
522 template <class ELFT>
anatofuz
parents:
diff changeset
523 class AndroidPackedRelocationSection final : public RelocationBaseSection {
anatofuz
parents:
diff changeset
524 using Elf_Rel = typename ELFT::Rel;
anatofuz
parents:
diff changeset
525 using Elf_Rela = typename ELFT::Rela;
anatofuz
parents:
diff changeset
526
anatofuz
parents:
diff changeset
527 public:
anatofuz
parents:
diff changeset
528 AndroidPackedRelocationSection(StringRef name);
anatofuz
parents:
diff changeset
529
anatofuz
parents:
diff changeset
530 bool updateAllocSize() override;
anatofuz
parents:
diff changeset
531 size_t getSize() const override { return relocData.size(); }
anatofuz
parents:
diff changeset
532 void writeTo(uint8_t *buf) override {
anatofuz
parents:
diff changeset
533 memcpy(buf, relocData.data(), relocData.size());
anatofuz
parents:
diff changeset
534 }
anatofuz
parents:
diff changeset
535
anatofuz
parents:
diff changeset
536 private:
anatofuz
parents:
diff changeset
537 SmallVector<char, 0> relocData;
anatofuz
parents:
diff changeset
538 };
anatofuz
parents:
diff changeset
539
anatofuz
parents:
diff changeset
540 struct RelativeReloc {
anatofuz
parents:
diff changeset
541 uint64_t getOffset() const { return inputSec->getVA(offsetInSec); }
anatofuz
parents:
diff changeset
542
anatofuz
parents:
diff changeset
543 const InputSectionBase *inputSec;
anatofuz
parents:
diff changeset
544 uint64_t offsetInSec;
anatofuz
parents:
diff changeset
545 };
anatofuz
parents:
diff changeset
546
anatofuz
parents:
diff changeset
547 class RelrBaseSection : public SyntheticSection {
anatofuz
parents:
diff changeset
548 public:
anatofuz
parents:
diff changeset
549 RelrBaseSection();
anatofuz
parents:
diff changeset
550 bool isNeeded() const override { return !relocs.empty(); }
anatofuz
parents:
diff changeset
551 std::vector<RelativeReloc> relocs;
anatofuz
parents:
diff changeset
552 };
anatofuz
parents:
diff changeset
553
anatofuz
parents:
diff changeset
554 // RelrSection is used to encode offsets for relative relocations.
anatofuz
parents:
diff changeset
555 // Proposal for adding SHT_RELR sections to generic-abi is here:
anatofuz
parents:
diff changeset
556 // https://groups.google.com/forum/#!topic/generic-abi/bX460iggiKg
anatofuz
parents:
diff changeset
557 // For more details, see the comment in RelrSection::updateAllocSize().
anatofuz
parents:
diff changeset
558 template <class ELFT> class RelrSection final : public RelrBaseSection {
anatofuz
parents:
diff changeset
559 using Elf_Relr = typename ELFT::Relr;
anatofuz
parents:
diff changeset
560
anatofuz
parents:
diff changeset
561 public:
anatofuz
parents:
diff changeset
562 RelrSection();
anatofuz
parents:
diff changeset
563
anatofuz
parents:
diff changeset
564 bool updateAllocSize() override;
anatofuz
parents:
diff changeset
565 size_t getSize() const override { return relrRelocs.size() * this->entsize; }
anatofuz
parents:
diff changeset
566 void writeTo(uint8_t *buf) override {
anatofuz
parents:
diff changeset
567 memcpy(buf, relrRelocs.data(), getSize());
anatofuz
parents:
diff changeset
568 }
anatofuz
parents:
diff changeset
569
anatofuz
parents:
diff changeset
570 private:
anatofuz
parents:
diff changeset
571 std::vector<Elf_Relr> relrRelocs;
anatofuz
parents:
diff changeset
572 };
anatofuz
parents:
diff changeset
573
anatofuz
parents:
diff changeset
574 struct SymbolTableEntry {
anatofuz
parents:
diff changeset
575 Symbol *sym;
anatofuz
parents:
diff changeset
576 size_t strTabOffset;
anatofuz
parents:
diff changeset
577 };
anatofuz
parents:
diff changeset
578
anatofuz
parents:
diff changeset
579 class SymbolTableBaseSection : public SyntheticSection {
anatofuz
parents:
diff changeset
580 public:
anatofuz
parents:
diff changeset
581 SymbolTableBaseSection(StringTableSection &strTabSec);
anatofuz
parents:
diff changeset
582 void finalizeContents() override;
anatofuz
parents:
diff changeset
583 size_t getSize() const override { return getNumSymbols() * entsize; }
anatofuz
parents:
diff changeset
584 void addSymbol(Symbol *sym);
anatofuz
parents:
diff changeset
585 unsigned getNumSymbols() const { return symbols.size() + 1; }
anatofuz
parents:
diff changeset
586 size_t getSymbolIndex(Symbol *sym);
anatofuz
parents:
diff changeset
587 ArrayRef<SymbolTableEntry> getSymbols() const { return symbols; }
anatofuz
parents:
diff changeset
588
anatofuz
parents:
diff changeset
589 protected:
anatofuz
parents:
diff changeset
590 void sortSymTabSymbols();
anatofuz
parents:
diff changeset
591
anatofuz
parents:
diff changeset
592 // A vector of symbols and their string table offsets.
anatofuz
parents:
diff changeset
593 std::vector<SymbolTableEntry> symbols;
anatofuz
parents:
diff changeset
594
anatofuz
parents:
diff changeset
595 StringTableSection &strTabSec;
anatofuz
parents:
diff changeset
596
anatofuz
parents:
diff changeset
597 llvm::once_flag onceFlag;
anatofuz
parents:
diff changeset
598 llvm::DenseMap<Symbol *, size_t> symbolIndexMap;
anatofuz
parents:
diff changeset
599 llvm::DenseMap<OutputSection *, size_t> sectionIndexMap;
anatofuz
parents:
diff changeset
600 };
anatofuz
parents:
diff changeset
601
anatofuz
parents:
diff changeset
602 template <class ELFT>
anatofuz
parents:
diff changeset
603 class SymbolTableSection final : public SymbolTableBaseSection {
anatofuz
parents:
diff changeset
604 using Elf_Sym = typename ELFT::Sym;
anatofuz
parents:
diff changeset
605
anatofuz
parents:
diff changeset
606 public:
anatofuz
parents:
diff changeset
607 SymbolTableSection(StringTableSection &strTabSec);
anatofuz
parents:
diff changeset
608 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
609 };
anatofuz
parents:
diff changeset
610
anatofuz
parents:
diff changeset
611 class SymtabShndxSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
612 public:
anatofuz
parents:
diff changeset
613 SymtabShndxSection();
anatofuz
parents:
diff changeset
614
anatofuz
parents:
diff changeset
615 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
616 size_t getSize() const override;
anatofuz
parents:
diff changeset
617 bool isNeeded() const override;
anatofuz
parents:
diff changeset
618 void finalizeContents() override;
anatofuz
parents:
diff changeset
619 };
anatofuz
parents:
diff changeset
620
anatofuz
parents:
diff changeset
621 // Outputs GNU Hash section. For detailed explanation see:
anatofuz
parents:
diff changeset
622 // https://blogs.oracle.com/ali/entry/gnu_hash_elf_sections
anatofuz
parents:
diff changeset
623 class GnuHashTableSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
624 public:
anatofuz
parents:
diff changeset
625 GnuHashTableSection();
anatofuz
parents:
diff changeset
626 void finalizeContents() override;
anatofuz
parents:
diff changeset
627 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
628 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
629
anatofuz
parents:
diff changeset
630 // Adds symbols to the hash table.
anatofuz
parents:
diff changeset
631 // Sorts the input to satisfy GNU hash section requirements.
anatofuz
parents:
diff changeset
632 void addSymbols(std::vector<SymbolTableEntry> &symbols);
anatofuz
parents:
diff changeset
633
anatofuz
parents:
diff changeset
634 private:
anatofuz
parents:
diff changeset
635 // See the comment in writeBloomFilter.
anatofuz
parents:
diff changeset
636 enum { Shift2 = 26 };
anatofuz
parents:
diff changeset
637
anatofuz
parents:
diff changeset
638 void writeBloomFilter(uint8_t *buf);
anatofuz
parents:
diff changeset
639 void writeHashTable(uint8_t *buf);
anatofuz
parents:
diff changeset
640
anatofuz
parents:
diff changeset
641 struct Entry {
anatofuz
parents:
diff changeset
642 Symbol *sym;
anatofuz
parents:
diff changeset
643 size_t strTabOffset;
anatofuz
parents:
diff changeset
644 uint32_t hash;
anatofuz
parents:
diff changeset
645 uint32_t bucketIdx;
anatofuz
parents:
diff changeset
646 };
anatofuz
parents:
diff changeset
647
anatofuz
parents:
diff changeset
648 std::vector<Entry> symbols;
anatofuz
parents:
diff changeset
649 size_t maskWords;
anatofuz
parents:
diff changeset
650 size_t nBuckets = 0;
anatofuz
parents:
diff changeset
651 size_t size = 0;
anatofuz
parents:
diff changeset
652 };
anatofuz
parents:
diff changeset
653
anatofuz
parents:
diff changeset
654 class HashTableSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
655 public:
anatofuz
parents:
diff changeset
656 HashTableSection();
anatofuz
parents:
diff changeset
657 void finalizeContents() override;
anatofuz
parents:
diff changeset
658 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
659 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
660
anatofuz
parents:
diff changeset
661 private:
anatofuz
parents:
diff changeset
662 size_t size = 0;
anatofuz
parents:
diff changeset
663 };
anatofuz
parents:
diff changeset
664
anatofuz
parents:
diff changeset
665 // Used for PLT entries. It usually has a PLT header for lazy binding. Each PLT
anatofuz
parents:
diff changeset
666 // entry is associated with a JUMP_SLOT relocation, which may be resolved lazily
anatofuz
parents:
diff changeset
667 // at runtime.
anatofuz
parents:
diff changeset
668 //
anatofuz
parents:
diff changeset
669 // On PowerPC, this section contains lazy symbol resolvers. A branch instruction
anatofuz
parents:
diff changeset
670 // jumps to a PLT call stub, which will then jump to the target (BIND_NOW) or a
anatofuz
parents:
diff changeset
671 // lazy symbol resolver.
anatofuz
parents:
diff changeset
672 //
anatofuz
parents:
diff changeset
673 // On x86 when IBT is enabled, this section (.plt.sec) contains PLT call stubs.
anatofuz
parents:
diff changeset
674 // A call instruction jumps to a .plt.sec entry, which will then jump to the
anatofuz
parents:
diff changeset
675 // target (BIND_NOW) or a .plt entry.
anatofuz
parents:
diff changeset
676 class PltSection : public SyntheticSection {
anatofuz
parents:
diff changeset
677 public:
anatofuz
parents:
diff changeset
678 PltSection();
anatofuz
parents:
diff changeset
679 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
680 size_t getSize() const override;
anatofuz
parents:
diff changeset
681 bool isNeeded() const override;
anatofuz
parents:
diff changeset
682 void addSymbols();
anatofuz
parents:
diff changeset
683 void addEntry(Symbol &sym);
anatofuz
parents:
diff changeset
684 size_t getNumEntries() const { return entries.size(); }
anatofuz
parents:
diff changeset
685
anatofuz
parents:
diff changeset
686 size_t headerSize;
anatofuz
parents:
diff changeset
687
anatofuz
parents:
diff changeset
688 std::vector<const Symbol *> entries;
anatofuz
parents:
diff changeset
689 };
anatofuz
parents:
diff changeset
690
anatofuz
parents:
diff changeset
691 // Used for non-preemptible ifuncs. It does not have a header. Each entry is
anatofuz
parents:
diff changeset
692 // associated with an IRELATIVE relocation, which will be resolved eagerly at
anatofuz
parents:
diff changeset
693 // runtime. PltSection can only contain entries associated with JUMP_SLOT
anatofuz
parents:
diff changeset
694 // relocations, so IPLT entries are in a separate section.
anatofuz
parents:
diff changeset
695 class IpltSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
696 std::vector<const Symbol *> entries;
anatofuz
parents:
diff changeset
697
anatofuz
parents:
diff changeset
698 public:
anatofuz
parents:
diff changeset
699 IpltSection();
anatofuz
parents:
diff changeset
700 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
701 size_t getSize() const override;
anatofuz
parents:
diff changeset
702 bool isNeeded() const override { return !entries.empty(); }
anatofuz
parents:
diff changeset
703 void addSymbols();
anatofuz
parents:
diff changeset
704 void addEntry(Symbol &sym);
anatofuz
parents:
diff changeset
705 };
anatofuz
parents:
diff changeset
706
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
707 class PPC32GlinkSection : public PltSection {
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
708 public:
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
709 PPC32GlinkSection();
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
710 void writeTo(uint8_t *buf) override;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
711 size_t getSize() const override;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
712
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
713 std::vector<const Symbol *> canonical_plts;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
714 static constexpr size_t footerSize = 64;
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
715 };
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
716
150
anatofuz
parents:
diff changeset
717 // This is x86-only.
anatofuz
parents:
diff changeset
718 class IBTPltSection : public SyntheticSection {
anatofuz
parents:
diff changeset
719 public:
anatofuz
parents:
diff changeset
720 IBTPltSection();
anatofuz
parents:
diff changeset
721 void writeTo(uint8_t *Buf) override;
anatofuz
parents:
diff changeset
722 size_t getSize() const override;
anatofuz
parents:
diff changeset
723 };
anatofuz
parents:
diff changeset
724
anatofuz
parents:
diff changeset
725 class GdbIndexSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
726 public:
anatofuz
parents:
diff changeset
727 struct AddressEntry {
anatofuz
parents:
diff changeset
728 InputSection *section;
anatofuz
parents:
diff changeset
729 uint64_t lowAddress;
anatofuz
parents:
diff changeset
730 uint64_t highAddress;
anatofuz
parents:
diff changeset
731 uint32_t cuIndex;
anatofuz
parents:
diff changeset
732 };
anatofuz
parents:
diff changeset
733
anatofuz
parents:
diff changeset
734 struct CuEntry {
anatofuz
parents:
diff changeset
735 uint64_t cuOffset;
anatofuz
parents:
diff changeset
736 uint64_t cuLength;
anatofuz
parents:
diff changeset
737 };
anatofuz
parents:
diff changeset
738
anatofuz
parents:
diff changeset
739 struct NameAttrEntry {
anatofuz
parents:
diff changeset
740 llvm::CachedHashStringRef name;
anatofuz
parents:
diff changeset
741 uint32_t cuIndexAndAttrs;
anatofuz
parents:
diff changeset
742 };
anatofuz
parents:
diff changeset
743
anatofuz
parents:
diff changeset
744 struct GdbChunk {
anatofuz
parents:
diff changeset
745 InputSection *sec;
anatofuz
parents:
diff changeset
746 std::vector<AddressEntry> addressAreas;
anatofuz
parents:
diff changeset
747 std::vector<CuEntry> compilationUnits;
anatofuz
parents:
diff changeset
748 };
anatofuz
parents:
diff changeset
749
anatofuz
parents:
diff changeset
750 struct GdbSymbol {
anatofuz
parents:
diff changeset
751 llvm::CachedHashStringRef name;
anatofuz
parents:
diff changeset
752 std::vector<uint32_t> cuVector;
anatofuz
parents:
diff changeset
753 uint32_t nameOff;
anatofuz
parents:
diff changeset
754 uint32_t cuVectorOff;
anatofuz
parents:
diff changeset
755 };
anatofuz
parents:
diff changeset
756
anatofuz
parents:
diff changeset
757 GdbIndexSection();
anatofuz
parents:
diff changeset
758 template <typename ELFT> static GdbIndexSection *create();
anatofuz
parents:
diff changeset
759 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
760 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
761 bool isNeeded() const override;
anatofuz
parents:
diff changeset
762
anatofuz
parents:
diff changeset
763 private:
anatofuz
parents:
diff changeset
764 struct GdbIndexHeader {
anatofuz
parents:
diff changeset
765 llvm::support::ulittle32_t version;
anatofuz
parents:
diff changeset
766 llvm::support::ulittle32_t cuListOff;
anatofuz
parents:
diff changeset
767 llvm::support::ulittle32_t cuTypesOff;
anatofuz
parents:
diff changeset
768 llvm::support::ulittle32_t addressAreaOff;
anatofuz
parents:
diff changeset
769 llvm::support::ulittle32_t symtabOff;
anatofuz
parents:
diff changeset
770 llvm::support::ulittle32_t constantPoolOff;
anatofuz
parents:
diff changeset
771 };
anatofuz
parents:
diff changeset
772
anatofuz
parents:
diff changeset
773 void initOutputSize();
anatofuz
parents:
diff changeset
774 size_t computeSymtabSize() const;
anatofuz
parents:
diff changeset
775
anatofuz
parents:
diff changeset
776 // Each chunk contains information gathered from debug sections of a
anatofuz
parents:
diff changeset
777 // single object file.
anatofuz
parents:
diff changeset
778 std::vector<GdbChunk> chunks;
anatofuz
parents:
diff changeset
779
anatofuz
parents:
diff changeset
780 // A symbol table for this .gdb_index section.
anatofuz
parents:
diff changeset
781 std::vector<GdbSymbol> symbols;
anatofuz
parents:
diff changeset
782
anatofuz
parents:
diff changeset
783 size_t size;
anatofuz
parents:
diff changeset
784 };
anatofuz
parents:
diff changeset
785
anatofuz
parents:
diff changeset
786 // --eh-frame-hdr option tells linker to construct a header for all the
anatofuz
parents:
diff changeset
787 // .eh_frame sections. This header is placed to a section named .eh_frame_hdr
anatofuz
parents:
diff changeset
788 // and also to a PT_GNU_EH_FRAME segment.
anatofuz
parents:
diff changeset
789 // At runtime the unwinder then can find all the PT_GNU_EH_FRAME segments by
anatofuz
parents:
diff changeset
790 // calling dl_iterate_phdr.
anatofuz
parents:
diff changeset
791 // This section contains a lookup table for quick binary search of FDEs.
anatofuz
parents:
diff changeset
792 // Detailed info about internals can be found in Ian Lance Taylor's blog:
anatofuz
parents:
diff changeset
793 // http://www.airs.com/blog/archives/460 (".eh_frame")
anatofuz
parents:
diff changeset
794 // http://www.airs.com/blog/archives/462 (".eh_frame_hdr")
anatofuz
parents:
diff changeset
795 class EhFrameHeader final : public SyntheticSection {
anatofuz
parents:
diff changeset
796 public:
anatofuz
parents:
diff changeset
797 EhFrameHeader();
anatofuz
parents:
diff changeset
798 void write();
anatofuz
parents:
diff changeset
799 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
800 size_t getSize() const override;
anatofuz
parents:
diff changeset
801 bool isNeeded() const override;
anatofuz
parents:
diff changeset
802 };
anatofuz
parents:
diff changeset
803
anatofuz
parents:
diff changeset
804 // For more information about .gnu.version and .gnu.version_r see:
anatofuz
parents:
diff changeset
805 // https://www.akkadia.org/drepper/symbol-versioning
anatofuz
parents:
diff changeset
806
anatofuz
parents:
diff changeset
807 // The .gnu.version_d section which has a section type of SHT_GNU_verdef shall
anatofuz
parents:
diff changeset
808 // contain symbol version definitions. The number of entries in this section
anatofuz
parents:
diff changeset
809 // shall be contained in the DT_VERDEFNUM entry of the .dynamic section.
anatofuz
parents:
diff changeset
810 // The section shall contain an array of Elf_Verdef structures, optionally
anatofuz
parents:
diff changeset
811 // followed by an array of Elf_Verdaux structures.
anatofuz
parents:
diff changeset
812 class VersionDefinitionSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
813 public:
anatofuz
parents:
diff changeset
814 VersionDefinitionSection();
anatofuz
parents:
diff changeset
815 void finalizeContents() override;
anatofuz
parents:
diff changeset
816 size_t getSize() const override;
anatofuz
parents:
diff changeset
817 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
818
anatofuz
parents:
diff changeset
819 private:
anatofuz
parents:
diff changeset
820 enum { EntrySize = 28 };
anatofuz
parents:
diff changeset
821 void writeOne(uint8_t *buf, uint32_t index, StringRef name, size_t nameOff);
anatofuz
parents:
diff changeset
822 StringRef getFileDefName();
anatofuz
parents:
diff changeset
823
anatofuz
parents:
diff changeset
824 unsigned fileDefNameOff;
anatofuz
parents:
diff changeset
825 std::vector<unsigned> verDefNameOffs;
anatofuz
parents:
diff changeset
826 };
anatofuz
parents:
diff changeset
827
anatofuz
parents:
diff changeset
828 // The .gnu.version section specifies the required version of each symbol in the
anatofuz
parents:
diff changeset
829 // dynamic symbol table. It contains one Elf_Versym for each dynamic symbol
anatofuz
parents:
diff changeset
830 // table entry. An Elf_Versym is just a 16-bit integer that refers to a version
anatofuz
parents:
diff changeset
831 // identifier defined in the either .gnu.version_r or .gnu.version_d section.
anatofuz
parents:
diff changeset
832 // The values 0 and 1 are reserved. All other values are used for versions in
anatofuz
parents:
diff changeset
833 // the own object or in any of the dependencies.
anatofuz
parents:
diff changeset
834 class VersionTableSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
835 public:
anatofuz
parents:
diff changeset
836 VersionTableSection();
anatofuz
parents:
diff changeset
837 void finalizeContents() override;
anatofuz
parents:
diff changeset
838 size_t getSize() const override;
anatofuz
parents:
diff changeset
839 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
840 bool isNeeded() const override;
anatofuz
parents:
diff changeset
841 };
anatofuz
parents:
diff changeset
842
anatofuz
parents:
diff changeset
843 // The .gnu.version_r section defines the version identifiers used by
anatofuz
parents:
diff changeset
844 // .gnu.version. It contains a linked list of Elf_Verneed data structures. Each
anatofuz
parents:
diff changeset
845 // Elf_Verneed specifies the version requirements for a single DSO, and contains
anatofuz
parents:
diff changeset
846 // a reference to a linked list of Elf_Vernaux data structures which define the
anatofuz
parents:
diff changeset
847 // mapping from version identifiers to version names.
anatofuz
parents:
diff changeset
848 template <class ELFT>
anatofuz
parents:
diff changeset
849 class VersionNeedSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
850 using Elf_Verneed = typename ELFT::Verneed;
anatofuz
parents:
diff changeset
851 using Elf_Vernaux = typename ELFT::Vernaux;
anatofuz
parents:
diff changeset
852
anatofuz
parents:
diff changeset
853 struct Vernaux {
anatofuz
parents:
diff changeset
854 uint64_t hash;
anatofuz
parents:
diff changeset
855 uint32_t verneedIndex;
anatofuz
parents:
diff changeset
856 uint64_t nameStrTab;
anatofuz
parents:
diff changeset
857 };
anatofuz
parents:
diff changeset
858
anatofuz
parents:
diff changeset
859 struct Verneed {
anatofuz
parents:
diff changeset
860 uint64_t nameStrTab;
anatofuz
parents:
diff changeset
861 std::vector<Vernaux> vernauxs;
anatofuz
parents:
diff changeset
862 };
anatofuz
parents:
diff changeset
863
anatofuz
parents:
diff changeset
864 std::vector<Verneed> verneeds;
anatofuz
parents:
diff changeset
865
anatofuz
parents:
diff changeset
866 public:
anatofuz
parents:
diff changeset
867 VersionNeedSection();
anatofuz
parents:
diff changeset
868 void finalizeContents() override;
anatofuz
parents:
diff changeset
869 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
870 size_t getSize() const override;
anatofuz
parents:
diff changeset
871 bool isNeeded() const override;
anatofuz
parents:
diff changeset
872 };
anatofuz
parents:
diff changeset
873
anatofuz
parents:
diff changeset
874 // MergeSyntheticSection is a class that allows us to put mergeable sections
anatofuz
parents:
diff changeset
875 // with different attributes in a single output sections. To do that
anatofuz
parents:
diff changeset
876 // we put them into MergeSyntheticSection synthetic input sections which are
anatofuz
parents:
diff changeset
877 // attached to regular output sections.
anatofuz
parents:
diff changeset
878 class MergeSyntheticSection : public SyntheticSection {
anatofuz
parents:
diff changeset
879 public:
anatofuz
parents:
diff changeset
880 void addSection(MergeInputSection *ms);
anatofuz
parents:
diff changeset
881 std::vector<MergeInputSection *> sections;
anatofuz
parents:
diff changeset
882
anatofuz
parents:
diff changeset
883 protected:
anatofuz
parents:
diff changeset
884 MergeSyntheticSection(StringRef name, uint32_t type, uint64_t flags,
anatofuz
parents:
diff changeset
885 uint32_t alignment)
anatofuz
parents:
diff changeset
886 : SyntheticSection(flags, type, alignment, name) {}
anatofuz
parents:
diff changeset
887 };
anatofuz
parents:
diff changeset
888
anatofuz
parents:
diff changeset
889 class MergeTailSection final : public MergeSyntheticSection {
anatofuz
parents:
diff changeset
890 public:
anatofuz
parents:
diff changeset
891 MergeTailSection(StringRef name, uint32_t type, uint64_t flags,
anatofuz
parents:
diff changeset
892 uint32_t alignment);
anatofuz
parents:
diff changeset
893
anatofuz
parents:
diff changeset
894 size_t getSize() const override;
anatofuz
parents:
diff changeset
895 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
896 void finalizeContents() override;
anatofuz
parents:
diff changeset
897
anatofuz
parents:
diff changeset
898 private:
anatofuz
parents:
diff changeset
899 llvm::StringTableBuilder builder;
anatofuz
parents:
diff changeset
900 };
anatofuz
parents:
diff changeset
901
anatofuz
parents:
diff changeset
902 class MergeNoTailSection final : public MergeSyntheticSection {
anatofuz
parents:
diff changeset
903 public:
anatofuz
parents:
diff changeset
904 MergeNoTailSection(StringRef name, uint32_t type, uint64_t flags,
anatofuz
parents:
diff changeset
905 uint32_t alignment)
anatofuz
parents:
diff changeset
906 : MergeSyntheticSection(name, type, flags, alignment) {}
anatofuz
parents:
diff changeset
907
anatofuz
parents:
diff changeset
908 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
909 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
910 void finalizeContents() override;
anatofuz
parents:
diff changeset
911
anatofuz
parents:
diff changeset
912 private:
anatofuz
parents:
diff changeset
913 // We use the most significant bits of a hash as a shard ID.
anatofuz
parents:
diff changeset
914 // The reason why we don't want to use the least significant bits is
anatofuz
parents:
diff changeset
915 // because DenseMap also uses lower bits to determine a bucket ID.
anatofuz
parents:
diff changeset
916 // If we use lower bits, it significantly increases the probability of
anatofuz
parents:
diff changeset
917 // hash collisons.
anatofuz
parents:
diff changeset
918 size_t getShardId(uint32_t hash) {
anatofuz
parents:
diff changeset
919 assert((hash >> 31) == 0);
anatofuz
parents:
diff changeset
920 return hash >> (31 - llvm::countTrailingZeros(numShards));
anatofuz
parents:
diff changeset
921 }
anatofuz
parents:
diff changeset
922
anatofuz
parents:
diff changeset
923 // Section size
anatofuz
parents:
diff changeset
924 size_t size;
anatofuz
parents:
diff changeset
925
anatofuz
parents:
diff changeset
926 // String table contents
anatofuz
parents:
diff changeset
927 constexpr static size_t numShards = 32;
anatofuz
parents:
diff changeset
928 std::vector<llvm::StringTableBuilder> shards;
anatofuz
parents:
diff changeset
929 size_t shardOffsets[numShards];
anatofuz
parents:
diff changeset
930 };
anatofuz
parents:
diff changeset
931
anatofuz
parents:
diff changeset
932 // .MIPS.abiflags section.
anatofuz
parents:
diff changeset
933 template <class ELFT>
anatofuz
parents:
diff changeset
934 class MipsAbiFlagsSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
935 using Elf_Mips_ABIFlags = llvm::object::Elf_Mips_ABIFlags<ELFT>;
anatofuz
parents:
diff changeset
936
anatofuz
parents:
diff changeset
937 public:
anatofuz
parents:
diff changeset
938 static MipsAbiFlagsSection *create();
anatofuz
parents:
diff changeset
939
anatofuz
parents:
diff changeset
940 MipsAbiFlagsSection(Elf_Mips_ABIFlags flags);
anatofuz
parents:
diff changeset
941 size_t getSize() const override { return sizeof(Elf_Mips_ABIFlags); }
anatofuz
parents:
diff changeset
942 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
943
anatofuz
parents:
diff changeset
944 private:
anatofuz
parents:
diff changeset
945 Elf_Mips_ABIFlags flags;
anatofuz
parents:
diff changeset
946 };
anatofuz
parents:
diff changeset
947
anatofuz
parents:
diff changeset
948 // .MIPS.options section.
anatofuz
parents:
diff changeset
949 template <class ELFT> class MipsOptionsSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
950 using Elf_Mips_Options = llvm::object::Elf_Mips_Options<ELFT>;
anatofuz
parents:
diff changeset
951 using Elf_Mips_RegInfo = llvm::object::Elf_Mips_RegInfo<ELFT>;
anatofuz
parents:
diff changeset
952
anatofuz
parents:
diff changeset
953 public:
anatofuz
parents:
diff changeset
954 static MipsOptionsSection *create();
anatofuz
parents:
diff changeset
955
anatofuz
parents:
diff changeset
956 MipsOptionsSection(Elf_Mips_RegInfo reginfo);
anatofuz
parents:
diff changeset
957 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
958
anatofuz
parents:
diff changeset
959 size_t getSize() const override {
anatofuz
parents:
diff changeset
960 return sizeof(Elf_Mips_Options) + sizeof(Elf_Mips_RegInfo);
anatofuz
parents:
diff changeset
961 }
anatofuz
parents:
diff changeset
962
anatofuz
parents:
diff changeset
963 private:
anatofuz
parents:
diff changeset
964 Elf_Mips_RegInfo reginfo;
anatofuz
parents:
diff changeset
965 };
anatofuz
parents:
diff changeset
966
anatofuz
parents:
diff changeset
967 // MIPS .reginfo section.
anatofuz
parents:
diff changeset
968 template <class ELFT> class MipsReginfoSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
969 using Elf_Mips_RegInfo = llvm::object::Elf_Mips_RegInfo<ELFT>;
anatofuz
parents:
diff changeset
970
anatofuz
parents:
diff changeset
971 public:
anatofuz
parents:
diff changeset
972 static MipsReginfoSection *create();
anatofuz
parents:
diff changeset
973
anatofuz
parents:
diff changeset
974 MipsReginfoSection(Elf_Mips_RegInfo reginfo);
anatofuz
parents:
diff changeset
975 size_t getSize() const override { return sizeof(Elf_Mips_RegInfo); }
anatofuz
parents:
diff changeset
976 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
977
anatofuz
parents:
diff changeset
978 private:
anatofuz
parents:
diff changeset
979 Elf_Mips_RegInfo reginfo;
anatofuz
parents:
diff changeset
980 };
anatofuz
parents:
diff changeset
981
anatofuz
parents:
diff changeset
982 // This is a MIPS specific section to hold a space within the data segment
anatofuz
parents:
diff changeset
983 // of executable file which is pointed to by the DT_MIPS_RLD_MAP entry.
anatofuz
parents:
diff changeset
984 // See "Dynamic section" in Chapter 5 in the following document:
anatofuz
parents:
diff changeset
985 // ftp://www.linux-mips.org/pub/linux/mips/doc/ABI/mipsabi.pdf
anatofuz
parents:
diff changeset
986 class MipsRldMapSection : public SyntheticSection {
anatofuz
parents:
diff changeset
987 public:
anatofuz
parents:
diff changeset
988 MipsRldMapSection();
anatofuz
parents:
diff changeset
989 size_t getSize() const override { return config->wordsize; }
anatofuz
parents:
diff changeset
990 void writeTo(uint8_t *buf) override {}
anatofuz
parents:
diff changeset
991 };
anatofuz
parents:
diff changeset
992
anatofuz
parents:
diff changeset
993 // Representation of the combined .ARM.Exidx input sections. We process these
anatofuz
parents:
diff changeset
994 // as a SyntheticSection like .eh_frame as we need to merge duplicate entries
anatofuz
parents:
diff changeset
995 // and add terminating sentinel entries.
anatofuz
parents:
diff changeset
996 //
anatofuz
parents:
diff changeset
997 // The .ARM.exidx input sections after SHF_LINK_ORDER processing is done form
anatofuz
parents:
diff changeset
998 // a table that the unwinder can derive (Addresses are encoded as offsets from
anatofuz
parents:
diff changeset
999 // table):
anatofuz
parents:
diff changeset
1000 // | Address of function | Unwind instructions for function |
anatofuz
parents:
diff changeset
1001 // where the unwind instructions are either a small number of unwind or the
anatofuz
parents:
diff changeset
1002 // special EXIDX_CANTUNWIND entry representing no unwinding information.
anatofuz
parents:
diff changeset
1003 // When an exception is thrown from an address A, the unwinder searches the
anatofuz
parents:
diff changeset
1004 // table for the closest table entry with Address of function <= A. This means
anatofuz
parents:
diff changeset
1005 // that for two consecutive table entries:
anatofuz
parents:
diff changeset
1006 // | A1 | U1 |
anatofuz
parents:
diff changeset
1007 // | A2 | U2 |
anatofuz
parents:
diff changeset
1008 // The range of addresses described by U1 is [A1, A2)
anatofuz
parents:
diff changeset
1009 //
anatofuz
parents:
diff changeset
1010 // There are two cases where we need a linker generated table entry to fixup
anatofuz
parents:
diff changeset
1011 // the address ranges in the table
anatofuz
parents:
diff changeset
1012 // Case 1:
anatofuz
parents:
diff changeset
1013 // - A sentinel entry added with an address higher than all
anatofuz
parents:
diff changeset
1014 // executable sections. This was needed to work around libunwind bug pr31091.
anatofuz
parents:
diff changeset
1015 // - After address assignment we need to find the highest addressed executable
anatofuz
parents:
diff changeset
1016 // section and use the limit of that section so that the unwinder never
anatofuz
parents:
diff changeset
1017 // matches it.
anatofuz
parents:
diff changeset
1018 // Case 2:
anatofuz
parents:
diff changeset
1019 // - InputSections without a .ARM.exidx section (usually from Assembly)
anatofuz
parents:
diff changeset
1020 // need a table entry so that they terminate the range of the previously
anatofuz
parents:
diff changeset
1021 // function. This is pr40277.
anatofuz
parents:
diff changeset
1022 //
anatofuz
parents:
diff changeset
1023 // Instead of storing pointers to the .ARM.exidx InputSections from
anatofuz
parents:
diff changeset
1024 // InputObjects, we store pointers to the executable sections that need
anatofuz
parents:
diff changeset
1025 // .ARM.exidx sections. We can then use the dependentSections of these to
anatofuz
parents:
diff changeset
1026 // either find the .ARM.exidx section or know that we need to generate one.
anatofuz
parents:
diff changeset
1027 class ARMExidxSyntheticSection : public SyntheticSection {
anatofuz
parents:
diff changeset
1028 public:
anatofuz
parents:
diff changeset
1029 ARMExidxSyntheticSection();
anatofuz
parents:
diff changeset
1030
anatofuz
parents:
diff changeset
1031 // Add an input section to the ARMExidxSyntheticSection. Returns whether the
anatofuz
parents:
diff changeset
1032 // section needs to be removed from the main input section list.
anatofuz
parents:
diff changeset
1033 bool addSection(InputSection *isec);
anatofuz
parents:
diff changeset
1034
anatofuz
parents:
diff changeset
1035 size_t getSize() const override { return size; }
anatofuz
parents:
diff changeset
1036 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
1037 bool isNeeded() const override;
anatofuz
parents:
diff changeset
1038 // Sort and remove duplicate entries.
anatofuz
parents:
diff changeset
1039 void finalizeContents() override;
anatofuz
parents:
diff changeset
1040 InputSection *getLinkOrderDep() const;
anatofuz
parents:
diff changeset
1041
anatofuz
parents:
diff changeset
1042 static bool classof(const SectionBase *d);
anatofuz
parents:
diff changeset
1043
anatofuz
parents:
diff changeset
1044 // Links to the ARMExidxSections so we can transfer the relocations once the
anatofuz
parents:
diff changeset
1045 // layout is known.
anatofuz
parents:
diff changeset
1046 std::vector<InputSection *> exidxSections;
anatofuz
parents:
diff changeset
1047
anatofuz
parents:
diff changeset
1048 private:
173
0572611fdcc8 reorgnization done
Shinji KONO <kono@ie.u-ryukyu.ac.jp>
parents: 150
diff changeset
1049 size_t size = 0;
150
anatofuz
parents:
diff changeset
1050
anatofuz
parents:
diff changeset
1051 // Instead of storing pointers to the .ARM.exidx InputSections from
anatofuz
parents:
diff changeset
1052 // InputObjects, we store pointers to the executable sections that need
anatofuz
parents:
diff changeset
1053 // .ARM.exidx sections. We can then use the dependentSections of these to
anatofuz
parents:
diff changeset
1054 // either find the .ARM.exidx section or know that we need to generate one.
anatofuz
parents:
diff changeset
1055 std::vector<InputSection *> executableSections;
anatofuz
parents:
diff changeset
1056
anatofuz
parents:
diff changeset
1057 // The executable InputSection with the highest address to use for the
anatofuz
parents:
diff changeset
1058 // sentinel. We store separately from ExecutableSections as merging of
anatofuz
parents:
diff changeset
1059 // duplicate entries may mean this InputSection is removed from
anatofuz
parents:
diff changeset
1060 // ExecutableSections.
anatofuz
parents:
diff changeset
1061 InputSection *sentinel = nullptr;
anatofuz
parents:
diff changeset
1062 };
anatofuz
parents:
diff changeset
1063
anatofuz
parents:
diff changeset
1064 // A container for one or more linker generated thunks. Instances of these
anatofuz
parents:
diff changeset
1065 // thunks including ARM interworking and Mips LA25 PI to non-PI thunks.
anatofuz
parents:
diff changeset
1066 class ThunkSection : public SyntheticSection {
anatofuz
parents:
diff changeset
1067 public:
anatofuz
parents:
diff changeset
1068 // ThunkSection in OS, with desired outSecOff of Off
anatofuz
parents:
diff changeset
1069 ThunkSection(OutputSection *os, uint64_t off);
anatofuz
parents:
diff changeset
1070
anatofuz
parents:
diff changeset
1071 // Add a newly created Thunk to this container:
anatofuz
parents:
diff changeset
1072 // Thunk is given offset from start of this InputSection
anatofuz
parents:
diff changeset
1073 // Thunk defines a symbol in this InputSection that can be used as target
anatofuz
parents:
diff changeset
1074 // of a relocation
anatofuz
parents:
diff changeset
1075 void addThunk(Thunk *t);
anatofuz
parents:
diff changeset
1076 size_t getSize() const override;
anatofuz
parents:
diff changeset
1077 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
1078 InputSection *getTargetInputSection() const;
anatofuz
parents:
diff changeset
1079 bool assignOffsets();
anatofuz
parents:
diff changeset
1080
anatofuz
parents:
diff changeset
1081 // When true, round up reported size of section to 4 KiB. See comment
anatofuz
parents:
diff changeset
1082 // in addThunkSection() for more details.
anatofuz
parents:
diff changeset
1083 bool roundUpSizeForErrata = false;
anatofuz
parents:
diff changeset
1084
anatofuz
parents:
diff changeset
1085 private:
anatofuz
parents:
diff changeset
1086 std::vector<Thunk *> thunks;
anatofuz
parents:
diff changeset
1087 size_t size = 0;
anatofuz
parents:
diff changeset
1088 };
anatofuz
parents:
diff changeset
1089
anatofuz
parents:
diff changeset
1090 // Used to compute outSecOff of .got2 in each object file. This is needed to
anatofuz
parents:
diff changeset
1091 // synthesize PLT entries for PPC32 Secure PLT ABI.
anatofuz
parents:
diff changeset
1092 class PPC32Got2Section final : public SyntheticSection {
anatofuz
parents:
diff changeset
1093 public:
anatofuz
parents:
diff changeset
1094 PPC32Got2Section();
anatofuz
parents:
diff changeset
1095 size_t getSize() const override { return 0; }
anatofuz
parents:
diff changeset
1096 bool isNeeded() const override;
anatofuz
parents:
diff changeset
1097 void finalizeContents() override;
anatofuz
parents:
diff changeset
1098 void writeTo(uint8_t *buf) override {}
anatofuz
parents:
diff changeset
1099 };
anatofuz
parents:
diff changeset
1100
anatofuz
parents:
diff changeset
1101 // This section is used to store the addresses of functions that are called
anatofuz
parents:
diff changeset
1102 // in range-extending thunks on PowerPC64. When producing position dependent
anatofuz
parents:
diff changeset
1103 // code the addresses are link-time constants and the table is written out to
anatofuz
parents:
diff changeset
1104 // the binary. When producing position-dependent code the table is allocated and
anatofuz
parents:
diff changeset
1105 // filled in by the dynamic linker.
anatofuz
parents:
diff changeset
1106 class PPC64LongBranchTargetSection final : public SyntheticSection {
anatofuz
parents:
diff changeset
1107 public:
anatofuz
parents:
diff changeset
1108 PPC64LongBranchTargetSection();
anatofuz
parents:
diff changeset
1109 uint64_t getEntryVA(const Symbol *sym, int64_t addend);
anatofuz
parents:
diff changeset
1110 llvm::Optional<uint32_t> addEntry(const Symbol *sym, int64_t addend);
anatofuz
parents:
diff changeset
1111 size_t getSize() const override;
anatofuz
parents:
diff changeset
1112 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
1113 bool isNeeded() const override;
anatofuz
parents:
diff changeset
1114 void finalizeContents() override { finalized = true; }
anatofuz
parents:
diff changeset
1115
anatofuz
parents:
diff changeset
1116 private:
anatofuz
parents:
diff changeset
1117 std::vector<std::pair<const Symbol *, int64_t>> entries;
anatofuz
parents:
diff changeset
1118 llvm::DenseMap<std::pair<const Symbol *, int64_t>, uint32_t> entry_index;
anatofuz
parents:
diff changeset
1119 bool finalized = false;
anatofuz
parents:
diff changeset
1120 };
anatofuz
parents:
diff changeset
1121
anatofuz
parents:
diff changeset
1122 template <typename ELFT>
anatofuz
parents:
diff changeset
1123 class PartitionElfHeaderSection : public SyntheticSection {
anatofuz
parents:
diff changeset
1124 public:
anatofuz
parents:
diff changeset
1125 PartitionElfHeaderSection();
anatofuz
parents:
diff changeset
1126 size_t getSize() const override;
anatofuz
parents:
diff changeset
1127 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
1128 };
anatofuz
parents:
diff changeset
1129
anatofuz
parents:
diff changeset
1130 template <typename ELFT>
anatofuz
parents:
diff changeset
1131 class PartitionProgramHeadersSection : public SyntheticSection {
anatofuz
parents:
diff changeset
1132 public:
anatofuz
parents:
diff changeset
1133 PartitionProgramHeadersSection();
anatofuz
parents:
diff changeset
1134 size_t getSize() const override;
anatofuz
parents:
diff changeset
1135 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
1136 };
anatofuz
parents:
diff changeset
1137
anatofuz
parents:
diff changeset
1138 class PartitionIndexSection : public SyntheticSection {
anatofuz
parents:
diff changeset
1139 public:
anatofuz
parents:
diff changeset
1140 PartitionIndexSection();
anatofuz
parents:
diff changeset
1141 size_t getSize() const override;
anatofuz
parents:
diff changeset
1142 void finalizeContents() override;
anatofuz
parents:
diff changeset
1143 void writeTo(uint8_t *buf) override;
anatofuz
parents:
diff changeset
1144 };
anatofuz
parents:
diff changeset
1145
anatofuz
parents:
diff changeset
1146 InputSection *createInterpSection();
anatofuz
parents:
diff changeset
1147 MergeInputSection *createCommentSection();
anatofuz
parents:
diff changeset
1148 MergeSyntheticSection *createMergeSynthetic(StringRef name, uint32_t type,
anatofuz
parents:
diff changeset
1149 uint64_t flags, uint32_t alignment);
anatofuz
parents:
diff changeset
1150 template <class ELFT> void splitSections();
anatofuz
parents:
diff changeset
1151
anatofuz
parents:
diff changeset
1152 template <typename ELFT> void writeEhdr(uint8_t *buf, Partition &part);
anatofuz
parents:
diff changeset
1153 template <typename ELFT> void writePhdrs(uint8_t *buf, Partition &part);
anatofuz
parents:
diff changeset
1154
anatofuz
parents:
diff changeset
1155 Defined *addSyntheticLocal(StringRef name, uint8_t type, uint64_t value,
anatofuz
parents:
diff changeset
1156 uint64_t size, InputSectionBase &section);
anatofuz
parents:
diff changeset
1157
anatofuz
parents:
diff changeset
1158 void addVerneed(Symbol *ss);
anatofuz
parents:
diff changeset
1159
anatofuz
parents:
diff changeset
1160 // Linker generated per-partition sections.
anatofuz
parents:
diff changeset
1161 struct Partition {
anatofuz
parents:
diff changeset
1162 StringRef name;
anatofuz
parents:
diff changeset
1163 uint64_t nameStrTab;
anatofuz
parents:
diff changeset
1164
anatofuz
parents:
diff changeset
1165 SyntheticSection *elfHeader;
anatofuz
parents:
diff changeset
1166 SyntheticSection *programHeaders;
anatofuz
parents:
diff changeset
1167 std::vector<PhdrEntry *> phdrs;
anatofuz
parents:
diff changeset
1168
anatofuz
parents:
diff changeset
1169 ARMExidxSyntheticSection *armExidx;
anatofuz
parents:
diff changeset
1170 BuildIdSection *buildId;
anatofuz
parents:
diff changeset
1171 SyntheticSection *dynamic;
anatofuz
parents:
diff changeset
1172 StringTableSection *dynStrTab;
anatofuz
parents:
diff changeset
1173 SymbolTableBaseSection *dynSymTab;
anatofuz
parents:
diff changeset
1174 EhFrameHeader *ehFrameHdr;
anatofuz
parents:
diff changeset
1175 EhFrameSection *ehFrame;
anatofuz
parents:
diff changeset
1176 GnuHashTableSection *gnuHashTab;
anatofuz
parents:
diff changeset
1177 HashTableSection *hashTab;
anatofuz
parents:
diff changeset
1178 RelocationBaseSection *relaDyn;
anatofuz
parents:
diff changeset
1179 RelrBaseSection *relrDyn;
anatofuz
parents:
diff changeset
1180 VersionDefinitionSection *verDef;
anatofuz
parents:
diff changeset
1181 SyntheticSection *verNeed;
anatofuz
parents:
diff changeset
1182 VersionTableSection *verSym;
anatofuz
parents:
diff changeset
1183
anatofuz
parents:
diff changeset
1184 unsigned getNumber() const { return this - &partitions[0] + 1; }
anatofuz
parents:
diff changeset
1185 };
anatofuz
parents:
diff changeset
1186
anatofuz
parents:
diff changeset
1187 extern Partition *mainPart;
anatofuz
parents:
diff changeset
1188
anatofuz
parents:
diff changeset
1189 inline Partition &SectionBase::getPartition() const {
anatofuz
parents:
diff changeset
1190 assert(isLive());
anatofuz
parents:
diff changeset
1191 return partitions[partition - 1];
anatofuz
parents:
diff changeset
1192 }
anatofuz
parents:
diff changeset
1193
anatofuz
parents:
diff changeset
1194 // Linker generated sections which can be used as inputs and are not specific to
anatofuz
parents:
diff changeset
1195 // a partition.
anatofuz
parents:
diff changeset
1196 struct InStruct {
anatofuz
parents:
diff changeset
1197 InputSection *armAttributes;
anatofuz
parents:
diff changeset
1198 BssSection *bss;
anatofuz
parents:
diff changeset
1199 BssSection *bssRelRo;
anatofuz
parents:
diff changeset
1200 GotSection *got;
anatofuz
parents:
diff changeset
1201 GotPltSection *gotPlt;
anatofuz
parents:
diff changeset
1202 IgotPltSection *igotPlt;
anatofuz
parents:
diff changeset
1203 PPC64LongBranchTargetSection *ppc64LongBranchTarget;
anatofuz
parents:
diff changeset
1204 MipsGotSection *mipsGot;
anatofuz
parents:
diff changeset
1205 MipsRldMapSection *mipsRldMap;
anatofuz
parents:
diff changeset
1206 SyntheticSection *partEnd;
anatofuz
parents:
diff changeset
1207 SyntheticSection *partIndex;
anatofuz
parents:
diff changeset
1208 PltSection *plt;
anatofuz
parents:
diff changeset
1209 IpltSection *iplt;
anatofuz
parents:
diff changeset
1210 PPC32Got2Section *ppc32Got2;
anatofuz
parents:
diff changeset
1211 IBTPltSection *ibtPlt;
anatofuz
parents:
diff changeset
1212 RelocationBaseSection *relaPlt;
anatofuz
parents:
diff changeset
1213 RelocationBaseSection *relaIplt;
anatofuz
parents:
diff changeset
1214 StringTableSection *shStrTab;
anatofuz
parents:
diff changeset
1215 StringTableSection *strTab;
anatofuz
parents:
diff changeset
1216 SymbolTableBaseSection *symTab;
anatofuz
parents:
diff changeset
1217 SymtabShndxSection *symTabShndx;
anatofuz
parents:
diff changeset
1218 };
anatofuz
parents:
diff changeset
1219
anatofuz
parents:
diff changeset
1220 extern InStruct in;
anatofuz
parents:
diff changeset
1221
anatofuz
parents:
diff changeset
1222 } // namespace elf
anatofuz
parents:
diff changeset
1223 } // namespace lld
anatofuz
parents:
diff changeset
1224
anatofuz
parents:
diff changeset
1225 #endif