//===- X86PLT.h -----------------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #ifndef MCLD_TARGET_X86_PLT_H #define MCLD_TARGET_X86_PLT_H #include <mcld/Target/PLT.h> namespace { const uint8_t x86_32_dyn_plt0[] = { 0xff, 0xb3, 0x04, 0, 0, 0, // pushl 0x4(%ebx) 0xff, 0xa3, 0x08, 0, 0, 0, // jmp *0x8(%ebx) 0x0f, 0x1f, 0x4, 0 // nopl 0(%eax) }; const uint8_t x86_32_dyn_plt1[] = { 0xff, 0xa3, 0, 0, 0, 0, // jmp *sym@GOT(%ebx) 0x68, 0, 0, 0, 0, // pushl $offset 0xe9, 0, 0, 0, 0 // jmp plt0 }; const uint8_t x86_32_exec_plt0[] = { 0xff, 0x35, 0, 0, 0, 0, // pushl .got + 4 0xff, 0x25, 0, 0, 0, 0, // jmp *(.got + 8) 0x0f, 0x1f, 0x4, 0 // nopl 0(%eax) }; const uint8_t x86_32_exec_plt1[] = { 0xff, 0x25, 0, 0, 0, 0, // jmp *(sym in .got) 0x68, 0, 0, 0, 0, // pushl $offset 0xe9, 0, 0, 0, 0 // jmp plt0 }; const uint8_t x86_64_plt0[] = { 0xff, 0x35, 0x8, 0, 0, 0, // pushq GOT + 8(%rip) 0xff, 0x25, 0x16, 0, 0, 0, // jmq *GOT + 16(%rip) 0x0f, 0x1f, 0x40, 0 // nopl 0(%rax) }; const uint8_t x86_64_plt1[] = { 0xff, 0x25, 0, 0, 0, 0, // jmpq *sym@GOTPCREL(%rip) 0x68, 0, 0, 0, 0, // pushq $index 0xe9, 0, 0, 0, 0 // jmpq plt0 }; } // anonymous namespace namespace mcld { class X86_32GOTPLT; class GOTEntry; class LinkerConfig; //===----------------------------------------------------------------------===// // X86_32PLT Entry //===----------------------------------------------------------------------===// class X86_32DynPLT0 : public PLT::Entry<sizeof(x86_32_dyn_plt0)> { public: X86_32DynPLT0(SectionData& pParent); }; class X86_32DynPLT1 : public PLT::Entry<sizeof(x86_32_dyn_plt1)> { public: X86_32DynPLT1(SectionData& pParent); }; class X86_32ExecPLT0 : public PLT::Entry<sizeof(x86_32_exec_plt0)> { public: X86_32ExecPLT0(SectionData& pParent); }; class X86_32ExecPLT1 : public PLT::Entry<sizeof(x86_32_exec_plt1)> { public: X86_32ExecPLT1(SectionData& pParent); }; //===----------------------------------------------------------------------===// // X86_64PLT Entry //===----------------------------------------------------------------------===// class X86_64PLT0 : public PLT::Entry<sizeof(x86_64_plt0)> { public: X86_64PLT0(SectionData& pParent); }; class X86_64PLT1 : public PLT::Entry<sizeof(x86_64_plt1)> { public: X86_64PLT1(SectionData& pParent); }; //===----------------------------------------------------------------------===// // X86PLT //===----------------------------------------------------------------------===// /** \class X86PLT * \brief X86 Procedure Linkage Table */ class X86PLT : public PLT { public: X86PLT(LDSection& pSection, const LinkerConfig& pConfig, int got_size); ~X86PLT(); // finalizeSectionSize - set LDSection size void finalizeSectionSize(); // hasPLT1 - return if this PLT has any PLT1 entry bool hasPLT1() const; void reserveEntry(size_t pNum = 1) ; PLTEntryBase* consume(); virtual void applyPLT0() = 0; virtual void applyPLT1() = 0; unsigned int getPLT0Size() const { return m_PLT0Size; } unsigned int getPLT1Size() const { return m_PLT1Size; } protected: PLTEntryBase* getPLT0() const; protected: // the last consumed entry. SectionData::iterator m_Last; const uint8_t *m_PLT0; const uint8_t *m_PLT1; unsigned int m_PLT0Size; unsigned int m_PLT1Size; const LinkerConfig& m_Config; }; //===----------------------------------------------------------------------===// // X86_32PLT //===----------------------------------------------------------------------===// /** \class X86_32PLT * \brief X86_32 Procedure Linkage Table */ class X86_32PLT : public X86PLT { public: X86_32PLT(LDSection& pSection, X86_32GOTPLT& pGOTPLT, const LinkerConfig& pConfig); void applyPLT0(); void applyPLT1(); private: X86_32GOTPLT& m_GOTPLT; }; //===----------------------------------------------------------------------===// // X86_64PLT //===----------------------------------------------------------------------===// /** \class X86_64PLT * \brief X86_64 Procedure Linkage Table */ class X86_64PLT : public X86PLT { public: X86_64PLT(LDSection& pSection, X86_64GOTPLT& pGOTPLT, const LinkerConfig& pConfig); void applyPLT0(); void applyPLT1(); private: X86_64GOTPLT& m_GOTPLT; }; } // namespace of mcld #endif