//===- AArch64CA53Erratum843419Stub.cpp -----------------------------------===// // // The MCLinker Project // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "AArch64CA53Erratum843419Stub.h" #include "AArch64InsnHelpers.h" #include "mcld/Fragment/FragmentRef.h" #include "mcld/Fragment/Relocation.h" #include "mcld/IRBuilder.h" #include "mcld/LD/BranchIsland.h" #include "mcld/LD/LDSection.h" #include "mcld/LD/LDSymbol.h" #include "mcld/LD/ResolveInfo.h" #include "mcld/LD/SectionData.h" #include <llvm/ADT/StringExtras.h> #include <llvm/Support/ELF.h> #include <cassert> namespace mcld { //===----------------------------------------------------------------------===// // AArch64CA53Erratum843419Stub //===----------------------------------------------------------------------===// AArch64CA53Erratum843419Stub::AArch64CA53Erratum843419Stub() { } /// for doClone AArch64CA53Erratum843419Stub::AArch64CA53Erratum843419Stub( const uint32_t* pData, size_t pSize, const char* pName, const_fixup_iterator pBegin, const_fixup_iterator pEnd) : AArch64CA53ErratumStub(pData, pSize, pName, pBegin, pEnd) { } AArch64CA53Erratum843419Stub::~AArch64CA53Erratum843419Stub() { } bool AArch64CA53Erratum843419Stub::isErratum843419Sequence(unsigned insn1, unsigned insn2, unsigned insn3) { unsigned rt; unsigned rt2; bool is_pair; bool is_load; return AArch64InsnHelpers::isMemOp(insn2, rt, rt2, is_pair, is_load) && (!is_pair || (is_pair && !is_load)) && AArch64InsnHelpers::isLDSTUIMM(insn3) && (AArch64InsnHelpers::getRn(insn3) == AArch64InsnHelpers::getRd(insn1)); } bool AArch64CA53Erratum843419Stub::isMyDuty(const FragmentRef& pFragRef) const { if ((pFragRef.offset() + AArch64InsnHelpers::InsnSize * 3) > pFragRef.frag()->size()) { return false; } // The first instruction must be ending at 0xFF8 or 0xFFC. const uint64_t vma = pFragRef.frag()->getParent()->getSection().addr() + pFragRef.getOutputOffset(); const unsigned page_offset = (vma & 0xFFF); if ((page_offset != 0xFF8) && (page_offset != 0xFFC)) { return false; } ErratumSequence code; pFragRef.memcpy(&code, AArch64InsnHelpers::InsnSize * 3, 0); if (isErratum843419Sequence(code.insns[0], code.insns[1], code.insns[2])) { return true; } return false; } Stub* AArch64CA53Erratum843419Stub::doClone() { return new AArch64CA53Erratum843419Stub(getData(), size(), "erratum_843419_veneer", fixup_begin(), fixup_end()); } } // namespace mcld