//===- AArch64CA53ErratumStub.cpp -----------------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "AArch64CA53ErratumStub.h" #include "AArch64InsnHelpers.h" #include "AArch64LDBackend.h" #include "AArch64RelocationHelpers.h" #include "AArch64Relocator.h" #include "mcld/Fragment/FragmentRef.h" #include "mcld/Fragment/Relocation.h" #include "mcld/IRBuilder.h" #include "mcld/LD/BranchIsland.h" #include "mcld/LD/LDSymbol.h" #include "mcld/LD/ResolveInfo.h" #include <llvm/ADT/StringExtras.h> #include <llvm/Support/ELF.h> #include <cassert> namespace mcld { //===----------------------------------------------------------------------===// // AArch64CA53ErratumStub //===----------------------------------------------------------------------===// const uint32_t AArch64CA53ErratumStub::TEMPLATE[] = { 0x00000000, // Placeholder for erratum insn 0x00000000, // Palceholder for branch instruction }; AArch64CA53ErratumStub::AArch64CA53ErratumStub() : m_pData(NULL), m_Name("erratum__prototype"), m_Size(0x0) { m_pData = TEMPLATE; m_Size = sizeof(TEMPLATE); addFixup(0x0, 0, AArch64Relocator::R_AARCH64_REWRITE_INSN); addFixup(0x4, 0, llvm::ELF::R_AARCH64_JUMP26); } /// for doClone AArch64CA53ErratumStub::AArch64CA53ErratumStub(const uint32_t* pData, size_t pSize, const char* pName, const_fixup_iterator pBegin, const_fixup_iterator pEnd) : m_pData(pData), m_Name(pName), m_Size(pSize) { for (const_fixup_iterator it = pBegin, ie = pEnd; it != ie; ++it) { addFixup(**it); } } AArch64CA53ErratumStub::~AArch64CA53ErratumStub() { } bool AArch64CA53ErratumStub::isMyDuty(const FragmentRef& pFragRef) const { return false; } void AArch64CA53ErratumStub::applyFixup(FragmentRef& pSrcFragRef, IRBuilder& pBuilder, BranchIsland& pIsland) { assert(isMyDuty(pSrcFragRef)); // Build stub symbol name. std::string sym_name("__"); sym_name.append(name()) .append(llvm::utohexstr(pIsland.numOfStubs())) .append("@") .append(pIsland.name()); // Create LDSymbol for the stub. LDSymbol* stub_sym = pBuilder.AddSymbol<IRBuilder::Force, IRBuilder::Unresolve>( sym_name, ResolveInfo::NoType, ResolveInfo::Define, ResolveInfo::Local, size(), initSymValue(), FragmentRef::Create(*this, initSymValue()), ResolveInfo::Default); setSymInfo(stub_sym->resolveInfo()); // Create the target symbol of the stub to the next instruction of erratum // pattarn. FragmentRef* target = FragmentRef::Create(*pSrcFragRef.frag(), pSrcFragRef.offset() + getErratumInsnOffset() + AArch64InsnHelpers::InsnSize); ResolveInfo* target_info = pBuilder.CreateLocalSymbol(*target); // Apply the fixups. fixup_iterator it = fixup_begin(); // Rewrite the first instruction as the erratum instruction. Relocation* reloc = Relocation::Create((*it)->type(), *(FragmentRef::Create(*this, (*it)->offset())), (*it)->addend()); reloc->setSymInfo(target_info); std::unique_ptr<unsigned[]> code(new unsigned[getErratumSequenceSize() / 4]); pSrcFragRef.memcpy(code.get(), getErratumSequenceSize(), 0); reloc->target() = code[getErratumInsnOffset() / AArch64InsnHelpers::InsnSize]; pIsland.addRelocation(*reloc); // Construct the second instruction as a branch to target. ++it; reloc = Relocation::Create((*it)->type(), *(FragmentRef::Create(*this, (*it)->offset())), (*it)->addend()); reloc->setSymInfo(target_info); reloc->target() = AArch64InsnHelpers::buildBranchInsn(); pIsland.addRelocation(*reloc); assert((++it) == fixup_end()); } const std::string& AArch64CA53ErratumStub::name() const { return m_Name; } const uint32_t* AArch64CA53ErratumStub::getData() const { return m_pData; } const uint8_t* AArch64CA53ErratumStub::getContent() const { return reinterpret_cast<const uint8_t*>(m_pData); } size_t AArch64CA53ErratumStub::size() const { return m_Size; } size_t AArch64CA53ErratumStub::alignment() const { return 8; } } // namespace mcld