blob: 7dea3784e73fcc8767a347f5ac28a61521bfc352 [file] [log] [blame]
Rafael Espindola5805c4f2015-09-21 21:38:08 +00001//===- OutputSections.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_OUTPUT_SECTIONS_H
11#define LLD_ELF_OUTPUT_SECTIONS_H
12
13#include "lld/Core/LLVM.h"
14
Rafael Espindolac159c962015-10-19 21:00:02 +000015#include "llvm/ADT/MapVector.h"
Rafael Espindola5805c4f2015-09-21 21:38:08 +000016#include "llvm/MC/StringTableBuilder.h"
17#include "llvm/Object/ELF.h"
18
Davide Italiano85121bb2015-09-25 03:56:11 +000019#include "Config.h"
20
Rafael Espindola5805c4f2015-09-21 21:38:08 +000021#include <type_traits>
22
23namespace lld {
24namespace elf2 {
25
26class SymbolBody;
Rui Ueyama3ce825e2015-10-09 21:07:25 +000027template <class ELFT> class SymbolTable;
Rafael Espindola5805c4f2015-09-21 21:38:08 +000028template <class ELFT> class SymbolTableSection;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +000029template <class ELFT> class StringTableSection;
Rafael Espindola5805c4f2015-09-21 21:38:08 +000030template <class ELFT> class InputSection;
Rafael Espindolac159c962015-10-19 21:00:02 +000031template <class ELFT> class MergeInputSection;
Rafael Espindola5805c4f2015-09-21 21:38:08 +000032template <class ELFT> class OutputSection;
33template <class ELFT> class ObjectFile;
34template <class ELFT> class DefinedRegular;
Rafael Espindolacd076f02015-09-25 18:19:03 +000035template <class ELFT> class ELFSymbolBody;
Rafael Espindola5805c4f2015-09-21 21:38:08 +000036
Rafael Espindola5805c4f2015-09-21 21:38:08 +000037template <class ELFT>
Rafael Espindola932efcf2015-10-19 20:24:44 +000038static inline typename llvm::object::ELFFile<ELFT>::uintX_t
39getAddend(const typename llvm::object::ELFFile<ELFT>::Elf_Rel &Rel) {
40 return 0;
41}
Rafael Espindola5805c4f2015-09-21 21:38:08 +000042
43template <class ELFT>
Rafael Espindola932efcf2015-10-19 20:24:44 +000044static inline typename llvm::object::ELFFile<ELFT>::uintX_t
45getAddend(const typename llvm::object::ELFFile<ELFT>::Elf_Rela &Rel) {
46 return Rel.r_addend;
47}
48
49template <class ELFT>
50typename llvm::object::ELFFile<ELFT>::uintX_t getSymVA(const SymbolBody &S);
51
52template <class ELFT, bool IsRela>
Rafael Espindola5805c4f2015-09-21 21:38:08 +000053typename llvm::object::ELFFile<ELFT>::uintX_t
Rui Ueyama126d08f2015-10-12 20:28:22 +000054getLocalRelTarget(const ObjectFile<ELFT> &File,
Rafael Espindola932efcf2015-10-19 20:24:44 +000055 const llvm::object::Elf_Rel_Impl<ELFT, IsRela> &Rel);
Rafael Espindolacc6ebb82015-10-14 18:42:16 +000056bool canBePreempted(const SymbolBody *Body, bool NeedsGot);
Rafael Espindola4f674ed2015-10-05 15:24:04 +000057template <class ELFT> bool includeInSymtab(const SymbolBody &B);
58
Rafael Espindola05a3dd22015-09-22 23:38:23 +000059bool includeInDynamicSymtab(const SymbolBody &B);
Rafael Espindolad1cf4212015-10-05 16:25:43 +000060
61template <class ELFT>
62bool shouldKeepInSymtab(
Rafael Espindola444576d2015-10-09 19:25:07 +000063 const ObjectFile<ELFT> &File, StringRef Name,
64 const typename llvm::object::ELFFile<ELFT>::Elf_Sym &Sym);
Davide Italiano85121bb2015-09-25 03:56:11 +000065
Rafael Espindola71675852015-09-22 00:16:19 +000066// This represents a section in an output file.
67// Different sub classes represent different types of sections. Some contain
68// input sections, others are created by the linker.
69// The writer creates multiple OutputSections and assign them unique,
Rafael Espindola5805c4f2015-09-21 21:38:08 +000070// non-overlapping file offsets and VAs.
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +000071template <class ELFT> class OutputSectionBase {
Rafael Espindola5805c4f2015-09-21 21:38:08 +000072public:
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +000073 typedef typename llvm::object::ELFFile<ELFT>::uintX_t uintX_t;
74 typedef typename llvm::object::ELFFile<ELFT>::Elf_Shdr Elf_Shdr;
Rafael Espindola5805c4f2015-09-21 21:38:08 +000075
76 OutputSectionBase(StringRef Name, uint32_t sh_type, uintX_t sh_flags);
77 void setVA(uintX_t VA) { Header.sh_addr = VA; }
78 uintX_t getVA() const { return Header.sh_addr; }
79 void setFileOffset(uintX_t Off) { Header.sh_offset = Off; }
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +000080 void writeHeaderTo(Elf_Shdr *SHdr);
Rafael Espindola5805c4f2015-09-21 21:38:08 +000081 StringRef getName() { return Name; }
82 void setNameOffset(uintX_t Offset) { Header.sh_name = Offset; }
83
Rui Ueyama2317d0d2015-10-15 20:55:22 +000084 unsigned SectionIndex;
Rafael Espindola5805c4f2015-09-21 21:38:08 +000085
86 // Returns the size of the section in the output file.
Rafael Espindola77572242015-10-02 19:37:55 +000087 uintX_t getSize() const { return Header.sh_size; }
Rafael Espindola5805c4f2015-09-21 21:38:08 +000088 void setSize(uintX_t Val) { Header.sh_size = Val; }
89 uintX_t getFlags() { return Header.sh_flags; }
90 uintX_t getFileOff() { return Header.sh_offset; }
91 uintX_t getAlign() {
92 // The ELF spec states that a value of 0 means the section has no alignment
93 // constraits.
94 return std::max<uintX_t>(Header.sh_addralign, 1);
95 }
96 uint32_t getType() { return Header.sh_type; }
97
Rafael Espindola5805c4f2015-09-21 21:38:08 +000098 virtual void finalize() {}
99 virtual void writeTo(uint8_t *Buf) = 0;
100
101protected:
102 StringRef Name;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000103 Elf_Shdr Header;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000104 ~OutputSectionBase() = default;
105};
106
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000107template <class ELFT> class GotSection final : public OutputSectionBase<ELFT> {
108 typedef OutputSectionBase<ELFT> Base;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000109 typedef typename Base::uintX_t uintX_t;
110
111public:
Rui Ueyama15ef5e12015-10-07 19:18:16 +0000112 GotSection();
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000113 void finalize() override {
Rui Ueyama5f551ae2015-10-14 14:02:06 +0000114 this->Header.sh_size = Entries.size() * sizeof(uintX_t);
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000115 }
Rafael Espindolaa6627382015-10-06 23:56:53 +0000116 void writeTo(uint8_t *Buf) override;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000117 void addEntry(SymbolBody *Sym);
118 bool empty() const { return Entries.empty(); }
119 uintX_t getEntryAddr(const SymbolBody &B) const;
120
121private:
122 std::vector<const SymbolBody *> Entries;
123};
124
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000125template <class ELFT> class PltSection final : public OutputSectionBase<ELFT> {
126 typedef OutputSectionBase<ELFT> Base;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000127 typedef typename Base::uintX_t uintX_t;
128
129public:
Rui Ueyama15ef5e12015-10-07 19:18:16 +0000130 PltSection();
Hal Finkel6c2a3b82015-10-08 21:51:31 +0000131 void finalize() override;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000132 void writeTo(uint8_t *Buf) override;
133 void addEntry(SymbolBody *Sym);
134 bool empty() const { return Entries.empty(); }
135 uintX_t getEntryAddr(const SymbolBody &B) const;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000136
137private:
138 std::vector<const SymbolBody *> Entries;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000139};
140
141template <class ELFT> struct DynamicReloc {
142 typedef typename llvm::object::ELFFile<ELFT>::Elf_Rel Elf_Rel;
143 const InputSection<ELFT> &C;
144 const Elf_Rel &RI;
145};
146
147template <class ELFT>
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000148class SymbolTableSection final : public OutputSectionBase<ELFT> {
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000149public:
150 typedef typename llvm::object::ELFFile<ELFT>::Elf_Shdr Elf_Shdr;
151 typedef typename llvm::object::ELFFile<ELFT>::Elf_Sym Elf_Sym;
152 typedef typename llvm::object::ELFFile<ELFT>::Elf_Sym_Range Elf_Sym_Range;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000153 typedef typename llvm::object::ELFFile<ELFT>::uintX_t uintX_t;
Rui Ueyama3ce825e2015-10-09 21:07:25 +0000154 SymbolTableSection(SymbolTable<ELFT> &Table,
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000155 StringTableSection<ELFT> &StrTabSec);
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000156
Rui Ueyama0db335f2015-10-07 16:58:54 +0000157 void finalize() override;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000158 void writeTo(uint8_t *Buf) override;
Rui Ueyama0db335f2015-10-07 16:58:54 +0000159 void addSymbol(StringRef Name, bool isLocal = false);
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000160 StringTableSection<ELFT> &getStrTabSec() const { return StrTabSec; }
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000161 unsigned getNumSymbols() const { return NumVisible + 1; }
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000162
163private:
Rui Ueyama8ddfa812015-09-30 00:32:10 +0000164 void writeLocalSymbols(uint8_t *&Buf);
Igor Kudrinea6a8352015-10-19 08:01:51 +0000165 void writeGlobalSymbols(uint8_t *Buf);
Rui Ueyama8ddfa812015-09-30 00:32:10 +0000166
Rui Ueyama3ce825e2015-10-09 21:07:25 +0000167 SymbolTable<ELFT> &Table;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000168 StringTableSection<ELFT> &StrTabSec;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000169 unsigned NumVisible = 0;
170 unsigned NumLocals = 0;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000171};
172
173template <class ELFT>
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000174class RelocationSection final : public OutputSectionBase<ELFT> {
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000175 typedef typename llvm::object::ELFFile<ELFT>::Elf_Rel Elf_Rel;
176 typedef typename llvm::object::ELFFile<ELFT>::Elf_Rela Elf_Rela;
Rafael Espindola3c83e2b2015-10-05 21:09:37 +0000177 typedef typename llvm::object::ELFFile<ELFT>::uintX_t uintX_t;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000178
179public:
Rui Ueyamac58656c2015-10-13 16:59:30 +0000180 RelocationSection(bool IsRela);
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000181 void addReloc(const DynamicReloc<ELFT> &Reloc) { Relocs.push_back(Reloc); }
182 void finalize() override;
183 void writeTo(uint8_t *Buf) override;
184 bool hasRelocs() const { return !Relocs.empty(); }
185 bool isRela() const { return IsRela; }
186
187private:
188 std::vector<DynamicReloc<ELFT>> Relocs;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000189 const bool IsRela;
190};
191
192template <class ELFT>
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000193class OutputSection final : public OutputSectionBase<ELFT> {
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000194public:
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000195 typedef typename llvm::object::ELFFile<ELFT>::Elf_Shdr Elf_Shdr;
196 typedef typename llvm::object::ELFFile<ELFT>::Elf_Sym Elf_Sym;
197 typedef typename llvm::object::ELFFile<ELFT>::Elf_Rel Elf_Rel;
198 typedef typename llvm::object::ELFFile<ELFT>::Elf_Rela Elf_Rela;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000199 typedef typename llvm::object::ELFFile<ELFT>::uintX_t uintX_t;
Rui Ueyama15ef5e12015-10-07 19:18:16 +0000200 OutputSection(StringRef Name, uint32_t sh_type, uintX_t sh_flags);
Rafael Espindola71675852015-09-22 00:16:19 +0000201 void addSection(InputSection<ELFT> *C);
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000202 void writeTo(uint8_t *Buf) override;
203
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000204private:
Rafael Espindola71675852015-09-22 00:16:19 +0000205 std::vector<InputSection<ELFT> *> Sections;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000206};
207
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000208template <class ELFT>
Rafael Espindolac159c962015-10-19 21:00:02 +0000209class MergeOutputSection final : public OutputSectionBase<ELFT> {
210 typedef typename OutputSectionBase<ELFT>::uintX_t uintX_t;
211
212public:
213 MergeOutputSection(StringRef Name, uint32_t sh_type, uintX_t sh_flags);
214 void addSection(MergeInputSection<ELFT> *S);
215 void writeTo(uint8_t *Buf) override;
216
217 unsigned getOffset(ArrayRef<uint8_t> Val);
218
219private:
220 // This map is used to find if we already have an entry for a given value and,
221 // if so, at what offset it is.
222 llvm::MapVector<ArrayRef<uint8_t>, uintX_t> Offsets;
223};
224
225template <class ELFT>
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000226class InterpSection final : public OutputSectionBase<ELFT> {
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000227public:
228 InterpSection();
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000229 void writeTo(uint8_t *Buf);
230};
231
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000232template <class ELFT>
233class StringTableSection final : public OutputSectionBase<ELFT> {
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000234public:
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000235 typedef typename llvm::object::ELFFile<ELFT>::uintX_t uintX_t;
Rafael Espindola35c6af32015-09-25 17:19:10 +0000236 StringTableSection(bool Dynamic);
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000237 void add(StringRef S) { StrTabBuilder.add(S); }
238 size_t getFileOff(StringRef S) const { return StrTabBuilder.getOffset(S); }
239 StringRef data() const { return StrTabBuilder.data(); }
240 void writeTo(uint8_t *Buf) override;
241
242 void finalize() override {
243 StrTabBuilder.finalize(llvm::StringTableBuilder::ELF);
244 this->Header.sh_size = StrTabBuilder.data().size();
245 }
246
247 bool isDynamic() const { return Dynamic; }
248
249private:
250 const bool Dynamic;
251 llvm::StringTableBuilder StrTabBuilder;
252};
253
254template <class ELFT>
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000255class HashTableSection final : public OutputSectionBase<ELFT> {
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000256 typedef typename llvm::object::ELFFile<ELFT>::Elf_Word Elf_Word;
257
258public:
Rui Ueyama15ef5e12015-10-07 19:18:16 +0000259 HashTableSection();
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000260 void addSymbol(SymbolBody *S);
Rui Ueyama0db335f2015-10-07 16:58:54 +0000261 void finalize() override;
262 void writeTo(uint8_t *Buf) override;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000263
264private:
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000265 std::vector<uint32_t> Hashes;
266};
267
268template <class ELFT>
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000269class DynamicSection final : public OutputSectionBase<ELFT> {
270 typedef OutputSectionBase<ELFT> Base;
271 typedef typename llvm::object::ELFFile<ELFT>::Elf_Dyn Elf_Dyn;
Rui Ueyama2dfd74f2015-09-30 21:57:53 +0000272 typedef typename llvm::object::ELFFile<ELFT>::Elf_Rel Elf_Rel;
273 typedef typename llvm::object::ELFFile<ELFT>::Elf_Rela Elf_Rela;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000274 typedef typename llvm::object::ELFFile<ELFT>::Elf_Shdr Elf_Shdr;
Rui Ueyama2dfd74f2015-09-30 21:57:53 +0000275 typedef typename llvm::object::ELFFile<ELFT>::Elf_Sym Elf_Sym;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000276
277public:
Rui Ueyama3ce825e2015-10-09 21:07:25 +0000278 DynamicSection(SymbolTable<ELFT> &SymTab);
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000279 void finalize() override;
280 void writeTo(uint8_t *Buf) override;
281
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000282 OutputSectionBase<ELFT> *PreInitArraySec = nullptr;
283 OutputSectionBase<ELFT> *InitArraySec = nullptr;
284 OutputSectionBase<ELFT> *FiniArraySec = nullptr;
Rafael Espindola77572242015-10-02 19:37:55 +0000285
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000286private:
Rui Ueyama3ce825e2015-10-09 21:07:25 +0000287 SymbolTable<ELFT> &SymTab;
Igor Kudrinb1f2b512015-10-05 10:29:46 +0000288 const ELFSymbolBody<ELFT> *InitSym = nullptr;
289 const ELFSymbolBody<ELFT> *FiniSym = nullptr;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000290};
Rui Ueyama15ef5e12015-10-07 19:18:16 +0000291
292// All output sections that are hadnled by the linker specially are
293// globally accessible. Writer initializes them, so don't use them
294// until Writer is initialized.
295template <class ELFT> struct Out {
296 static DynamicSection<ELFT> *Dynamic;
297 static GotSection<ELFT> *Got;
298 static HashTableSection<ELFT> *HashTab;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000299 static InterpSection<ELFT> *Interp;
Rui Ueyama15ef5e12015-10-07 19:18:16 +0000300 static OutputSection<ELFT> *Bss;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000301 static OutputSectionBase<ELFT> *Opd;
Hal Finkeldaedc122015-10-12 23:16:53 +0000302 static uint8_t *OpdBuf;
Rui Ueyama15ef5e12015-10-07 19:18:16 +0000303 static PltSection<ELFT> *Plt;
304 static RelocationSection<ELFT> *RelaDyn;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000305 static StringTableSection<ELFT> *DynStrTab;
306 static StringTableSection<ELFT> *StrTab;
Rui Ueyama15ef5e12015-10-07 19:18:16 +0000307 static SymbolTableSection<ELFT> *DynSymTab;
308 static SymbolTableSection<ELFT> *SymTab;
309};
Rui Ueyamad888d102015-10-09 19:34:55 +0000310
311template <class ELFT> DynamicSection<ELFT> *Out<ELFT>::Dynamic;
312template <class ELFT> GotSection<ELFT> *Out<ELFT>::Got;
313template <class ELFT> HashTableSection<ELFT> *Out<ELFT>::HashTab;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000314template <class ELFT> InterpSection<ELFT> *Out<ELFT>::Interp;
Rui Ueyamad888d102015-10-09 19:34:55 +0000315template <class ELFT> OutputSection<ELFT> *Out<ELFT>::Bss;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000316template <class ELFT> OutputSectionBase<ELFT> *Out<ELFT>::Opd;
Hal Finkeldaedc122015-10-12 23:16:53 +0000317template <class ELFT> uint8_t *Out<ELFT>::OpdBuf;
Rui Ueyamad888d102015-10-09 19:34:55 +0000318template <class ELFT> PltSection<ELFT> *Out<ELFT>::Plt;
319template <class ELFT> RelocationSection<ELFT> *Out<ELFT>::RelaDyn;
Rui Ueyamac7cc6ec2015-10-15 22:27:29 +0000320template <class ELFT> StringTableSection<ELFT> *Out<ELFT>::DynStrTab;
321template <class ELFT> StringTableSection<ELFT> *Out<ELFT>::StrTab;
Rui Ueyamad888d102015-10-09 19:34:55 +0000322template <class ELFT> SymbolTableSection<ELFT> *Out<ELFT>::DynSymTab;
323template <class ELFT> SymbolTableSection<ELFT> *Out<ELFT>::SymTab;
Rafael Espindola5805c4f2015-09-21 21:38:08 +0000324}
325}
326#endif