//===- InputFiles.h ---------------------------------------------*- C++ -*-===// // // The LLVM Linker // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef LLD_ELF_INPUT_FILES_H #define LLD_ELF_INPUT_FILES_H #include "Config.h" #include "InputSection.h" #include "Error.h" #include "Symbols.h" #include "lld/Core/LLVM.h" #include "llvm/ADT/DenseSet.h" #include "llvm/ADT/STLExtras.h" #include "llvm/Object/Archive.h" #include "llvm/Object/ELF.h" namespace lld { namespace elf2 { using llvm::object::Archive; class InputFile; class Lazy; class SymbolBody; // The root class of input files. class InputFile { public: enum Kind { ObjectKind, SharedKind, ArchiveKind }; Kind kind() const { return FileKind; } StringRef getName() const { return MB.getBufferIdentifier(); } protected: InputFile(Kind K, MemoryBufferRef M) : MB(M), FileKind(K) {} MemoryBufferRef MB; private: const Kind FileKind; }; template class ELFFileBase : public InputFile { public: typedef typename llvm::object::ELFFile::Elf_Shdr Elf_Shdr; typedef typename llvm::object::ELFFile::Elf_Sym Elf_Sym; typedef typename llvm::object::ELFFile::Elf_Word Elf_Word; typedef typename llvm::object::ELFFile::Elf_Sym_Range Elf_Sym_Range; ELFFileBase(Kind K, MemoryBufferRef M); static bool classof(const InputFile *F) { Kind K = F->kind(); return K == ObjectKind || K == SharedKind; } static ELFKind getELFKind(); const llvm::object::ELFFile &getObj() const { return ELFObj; } llvm::object::ELFFile &getObj() { return ELFObj; } uint16_t getEMachine() const { return getObj().getHeader()->e_machine; } uint8_t getOSABI() const { return getObj().getHeader()->e_ident[llvm::ELF::EI_OSABI]; } StringRef getStringTable() const { return StringTable; } uint32_t getSectionIndex(const Elf_Sym &Sym) const; protected: llvm::object::ELFFile ELFObj; const Elf_Shdr *Symtab = nullptr; ArrayRef SymtabSHNDX; StringRef StringTable; void initStringTable(); Elf_Sym_Range getNonLocalSymbols(); Elf_Sym_Range getSymbolsHelper(bool); }; // .o file. template class ObjectFile : public ELFFileBase { typedef ELFFileBase Base; typedef typename llvm::object::ELFFile::Elf_Sym Elf_Sym; typedef typename llvm::object::ELFFile::Elf_Shdr Elf_Shdr; typedef typename llvm::object::ELFFile::Elf_Sym_Range Elf_Sym_Range; typedef typename llvm::object::ELFFile::Elf_Word Elf_Word; typedef typename llvm::object::ELFFile::uintX_t uintX_t; // uint32 in ELFT's byte order typedef llvm::support::detail::packed_endian_specific_integral< uint32_t, ELFT::TargetEndianness, 2> uint32_X; StringRef getShtGroupSignature(const Elf_Shdr &Sec); ArrayRef getShtGroupEntries(const Elf_Shdr &Sec); public: static bool classof(const InputFile *F) { return F->kind() == Base::ObjectKind; } ArrayRef getSymbols() { return SymbolBodies; } explicit ObjectFile(MemoryBufferRef M); void parse(llvm::DenseSet &ComdatGroups); ArrayRef *> getSections() const { return Sections; } InputSectionBase *getSection(const Elf_Sym &Sym) const; SymbolBody *getSymbolBody(uint32_t SymbolIndex) const { uint32_t FirstNonLocal = this->Symtab->sh_info; if (SymbolIndex < FirstNonLocal) return nullptr; return SymbolBodies[SymbolIndex - FirstNonLocal]; } Elf_Sym_Range getLocalSymbols(); const Elf_Sym *getLocalSymbol(uintX_t SymIndex); const Elf_Shdr *getSymbolTable() const { return this->Symtab; }; // Get MIPS GP0 value defined by this file. This value represents the gp value // used to create the relocatable object and required to support // R_MIPS_GPREL16 / R_MIPS_GPREL32 relocations. uint32_t getMipsGp0() const; private: void initializeSections(llvm::DenseSet &ComdatGroups); void initializeSymbols(); InputSectionBase *createInputSection(const Elf_Shdr &Sec); SymbolBody *createSymbolBody(StringRef StringTable, const Elf_Sym *Sym); // List of all sections defined by this file. std::vector *> Sections; // List of all symbols referenced or defined by this file. std::vector SymbolBodies; // MIPS .reginfo section defined by this file. MipsReginfoInputSection *MipsReginfo = nullptr; llvm::BumpPtrAllocator Alloc; llvm::SpecificBumpPtrAllocator> MAlloc; llvm::SpecificBumpPtrAllocator> EHAlloc; }; class ArchiveFile : public InputFile { public: explicit ArchiveFile(MemoryBufferRef M) : InputFile(ArchiveKind, M) {} static bool classof(const InputFile *F) { return F->kind() == ArchiveKind; } void parse(); // Returns a memory buffer for a given symbol. An empty memory buffer // is returned if we have already returned the same memory buffer. // (So that we don't instantiate same members more than once.) MemoryBufferRef getMember(const Archive::Symbol *Sym); llvm::MutableArrayRef getLazySymbols() { return LazySymbols; } private: std::unique_ptr File; std::vector LazySymbols; llvm::DenseSet Seen; }; // .so file. template class SharedFile : public ELFFileBase { typedef ELFFileBase Base; typedef typename llvm::object::ELFFile::Elf_Shdr Elf_Shdr; typedef typename llvm::object::ELFFile::Elf_Sym Elf_Sym; typedef typename llvm::object::ELFFile::Elf_Word Elf_Word; typedef typename llvm::object::ELFFile::Elf_Sym_Range Elf_Sym_Range; std::vector> SymbolBodies; std::vector Undefs; StringRef SoName; public: StringRef getSoName() const { return SoName; } llvm::MutableArrayRef> getSharedSymbols() { return SymbolBodies; } const Elf_Shdr *getSection(const Elf_Sym &Sym) const; llvm::ArrayRef getUndefinedSymbols() { return Undefs; } static bool classof(const InputFile *F) { return F->kind() == Base::SharedKind; } explicit SharedFile(MemoryBufferRef M); void parseSoName(); void parseRest(); // Used for --as-needed bool AsNeeded = false; bool IsUsed = false; bool isNeeded() const { return !AsNeeded || IsUsed; } }; std::unique_ptr createObjectFile(MemoryBufferRef MB); std::unique_ptr createSharedFile(MemoryBufferRef MB); } // namespace elf2 } // namespace lld #endif