| //===- MipsRelocator.h --------------------------------------------===// |
| // |
| // The MCLinker Project |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| #ifndef TARGET_MIPS_MIPSRELOCATOR_H_ |
| #define TARGET_MIPS_MIPSRELOCATOR_H_ |
| |
| #include "mcld/LD/Relocator.h" |
| #include "mcld/Support/GCFactory.h" |
| #include "mcld/Target/KeyEntryMap.h" |
| #include "MipsLDBackend.h" |
| |
| #include <llvm/ADT/DenseMapInfo.h> |
| |
| namespace mcld { |
| |
| class MipsRelocationInfo; |
| |
| /** \class MipsRelocator |
| * \brief MipsRelocator creates and destroys the Mips relocations. |
| */ |
| class MipsRelocator : public Relocator { |
| public: |
| enum ReservedEntryType { |
| None = 0, // no reserved entry |
| ReserveRel = 1, // reserve a dynamic relocation entry |
| ReserveGot = 2, // reserve a GOT entry |
| ReservePLT = 4 // reserve a PLT entry |
| }; |
| |
| typedef KeyEntryMap<ResolveInfo, PLTEntryBase> SymPLTMap; |
| typedef KeyEntryMap<ResolveInfo, Fragment> SymGOTPLTMap; |
| |
| public: |
| MipsRelocator(MipsGNULDBackend& pParent, const LinkerConfig& pConfig); |
| |
| /// scanRelocation - determine the empty entries are needed or not and |
| /// create the empty entries if needed. |
| /// For Mips, the GOT, GP, and dynamic relocation entries are check to create. |
| void scanRelocation(Relocation& pReloc, |
| IRBuilder& pBuilder, |
| Module& pModule, |
| LDSection& pSection, |
| Input& pInput); |
| |
| /// initializeScan - do initialization before scan relocations in pInput |
| /// @return - return true for initialization success |
| bool initializeScan(Input& pInput); |
| |
| /// finalizeScan - do finalization after scan relocations in pInput |
| /// @return - return true for finalization success |
| bool finalizeScan(Input& pInput); |
| |
| /// initializeApply - do initialization before apply relocations in pInput |
| /// @return - return true for initialization success |
| bool initializeApply(Input& pInput); |
| |
| /// finalizeApply - do finalization after apply relocations in pInput |
| /// @return - return true for finalization success |
| bool finalizeApply(Input& pInput); |
| |
| Result applyRelocation(Relocation& pReloc); |
| |
| /// getDebugStringOffset - get the offset from the relocation target. This is |
| /// used to get the debug string offset. |
| uint32_t getDebugStringOffset(Relocation& pReloc) const; |
| |
| /// applyDebugStringOffset - apply the relocation target to specific offset. |
| /// This is used to set the debug string offset. |
| void applyDebugStringOffset(Relocation& pReloc, uint32_t pOffset); |
| |
| const Input& getApplyingInput() const { return *m_pApplyingInput; } |
| |
| MipsGNULDBackend& getTarget() { return m_Target; } |
| |
| const MipsGNULDBackend& getTarget() const { return m_Target; } |
| |
| /// postponeRelocation - save R_MIPS_LO16 paired relocations |
| /// like R_MISP_HI16 and R_MIPS_GOT16 for a future processing. |
| void postponeRelocation(Relocation& pReloc); |
| |
| /// applyPostponedRelocations - apply all postponed relocations |
| /// paired with the R_MIPS_LO16 one. |
| void applyPostponedRelocations(MipsRelocationInfo& pLo16Reloc); |
| |
| /// isGpDisp - return true if relocation is against _gp_disp symbol. |
| bool isGpDisp(const Relocation& pReloc) const; |
| |
| /// getGPAddress - return address of _gp symbol. |
| Address getGPAddress(); |
| |
| /// getTPOffset - return TP_OFFSET against the SHF_TLS |
| /// section in the processing input. |
| Address getTPOffset(); |
| |
| /// getDTPOffset - return DTP_OFFSET against the SHF_TLS |
| /// section in the processing input. |
| Address getDTPOffset(); |
| |
| /// getGP0 - the gp value used to create the relocatable objects |
| /// in the processing input. |
| Address getGP0(); |
| |
| /// getLocalGOTEntry - initialize and return a local GOT entry |
| /// for this relocation. |
| Fragment& getLocalGOTEntry(MipsRelocationInfo& pReloc, |
| Relocation::DWord entryValue); |
| |
| /// getGlobalGOTEntry - initialize and return a global GOT entry |
| /// for this relocation. |
| Fragment& getGlobalGOTEntry(MipsRelocationInfo& pReloc); |
| |
| /// getTLSGOTEntry - initialize and return a TLS GOT entry |
| /// for this relocation. |
| Fragment& getTLSGOTEntry(MipsRelocationInfo& pReloc); |
| |
| /// getGOTOffset - return offset of corresponded GOT entry. |
| Address getGOTOffset(MipsRelocationInfo& pReloc); |
| |
| /// getTLSGOTOffset - return offset of corresponded TLS GOT entry. |
| Address getTLSGOTOffset(MipsRelocationInfo& pReloc); |
| |
| /// createDynRel - initialize dynamic relocation for the relocation. |
| void createDynRel(MipsRelocationInfo& pReloc); |
| |
| /// calcAHL - calculate combined addend used |
| /// by R_MIPS_HI16 and R_MIPS_GOT16 relocations. |
| uint64_t calcAHL(const MipsRelocationInfo& pHiReloc); |
| |
| /// isN64ABI - check current ABI |
| bool isN64ABI() const; |
| |
| const char* getName(Relocation::Type pType) const; |
| |
| const SymPLTMap& getSymPLTMap() const { return m_SymPLTMap; } |
| SymPLTMap& getSymPLTMap() { return m_SymPLTMap; } |
| |
| const SymGOTPLTMap& getSymGOTPLTMap() const { return m_SymGOTPLTMap; } |
| SymGOTPLTMap& getSymGOTPLTMap() { return m_SymGOTPLTMap; } |
| |
| protected: |
| /// setupRelDynEntry - create dynamic relocation entry. |
| virtual void setupRel32DynEntry(FragmentRef& pFragRef, ResolveInfo* pSym) = 0; |
| /// setupTLSDynEntry - create DTPMOD / DTPREL relocation entries |
| virtual void setupTLSDynEntry(Fragment& pFrag, ResolveInfo* pSym, |
| Relocation::Type pType) = 0; |
| |
| /// isLocalReloc - handle relocation as a local symbol |
| bool isLocalReloc(ResolveInfo& pSym) const; |
| |
| /// setupRelDynEntry - create dynamic relocation entry with specified type. |
| void setupRelDynEntry(FragmentRef& pFragRef, ResolveInfo* pSym, |
| Relocation::Type pType); |
| |
| private: |
| typedef llvm::DenseSet<Relocation*> RelocationSet; |
| typedef llvm::DenseMap<const ResolveInfo*, RelocationSet> SymRelocSetMap; |
| |
| private: |
| MipsGNULDBackend& m_Target; |
| SymPLTMap m_SymPLTMap; |
| SymGOTPLTMap m_SymGOTPLTMap; |
| Input* m_pApplyingInput; |
| SymRelocSetMap m_PostponedRelocs; |
| MipsRelocationInfo* m_CurrentLo16Reloc; |
| |
| private: |
| void scanLocalReloc(MipsRelocationInfo& pReloc, |
| IRBuilder& pBuilder, |
| const LDSection& pSection); |
| |
| void scanGlobalReloc(MipsRelocationInfo& pReloc, |
| IRBuilder& pBuilder, |
| const LDSection& pSection); |
| |
| /// isPostponed - relocation applying needs to be postponed. |
| bool isPostponed(const Relocation& pReloc) const; |
| |
| /// addCopyReloc - add a copy relocation into .rel.dyn for pSym |
| /// @param pSym - A resolved copy symbol that defined in BSS section |
| void addCopyReloc(ResolveInfo& pSym); |
| |
| /// defineSymbolforCopyReloc - allocate a space in BSS section and |
| /// and force define the copy of pSym to BSS section |
| /// @return the output LDSymbol of the copy symbol |
| LDSymbol& defineSymbolforCopyReloc(IRBuilder& pBuilder, |
| const ResolveInfo& pSym); |
| |
| /// isRel - returns true if REL relocation record format is expected |
| bool isRel() const; |
| }; |
| |
| /** \class Mips32Relocator |
| * \brief Mips32Relocator creates and destroys the Mips 32-bit relocations. |
| */ |
| class Mips32Relocator : public MipsRelocator { |
| public: |
| Mips32Relocator(Mips32GNULDBackend& pParent, const LinkerConfig& pConfig); |
| |
| private: |
| // MipsRelocator |
| void setupRel32DynEntry(FragmentRef& pFragRef, ResolveInfo* pSym); |
| void setupTLSDynEntry(Fragment& pFrag, ResolveInfo* pSym, |
| Relocation::Type pType); |
| Size getSize(Relocation::Type pType) const; |
| }; |
| |
| /** \class Mips64Relocator |
| * \brief Mips64Relocator creates and destroys the Mips 64-bit relocations. |
| */ |
| class Mips64Relocator : public MipsRelocator { |
| public: |
| Mips64Relocator(Mips64GNULDBackend& pParent, const LinkerConfig& pConfig); |
| |
| private: |
| // MipsRelocator |
| void setupRel32DynEntry(FragmentRef& pFragRef, ResolveInfo* pSym); |
| void setupTLSDynEntry(Fragment& pFrag, ResolveInfo* pSym, |
| Relocation::Type pType); |
| Size getSize(Relocation::Type pType) const; |
| }; |
| |
| } // namespace mcld |
| |
| #endif // TARGET_MIPS_MIPSRELOCATOR_H_ |