//===- MipsGOT.h ----------------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef TARGET_MIPS_MIPSGOT_H_ #define TARGET_MIPS_MIPSGOT_H_ #include "mcld/ADT/SizeTraits.h" #include "mcld/Fragment/Relocation.h" #include "mcld/Support/MemoryRegion.h" #include "mcld/Target/GOT.h" #include <llvm/ADT/DenseMap.h> #include <llvm/ADT/DenseSet.h> #include <map> #include <set> #include <vector> namespace mcld { class Input; class LDSection; class LDSymbol; class OutputRelocSection; /** \class MipsGOT * \brief Mips Global Offset Table. */ class MipsGOT : public GOT { public: explicit MipsGOT(LDSection& pSection); /// Assign value to the GOT entry. virtual void setEntryValue(Fragment* entry, uint64_t pValue) = 0; /// Emit the global offset table. virtual uint64_t emit(MemoryRegion& pRegion) = 0; /// Address of _gp_disp symbol. uint64_t getGPDispAddress() const; void initializeScan(const Input& pInput); void finalizeScan(const Input& pInput); bool reserveLocalEntry(ResolveInfo& pInfo, int reloc, Relocation::DWord pAddend); bool reserveGlobalEntry(ResolveInfo& pInfo); bool reserveTLSGdEntry(ResolveInfo& pInfo); bool reserveTLSGotEntry(ResolveInfo& pInfo); bool reserveTLSLdmEntry(); size_t getLocalNum() const; ///< number of local symbols in primary GOT size_t getGlobalNum() const; ///< total number of global symbols bool isPrimaryGOTConsumed(); Fragment* consumeLocal(); Fragment* consumeGlobal(); Fragment* consumeTLS(Relocation::Type pType); uint64_t getGPAddr(const Input& pInput) const; uint64_t getGPRelOffset(const Input& pInput, const Fragment& pEntry) const; void recordGlobalEntry(const ResolveInfo* pInfo, Fragment* pEntry); Fragment* lookupGlobalEntry(const ResolveInfo* pInfo); void recordTLSEntry(const ResolveInfo* pInfo, Fragment* pEntry, Relocation::Type pType); Fragment* lookupTLSEntry(const ResolveInfo* pInfo, Relocation::Type pType); void recordLocalEntry(const ResolveInfo* pInfo, Relocation::DWord pAddend, Fragment* pEntry); Fragment* lookupLocalEntry(const ResolveInfo* pInfo, Relocation::DWord pAddend); /// hasGOT1 - return if this got section has any GOT1 entry bool hasGOT1() const; bool hasMultipleGOT() const; /// Create GOT entries and reserve dynrel entries. void finalizeScanning(OutputRelocSection& pRelDyn); /// Compare two symbols to define order in the .dynsym. bool dynSymOrderCompare(const LDSymbol* pX, const LDSymbol* pY) const; protected: /// Create GOT entry. virtual Fragment* createEntry(uint64_t pValue, SectionData* pParent) = 0; /// Size of GOT entry. virtual size_t getEntrySize() const = 0; /// Reserve GOT header entries. virtual void reserveHeader() = 0; private: /** \class GOTMultipart * \brief GOTMultipart counts local and global entries in the GOT. */ struct GOTMultipart { explicit GOTMultipart(size_t local = 0, size_t global = 0); typedef llvm::DenseSet<const Input*> InputSetType; size_t m_LocalNum; ///< number of reserved local entries size_t m_GlobalNum; ///< number of reserved global entries size_t m_TLSNum; ///< number of reserved TLS entries size_t m_TLSDynNum; ///< number of reserved TLS related dynamic relocations size_t m_ConsumedLocal; ///< consumed local entries size_t m_ConsumedGlobal; ///< consumed global entries size_t m_ConsumedTLS; ///< consumed TLS entries Fragment* m_pLastLocal; ///< the last consumed local entry Fragment* m_pLastGlobal; ///< the last consumed global entry Fragment* m_pLastTLS; ///< the last consumed TLS entry InputSetType m_Inputs; bool isConsumed() const; void consumeLocal(); void consumeGlobal(); void consumeTLS(Relocation::Type pType); }; /** \class LocalEntry * \brief LocalEntry local GOT entry descriptor. */ struct LocalEntry { const ResolveInfo* m_pInfo; Relocation::DWord m_Addend; bool m_IsGot16; LocalEntry(const ResolveInfo* pInfo, Relocation::DWord addend, bool isGot16); bool operator<(const LocalEntry& O) const; }; typedef std::vector<GOTMultipart> MultipartListType; // Set of global symbols. typedef llvm::DenseSet<const ResolveInfo*> SymbolSetType; // Map of symbols. If value is true, the symbol is referenced // in the current input only. If value is false, the symbol // is referenced in the other modules merged to the current GOT. typedef llvm::DenseMap<const ResolveInfo*, bool> SymbolUniqueMapType; // Set of local symbols. typedef std::set<LocalEntry> LocalSymbolSetType; MultipartListType m_MultipartList; ///< list of GOT's descriptors const Input* m_pInput; ///< current input // Global symbols merged to the current GOT // except symbols from the current input. SymbolSetType m_MergedGlobalSymbols; // Global symbols from the current input. SymbolUniqueMapType m_InputGlobalSymbols; // Set of symbols referenced by TLS GD relocations. SymbolSetType m_InputTLSGdSymbols; // Set of symbols referenced by TLS GOTTPREL relocation. SymbolSetType m_InputTLSGotSymbols; // There is a symbol referenced by TLS LDM relocations. bool m_HasTLSLdmSymbol; // Local symbols merged to the current GOT // except symbols from the current input. LocalSymbolSetType m_MergedLocalSymbols; // Local symbols from the current input. LocalSymbolSetType m_InputLocalSymbols; size_t m_CurrentGOTPart; typedef llvm::DenseMap<const LDSymbol*, unsigned> SymbolOrderMapType; SymbolOrderMapType m_SymbolOrderMap; void initGOTList(); void changeInput(); bool isGOTFull() const; void split(); void reserve(size_t pNum); private: struct GotEntryKey { size_t m_GOTPage; const ResolveInfo* m_pInfo; Relocation::DWord m_Addend; bool operator<(const GotEntryKey& key) const { if (m_GOTPage != key.m_GOTPage) return m_GOTPage < key.m_GOTPage; if (m_pInfo != key.m_pInfo) return m_pInfo < key.m_pInfo; return m_Addend < key.m_Addend; } }; typedef std::map<GotEntryKey, Fragment*> GotEntryMapType; GotEntryMapType m_GotLocalEntriesMap; GotEntryMapType m_GotGlobalEntriesMap; GotEntryMapType m_GotTLSGdEntriesMap; GotEntryMapType m_GotTLSGotEntriesMap; Fragment* m_GotTLSLdmEntry; }; /** \class Mips32GOT * \brief Mips 32-bit Global Offset Table. */ class Mips32GOT : public MipsGOT { public: explicit Mips32GOT(LDSection& pSection); private: typedef GOT::Entry<4> Mips32GOTEntry; // MipsGOT virtual void setEntryValue(Fragment* entry, uint64_t pValue); virtual uint64_t emit(MemoryRegion& pRegion); virtual Fragment* createEntry(uint64_t pValue, SectionData* pParent); virtual size_t getEntrySize() const; virtual void reserveHeader(); }; /** \class Mips64GOT * \brief Mips 64-bit Global Offset Table. */ class Mips64GOT : public MipsGOT { public: explicit Mips64GOT(LDSection& pSection); private: typedef GOT::Entry<8> Mips64GOTEntry; // MipsGOT virtual void setEntryValue(Fragment* entry, uint64_t pValue); virtual uint64_t emit(MemoryRegion& pRegion); virtual Fragment* createEntry(uint64_t pValue, SectionData* pParent); virtual size_t getEntrySize() const; virtual void reserveHeader(); }; } // namespace mcld #endif // TARGET_MIPS_MIPSGOT_H_