|  | //===-- AVRExpandPseudoInsts.cpp - Expand pseudo instructions -------------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // This file contains a pass that expands pseudo instructions into target | 
|  | // instructions. This pass should be run after register allocation but before | 
|  | // the post-regalloc scheduling pass. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "AVR.h" | 
|  | #include "AVRInstrInfo.h" | 
|  | #include "AVRTargetMachine.h" | 
|  | #include "MCTargetDesc/AVRMCTargetDesc.h" | 
|  |  | 
|  | #include "llvm/CodeGen/MachineFunctionPass.h" | 
|  | #include "llvm/CodeGen/MachineInstrBuilder.h" | 
|  | #include "llvm/CodeGen/MachineRegisterInfo.h" | 
|  | #include "llvm/CodeGen/TargetRegisterInfo.h" | 
|  |  | 
|  | using namespace llvm; | 
|  |  | 
|  | #define AVR_EXPAND_PSEUDO_NAME "AVR pseudo instruction expansion pass" | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | /// Expands "placeholder" instructions marked as pseudo into | 
|  | /// actual AVR instructions. | 
|  | class AVRExpandPseudo : public MachineFunctionPass { | 
|  | public: | 
|  | static char ID; | 
|  |  | 
|  | AVRExpandPseudo() : MachineFunctionPass(ID) { | 
|  | initializeAVRExpandPseudoPass(*PassRegistry::getPassRegistry()); | 
|  | } | 
|  |  | 
|  | bool runOnMachineFunction(MachineFunction &MF) override; | 
|  |  | 
|  | StringRef getPassName() const override { return AVR_EXPAND_PSEUDO_NAME; } | 
|  |  | 
|  | private: | 
|  | typedef MachineBasicBlock Block; | 
|  | typedef Block::iterator BlockIt; | 
|  |  | 
|  | const AVRRegisterInfo *TRI; | 
|  | const TargetInstrInfo *TII; | 
|  |  | 
|  | bool expandMBB(Block &MBB); | 
|  | bool expandMI(Block &MBB, BlockIt MBBI); | 
|  | template <unsigned OP> bool expand(Block &MBB, BlockIt MBBI); | 
|  |  | 
|  | MachineInstrBuilder buildMI(Block &MBB, BlockIt MBBI, unsigned Opcode) { | 
|  | return BuildMI(MBB, MBBI, MBBI->getDebugLoc(), TII->get(Opcode)); | 
|  | } | 
|  |  | 
|  | MachineInstrBuilder buildMI(Block &MBB, BlockIt MBBI, unsigned Opcode, | 
|  | Register DstReg) { | 
|  | return BuildMI(MBB, MBBI, MBBI->getDebugLoc(), TII->get(Opcode), DstReg); | 
|  | } | 
|  |  | 
|  | MachineRegisterInfo &getRegInfo(Block &MBB) { | 
|  | return MBB.getParent()->getRegInfo(); | 
|  | } | 
|  |  | 
|  | bool expandArith(unsigned OpLo, unsigned OpHi, Block &MBB, BlockIt MBBI); | 
|  | bool expandLogic(unsigned Op, Block &MBB, BlockIt MBBI); | 
|  | bool expandLogicImm(unsigned Op, Block &MBB, BlockIt MBBI); | 
|  | bool isLogicImmOpRedundant(unsigned Op, unsigned ImmVal) const; | 
|  | bool isLogicRegOpUndef(unsigned Op, unsigned ImmVal) const; | 
|  |  | 
|  | template <typename Func> bool expandAtomic(Block &MBB, BlockIt MBBI, Func f); | 
|  |  | 
|  | template <typename Func> | 
|  | bool expandAtomicBinaryOp(unsigned Opcode, Block &MBB, BlockIt MBBI, Func f); | 
|  |  | 
|  | bool expandAtomicBinaryOp(unsigned Opcode, Block &MBB, BlockIt MBBI); | 
|  |  | 
|  | /// Specific shift implementation for int8. | 
|  | bool expandLSLB7Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandLSRB7Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandASRB6Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandASRB7Rd(Block &MBB, BlockIt MBBI); | 
|  |  | 
|  | /// Specific shift implementation for int16. | 
|  | bool expandLSLW4Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandLSRW4Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandASRW7Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandLSLW8Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandLSRW8Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandASRW8Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandLSLW12Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandLSRW12Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandASRW14Rd(Block &MBB, BlockIt MBBI); | 
|  | bool expandASRW15Rd(Block &MBB, BlockIt MBBI); | 
|  |  | 
|  | // Common implementation of LPMWRdZ and ELPMWRdZ. | 
|  | bool expandLPMWELPMW(Block &MBB, BlockIt MBBI, bool IsELPM); | 
|  | // Common implementation of LPMBRdZ and ELPMBRdZ. | 
|  | bool expandLPMBELPMB(Block &MBB, BlockIt MBBI, bool IsELPM); | 
|  | // Common implementation of ROLBRdR1 and ROLBRdR17. | 
|  | bool expandROLBRd(Block &MBB, BlockIt MBBI); | 
|  | }; | 
|  |  | 
|  | char AVRExpandPseudo::ID = 0; | 
|  |  | 
|  | bool AVRExpandPseudo::expandMBB(MachineBasicBlock &MBB) { | 
|  | bool Modified = false; | 
|  |  | 
|  | BlockIt MBBI = MBB.begin(), E = MBB.end(); | 
|  | while (MBBI != E) { | 
|  | BlockIt NMBBI = std::next(MBBI); | 
|  | Modified |= expandMI(MBB, MBBI); | 
|  | MBBI = NMBBI; | 
|  | } | 
|  |  | 
|  | return Modified; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::runOnMachineFunction(MachineFunction &MF) { | 
|  | bool Modified = false; | 
|  |  | 
|  | const AVRSubtarget &STI = MF.getSubtarget<AVRSubtarget>(); | 
|  | TRI = STI.getRegisterInfo(); | 
|  | TII = STI.getInstrInfo(); | 
|  |  | 
|  | for (Block &MBB : MF) { | 
|  | bool ContinueExpanding = true; | 
|  | unsigned ExpandCount = 0; | 
|  |  | 
|  | // Continue expanding the block until all pseudos are expanded. | 
|  | do { | 
|  | assert(ExpandCount < 10 && "pseudo expand limit reached"); | 
|  | (void)ExpandCount; | 
|  |  | 
|  | bool BlockModified = expandMBB(MBB); | 
|  | Modified |= BlockModified; | 
|  | ExpandCount++; | 
|  |  | 
|  | ContinueExpanding = BlockModified; | 
|  | } while (ContinueExpanding); | 
|  | } | 
|  |  | 
|  | return Modified; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandArith(unsigned OpLo, unsigned OpHi, Block &MBB, | 
|  | BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg, DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(2).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool SrcIsKill = MI.getOperand(2).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBHI->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLogic(unsigned Op, Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg, DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(2).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool SrcIsKill = MI.getOperand(2).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, Op) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | // SREG is always implicitly dead | 
|  | MIBLO->getOperand(3).setIsDead(); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, Op) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::isLogicImmOpRedundant(unsigned Op, | 
|  | unsigned ImmVal) const { | 
|  |  | 
|  | // ANDI Rd, 0xff is redundant. | 
|  | if (Op == AVR::ANDIRdK && ImmVal == 0xff) | 
|  | return true; | 
|  |  | 
|  | // ORI Rd, 0x0 is redundant. | 
|  | if (Op == AVR::ORIRdK && ImmVal == 0x0) | 
|  | return true; | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::isLogicRegOpUndef(unsigned Op, unsigned ImmVal) const { | 
|  | // ANDI Rd, 0x00 clears all input bits. | 
|  | if (Op == AVR::ANDIRdK && ImmVal == 0x00) | 
|  | return true; | 
|  |  | 
|  | // ORI Rd, 0xff sets all input bits. | 
|  | if (Op == AVR::ORIRdK && ImmVal == 0xff) | 
|  | return true; | 
|  |  | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLogicImm(unsigned Op, Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | unsigned Lo8 = Imm & 0xff; | 
|  | unsigned Hi8 = (Imm >> 8) & 0xff; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | if (!isLogicImmOpRedundant(Op, Lo8)) { | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, Op) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Lo8); | 
|  |  | 
|  | // SREG is always implicitly dead | 
|  | MIBLO->getOperand(3).setIsDead(); | 
|  |  | 
|  | if (isLogicRegOpUndef(Op, Lo8)) | 
|  | MIBLO->getOperand(1).setIsUndef(true); | 
|  | } | 
|  |  | 
|  | if (!isLogicImmOpRedundant(Op, Hi8)) { | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, Op) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Hi8); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  |  | 
|  | if (isLogicRegOpUndef(Op, Hi8)) | 
|  | MIBHI->getOperand(1).setIsUndef(true); | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ADDWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | return expandArith(AVR::ADDRdRr, AVR::ADCRdRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ADCWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | return expandArith(AVR::ADCRdRr, AVR::ADCRdRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::SUBWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | return expandArith(AVR::SUBRdRr, AVR::SBCRdRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::SUBIWRdK>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, AVR::SUBIRdK) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, AVR::SBCIRdK) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | switch (MI.getOperand(2).getType()) { | 
|  | case MachineOperand::MO_GlobalAddress: { | 
|  | const GlobalValue *GV = MI.getOperand(2).getGlobal(); | 
|  | int64_t Offs = MI.getOperand(2).getOffset(); | 
|  | unsigned TF = MI.getOperand(2).getTargetFlags(); | 
|  | MIBLO.addGlobalAddress(GV, Offs, TF | AVRII::MO_NEG | AVRII::MO_LO); | 
|  | MIBHI.addGlobalAddress(GV, Offs, TF | AVRII::MO_NEG | AVRII::MO_HI); | 
|  | break; | 
|  | } | 
|  | case MachineOperand::MO_Immediate: { | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | MIBLO.addImm(Imm & 0xff); | 
|  | MIBHI.addImm((Imm >> 8) & 0xff); | 
|  | break; | 
|  | } | 
|  | default: | 
|  | llvm_unreachable("Unknown operand type!"); | 
|  | } | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBHI->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::SBCWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | return expandArith(AVR::SBCRdRr, AVR::SBCRdRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::SBCIWRdK>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | unsigned Lo8 = Imm & 0xff; | 
|  | unsigned Hi8 = (Imm >> 8) & 0xff; | 
|  | unsigned OpLo = AVR::SBCIRdK; | 
|  | unsigned OpHi = AVR::SBCIRdK; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Lo8); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBLO->getOperand(4).setIsKill(); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Hi8); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBHI->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ANDWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLogic(AVR::ANDRdRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ANDIWRdK>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLogicImm(AVR::ANDIRdK, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ORWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLogic(AVR::ORRdRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ORIWRdK>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLogicImm(AVR::ORIRdK, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::EORWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLogic(AVR::EORRdRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::COMWRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | unsigned OpLo = AVR::COMRd; | 
|  | unsigned OpHi = AVR::COMRd; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | // SREG is always implicitly dead | 
|  | MIBLO->getOperand(2).setIsDead(); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(2).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::NEGWRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register ZeroReg = MI.getOperand(2).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Do NEG on the upper byte. | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, AVR::NEGRd) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  | // SREG is always implicitly dead | 
|  | MIBHI->getOperand(2).setIsDead(); | 
|  |  | 
|  | // Do NEG on the lower byte. | 
|  | buildMI(MBB, MBBI, AVR::NEGRd) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | // Do an extra SBC. | 
|  | auto MISBCI = | 
|  | buildMI(MBB, MBBI, AVR::SBCRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(ZeroReg); | 
|  | if (ImpIsDead) | 
|  | MISBCI->getOperand(3).setIsDead(); | 
|  | // SREG is always implicitly killed | 
|  | MISBCI->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::CPWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg, DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool DstIsKill = MI.getOperand(0).isKill(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | unsigned OpLo = AVR::CPRdRr; | 
|  | unsigned OpHi = AVR::CPCRdRr; | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Low part | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | auto MIBHI = buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(2).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBHI->getOperand(3).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::CPCWRdRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg, DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool DstIsKill = MI.getOperand(0).isKill(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | unsigned OpLo = AVR::CPCRdRr; | 
|  | unsigned OpHi = AVR::CPCRdRr; | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | auto MIBLO = buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBLO->getOperand(3).setIsKill(); | 
|  |  | 
|  | auto MIBHI = buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(2).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBHI->getOperand(3).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LDIWRdK>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | unsigned OpLo = AVR::LDIRdK; | 
|  | unsigned OpHi = AVR::LDIRdK; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)); | 
|  |  | 
|  | switch (MI.getOperand(1).getType()) { | 
|  | case MachineOperand::MO_GlobalAddress: { | 
|  | const GlobalValue *GV = MI.getOperand(1).getGlobal(); | 
|  | int64_t Offs = MI.getOperand(1).getOffset(); | 
|  | unsigned TF = MI.getOperand(1).getTargetFlags(); | 
|  |  | 
|  | MIBLO.addGlobalAddress(GV, Offs, TF | AVRII::MO_LO); | 
|  | MIBHI.addGlobalAddress(GV, Offs, TF | AVRII::MO_HI); | 
|  | break; | 
|  | } | 
|  | case MachineOperand::MO_BlockAddress: { | 
|  | const BlockAddress *BA = MI.getOperand(1).getBlockAddress(); | 
|  | unsigned TF = MI.getOperand(1).getTargetFlags(); | 
|  |  | 
|  | MIBLO.add(MachineOperand::CreateBA(BA, TF | AVRII::MO_LO)); | 
|  | MIBHI.add(MachineOperand::CreateBA(BA, TF | AVRII::MO_HI)); | 
|  | break; | 
|  | } | 
|  | case MachineOperand::MO_Immediate: { | 
|  | unsigned Imm = MI.getOperand(1).getImm(); | 
|  |  | 
|  | MIBLO.addImm(Imm & 0xff); | 
|  | MIBHI.addImm((Imm >> 8) & 0xff); | 
|  | break; | 
|  | } | 
|  | default: | 
|  | llvm_unreachable("Unknown operand type!"); | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LDSWRdK>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | unsigned OpLo = AVR::LDSRdK; | 
|  | unsigned OpHi = AVR::LDSRdK; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)); | 
|  |  | 
|  | switch (MI.getOperand(1).getType()) { | 
|  | case MachineOperand::MO_GlobalAddress: { | 
|  | const GlobalValue *GV = MI.getOperand(1).getGlobal(); | 
|  | int64_t Offs = MI.getOperand(1).getOffset(); | 
|  | unsigned TF = MI.getOperand(1).getTargetFlags(); | 
|  |  | 
|  | MIBLO.addGlobalAddress(GV, Offs, TF); | 
|  | MIBHI.addGlobalAddress(GV, Offs + 1, TF); | 
|  | break; | 
|  | } | 
|  | case MachineOperand::MO_Immediate: { | 
|  | unsigned Imm = MI.getOperand(1).getImm(); | 
|  |  | 
|  | MIBLO.addImm(Imm); | 
|  | MIBHI.addImm(Imm + 1); | 
|  | break; | 
|  | } | 
|  | default: | 
|  | llvm_unreachable("Unknown operand type!"); | 
|  | } | 
|  |  | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LDWRdPtr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool DstIsKill = MI.getOperand(0).isKill(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  |  | 
|  | // DstReg has an earlyclobber so the register allocator will allocate them in | 
|  | // separate registers. | 
|  | assert(DstReg != SrcReg && "Dst and Src registers are the same!"); | 
|  |  | 
|  | if (STI.hasTinyEncoding()) { | 
|  | // Handle this case in the expansion of LDDWRdPtrQ because it is very | 
|  | // similar. | 
|  | buildMI(MBB, MBBI, AVR::LDDWRdPtrQ) | 
|  | .addDef(DstReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(0) | 
|  | .setMemRefs(MI.memoperands()); | 
|  |  | 
|  | } else { | 
|  | Register DstLoReg, DstHiReg; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Load low byte. | 
|  | buildMI(MBB, MBBI, AVR::LDRdPtr) | 
|  | .addReg(DstLoReg, RegState::Define) | 
|  | .addReg(SrcReg) | 
|  | .setMemRefs(MI.memoperands()); | 
|  |  | 
|  | // Load high byte. | 
|  | buildMI(MBB, MBBI, AVR::LDDRdPtrQ) | 
|  | .addReg(DstHiReg, RegState::Define) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(1) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LDWRdPtrPi>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsDead = MI.getOperand(1).isKill(); | 
|  | unsigned OpLo = AVR::LDRdPtrPi; | 
|  | unsigned OpHi = AVR::LDRdPtrPi; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | assert(DstReg != SrcReg && "SrcReg and DstReg cannot be the same"); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(SrcReg, RegState::Define) | 
|  | .addReg(SrcReg, RegState::Kill); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(SrcReg, RegState::Define | getDeadRegState(SrcIsDead)) | 
|  | .addReg(SrcReg, RegState::Kill); | 
|  |  | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LDWRdPtrPd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsDead = MI.getOperand(1).isKill(); | 
|  | unsigned OpLo = AVR::LDRdPtrPd; | 
|  | unsigned OpHi = AVR::LDRdPtrPd; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | assert(DstReg != SrcReg && "SrcReg and DstReg cannot be the same"); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(SrcReg, RegState::Define) | 
|  | .addReg(SrcReg, RegState::Kill); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(SrcReg, RegState::Define | getDeadRegState(SrcIsDead)) | 
|  | .addReg(SrcReg, RegState::Kill); | 
|  |  | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LDDWRdPtrQ>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | bool DstIsKill = MI.getOperand(0).isKill(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  |  | 
|  | // Since we add 1 to the Imm value for the high byte below, and 63 is the | 
|  | // highest Imm value allowed for the instruction, 62 is the limit here. | 
|  | assert(Imm <= 62 && "Offset is out of range"); | 
|  |  | 
|  | // DstReg has an earlyclobber so the register allocator will allocate them in | 
|  | // separate registers. | 
|  | assert(DstReg != SrcReg && "Dst and Src registers are the same!"); | 
|  |  | 
|  | if (STI.hasTinyEncoding()) { | 
|  | // Reduced tiny cores don't support load/store with displacement. However, | 
|  | // they do support postincrement. So we'll simply adjust the pointer before | 
|  | // and after and use postincrement to load multiple registers. | 
|  |  | 
|  | // Add offset. The offset can be 0 when expanding this instruction from the | 
|  | // more specific LDWRdPtr instruction. | 
|  | if (Imm != 0) { | 
|  | buildMI(MBB, MBBI, AVR::SUBIWRdK, SrcReg) | 
|  | .addReg(SrcReg) | 
|  | .addImm(0x10000 - Imm); | 
|  | } | 
|  |  | 
|  | // Do a word load with postincrement. This will be lowered to a two byte | 
|  | // load. | 
|  | buildMI(MBB, MBBI, AVR::LDWRdPtrPi) | 
|  | .addDef(DstReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(0) | 
|  | .setMemRefs(MI.memoperands()); | 
|  |  | 
|  | // If the pointer is used after the store instruction, subtract the new | 
|  | // offset (with 2 added after the postincrement instructions) so it is the | 
|  | // same as before. | 
|  | if (!SrcIsKill) { | 
|  | buildMI(MBB, MBBI, AVR::SUBIWRdK, SrcReg).addReg(SrcReg).addImm(Imm + 2); | 
|  | } | 
|  | } else { | 
|  | Register DstLoReg, DstHiReg; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Load low byte. | 
|  | buildMI(MBB, MBBI, AVR::LDDRdPtrQ) | 
|  | .addReg(DstLoReg, RegState::Define) | 
|  | .addReg(SrcReg) | 
|  | .addImm(Imm) | 
|  | .setMemRefs(MI.memoperands()); | 
|  |  | 
|  | // Load high byte. | 
|  | buildMI(MBB, MBBI, AVR::LDDRdPtrQ) | 
|  | .addReg(DstHiReg, RegState::Define) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Imm + 1) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLPMWELPMW(Block &MBB, BlockIt MBBI, bool IsELPM) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  | bool IsLPMRn = IsELPM ? STI.hasELPMX() : STI.hasLPMX(); | 
|  |  | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  |  | 
|  | // Set the I/O register RAMPZ for ELPM. | 
|  | if (IsELPM) { | 
|  | Register Bank = MI.getOperand(2).getReg(); | 
|  | // out RAMPZ, rtmp | 
|  | buildMI(MBB, MBBI, AVR::OUTARr).addImm(STI.getIORegRAMPZ()).addReg(Bank); | 
|  | } | 
|  |  | 
|  | // This is enforced by the @earlyclobber constraint. | 
|  | assert(DstReg != SrcReg && "SrcReg and DstReg cannot be the same"); | 
|  |  | 
|  | if (IsLPMRn) { | 
|  | unsigned OpLo = IsELPM ? AVR::ELPMRdZPi : AVR::LPMRdZPi; | 
|  | unsigned OpHi = IsELPM ? AVR::ELPMRdZ : AVR::LPMRdZ; | 
|  | // Load low byte. | 
|  | auto MIBLO = buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define) | 
|  | .addReg(SrcReg); | 
|  | // Load high byte. | 
|  | auto MIBHI = buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)); | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  | } else { | 
|  | unsigned Opc = IsELPM ? AVR::ELPM : AVR::LPM; | 
|  | // Load low byte, and copy to the low destination register. | 
|  | auto MIBLO = buildMI(MBB, MBBI, Opc); | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstLoReg, RegState::Define) | 
|  | .addReg(AVR::R0, RegState::Kill); | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | // Increase the Z register by 1. | 
|  | if (STI.hasADDSUBIW()) { | 
|  | // adiw r31:r30, 1 | 
|  | auto MIINC = buildMI(MBB, MBBI, AVR::ADIWRdK) | 
|  | .addReg(SrcReg, RegState::Define) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(1); | 
|  | MIINC->getOperand(3).setIsDead(); | 
|  | } else { | 
|  | // subi r30, 255 | 
|  | // sbci r31, 255 | 
|  | buildMI(MBB, MBBI, AVR::SUBIRdK) | 
|  | .addReg(SrcLoReg, RegState::Define) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(255); | 
|  | auto MIZHI = buildMI(MBB, MBBI, AVR::SBCIRdK) | 
|  | .addReg(SrcHiReg, RegState::Define) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(255); | 
|  | MIZHI->getOperand(3).setIsDead(); | 
|  | MIZHI->getOperand(4).setIsKill(); | 
|  | } | 
|  | // Load high byte, and copy to the high destination register. | 
|  | auto MIBHI = buildMI(MBB, MBBI, Opc); | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstHiReg, RegState::Define) | 
|  | .addReg(AVR::R0, RegState::Kill); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  | } | 
|  |  | 
|  | // Restore the Z register if it is not killed. | 
|  | if (!SrcIsKill) { | 
|  | if (STI.hasADDSUBIW()) { | 
|  | // sbiw r31:r30, 1 | 
|  | auto MIDEC = buildMI(MBB, MBBI, AVR::SBIWRdK) | 
|  | .addReg(SrcReg, RegState::Define) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(1); | 
|  | MIDEC->getOperand(3).setIsDead(); | 
|  | } else { | 
|  | // subi r30, 1 | 
|  | // sbci r31, 0 | 
|  | buildMI(MBB, MBBI, AVR::SUBIRdK) | 
|  | .addReg(SrcLoReg, RegState::Define) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(1); | 
|  | auto MIZHI = buildMI(MBB, MBBI, AVR::SBCIRdK) | 
|  | .addReg(SrcHiReg, RegState::Define) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(0); | 
|  | MIZHI->getOperand(3).setIsDead(); | 
|  | MIZHI->getOperand(4).setIsKill(); | 
|  | } | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LPMWRdZ>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLPMWELPMW(MBB, MBBI, false); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ELPMWRdZ>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLPMWELPMW(MBB, MBBI, true); | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLPMBELPMB(Block &MBB, BlockIt MBBI, bool IsELPM) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  | bool IsLPMRn = IsELPM ? STI.hasELPMX() : STI.hasLPMX(); | 
|  |  | 
|  | // Set the I/O register RAMPZ for ELPM (out RAMPZ, rtmp). | 
|  | if (IsELPM) { | 
|  | Register BankReg = MI.getOperand(2).getReg(); | 
|  | buildMI(MBB, MBBI, AVR::OUTARr).addImm(STI.getIORegRAMPZ()).addReg(BankReg); | 
|  | } | 
|  |  | 
|  | // Load byte. | 
|  | if (IsLPMRn) { | 
|  | unsigned Opc = IsELPM ? AVR::ELPMRdZ : AVR::LPMRdZ; | 
|  | auto MILB = buildMI(MBB, MBBI, Opc) | 
|  | .addReg(DstReg, RegState::Define) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)); | 
|  | MILB.setMemRefs(MI.memoperands()); | 
|  | } else { | 
|  | // For the basic ELPM/LPM instruction, its operand[0] is the implicit | 
|  | // 'Z' register, and its operand[1] is the implicit 'R0' register. | 
|  | unsigned Opc = IsELPM ? AVR::ELPM : AVR::LPM; | 
|  | auto MILB = buildMI(MBB, MBBI, Opc); | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstReg, RegState::Define) | 
|  | .addReg(AVR::R0, RegState::Kill); | 
|  | MILB.setMemRefs(MI.memoperands()); | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ELPMBRdZ>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLPMBELPMB(MBB, MBBI, true); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LPMBRdZ>(Block &MBB, BlockIt MBBI) { | 
|  | return expandLPMBELPMB(MBB, MBBI, false); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LPMWRdZPi>(Block &MBB, BlockIt MBBI) { | 
|  | llvm_unreachable("16-bit LPMPi is unimplemented"); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ELPMBRdZPi>(Block &MBB, BlockIt MBBI) { | 
|  | llvm_unreachable("8-bit ELPMPi is unimplemented"); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ELPMWRdZPi>(Block &MBB, BlockIt MBBI) { | 
|  | llvm_unreachable("16-bit ELPMPi is unimplemented"); | 
|  | } | 
|  |  | 
|  | template <typename Func> | 
|  | bool AVRExpandPseudo::expandAtomic(Block &MBB, BlockIt MBBI, Func f) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  |  | 
|  | // Store the SREG. | 
|  | buildMI(MBB, MBBI, AVR::INRdA) | 
|  | .addReg(STI.getTmpRegister(), RegState::Define) | 
|  | .addImm(STI.getIORegSREG()); | 
|  |  | 
|  | // Disable exceptions. | 
|  | buildMI(MBB, MBBI, AVR::BCLRs).addImm(7); // CLI | 
|  |  | 
|  | f(MI); | 
|  |  | 
|  | // Restore the status reg. | 
|  | buildMI(MBB, MBBI, AVR::OUTARr) | 
|  | .addImm(STI.getIORegSREG()) | 
|  | .addReg(STI.getTmpRegister()); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <typename Func> | 
|  | bool AVRExpandPseudo::expandAtomicBinaryOp(unsigned Opcode, Block &MBB, | 
|  | BlockIt MBBI, Func f) { | 
|  | return expandAtomic(MBB, MBBI, [&](MachineInstr &MI) { | 
|  | auto Op1 = MI.getOperand(0); | 
|  | auto Op2 = MI.getOperand(1); | 
|  |  | 
|  | MachineInstr &NewInst = | 
|  | *buildMI(MBB, MBBI, Opcode).add(Op1).add(Op2).getInstr(); | 
|  | f(NewInst); | 
|  | }); | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandAtomicBinaryOp(unsigned Opcode, Block &MBB, | 
|  | BlockIt MBBI) { | 
|  | return expandAtomicBinaryOp(Opcode, MBB, MBBI, [](MachineInstr &MI) {}); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::AtomicLoad8>(Block &MBB, BlockIt MBBI) { | 
|  | return expandAtomicBinaryOp(AVR::LDRdPtr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::AtomicLoad16>(Block &MBB, BlockIt MBBI) { | 
|  | return expandAtomicBinaryOp(AVR::LDWRdPtr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::AtomicStore8>(Block &MBB, BlockIt MBBI) { | 
|  | return expandAtomicBinaryOp(AVR::STPtrRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::AtomicStore16>(Block &MBB, BlockIt MBBI) { | 
|  | return expandAtomicBinaryOp(AVR::STWPtrRr, MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::AtomicFence>(Block &MBB, BlockIt MBBI) { | 
|  | // On AVR, there is only one core and so atomic fences do nothing. | 
|  | MBBI->eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::STSWKRr>(Block &MBB, BlockIt MBBI) { | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  |  | 
|  | auto MIB0 = buildMI(MBB, MBBI, AVR::STSKRr); | 
|  | auto MIB1 = buildMI(MBB, MBBI, AVR::STSKRr); | 
|  |  | 
|  | switch (MI.getOperand(0).getType()) { | 
|  | case MachineOperand::MO_GlobalAddress: { | 
|  | const GlobalValue *GV = MI.getOperand(0).getGlobal(); | 
|  | int64_t Offs = MI.getOperand(0).getOffset(); | 
|  | unsigned TF = MI.getOperand(0).getTargetFlags(); | 
|  |  | 
|  | if (STI.hasLowByteFirst()) { | 
|  | // Write the low byte first for XMEGA devices. | 
|  | MIB0.addGlobalAddress(GV, Offs, TF); | 
|  | MIB1.addGlobalAddress(GV, Offs + 1, TF); | 
|  | } else { | 
|  | // Write the high byte first for traditional devices. | 
|  | MIB0.addGlobalAddress(GV, Offs + 1, TF); | 
|  | MIB1.addGlobalAddress(GV, Offs, TF); | 
|  | } | 
|  |  | 
|  | break; | 
|  | } | 
|  | case MachineOperand::MO_Immediate: { | 
|  | unsigned Imm = MI.getOperand(0).getImm(); | 
|  |  | 
|  | if (STI.hasLowByteFirst()) { | 
|  | // Write the low byte first for XMEGA devices. | 
|  | MIB0.addImm(Imm); | 
|  | MIB1.addImm(Imm + 1); | 
|  | } else { | 
|  | // Write the high byte first for traditional devices. | 
|  | MIB0.addImm(Imm + 1); | 
|  | MIB1.addImm(Imm); | 
|  | } | 
|  |  | 
|  | break; | 
|  | } | 
|  | default: | 
|  | llvm_unreachable("Unknown operand type!"); | 
|  | } | 
|  |  | 
|  | if (STI.hasLowByteFirst()) { | 
|  | // Write the low byte first for XMEGA devices. | 
|  | MIB0.addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | MIB1.addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | } else { | 
|  | // Write the high byte first for traditional devices. | 
|  | MIB0.addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | MIB1.addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::STWPtrRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool DstIsKill = MI.getOperand(0).isKill(); | 
|  | bool DstIsUndef = MI.getOperand(0).isUndef(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  |  | 
|  | //: TODO: need to reverse this order like inw and stsw? | 
|  |  | 
|  | if (STI.hasTinyEncoding()) { | 
|  | // Handle this case in the expansion of STDWPtrQRr because it is very | 
|  | // similar. | 
|  | buildMI(MBB, MBBI, AVR::STDWPtrQRr) | 
|  | .addReg(DstReg, | 
|  | getKillRegState(DstIsKill) | getUndefRegState(DstIsUndef)) | 
|  | .addImm(0) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  |  | 
|  | } else { | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  | if (STI.hasLowByteFirst()) { | 
|  | buildMI(MBB, MBBI, AVR::STPtrRr) | 
|  | .addReg(DstReg, getUndefRegState(DstIsUndef)) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | buildMI(MBB, MBBI, AVR::STDPtrQRr) | 
|  | .addReg(DstReg, getUndefRegState(DstIsUndef)) | 
|  | .addImm(1) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | } else { | 
|  | buildMI(MBB, MBBI, AVR::STDPtrQRr) | 
|  | .addReg(DstReg, getUndefRegState(DstIsUndef)) | 
|  | .addImm(1) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | buildMI(MBB, MBBI, AVR::STPtrRr) | 
|  | .addReg(DstReg, getUndefRegState(DstIsUndef)) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | } | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::STWPtrPiRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(2).getReg(); | 
|  | unsigned Imm = MI.getOperand(3).getImm(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsKill = MI.getOperand(2).isKill(); | 
|  | unsigned OpLo = AVR::STPtrPiRr; | 
|  | unsigned OpHi = AVR::STPtrPiRr; | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  |  | 
|  | assert(DstReg != SrcReg && "SrcReg and DstReg cannot be the same"); | 
|  |  | 
|  | auto MIBLO = buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstReg, RegState::Define) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Imm); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Imm); | 
|  |  | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::STWPtrPdRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(2).getReg(); | 
|  | unsigned Imm = MI.getOperand(3).getImm(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsKill = MI.getOperand(2).isKill(); | 
|  | unsigned OpLo = AVR::STPtrPdRr; | 
|  | unsigned OpHi = AVR::STPtrPdRr; | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  |  | 
|  | assert(DstReg != SrcReg && "SrcReg and DstReg cannot be the same"); | 
|  |  | 
|  | auto MIBHI = buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstReg, RegState::Define) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Imm); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(Imm); | 
|  |  | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::STDWPtrQRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  |  | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsKill = MI.getOperand(0).isKill(); | 
|  | unsigned Imm = MI.getOperand(1).getImm(); | 
|  | Register SrcReg = MI.getOperand(2).getReg(); | 
|  | bool SrcIsKill = MI.getOperand(2).isKill(); | 
|  |  | 
|  | // STD's maximum displacement is 63, so larger stores have to be split into a | 
|  | // set of operations. | 
|  | // For avrtiny chips, STD is not available at all so we always have to fall | 
|  | // back to manual pointer adjustments. | 
|  | if (Imm >= 63 || STI.hasTinyEncoding()) { | 
|  | // Add offset. The offset can be 0 when expanding this instruction from the | 
|  | // more specific STWPtrRr instruction. | 
|  | if (Imm != 0) { | 
|  | buildMI(MBB, MBBI, AVR::SUBIWRdK, DstReg) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addImm(0x10000 - Imm); | 
|  | } | 
|  |  | 
|  | // Do the store. This is a word store, that will be expanded further. | 
|  | buildMI(MBB, MBBI, AVR::STWPtrPiRr, DstReg) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)) | 
|  | .addImm(0) | 
|  | .setMemRefs(MI.memoperands()); | 
|  |  | 
|  | // If the pointer is used after the store instruction, subtract the new | 
|  | // offset (with 2 added after the postincrement instructions) so it is the | 
|  | // same as before. | 
|  | if (!DstIsKill) { | 
|  | buildMI(MBB, MBBI, AVR::SUBIWRdK, DstReg) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addImm(Imm + 2); | 
|  | } | 
|  | } else { | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  |  | 
|  | if (STI.hasLowByteFirst()) { | 
|  | buildMI(MBB, MBBI, AVR::STDPtrQRr) | 
|  | .addReg(DstReg) | 
|  | .addImm(Imm) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | buildMI(MBB, MBBI, AVR::STDPtrQRr) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)) | 
|  | .addImm(Imm + 1) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | } else { | 
|  | buildMI(MBB, MBBI, AVR::STDPtrQRr) | 
|  | .addReg(DstReg) | 
|  | .addImm(Imm + 1) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | buildMI(MBB, MBBI, AVR::STDPtrQRr) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)) | 
|  | .addImm(Imm) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .setMemRefs(MI.memoperands()); | 
|  | } | 
|  | } | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::STDSPQRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | const MachineFunction &MF = *MBB.getParent(); | 
|  | const AVRSubtarget &STI = MF.getSubtarget<AVRSubtarget>(); | 
|  |  | 
|  | assert(MI.getOperand(0).getReg() == AVR::SP && | 
|  | "SP is expected as base pointer"); | 
|  |  | 
|  | assert(STI.getFrameLowering()->hasReservedCallFrame(MF) && | 
|  | "unexpected STDSPQRr pseudo instruction"); | 
|  | (void)STI; | 
|  |  | 
|  | MI.setDesc(TII->get(AVR::STDPtrQRr)); | 
|  | MI.getOperand(0).setReg(AVR::R29R28); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::STDWSPQRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | const MachineFunction &MF = *MBB.getParent(); | 
|  | const AVRSubtarget &STI = MF.getSubtarget<AVRSubtarget>(); | 
|  |  | 
|  | assert(MI.getOperand(0).getReg() == AVR::SP && | 
|  | "SP is expected as base pointer"); | 
|  |  | 
|  | assert(STI.getFrameLowering()->hasReservedCallFrame(MF) && | 
|  | "unexpected STDWSPQRr pseudo instruction"); | 
|  | (void)STI; | 
|  |  | 
|  | MI.setDesc(TII->get(AVR::STDWPtrQRr)); | 
|  | MI.getOperand(0).setReg(AVR::R29R28); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::INWRdA>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | unsigned Imm = MI.getOperand(1).getImm(); | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | unsigned OpLo = AVR::INRdA; | 
|  | unsigned OpHi = AVR::INRdA; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Since we add 1 to the Imm value for the high byte below, and 63 is the | 
|  | // highest Imm value allowed for the instruction, 62 is the limit here. | 
|  | assert(Imm <= 62 && "Address is out of range"); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addImm(Imm); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addImm(Imm + 1); | 
|  |  | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::OUTWARr>(Block &MBB, BlockIt MBBI) { | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | unsigned Imm = MI.getOperand(0).getImm(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  |  | 
|  | // Since we add 1 to the Imm value for the high byte below, and 63 is the | 
|  | // highest Imm value allowed for the instruction, 62 is the limit here. | 
|  | assert(Imm <= 62 && "Address is out of range"); | 
|  |  | 
|  | // 16 bit I/O writes need the high byte first on normal AVR devices, | 
|  | // and in reverse order for the XMEGA/XMEGA3/XMEGAU families. | 
|  | auto MIBHI = buildMI(MBB, MBBI, AVR::OUTARr) | 
|  | .addImm(STI.hasLowByteFirst() ? Imm : Imm + 1) | 
|  | .addReg(STI.hasLowByteFirst() ? SrcLoReg : SrcHiReg, | 
|  | getKillRegState(SrcIsKill)); | 
|  | auto MIBLO = buildMI(MBB, MBBI, AVR::OUTARr) | 
|  | .addImm(STI.hasLowByteFirst() ? Imm + 1 : Imm) | 
|  | .addReg(STI.hasLowByteFirst() ? SrcHiReg : SrcLoReg, | 
|  | getKillRegState(SrcIsKill)); | 
|  |  | 
|  | MIBLO.setMemRefs(MI.memoperands()); | 
|  | MIBHI.setMemRefs(MI.memoperands()); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::PUSHWRr>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | Register SrcReg = MI.getOperand(0).getReg(); | 
|  | bool SrcIsKill = MI.getOperand(0).isKill(); | 
|  | unsigned Flags = MI.getFlags(); | 
|  | unsigned OpLo = AVR::PUSHRr; | 
|  | unsigned OpHi = AVR::PUSHRr; | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  |  | 
|  | // Low part | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .setMIFlags(Flags); | 
|  |  | 
|  | // High part | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .setMIFlags(Flags); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::POPWRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | unsigned Flags = MI.getFlags(); | 
|  | unsigned OpLo = AVR::POPRd; | 
|  | unsigned OpHi = AVR::POPRd; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | buildMI(MBB, MBBI, OpHi, DstHiReg).setMIFlags(Flags); // High | 
|  | buildMI(MBB, MBBI, OpLo, DstLoReg).setMIFlags(Flags); // Low | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandROLBRd(Block &MBB, BlockIt MBBI) { | 
|  | // In AVR, the rotate instructions behave quite unintuitively. They rotate | 
|  | // bits through the carry bit in SREG, effectively rotating over 9 bits, | 
|  | // instead of 8. This is useful when we are dealing with numbers over | 
|  | // multiple registers, but when we actually need to rotate stuff, we have | 
|  | // to explicitly add the carry bit. | 
|  |  | 
|  | MachineInstr &MI = *MBBI; | 
|  | unsigned OpShift, OpCarry; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register ZeroReg = MI.getOperand(3).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | OpShift = AVR::ADDRdRr; | 
|  | OpCarry = AVR::ADCRdRr; | 
|  |  | 
|  | // add r16, r16 | 
|  | // adc r16, r1 | 
|  |  | 
|  | // Shift part | 
|  | buildMI(MBB, MBBI, OpShift) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(DstReg, RegState::Kill); | 
|  |  | 
|  | // Add the carry bit | 
|  | auto MIB = buildMI(MBB, MBBI, OpCarry) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)) | 
|  | .addReg(ZeroReg); | 
|  |  | 
|  | MIB->getOperand(3).setIsDead(); // SREG is always dead | 
|  | MIB->getOperand(4).setIsKill(); // SREG is always implicitly killed | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ROLBRdR1>(Block &MBB, BlockIt MBBI) { | 
|  | return expandROLBRd(MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ROLBRdR17>(Block &MBB, BlockIt MBBI) { | 
|  | return expandROLBRd(MBB, MBBI); | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::RORBRd>(Block &MBB, BlockIt MBBI) { | 
|  | // In AVR, the rotate instructions behave quite unintuitively. They rotate | 
|  | // bits through the carry bit in SREG, effectively rotating over 9 bits, | 
|  | // instead of 8. This is useful when we are dealing with numbers over | 
|  | // multiple registers, but when we actually need to rotate stuff, we have | 
|  | // to explicitly add the carry bit. | 
|  |  | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  |  | 
|  | // bst r16, 0 | 
|  | // ror r16 | 
|  | // bld r16, 7 | 
|  |  | 
|  | // Move the lowest bit from DstReg into the T bit | 
|  | buildMI(MBB, MBBI, AVR::BST).addReg(DstReg).addImm(0); | 
|  |  | 
|  | // Rotate to the right | 
|  | buildMI(MBB, MBBI, AVR::RORRd, DstReg).addReg(DstReg); | 
|  |  | 
|  | // Move the T bit into the highest bit of DstReg. | 
|  | buildMI(MBB, MBBI, AVR::BLD, DstReg).addReg(DstReg).addImm(7); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LSLWRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | unsigned OpLo = AVR::ADDRdRr; // ADD Rd, Rd <==> LSL Rd | 
|  | unsigned OpHi = AVR::ADCRdRr; // ADC Rd, Rd <==> ROL Rd | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Low part | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBHI->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LSLWHiRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // add hireg, hireg <==> lsl hireg | 
|  | auto MILSL = | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MILSL->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLSLW4Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // swap Rh | 
|  | // swap Rl | 
|  | buildMI(MBB, MBBI, AVR::SWAPRd) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  | buildMI(MBB, MBBI, AVR::SWAPRd) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, RegState::Kill); | 
|  |  | 
|  | // andi Rh, 0xf0 | 
|  | auto MI0 = | 
|  | buildMI(MBB, MBBI, AVR::ANDIRdK) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addImm(0xf0); | 
|  | // SREG is implicitly dead. | 
|  | MI0->getOperand(3).setIsDead(); | 
|  |  | 
|  | // eor Rh, Rl | 
|  | auto MI1 = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addReg(DstLoReg); | 
|  | // SREG is implicitly dead. | 
|  | MI1->getOperand(3).setIsDead(); | 
|  |  | 
|  | // andi Rl, 0xf0 | 
|  | auto MI2 = | 
|  | buildMI(MBB, MBBI, AVR::ANDIRdK) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addImm(0xf0); | 
|  | // SREG is implicitly dead. | 
|  | MI2->getOperand(3).setIsDead(); | 
|  |  | 
|  | // eor Rh, Rl | 
|  | auto MI3 = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstLoReg); | 
|  | if (ImpIsDead) | 
|  | MI3->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLSLW8Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // mov Rh, Rl | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg); | 
|  |  | 
|  | // clr Rl | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  | if (ImpIsDead) | 
|  | MIBLO->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLSLW12Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // mov Rh, Rl | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg); | 
|  |  | 
|  | // swap Rh | 
|  | buildMI(MBB, MBBI, AVR::SWAPRd) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  |  | 
|  | // andi Rh, 0xf0 | 
|  | auto MI0 = | 
|  | buildMI(MBB, MBBI, AVR::ANDIRdK) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addImm(0xf0); | 
|  | // SREG is implicitly dead. | 
|  | MI0->getOperand(3).setIsDead(); | 
|  |  | 
|  | // clr Rl | 
|  | auto MI1 = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  | if (ImpIsDead) | 
|  | MI1->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LSLWNRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | switch (Imm) { | 
|  | case 4: | 
|  | return expandLSLW4Rd(MBB, MBBI); | 
|  | case 8: | 
|  | return expandLSLW8Rd(MBB, MBBI); | 
|  | case 12: | 
|  | return expandLSLW12Rd(MBB, MBBI); | 
|  | default: | 
|  | llvm_unreachable("unimplemented lslwn"); | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LSRWRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | unsigned OpLo = AVR::RORRd; | 
|  | unsigned OpHi = AVR::LSRRd; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // High part | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBLO->getOperand(2).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBLO->getOperand(3).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LSRWLoRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // lsr loreg | 
|  | auto MILSR = | 
|  | buildMI(MBB, MBBI, AVR::LSRRd) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MILSR->getOperand(2).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLSRW4Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // swap Rh | 
|  | // swap Rl | 
|  | buildMI(MBB, MBBI, AVR::SWAPRd) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  | buildMI(MBB, MBBI, AVR::SWAPRd) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, RegState::Kill); | 
|  |  | 
|  | // andi Rl, 0xf | 
|  | auto MI0 = | 
|  | buildMI(MBB, MBBI, AVR::ANDIRdK) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, RegState::Kill) | 
|  | .addImm(0xf); | 
|  | // SREG is implicitly dead. | 
|  | MI0->getOperand(3).setIsDead(); | 
|  |  | 
|  | // eor Rl, Rh | 
|  | auto MI1 = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, RegState::Kill) | 
|  | .addReg(DstHiReg); | 
|  | // SREG is implicitly dead. | 
|  | MI1->getOperand(3).setIsDead(); | 
|  |  | 
|  | // andi Rh, 0xf | 
|  | auto MI2 = | 
|  | buildMI(MBB, MBBI, AVR::ANDIRdK) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addImm(0xf); | 
|  | // SREG is implicitly dead. | 
|  | MI2->getOperand(3).setIsDead(); | 
|  |  | 
|  | // eor Rl, Rh | 
|  | auto MI3 = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstHiReg); | 
|  | if (ImpIsDead) | 
|  | MI3->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLSRW8Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Move upper byte to lower byte. | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg); | 
|  |  | 
|  | // Clear upper byte. | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLSRW12Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Move upper byte to lower byte. | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg); | 
|  |  | 
|  | // swap Rl | 
|  | buildMI(MBB, MBBI, AVR::SWAPRd) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, RegState::Kill); | 
|  |  | 
|  | // andi Rl, 0xf | 
|  | auto MI0 = | 
|  | buildMI(MBB, MBBI, AVR::ANDIRdK) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addImm(0xf); | 
|  | // SREG is implicitly dead. | 
|  | MI0->getOperand(3).setIsDead(); | 
|  |  | 
|  | // Clear upper byte. | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LSRWNRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | switch (Imm) { | 
|  | case 4: | 
|  | return expandLSRW4Rd(MBB, MBBI); | 
|  | case 8: | 
|  | return expandLSRW8Rd(MBB, MBBI); | 
|  | case 12: | 
|  | return expandLSRW12Rd(MBB, MBBI); | 
|  | default: | 
|  | llvm_unreachable("unimplemented lsrwn"); | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::RORWRd>(Block &MBB, BlockIt MBBI) { | 
|  | llvm_unreachable("RORW unimplemented"); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ROLWRd>(Block &MBB, BlockIt MBBI) { | 
|  | llvm_unreachable("ROLW unimplemented"); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ASRWRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | unsigned OpLo = AVR::RORRd; | 
|  | unsigned OpHi = AVR::ASRRd; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // High part | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | auto MIBLO = | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBLO->getOperand(2).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIBLO->getOperand(3).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ASRWLoRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // asr loreg | 
|  | auto MIASR = | 
|  | buildMI(MBB, MBBI, AVR::ASRRd) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIASR->getOperand(2).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandASRW7Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // lsl r24 | 
|  | // mov r24,r25 | 
|  | // rol r24 | 
|  | // sbc r25,r25 | 
|  |  | 
|  | // lsl r24 <=> add r24, r24 | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, RegState::Kill) | 
|  | .addReg(DstLoReg, RegState::Kill); | 
|  |  | 
|  | // mov r24, r25 | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg); | 
|  |  | 
|  | // rol r24 <=> adc r24, r24 | 
|  | buildMI(MBB, MBBI, AVR::ADCRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | // sbc r25, r25 | 
|  | auto MISBC = | 
|  | buildMI(MBB, MBBI, AVR::SBCRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MISBC->getOperand(3).setIsDead(); | 
|  | // SREG is always implicitly killed | 
|  | MISBC->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandASRW8Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Move upper byte to lower byte. | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg); | 
|  |  | 
|  | // Move the sign bit to the C flag. | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  |  | 
|  | // Set upper byte to 0 or -1. | 
|  | auto MIBHI = | 
|  | buildMI(MBB, MBBI, AVR::SBCRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstHiReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIBHI->getOperand(3).setIsDead(); | 
|  | // SREG is always implicitly killed | 
|  | MIBHI->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  | bool AVRExpandPseudo::expandASRW14Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // lsl r25 | 
|  | // sbc r24, r24 | 
|  | // lsl r25 | 
|  | // mov r25, r24 | 
|  | // rol r24 | 
|  |  | 
|  | // lsl r25 <=> add r25, r25 | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  |  | 
|  | // sbc r24, r24 | 
|  | buildMI(MBB, MBBI, AVR::SBCRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, RegState::Kill) | 
|  | .addReg(DstLoReg, RegState::Kill); | 
|  |  | 
|  | // lsl r25 <=> add r25, r25 | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  |  | 
|  | // mov r25, r24 | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg); | 
|  |  | 
|  | // rol r24 <=> adc r24, r24 | 
|  | auto MIROL = | 
|  | buildMI(MBB, MBBI, AVR::ADCRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstLoReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIROL->getOperand(3).setIsDead(); | 
|  | // SREG is always implicitly killed | 
|  | MIROL->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandASRW15Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // lsl r25 | 
|  | // sbc r25, r25 | 
|  | // mov r24, r25 | 
|  |  | 
|  | // lsl r25 <=> add r25, r25 | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) | 
|  | .addReg(DstHiReg, RegState::Define) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  |  | 
|  | // sbc r25, r25 | 
|  | auto MISBC = | 
|  | buildMI(MBB, MBBI, AVR::SBCRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  | if (ImpIsDead) | 
|  | MISBC->getOperand(3).setIsDead(); | 
|  | // SREG is always implicitly killed | 
|  | MISBC->getOperand(4).setIsKill(); | 
|  |  | 
|  | // mov r24, r25 | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ASRWNRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | switch (Imm) { | 
|  | case 7: | 
|  | return expandASRW7Rd(MBB, MBBI); | 
|  | case 8: | 
|  | return expandASRW8Rd(MBB, MBBI); | 
|  | case 14: | 
|  | return expandASRW14Rd(MBB, MBBI); | 
|  | case 15: | 
|  | return expandASRW15Rd(MBB, MBBI); | 
|  | default: | 
|  | llvm_unreachable("unimplemented asrwn"); | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLSLB7Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  |  | 
|  | // ror r24 | 
|  | // clr r24 | 
|  | // ror r24 | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::RORRd) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | ->getOperand(3) | 
|  | .setIsUndef(true); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(DstReg, RegState::Kill); | 
|  |  | 
|  | auto MIRRC = | 
|  | buildMI(MBB, MBBI, AVR::RORRd) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIRRC->getOperand(2).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIRRC->getOperand(3).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LSLBNRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | switch (Imm) { | 
|  | case 7: | 
|  | return expandLSLB7Rd(MBB, MBBI); | 
|  | default: | 
|  | llvm_unreachable("unimplemented lslbn"); | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandLSRB7Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  |  | 
|  | // rol r24 | 
|  | // clr r24 | 
|  | // rol r24 | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::ADCRdRr) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | ->getOperand(4) | 
|  | .setIsUndef(true); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(DstReg, RegState::Kill); | 
|  |  | 
|  | auto MIRRC = | 
|  | buildMI(MBB, MBBI, AVR::ADCRdRr) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIRRC->getOperand(3).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIRRC->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::LSRBNRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | switch (Imm) { | 
|  | case 7: | 
|  | return expandLSRB7Rd(MBB, MBBI); | 
|  | default: | 
|  | llvm_unreachable("unimplemented lsrbn"); | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandASRB6Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  |  | 
|  | // bst r24, 6 | 
|  | // lsl r24 | 
|  | // sbc r24, r24 | 
|  | // bld r24, 0 | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::BST) | 
|  | .addReg(DstReg) | 
|  | .addImm(6) | 
|  | ->getOperand(2) | 
|  | .setIsUndef(true); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) // LSL Rd <==> ADD Rd, Rd | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(DstReg, RegState::Kill); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::SBCRdRr) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(DstReg, RegState::Kill); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::BLD) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)) | 
|  | .addImm(0) | 
|  | ->getOperand(3) | 
|  | .setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandASRB7Rd(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool DstIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(3).isDead(); | 
|  |  | 
|  | // lsl r24 | 
|  | // sbc r24, r24 | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, RegState::Kill) | 
|  | .addReg(DstReg, RegState::Kill); | 
|  |  | 
|  | auto MIRRC = | 
|  | buildMI(MBB, MBBI, AVR::SBCRdRr) | 
|  | .addReg(DstReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)) | 
|  | .addReg(DstReg, getKillRegState(DstIsKill)); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | MIRRC->getOperand(3).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | MIRRC->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::ASRBNRd>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | unsigned Imm = MI.getOperand(2).getImm(); | 
|  | switch (Imm) { | 
|  | case 6: | 
|  | return expandASRB6Rd(MBB, MBBI); | 
|  | case 7: | 
|  | return expandASRB7Rd(MBB, MBBI); | 
|  | default: | 
|  | llvm_unreachable("unimplemented asrbn"); | 
|  | return false; | 
|  | } | 
|  | } | 
|  |  | 
|  | template <> bool AVRExpandPseudo::expand<AVR::SEXT>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | // sext R17:R16, R17 | 
|  | // mov     r16, r17 | 
|  | // lsl     r17 | 
|  | // sbc     r17, r17 | 
|  | // sext R17:R16, R13 | 
|  | // mov     r16, r13 | 
|  | // mov     r17, r13 | 
|  | // lsl     r17 | 
|  | // sbc     r17, r17 | 
|  | // sext R17:R16, R16 | 
|  | // mov     r17, r16 | 
|  | // lsl     r17 | 
|  | // sbc     r17, r17 | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | if (SrcReg != DstLoReg) | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(SrcReg); | 
|  |  | 
|  | if (SrcReg != DstHiReg) { | 
|  | auto MOV = buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstHiReg, RegState::Define) | 
|  | .addReg(SrcReg); | 
|  | if (SrcReg != DstLoReg && SrcIsKill) | 
|  | MOV->getOperand(1).setIsKill(); | 
|  | } | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::ADDRdRr) // LSL Rd <==> ADD Rd, Rr | 
|  | .addReg(DstHiReg, RegState::Define) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  |  | 
|  | auto SBC = | 
|  | buildMI(MBB, MBBI, AVR::SBCRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill) | 
|  | .addReg(DstHiReg, RegState::Kill); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | SBC->getOperand(3).setIsDead(); | 
|  |  | 
|  | // SREG is always implicitly killed | 
|  | SBC->getOperand(4).setIsKill(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> bool AVRExpandPseudo::expand<AVR::ZEXT>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | // zext R25:R24, R20 | 
|  | // mov      R24, R20 | 
|  | // eor      R25, R25 | 
|  | // zext R25:R24, R24 | 
|  | // eor      R25, R25 | 
|  | // zext R25:R24, R25 | 
|  | // mov      R24, R25 | 
|  | // eor      R25, R25 | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | bool ImpIsDead = MI.getOperand(2).isDead(); | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | if (SrcReg != DstLoReg) { | 
|  | buildMI(MBB, MBBI, AVR::MOVRdRr) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(SrcReg, getKillRegState(SrcIsKill)); | 
|  | } | 
|  |  | 
|  | auto EOR = | 
|  | buildMI(MBB, MBBI, AVR::EORRdRr) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addReg(DstHiReg, RegState::Kill | RegState::Undef) | 
|  | .addReg(DstHiReg, RegState::Kill | RegState::Undef); | 
|  |  | 
|  | if (ImpIsDead) | 
|  | EOR->getOperand(3).setIsDead(); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::SPREAD>(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register DstLoReg, DstHiReg; | 
|  | Register DstReg = MI.getOperand(0).getReg(); | 
|  | bool DstIsDead = MI.getOperand(0).isDead(); | 
|  | unsigned Flags = MI.getFlags(); | 
|  | unsigned OpLo = AVR::INRdA; | 
|  | unsigned OpHi = AVR::INRdA; | 
|  | TRI->splitReg(DstReg, DstLoReg, DstHiReg); | 
|  |  | 
|  | // Low part | 
|  | buildMI(MBB, MBBI, OpLo) | 
|  | .addReg(DstLoReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addImm(0x3d) | 
|  | .setMIFlags(Flags); | 
|  |  | 
|  | // High part | 
|  | buildMI(MBB, MBBI, OpHi) | 
|  | .addReg(DstHiReg, RegState::Define | getDeadRegState(DstIsDead)) | 
|  | .addImm(0x3e) | 
|  | .setMIFlags(Flags); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | template <> | 
|  | bool AVRExpandPseudo::expand<AVR::SPWRITE>(Block &MBB, BlockIt MBBI) { | 
|  | const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); | 
|  | MachineInstr &MI = *MBBI; | 
|  | Register SrcLoReg, SrcHiReg; | 
|  | Register SrcReg = MI.getOperand(1).getReg(); | 
|  | bool SrcIsKill = MI.getOperand(1).isKill(); | 
|  | unsigned Flags = MI.getFlags(); | 
|  | TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::INRdA) | 
|  | .addReg(STI.getTmpRegister(), RegState::Define) | 
|  | .addImm(STI.getIORegSREG()) | 
|  | .setMIFlags(Flags); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::BCLRs).addImm(0x07).setMIFlags(Flags); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::OUTARr) | 
|  | .addImm(0x3e) | 
|  | .addReg(SrcHiReg, getKillRegState(SrcIsKill)) | 
|  | .setMIFlags(Flags); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::OUTARr) | 
|  | .addImm(STI.getIORegSREG()) | 
|  | .addReg(STI.getTmpRegister(), RegState::Kill) | 
|  | .setMIFlags(Flags); | 
|  |  | 
|  | buildMI(MBB, MBBI, AVR::OUTARr) | 
|  | .addImm(0x3d) | 
|  | .addReg(SrcLoReg, getKillRegState(SrcIsKill)) | 
|  | .setMIFlags(Flags); | 
|  |  | 
|  | MI.eraseFromParent(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool AVRExpandPseudo::expandMI(Block &MBB, BlockIt MBBI) { | 
|  | MachineInstr &MI = *MBBI; | 
|  | int Opcode = MBBI->getOpcode(); | 
|  |  | 
|  | #define EXPAND(Op)                                                             \ | 
|  | case Op:                                                                     \ | 
|  | return expand<Op>(MBB, MI) | 
|  |  | 
|  | switch (Opcode) { | 
|  | EXPAND(AVR::ADDWRdRr); | 
|  | EXPAND(AVR::ADCWRdRr); | 
|  | EXPAND(AVR::SUBWRdRr); | 
|  | EXPAND(AVR::SUBIWRdK); | 
|  | EXPAND(AVR::SBCWRdRr); | 
|  | EXPAND(AVR::SBCIWRdK); | 
|  | EXPAND(AVR::ANDWRdRr); | 
|  | EXPAND(AVR::ANDIWRdK); | 
|  | EXPAND(AVR::ORWRdRr); | 
|  | EXPAND(AVR::ORIWRdK); | 
|  | EXPAND(AVR::EORWRdRr); | 
|  | EXPAND(AVR::COMWRd); | 
|  | EXPAND(AVR::NEGWRd); | 
|  | EXPAND(AVR::CPWRdRr); | 
|  | EXPAND(AVR::CPCWRdRr); | 
|  | EXPAND(AVR::LDIWRdK); | 
|  | EXPAND(AVR::LDSWRdK); | 
|  | EXPAND(AVR::LDWRdPtr); | 
|  | EXPAND(AVR::LDWRdPtrPi); | 
|  | EXPAND(AVR::LDWRdPtrPd); | 
|  | case AVR::LDDWRdYQ: //: FIXME: remove this once PR13375 gets fixed | 
|  | EXPAND(AVR::LDDWRdPtrQ); | 
|  | EXPAND(AVR::LPMBRdZ); | 
|  | EXPAND(AVR::LPMWRdZ); | 
|  | EXPAND(AVR::LPMWRdZPi); | 
|  | EXPAND(AVR::ELPMBRdZ); | 
|  | EXPAND(AVR::ELPMWRdZ); | 
|  | EXPAND(AVR::ELPMBRdZPi); | 
|  | EXPAND(AVR::ELPMWRdZPi); | 
|  | EXPAND(AVR::AtomicLoad8); | 
|  | EXPAND(AVR::AtomicLoad16); | 
|  | EXPAND(AVR::AtomicStore8); | 
|  | EXPAND(AVR::AtomicStore16); | 
|  | EXPAND(AVR::AtomicFence); | 
|  | EXPAND(AVR::STSWKRr); | 
|  | EXPAND(AVR::STWPtrRr); | 
|  | EXPAND(AVR::STWPtrPiRr); | 
|  | EXPAND(AVR::STWPtrPdRr); | 
|  | EXPAND(AVR::STDWPtrQRr); | 
|  | EXPAND(AVR::STDSPQRr); | 
|  | EXPAND(AVR::STDWSPQRr); | 
|  | EXPAND(AVR::INWRdA); | 
|  | EXPAND(AVR::OUTWARr); | 
|  | EXPAND(AVR::PUSHWRr); | 
|  | EXPAND(AVR::POPWRd); | 
|  | EXPAND(AVR::ROLBRdR1); | 
|  | EXPAND(AVR::ROLBRdR17); | 
|  | EXPAND(AVR::RORBRd); | 
|  | EXPAND(AVR::LSLWRd); | 
|  | EXPAND(AVR::LSRWRd); | 
|  | EXPAND(AVR::RORWRd); | 
|  | EXPAND(AVR::ROLWRd); | 
|  | EXPAND(AVR::ASRWRd); | 
|  | EXPAND(AVR::LSLWHiRd); | 
|  | EXPAND(AVR::LSRWLoRd); | 
|  | EXPAND(AVR::ASRWLoRd); | 
|  | EXPAND(AVR::LSLWNRd); | 
|  | EXPAND(AVR::LSRWNRd); | 
|  | EXPAND(AVR::ASRWNRd); | 
|  | EXPAND(AVR::LSLBNRd); | 
|  | EXPAND(AVR::LSRBNRd); | 
|  | EXPAND(AVR::ASRBNRd); | 
|  | EXPAND(AVR::SEXT); | 
|  | EXPAND(AVR::ZEXT); | 
|  | EXPAND(AVR::SPREAD); | 
|  | EXPAND(AVR::SPWRITE); | 
|  | } | 
|  | #undef EXPAND | 
|  | return false; | 
|  | } | 
|  |  | 
|  | } // end of anonymous namespace | 
|  |  | 
|  | INITIALIZE_PASS(AVRExpandPseudo, "avr-expand-pseudo", AVR_EXPAND_PSEUDO_NAME, | 
|  | false, false) | 
|  | namespace llvm { | 
|  |  | 
|  | FunctionPass *createAVRExpandPseudoPass() { return new AVRExpandPseudo(); } | 
|  |  | 
|  | } // end of namespace llvm |