//===- AArch64PLT.cpp -----------------------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "AArch64GOT.h" #include "AArch64PLT.h" #include "AArch64RelocationHelpers.h" #include "mcld/LD/LDSection.h" #include "mcld/Support/MsgHandling.h" #include <llvm/Support/Casting.h> #include <new> namespace mcld { AArch64PLT0::AArch64PLT0(SectionData& pParent) : PLT::Entry<sizeof(aarch64_plt0)>(pParent) { } AArch64PLT1::AArch64PLT1(SectionData& pParent) : PLT::Entry<sizeof(aarch64_plt1)>(pParent) { } //===----------------------------------------------------------------------===// // AArch64PLT AArch64PLT::AArch64PLT(LDSection& pSection, AArch64GOT& pGOTPLT) : PLT(pSection), m_GOT(pGOTPLT) { new AArch64PLT0(*m_pSectionData); } AArch64PLT::~AArch64PLT() { } bool AArch64PLT::hasPLT1() const { return (m_pSectionData->size() > 1); } void AArch64PLT::finalizeSectionSize() { uint64_t size = (m_pSectionData->size() - 1) * sizeof(aarch64_plt1) + sizeof(aarch64_plt0); m_Section.setSize(size); uint32_t offset = 0; SectionData::iterator frag, fragEnd = m_pSectionData->end(); for (frag = m_pSectionData->begin(); frag != fragEnd; ++frag) { frag->setOffset(offset); offset += frag->size(); } } AArch64PLT1* AArch64PLT::create() { AArch64PLT1* plt1_entry = new (std::nothrow) AArch64PLT1(*m_pSectionData); if (!plt1_entry) fatal(diag::fail_allocate_memory_plt); return plt1_entry; } void AArch64PLT::applyPLT0() { // malloc plt0 iterator first = m_pSectionData->getFragmentList().begin(); assert(first != m_pSectionData->getFragmentList().end() && "FragmentList is empty, applyPLT0 failed!"); AArch64PLT0* plt0 = &(llvm::cast<AArch64PLT0>(*first)); uint32_t* data = NULL; data = static_cast<uint32_t*>(malloc(AArch64PLT0::EntrySize)); if (data == NULL) fatal(diag::fail_allocate_memory_plt); memcpy(data, aarch64_plt0, AArch64PLT0::EntrySize); // apply plt0 uint64_t plt_base = m_Section.addr(); assert(plt_base && ".plt base address is NULL!"); uint64_t got_base = m_GOT.addr(); assert(got_base && ".got base address is NULL!"); // apply 2nd instruction // get the address of got entry 2 uint64_t got_ent2_base = got_base + sizeof(AArch64GOTEntry::EntrySize) * 2; // compute the immediate AArch64Relocator::DWord imm = helper_get_page_address(got_ent2_base) - helper_get_page_address(plt_base + (sizeof(AArch64PLT0::EntrySize) * 8)); data[1] = helper_reencode_adr_imm(data[1], imm >> 12); // apply 3rd instruction data[2] = helper_reencode_add_imm(data[2], helper_get_page_offset(got_ent2_base) >> 3); // apply 4th instruction data[3] = helper_reencode_add_imm(data[3], helper_get_page_offset(got_ent2_base)); plt0->setValue(reinterpret_cast<unsigned char*>(data)); } void AArch64PLT::applyPLT1() { uint64_t plt_base = m_Section.addr(); assert(plt_base && ".plt base address is NULL!"); uint64_t got_base = m_GOT.addr(); assert(got_base && ".got base address is NULL!"); AArch64PLT::iterator it = m_pSectionData->begin(); AArch64PLT::iterator ie = m_pSectionData->end(); assert(it != ie && "FragmentList is empty, applyPLT1 failed!"); uint32_t GOTEntrySize = AArch64GOTEntry::EntrySize; // first gotplt1 address uint32_t GOTEntryAddress = got_base + GOTEntrySize * 3; // first plt1 address uint32_t PLTEntryAddress = plt_base + AArch64PLT0::EntrySize; ++it; // skip PLT0 uint32_t PLT1EntrySize = AArch64PLT1::EntrySize; AArch64PLT1* plt1 = NULL; uint32_t* Out = NULL; while (it != ie) { plt1 = &(llvm::cast<AArch64PLT1>(*it)); Out = static_cast<uint32_t*>(malloc(AArch64PLT1::EntrySize)); memcpy(Out, aarch64_plt1, AArch64PLT1::EntrySize); // apply 1st instruction AArch64Relocator::DWord imm = helper_get_page_address(GOTEntryAddress) - helper_get_page_address(PLTEntryAddress); Out[0] = helper_reencode_adr_imm(Out[0], imm >> 12); // apply 2nd instruction Out[1] = helper_reencode_add_imm( Out[1], helper_get_page_offset(GOTEntryAddress) >> 3); // apply 3rd instruction Out[2] = helper_reencode_add_imm(Out[2], helper_get_page_offset(GOTEntryAddress)); plt1->setValue(reinterpret_cast<unsigned char*>(Out)); ++it; GOTEntryAddress += GOTEntrySize; PLTEntryAddress += PLT1EntrySize; } m_GOT.applyGOTPLT(plt_base); } uint64_t AArch64PLT::emit(MemoryRegion& pRegion) { uint64_t result = 0x0; iterator it = begin(); unsigned char* buffer = pRegion.begin(); memcpy(buffer, llvm::cast<AArch64PLT0>((*it)).getValue(), AArch64PLT0::EntrySize); result += AArch64PLT0::EntrySize; ++it; AArch64PLT1* plt1 = NULL; AArch64PLT::iterator ie = end(); while (it != ie) { plt1 = &(llvm::cast<AArch64PLT1>(*it)); memcpy(buffer + result, plt1->getValue(), AArch64PLT1::EntrySize); result += AArch64PLT1::EntrySize; ++it; } return result; } } // namespace mcld