Line data Source code
1 : //===- ELF.h - ELF object file implementation -------------------*- C++ -*-===//
2 : //
3 : // The LLVM Compiler Infrastructure
4 : //
5 : // This file is distributed under the University of Illinois Open Source
6 : // License. See LICENSE.TXT for details.
7 : //
8 : //===----------------------------------------------------------------------===//
9 : //
10 : // This file declares the ELFFile template class.
11 : //
12 : //===----------------------------------------------------------------------===//
13 :
14 : #ifndef LLVM_OBJECT_ELF_H
15 : #define LLVM_OBJECT_ELF_H
16 :
17 : #include "llvm/ADT/ArrayRef.h"
18 : #include "llvm/ADT/SmallVector.h"
19 : #include "llvm/ADT/StringRef.h"
20 : #include "llvm/BinaryFormat/ELF.h"
21 : #include "llvm/Object/ELFTypes.h"
22 : #include "llvm/Object/Error.h"
23 : #include "llvm/Support/Endian.h"
24 : #include "llvm/Support/Error.h"
25 : #include <cassert>
26 : #include <cstddef>
27 : #include <cstdint>
28 : #include <limits>
29 : #include <utility>
30 :
31 : namespace llvm {
32 : namespace object {
33 :
34 : StringRef getELFRelocationTypeName(uint32_t Machine, uint32_t Type);
35 : uint32_t getELFRelrRelocationType(uint32_t Machine);
36 : StringRef getELFSectionTypeName(uint32_t Machine, uint32_t Type);
37 :
38 : // Subclasses of ELFFile may need this for template instantiation
39 : inline std::pair<unsigned char, unsigned char>
40 : getElfArchType(StringRef Object) {
41 9708 : if (Object.size() < ELF::EI_NIDENT)
42 : return std::make_pair((uint8_t)ELF::ELFCLASSNONE,
43 : (uint8_t)ELF::ELFDATANONE);
44 9708 : return std::make_pair((uint8_t)Object[ELF::EI_CLASS],
45 9708 : (uint8_t)Object[ELF::EI_DATA]);
46 : }
47 :
48 : static inline Error createError(StringRef Err) {
49 : return make_error<StringError>(Err, object_error::parse_failed);
50 : }
51 :
52 : template <class ELFT>
53 : class ELFFile {
54 : public:
55 : LLVM_ELF_IMPORT_TYPES_ELFT(ELFT)
56 : using uintX_t = typename ELFT::uint;
57 : using Elf_Ehdr = typename ELFT::Ehdr;
58 : using Elf_Shdr = typename ELFT::Shdr;
59 : using Elf_Sym = typename ELFT::Sym;
60 : using Elf_Dyn = typename ELFT::Dyn;
61 : using Elf_Phdr = typename ELFT::Phdr;
62 : using Elf_Rel = typename ELFT::Rel;
63 : using Elf_Rela = typename ELFT::Rela;
64 : using Elf_Relr = typename ELFT::Relr;
65 : using Elf_Verdef = typename ELFT::Verdef;
66 : using Elf_Verdaux = typename ELFT::Verdaux;
67 : using Elf_Verneed = typename ELFT::Verneed;
68 : using Elf_Vernaux = typename ELFT::Vernaux;
69 : using Elf_Versym = typename ELFT::Versym;
70 : using Elf_Hash = typename ELFT::Hash;
71 : using Elf_GnuHash = typename ELFT::GnuHash;
72 : using Elf_Nhdr = typename ELFT::Nhdr;
73 : using Elf_Note = typename ELFT::Note;
74 : using Elf_Note_Iterator = typename ELFT::NoteIterator;
75 : using Elf_Dyn_Range = typename ELFT::DynRange;
76 : using Elf_Shdr_Range = typename ELFT::ShdrRange;
77 : using Elf_Sym_Range = typename ELFT::SymRange;
78 : using Elf_Rel_Range = typename ELFT::RelRange;
79 : using Elf_Rela_Range = typename ELFT::RelaRange;
80 : using Elf_Relr_Range = typename ELFT::RelrRange;
81 : using Elf_Phdr_Range = typename ELFT::PhdrRange;
82 :
83 0 : const uint8_t *base() const {
84 0 : return reinterpret_cast<const uint8_t *>(Buf.data());
85 : }
86 0 :
87 0 : size_t getBufSize() const { return Buf.size(); }
88 :
89 0 : private:
90 0 : StringRef Buf;
91 :
92 0 : ELFFile(StringRef Object);
93 0 :
94 : public:
95 0 : const Elf_Ehdr *getHeader() const {
96 0 : return reinterpret_cast<const Elf_Ehdr *>(base());
97 : }
98 :
99 0 : template <typename T>
100 : Expected<const T *> getEntry(uint32_t Section, uint32_t Entry) const;
101 : template <typename T>
102 : Expected<const T *> getEntry(const Elf_Shdr *Section, uint32_t Entry) const;
103 :
104 : Expected<StringRef> getStringTable(const Elf_Shdr *Section) const;
105 : Expected<StringRef> getStringTableForSymtab(const Elf_Shdr &Section) const;
106 : Expected<StringRef> getStringTableForSymtab(const Elf_Shdr &Section,
107 0 : Elf_Shdr_Range Sections) const;
108 0 :
109 : Expected<ArrayRef<Elf_Word>> getSHNDXTable(const Elf_Shdr &Section) const;
110 0 : Expected<ArrayRef<Elf_Word>> getSHNDXTable(const Elf_Shdr &Section,
111 0 : Elf_Shdr_Range Sections) const;
112 :
113 0 : StringRef getRelocationTypeName(uint32_t Type) const;
114 0 : void getRelocationTypeName(uint32_t Type,
115 : SmallVectorImpl<char> &Result) const;
116 0 : uint32_t getRelrRelocationType() const;
117 0 :
118 : const char *getDynamicTagAsString(unsigned Arch, uint64_t Type) const;
119 0 : const char *getDynamicTagAsString(uint64_t Type) const;
120 0 :
121 : /// Get the symbol for a given relocation.
122 : Expected<const Elf_Sym *> getRelocationSymbol(const Elf_Rel *Rel,
123 : const Elf_Shdr *SymTab) const;
124 :
125 : static Expected<ELFFile> create(StringRef Object);
126 :
127 : bool isMipsELF64() const {
128 37451 : return getHeader()->e_machine == ELF::EM_MIPS &&
129 1241 : getHeader()->getFileClass() == ELF::ELFCLASS64;
130 : }
131 :
132 : bool isMips64EL() const {
133 644 : return isMipsELF64() &&
134 644 : getHeader()->getDataEncoding() == ELF::ELFDATA2LSB;
135 : }
136 :
137 : Expected<Elf_Shdr_Range> sections() const;
138 :
139 : Expected<Elf_Dyn_Range> dynamicEntries() const;
140 :
141 : Expected<const uint8_t *> toMappedAddr(uint64_t VAddr) const;
142 :
143 : Expected<Elf_Sym_Range> symbols(const Elf_Shdr *Sec) const {
144 26767744 : if (!Sec)
145 : return makeArrayRef<Elf_Sym>(nullptr, nullptr);
146 26696855 : return getSectionContentsAsArray<Elf_Sym>(Sec);
147 : }
148 :
149 : Expected<Elf_Rela_Range> relas(const Elf_Shdr *Sec) const {
150 1748 : return getSectionContentsAsArray<Elf_Rela>(Sec);
151 0 : }
152 905 :
153 79 : Expected<Elf_Rel_Range> rels(const Elf_Shdr *Sec) const {
154 623 : return getSectionContentsAsArray<Elf_Rel>(Sec);
155 0 : }
156 0 :
157 0 : Expected<Elf_Relr_Range> relrs(const Elf_Shdr *Sec) const {
158 13 : return getSectionContentsAsArray<Elf_Relr>(Sec);
159 0 : }
160 0 :
161 0 : Expected<std::vector<Elf_Rela>> decode_relrs(Elf_Relr_Range relrs) const;
162 :
163 0 : Expected<std::vector<Elf_Rela>> android_relas(const Elf_Shdr *Sec) const;
164 0 :
165 0 : /// Iterate over program header table.
166 2772 : Expected<Elf_Phdr_Range> program_headers() const {
167 2772 : if (getHeader()->e_phnum && getHeader()->e_phentsize != sizeof(Elf_Phdr))
168 2 : return createError("invalid e_phentsize");
169 2770 : if (getHeader()->e_phoff +
170 5540 : (getHeader()->e_phnum * getHeader()->e_phentsize) >
171 : getBufSize())
172 1 : return createError("program headers longer than binary");
173 2769 : auto *Begin =
174 714 : reinterpret_cast<const Elf_Phdr *>(base() + getHeader()->e_phoff);
175 2769 : return makeArrayRef(Begin, Begin + getHeader()->e_phnum);
176 0 : }
177 141 :
178 141 : /// Get an iterator over notes in a program header.
179 0 : ///
180 141 : /// The program header must be of type \c PT_NOTE.
181 282 : ///
182 0 : /// \param Phdr the program header to iterate over.
183 0 : /// \param Err [out] an error to support fallible iteration, which should
184 141 : /// be checked after iteration ends.
185 1 : Elf_Note_Iterator notes_begin(const Elf_Phdr &Phdr, Error &Err) const {
186 141 : if (Phdr.p_type != ELF::PT_NOTE) {
187 : Err = createError("attempt to iterate notes of non-note program header");
188 1711 : return Elf_Note_Iterator(Err);
189 1711 : }
190 2 : if (Phdr.p_offset + Phdr.p_filesz > getBufSize()) {
191 1709 : Err = createError("invalid program header offset/size");
192 3418 : return Elf_Note_Iterator(Err);
193 : }
194 1 : return Elf_Note_Iterator(base() + Phdr.p_offset, Phdr.p_filesz, Err);
195 1708 : }
196 3 :
197 1708 : /// Get an iterator over notes in a section.
198 : ///
199 174 : /// The section must be of type \c SHT_NOTE.
200 22417 : ///
201 0 : /// \param Shdr the section to iterate over.
202 22417 : /// \param Err [out] an error to support fallible iteration, which should
203 348 : /// be checked after iteration ends.
204 0 : Elf_Note_Iterator notes_begin(const Elf_Shdr &Shdr, Error &Err) const {
205 0 : if (Shdr.sh_type != ELF::SHT_NOTE) {
206 174 : Err = createError("attempt to iterate notes of non-note section");
207 172 : return Elf_Note_Iterator(Err);
208 174 : }
209 0 : if (Shdr.sh_offset + Shdr.sh_size > getBufSize()) {
210 746 : Err = createError("invalid section offset/size");
211 746 : return Elf_Note_Iterator(Err);
212 0 : }
213 746 : return Elf_Note_Iterator(base() + Shdr.sh_offset, Shdr.sh_size, Err);
214 1492 : }
215 0 :
216 0 : /// Get the end iterator for notes.
217 746 : Elf_Note_Iterator notes_end() const {
218 538 : return Elf_Note_Iterator();
219 746 : }
220 0 :
221 : /// Get an iterator range over notes of a program header.
222 0 : ///
223 : /// The program header must be of type \c PT_NOTE.
224 : ///
225 0 : /// \param Phdr the program header to iterate over.
226 0 : /// \param Err [out] an error to support fallible iteration, which should
227 : /// be checked after iteration ends.
228 0 : iterator_range<Elf_Note_Iterator> notes(const Elf_Phdr &Phdr,
229 0 : Error &Err) const {
230 0 : return make_range(notes_begin(Phdr, Err), notes_end());
231 0 : }
232 0 :
233 : /// Get an iterator range over notes of a section.
234 0 : ///
235 0 : /// The section must be of type \c SHT_NOTE.
236 : ///
237 0 : /// \param Shdr the section to iterate over.
238 0 : /// \param Err [out] an error to support fallible iteration, which should
239 : /// be checked after iteration ends.
240 0 : iterator_range<Elf_Note_Iterator> notes(const Elf_Shdr &Shdr,
241 0 : Error &Err) const {
242 0 : return make_range(notes_begin(Shdr, Err), notes_end());
243 : }
244 0 :
245 0 : Expected<StringRef> getSectionStringTable(Elf_Shdr_Range Sections) const;
246 0 : Expected<uint32_t> getSectionIndex(const Elf_Sym *Sym, Elf_Sym_Range Syms,
247 0 : ArrayRef<Elf_Word> ShndxTable) const;
248 0 : Expected<const Elf_Shdr *> getSection(const Elf_Sym *Sym,
249 0 : const Elf_Shdr *SymTab,
250 0 : ArrayRef<Elf_Word> ShndxTable) const;
251 0 : Expected<const Elf_Shdr *> getSection(const Elf_Sym *Sym,
252 0 : Elf_Sym_Range Symtab,
253 0 : ArrayRef<Elf_Word> ShndxTable) const;
254 0 : Expected<const Elf_Shdr *> getSection(uint32_t Index) const;
255 : Expected<const Elf_Shdr *> getSection(const StringRef SectionName) const;
256 0 :
257 0 : Expected<const Elf_Sym *> getSymbol(const Elf_Shdr *Sec,
258 0 : uint32_t Index) const;
259 :
260 0 : Expected<StringRef> getSectionName(const Elf_Shdr *Section) const;
261 0 : Expected<StringRef> getSectionName(const Elf_Shdr *Section,
262 0 : StringRef DotShstrtab) const;
263 0 : template <typename T>
264 0 : Expected<ArrayRef<T>> getSectionContentsAsArray(const Elf_Shdr *Sec) const;
265 : Expected<ArrayRef<uint8_t>> getSectionContents(const Elf_Shdr *Sec) const;
266 0 : };
267 0 :
268 0 : using ELF32LEFile = ELFFile<ELF32LE>;
269 0 : using ELF64LEFile = ELFFile<ELF64LE>;
270 0 : using ELF32BEFile = ELFFile<ELF32BE>;
271 0 : using ELF64BEFile = ELFFile<ELF64BE>;
272 :
273 0 : template <class ELFT>
274 0 : inline Expected<const typename ELFT::Shdr *>
275 166967 : getSection(typename ELFT::ShdrRange Sections, uint32_t Index) {
276 166967 : if (Index >= Sections.size())
277 0 : return createError("invalid section index");
278 166993 : return &Sections[Index];
279 26 : }
280 5310 :
281 5336 : template <class ELFT>
282 52 : inline Expected<uint32_t>
283 5310 : getExtendedSymbolTableIndex(const typename ELFT::Sym *Sym,
284 0 : const typename ELFT::Sym *FirstSym,
285 10815 : ArrayRef<typename ELFT::Word> ShndxTable) {
286 10790 : assert(Sym->st_shndx == ELF::SHN_XINDEX);
287 26 : unsigned Index = Sym - FirstSym;
288 10789 : if (Index >= ShndxTable.size())
289 0 : return createError("index past the end of the symbol table");
290 43437 :
291 43437 : // The size of the table was checked in getSHNDXTable.
292 28 : return ShndxTable[Index];
293 43465 : }
294 0 :
295 107431 : template <class ELFT>
296 107431 : Expected<uint32_t>
297 56 : ELFFile<ELFT>::getSectionIndex(const Elf_Sym *Sym, Elf_Sym_Range Syms,
298 107431 : ArrayRef<Elf_Word> ShndxTable) const {
299 : uint32_t Index = Sym->st_shndx;
300 25 : if (Index == ELF::SHN_XINDEX) {
301 53 : auto ErrorOrIndex = getExtendedSymbolTableIndex<ELFT>(
302 0 : Sym, Syms.begin(), ShndxTable);
303 289 : if (!ErrorOrIndex)
304 50 : return ErrorOrIndex.takeError();
305 0 : return *ErrorOrIndex;
306 0 : }
307 289 : if (Index == ELF::SHN_UNDEF || Index >= ELF::SHN_LORESERVE)
308 264 : return 0;
309 25 : return Index;
310 : }
311 0 :
312 264 : template <class ELFT>
313 0 : Expected<const typename ELFT::Shdr *>
314 0 : ELFFile<ELFT>::getSection(const Elf_Sym *Sym, const Elf_Shdr *SymTab,
315 0 : ArrayRef<Elf_Word> ShndxTable) const {
316 0 : auto SymsOrErr = symbols(SymTab);
317 0 : if (!SymsOrErr)
318 0 : return SymsOrErr.takeError();
319 468 : return getSection(Sym, *SymsOrErr, ShndxTable);
320 468 : }
321 0 :
322 469 : template <class ELFT>
323 1 : Expected<const typename ELFT::Shdr *>
324 4 : ELFFile<ELFT>::getSection(const Elf_Sym *Sym, Elf_Sym_Range Symbols,
325 297 : ArrayRef<Elf_Word> ShndxTable) const {
326 30 : auto IndexOrErr = getSectionIndex(Sym, Symbols, ShndxTable);
327 4 : if (!IndexOrErr)
328 0 : return IndexOrErr.takeError();
329 267 : uint32_t Index = *IndexOrErr;
330 323 : if (Index == 0)
331 1 : return nullptr;
332 2 : return getSection(Index);
333 : }
334 748 :
335 456 : template <class ELFT>
336 0 : inline Expected<const typename ELFT::Sym *>
337 456 : getSymbol(typename ELFT::SymRange Symbols, uint32_t Index) {
338 0 : if (Index >= Symbols.size())
339 6 : return createError("invalid symbol index");
340 6 : return &Symbols[Index];
341 0 : }
342 6 :
343 0 : template <class ELFT>
344 : Expected<const typename ELFT::Sym *>
345 0 : ELFFile<ELFT>::getSymbol(const Elf_Shdr *Sec, uint32_t Index) const {
346 0 : auto SymtabOrErr = symbols(Sec);
347 0 : if (!SymtabOrErr)
348 : return SymtabOrErr.takeError();
349 0 : return object::getSymbol<ELFT>(*SymtabOrErr, Index);
350 0 : }
351 0 :
352 0 : template <class ELFT>
353 0 : template <typename T>
354 0 : Expected<ArrayRef<T>>
355 335830 : ELFFile<ELFT>::getSectionContentsAsArray(const Elf_Shdr *Sec) const {
356 0 : if (Sec->sh_entsize != sizeof(T) && sizeof(T) != 1)
357 0 : return createError("invalid sh_entsize");
358 0 :
359 0 : uintX_t Offset = Sec->sh_offset;
360 : uintX_t Size = Sec->sh_size;
361 87033 :
362 0 : if (Size % sizeof(T))
363 87033 : return createError("size is not a multiple of sh_entsize");
364 422863 : if ((std::numeric_limits<uintX_t>::max() - Offset < Size) ||
365 336094 : Offset + Size > Buf.size())
366 0 : return createError("invalid section offset");
367 264 :
368 0 : if (Offset % alignof(T))
369 0 : return createError("unaligned data");
370 :
371 422599 : const T *Start = reinterpret_cast<const T *>(base() + Offset);
372 1762 : return makeArrayRef(Start, Size / sizeof(T));
373 : }
374 336 :
375 48111 : template <class ELFT>
376 0 : Expected<ArrayRef<uint8_t>>
377 10551 : ELFFile<ELFT>::getSectionContents(const Elf_Shdr *Sec) const {
378 10551 : return getSectionContentsAsArray<uint8_t>(Sec);
379 0 : }
380 0 :
381 0 : template <class ELFT>
382 : StringRef ELFFile<ELFT>::getRelocationTypeName(uint32_t Type) const {
383 336 : return getELFRelocationTypeName(getHeader()->e_machine, Type);
384 37896 : }
385 48111 :
386 29 : template <class ELFT>
387 0 : void ELFFile<ELFT>::getRelocationTypeName(uint32_t Type,
388 : SmallVectorImpl<char> &Result) const {
389 73565 : if (!isMipsELF64()) {
390 336 : StringRef Name = getRelocationTypeName(Type);
391 111124 : Result.append(Name.begin(), Name.end());
392 95244 : } else {
393 333996 : // The Mips N64 ABI allows up to three operations to be specified per
394 1483 : // relocation record. Unfortunately there's no easy way to test for the
395 264 : // presence of N64 ELFs as they have no special flag that identifies them
396 : // as being N64. We can safely assume at the moment that all Mips
397 : // ELFCLASS64 ELFs are N64. New Mips64 ABIs should provide enough
398 : // information to disambiguate between old vs new ABIs.
399 73301 : uint8_t Type1 = (Type >> 0) & 0xFF;
400 : uint8_t Type2 = (Type >> 8) & 0xFF;
401 : uint8_t Type3 = (Type >> 16) & 0xFF;
402 333732 :
403 335872 : // Concat all three relocation type names.
404 1483 : StringRef Name = getRelocationTypeName(Type1);
405 657 : Result.append(Name.begin(), Name.end());
406 657 :
407 0 : Name = getRelocationTypeName(Type2);
408 : Result.append(1, '/');
409 333732 : Result.append(Name.begin(), Name.end());
410 1483 :
411 : Name = getRelocationTypeName(Type3);
412 668 : Result.append(1, '/');
413 2883 : Result.append(Name.begin(), Name.end());
414 : }
415 0 : }
416 0 :
417 2260 : template <class ELFT>
418 : uint32_t ELFFile<ELFT>::getRelrRelocationType() const {
419 404095 : return getELFRelrRelocationType(getHeader()->e_machine);
420 75332 : }
421 669 :
422 75647 : template <class ELFT>
423 2226 : Expected<const typename ELFT::Sym *>
424 913 : ELFFile<ELFT>::getRelocationSymbol(const Elf_Rel *Rel,
425 913 : const Elf_Shdr *SymTab) const {
426 318 : uint32_t Index = Rel->getSymbol(isMips64EL());
427 3173 : if (Index == 0)
428 329749 : return nullptr;
429 401483 : return getEntry<Elf_Sym>(SymTab, Index);
430 71734 : }
431 1095 :
432 83905 : template <class ELFT>
433 : Expected<StringRef>
434 734 : ELFFile<ELFT>::getSectionStringTable(Elf_Shdr_Range Sections) const {
435 329647 : uint32_t Index = getHeader()->e_shstrndx;
436 353 : if (Index == ELF::SHN_XINDEX)
437 734 : Index = Sections[0].sh_link;
438 4 :
439 1935 : if (!Index) // no section string table.
440 2861 : return "";
441 15493 : if (Index >= Sections.size())
442 16165 : return createError("invalid section index");
443 1 : return getStringTable(&Sections[Index]);
444 1 : }
445 :
446 1 : template <class ELFT> ELFFile<ELFT>::ELFFile(StringRef Object) : Buf(Object) {}
447 1369 :
448 15597 : template <class ELFT>
449 989 : Expected<ELFFile<ELFT>> ELFFile<ELFT>::create(StringRef Object) {
450 989 : if (sizeof(Elf_Ehdr) > Object.size())
451 19826 : return createError("Invalid buffer");
452 271 : return ELFFile(Object);
453 103 : }
454 335834 :
455 0 : template <class ELFT>
456 270 : Expected<typename ELFT::ShdrRange> ELFFile<ELFT>::sections() const {
457 79 : const uintX_t SectionTableOffset = getHeader()->e_shoff;
458 79 : if (SectionTableOffset == 0)
459 0 : return ArrayRef<Elf_Shdr>();
460 19524 :
461 19453 : if (getHeader()->e_shentsize != sizeof(Elf_Shdr))
462 65 : return createError(
463 0 : "invalid section header entry size (e_shentsize) in ELF header");
464 8 :
465 65 : const uint64_t FileSize = Buf.size();
466 8 :
467 19595 : if (SectionTableOffset + sizeof(Elf_Shdr) > FileSize)
468 19522 : return createError("section header table goes past the end of the file");
469 271 :
470 69 : // Invalid address alignment of section headers
471 0 : if (SectionTableOffset & (alignof(Elf_Shdr) - 1))
472 168 : return createError("invalid alignment of section headers");
473 279 :
474 447 : const Elf_Shdr *First =
475 169 : reinterpret_cast<const Elf_Shdr *>(base() + SectionTableOffset);
476 0 :
477 170 : uintX_t NumSections = getHeader()->e_shnum;
478 440 : if (NumSections == 0)
479 0 : NumSections = First->sh_size;
480 154 :
481 : if (NumSections > UINT64_MAX / sizeof(Elf_Shdr))
482 1 : return createError("section table goes past the end of file");
483 0 :
484 1 : const uint64_t SectionTableSize = NumSections * sizeof(Elf_Shdr);
485 1 :
486 0 : // Section table goes past end of file!
487 1 : if (SectionTableOffset + SectionTableSize > FileSize)
488 1 : return createError("section table goes past the end of file");
489 0 :
490 1 : return makeArrayRef(First, NumSections);
491 0 : }
492 102 :
493 51 : template <class ELFT>
494 153 : template <typename T>
495 11761 : Expected<const T *> ELFFile<ELFT>::getEntry(uint32_t Section,
496 5361 : uint32_t Entry) const {
497 102 : auto SecOrErr = getSection(Section);
498 102 : if (!SecOrErr)
499 : return SecOrErr.takeError();
500 94 : return getEntry<T>(*SecOrErr, Entry);
501 : }
502 5361 :
503 0 : template <class ELFT>
504 11659 : template <typename T>
505 12384 : Expected<const T *> ELFFile<ELFT>::getEntry(const Elf_Shdr *Section,
506 0 : uint32_t Entry) const {
507 725 : if (sizeof(T) != Section->sh_entsize)
508 725 : return createError("invalid sh_entsize");
509 0 : size_t Pos = Section->sh_offset + Entry * sizeof(T);
510 0 : if (Pos + sizeof(T) > Buf.size())
511 11659 : return createError("invalid section offset");
512 6609 : return reinterpret_cast<const T *>(base() + Pos);
513 : }
514 217 :
515 942 : template <class ELFT>
516 0 : Expected<const typename ELFT::Shdr *>
517 65 : ELFFile<ELFT>::getSection(uint32_t Index) const {
518 65 : auto TableOrErr = sections();
519 16 : if (!TableOrErr)
520 57 : return TableOrErr.takeError();
521 168 : return object::getSection<ELFT>(*TableOrErr, Index);
522 16 : }
523 152 :
524 152 : template <class ELFT>
525 335835 : Expected<const typename ELFT::Shdr *>
526 335835 : ELFFile<ELFT>::getSection(const StringRef SectionName) const {
527 0 : auto TableOrErr = sections();
528 5 : if (!TableOrErr)
529 16 : return TableOrErr.takeError();
530 17899 : for (auto &Sec : *TableOrErr) {
531 17899 : auto SecNameOrErr = getSectionName(&Sec);
532 17747 : if (!SecNameOrErr)
533 487 : return SecNameOrErr.takeError();
534 223 : if (*SecNameOrErr == SectionName)
535 18016 : return &Sec;
536 269 : }
537 17747 : return createError("invalid section name");
538 5 : }
539 17747 :
540 223 : template <class ELFT>
541 1035 : Expected<StringRef>
542 1258 : ELFFile<ELFT>::getStringTable(const Elf_Shdr *Section) const {
543 1679 : if (Section->sh_type != ELF::SHT_STRTAB)
544 0 : return createError("invalid sh_type for string table, expected SHT_STRTAB");
545 0 : auto V = getSectionContentsAsArray<char>(Section);
546 1035 : if (!V)
547 155 : return V.takeError();
548 1035 : ArrayRef<char> Data = *V;
549 378 : if (Data.empty())
550 1413 : return createError("empty string table");
551 0 : if (Data.back() != '\0')
552 1720 : return createError("string table non-null terminated");
553 1725 : return StringRef(Data.begin(), Data.size());
554 1553 : }
555 5 :
556 : template <class ELFT>
557 1713 : Expected<ArrayRef<typename ELFT::Word>>
558 : ELFFile<ELFT>::getSHNDXTable(const Elf_Shdr &Section) const {
559 1720 : auto SectionsOrErr = sections();
560 157 : if (!SectionsOrErr)
561 1853 : return SectionsOrErr.takeError();
562 167 : return getSHNDXTable(Section, *SectionsOrErr);
563 119234347 : }
564 119234347 :
565 10129 : template <class ELFT>
566 119223933 : Expected<ArrayRef<typename ELFT::Word>>
567 5 : ELFFile<ELFT>::getSHNDXTable(const Elf_Shdr &Section,
568 13158 : Elf_Shdr_Range Sections) const {
569 3039 : assert(Section.sh_type == ELF::SHT_SYMTAB_SHNDX);
570 10124 : auto VOrErr = getSectionContentsAsArray<Elf_Word>(&Section);
571 3544 : if (!VOrErr)
572 10354 : return VOrErr.takeError();
573 119209348 : ArrayRef<Elf_Word> V = *VOrErr;
574 119214383 : auto SymTableOrErr = object::getSection<ELFT>(Sections, Section.sh_link);
575 5035 : if (!SymTableOrErr)
576 119214383 : return SymTableOrErr.takeError();
577 0 : const Elf_Shdr &SymTable = **SymTableOrErr;
578 3875 : if (SymTable.sh_type != ELF::SHT_SYMTAB &&
579 8112 : SymTable.sh_type != ELF::SHT_DYNSYM)
580 73 : return createError("invalid sh_type");
581 8753 : if (V.size() != (SymTable.sh_size / sizeof(Elf_Sym)))
582 0 : return createError("invalid section contents size");
583 14401 : return V;
584 8641 : }
585 16 :
586 8641 : template <class ELFT>
587 598 : Expected<StringRef>
588 337 : ELFFile<ELFT>::getStringTableForSymtab(const Elf_Shdr &Sec) const {
589 0 : auto SectionsOrErr = sections();
590 691 : if (!SectionsOrErr)
591 3 : return SectionsOrErr.takeError();
592 264 : return getStringTableForSymtab(Sec, *SectionsOrErr);
593 : }
594 248 :
595 267 : template <class ELFT>
596 239675 : Expected<StringRef>
597 789 : ELFFile<ELFT>::getStringTableForSymtab(const Elf_Shdr &Sec,
598 239147 : Elf_Shdr_Range Sections) const {
599 830 :
600 678 : if (Sec.sh_type != ELF::SHT_SYMTAB && Sec.sh_type != ELF::SHT_DYNSYM)
601 239141 : return createError(
602 155 : "invalid sh_type for symbol table, expected SHT_SYMTAB or SHT_DYNSYM");
603 525 : auto SectionOrErr = object::getSection<ELFT>(Sections, Sec.sh_link);
604 490 : if (!SectionOrErr)
605 : return SectionOrErr.takeError();
606 974 : return getStringTable(*SectionOrErr);
607 239150 : }
608 0 :
609 366 : template <class ELFT>
610 76 : Expected<StringRef>
611 290 : ELFFile<ELFT>::getSectionName(const Elf_Shdr *Section) const {
612 : auto SectionsOrErr = sections();
613 12 : if (!SectionsOrErr)
614 239141 : return SectionsOrErr.takeError();
615 9 : auto Table = getSectionStringTable(*SectionsOrErr);
616 810 : if (!Table)
617 239144 : return Table.takeError();
618 239945 : return getSectionName(Section, *Table);
619 801 : }
620 0 :
621 94022 : template <class ELFT>
622 737 : Expected<StringRef> ELFFile<ELFT>::getSectionName(const Elf_Shdr *Section,
623 9 : StringRef DotShstrtab) const {
624 239816 : uint32_t Offset = Section->sh_name;
625 76 : if (Offset == 0)
626 92 : return StringRef();
627 239141 : if (Offset >= DotShstrtab.size())
628 21 : return createError("invalid string offset");
629 21 : return StringRef(DotShstrtab.data() + Offset);
630 0 : }
631 16 :
632 8760 : /// This function returns the hash value for a symbol in the .dynsym section
633 0 : /// Name of the API remains consistent as specified in the libelf
634 8760 : /// REF : http://www.sco.com/developers/gabi/latest/ch5.dynamic.html#hash
635 5 : inline unsigned hashSysV(StringRef SymbolName) {
636 264 : unsigned h = 0, g;
637 24804 : for (char C : SymbolName) {
638 14343 : h = (h << 4) + C;
639 14338 : g = h & 0xf0000000L;
640 14074 : if (g != 0)
641 6129 : h ^= g >> 24;
642 14338 : h &= ~g;
643 8744 : }
644 269 : return h;
645 5 : }
646 155 :
647 0 : } // end namespace object
648 155 : } // end namespace llvm
649 22398 :
650 8744 : #endif // LLVM_OBJECT_ELF_H
|