| //===-- 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; |
| |
| 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 IsExt); |
| }; |
| |
| 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::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 (!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(); |
| } |
| |
| 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 IsExt) { |
| MachineInstr &MI = *MBBI; |
| Register DstLoReg, DstHiReg; |
| Register DstReg = MI.getOperand(0).getReg(); |
| Register SrcReg = MI.getOperand(1).getReg(); |
| bool SrcIsKill = MI.getOperand(1).isKill(); |
| unsigned OpLo = IsExt ? AVR::ELPMRdZPi : AVR::LPMRdZPi; |
| unsigned OpHi = IsExt ? AVR::ELPMRdZ : AVR::LPMRdZ; |
| TRI->splitReg(DstReg, DstLoReg, DstHiReg); |
| |
| // Set the I/O register RAMPZ for ELPM. |
| if (IsExt) { |
| const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); |
| 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"); |
| |
| // 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()); |
| |
| 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); |
| } |
| |
| template <> |
| bool AVRExpandPseudo::expand<AVR::ELPMBRdZ>(Block &MBB, BlockIt MBBI) { |
| MachineInstr &MI = *MBBI; |
| Register DstReg = MI.getOperand(0).getReg(); |
| Register SrcReg = MI.getOperand(1).getReg(); |
| Register BankReg = MI.getOperand(2).getReg(); |
| bool SrcIsKill = MI.getOperand(1).isKill(); |
| const AVRSubtarget &STI = MBB.getParent()->getSubtarget<AVRSubtarget>(); |
| |
| // Set the I/O register RAMPZ for ELPM (out RAMPZ, rtmp). |
| buildMI(MBB, MBBI, AVR::OUTARr).addImm(STI.getIORegRAMPZ()).addReg(BankReg); |
| |
| // Load byte. |
| auto MILB = buildMI(MBB, MBBI, AVR::ELPMRdZ) |
| .addReg(DstReg, RegState::Define) |
| .addReg(SrcReg, getKillRegState(SrcIsKill)); |
| |
| MILB.setMemRefs(MI.memoperands()); |
| |
| MI.eraseFromParent(); |
| return true; |
| } |
| |
| 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) { |
| MachineInstr &MI = *MBBI; |
| Register SrcLoReg, SrcHiReg; |
| Register SrcReg = MI.getOperand(1).getReg(); |
| bool SrcIsKill = MI.getOperand(1).isKill(); |
| unsigned OpLo = AVR::STSKRr; |
| unsigned OpHi = AVR::STSKRr; |
| TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); |
| |
| // Write the high byte first in case this address belongs to a special |
| // I/O address with a special temporary register. |
| auto MIBHI = buildMI(MBB, MBBI, OpHi); |
| auto MIBLO = buildMI(MBB, MBBI, OpLo); |
| |
| 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(); |
| |
| MIBLO.addGlobalAddress(GV, Offs, TF); |
| MIBHI.addGlobalAddress(GV, Offs + 1, TF); |
| break; |
| } |
| case MachineOperand::MO_Immediate: { |
| unsigned Imm = MI.getOperand(0).getImm(); |
| |
| MIBLO.addImm(Imm); |
| MIBHI.addImm(Imm + 1); |
| break; |
| } |
| default: |
| llvm_unreachable("Unknown operand type!"); |
| } |
| |
| MIBLO.addReg(SrcLoReg, getKillRegState(SrcIsKill)); |
| MIBHI.addReg(SrcHiReg, getKillRegState(SrcIsKill)); |
| |
| MIBLO.setMemRefs(MI.memoperands()); |
| MIBHI.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); |
| 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()); |
| } |
| |
| 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 { |
| unsigned OpLo = AVR::STDPtrQRr; |
| unsigned OpHi = AVR::STDPtrQRr; |
| Register SrcLoReg, SrcHiReg; |
| TRI->splitReg(SrcReg, SrcLoReg, SrcHiReg); |
| |
| auto MIBLO = buildMI(MBB, MBBI, OpLo) |
| .addReg(DstReg) |
| .addImm(Imm) |
| .addReg(SrcLoReg, getKillRegState(SrcIsKill)); |
| |
| auto MIBHI = buildMI(MBB, MBBI, OpHi) |
| .addReg(DstReg, getKillRegState(DstIsKill)) |
| .addImm(Imm + 1) |
| .addReg(SrcHiReg, getKillRegState(SrcIsKill)); |
| |
| MIBLO.setMemRefs(MI.memoperands()); |
| MIBHI.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) { |
| MachineInstr &MI = *MBBI; |
| Register SrcLoReg, SrcHiReg; |
| unsigned Imm = MI.getOperand(0).getImm(); |
| Register SrcReg = MI.getOperand(1).getReg(); |
| bool SrcIsKill = MI.getOperand(1).isKill(); |
| unsigned OpLo = AVR::OUTARr; |
| unsigned OpHi = AVR::OUTARr; |
| 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 |
| auto MIBHI = buildMI(MBB, MBBI, OpHi) |
| .addImm(Imm + 1) |
| .addReg(SrcHiReg, getKillRegState(SrcIsKill)); |
| |
| auto MIBLO = buildMI(MBB, MBBI, OpLo) |
| .addImm(Imm) |
| .addReg(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; |
| } |
| |
| template <> |
| bool AVRExpandPseudo::expand<AVR::ROLBRd>(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(2).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::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::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::ROLBRd); |
| 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 |