|  | //=- RISCVRedundantCopyElimination.cpp - Remove useless copy for RISC-V -----=// | 
|  | // | 
|  | // 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 pass removes unnecessary zero copies in BBs that are targets of | 
|  | // beqz/bnez instructions. For instance, the copy instruction in the code below | 
|  | // can be removed because the beqz jumps to BB#2 when a0 is zero. | 
|  | //  BB#1: | 
|  | //    beqz %a0, <BB#2> | 
|  | //  BB#2: | 
|  | //    %a0 = COPY %x0 | 
|  | // This pass should be run after register allocation. | 
|  | // | 
|  | // This pass is based on the earliest versions of | 
|  | // AArch64RedundantCopyElimination. | 
|  | // | 
|  | // FIXME: Support compares with constants other than zero? This is harder to | 
|  | // do on RISC-V since branches can't have immediates. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "RISCV.h" | 
|  | #include "RISCVInstrInfo.h" | 
|  | #include "llvm/ADT/Statistic.h" | 
|  | #include "llvm/CodeGen/MachineFunctionPass.h" | 
|  | #include "llvm/CodeGen/MachineRegisterInfo.h" | 
|  | #include "llvm/Support/Debug.h" | 
|  |  | 
|  | using namespace llvm; | 
|  |  | 
|  | #define DEBUG_TYPE "riscv-copyelim" | 
|  |  | 
|  | STATISTIC(NumCopiesRemoved, "Number of copies removed."); | 
|  |  | 
|  | namespace { | 
|  | class RISCVRedundantCopyElimination : public MachineFunctionPass { | 
|  | const MachineRegisterInfo *MRI; | 
|  | const TargetRegisterInfo *TRI; | 
|  | const TargetInstrInfo *TII; | 
|  |  | 
|  | public: | 
|  | static char ID; | 
|  | RISCVRedundantCopyElimination() : MachineFunctionPass(ID) {} | 
|  |  | 
|  | bool runOnMachineFunction(MachineFunction &MF) override; | 
|  | MachineFunctionProperties getRequiredProperties() const override { | 
|  | return MachineFunctionProperties().setNoVRegs(); | 
|  | } | 
|  |  | 
|  | StringRef getPassName() const override { | 
|  | return "RISC-V Redundant Copy Elimination"; | 
|  | } | 
|  |  | 
|  | private: | 
|  | bool optimizeBlock(MachineBasicBlock &MBB); | 
|  | }; | 
|  |  | 
|  | } // end anonymous namespace | 
|  |  | 
|  | char RISCVRedundantCopyElimination::ID = 0; | 
|  |  | 
|  | INITIALIZE_PASS(RISCVRedundantCopyElimination, "riscv-copyelim", | 
|  | "RISC-V Redundant Copy Elimination", false, false) | 
|  |  | 
|  | static bool | 
|  | guaranteesZeroRegInBlock(MachineBasicBlock &MBB, | 
|  | const SmallVectorImpl<MachineOperand> &Cond, | 
|  | MachineBasicBlock *TBB) { | 
|  | assert(Cond.size() == 3 && "Unexpected number of operands"); | 
|  | assert(TBB != nullptr && "Expected branch target basic block"); | 
|  | auto Opc = Cond[0].getImm(); | 
|  | if (Opc == RISCV::BEQ && Cond[2].isReg() && Cond[2].getReg() == RISCV::X0 && | 
|  | TBB == &MBB) | 
|  | return true; | 
|  | if (Opc == RISCV::BNE && Cond[2].isReg() && Cond[2].getReg() == RISCV::X0 && | 
|  | TBB != &MBB) | 
|  | return true; | 
|  | return false; | 
|  | } | 
|  |  | 
|  | bool RISCVRedundantCopyElimination::optimizeBlock(MachineBasicBlock &MBB) { | 
|  | // Check if the current basic block has a single predecessor. | 
|  | if (MBB.pred_size() != 1) | 
|  | return false; | 
|  |  | 
|  | // Check if the predecessor has two successors, implying the block ends in a | 
|  | // conditional branch. | 
|  | MachineBasicBlock *PredMBB = *MBB.pred_begin(); | 
|  | if (PredMBB->succ_size() != 2) | 
|  | return false; | 
|  |  | 
|  | MachineBasicBlock *TBB = nullptr, *FBB = nullptr; | 
|  | SmallVector<MachineOperand, 3> Cond; | 
|  | if (TII->analyzeBranch(*PredMBB, TBB, FBB, Cond, /*AllowModify*/ false) || | 
|  | Cond.empty()) | 
|  | return false; | 
|  |  | 
|  | // Is this a branch with X0? | 
|  | if (!guaranteesZeroRegInBlock(MBB, Cond, TBB)) | 
|  | return false; | 
|  |  | 
|  | Register TargetReg = Cond[1].getReg(); | 
|  | if (!TargetReg) | 
|  | return false; | 
|  |  | 
|  | bool Changed = false; | 
|  | MachineBasicBlock::iterator LastChange = MBB.begin(); | 
|  | // Remove redundant Copy instructions unless TargetReg is modified. | 
|  | for (MachineBasicBlock::iterator I = MBB.begin(), E = MBB.end(); I != E;) { | 
|  | MachineInstr *MI = &*I; | 
|  | ++I; | 
|  | if (MI->isCopy() && MI->getOperand(0).isReg() && | 
|  | MI->getOperand(1).isReg()) { | 
|  | Register DefReg = MI->getOperand(0).getReg(); | 
|  | Register SrcReg = MI->getOperand(1).getReg(); | 
|  |  | 
|  | if (SrcReg == RISCV::X0 && !MRI->isReserved(DefReg) && | 
|  | TargetReg == DefReg) { | 
|  | LLVM_DEBUG(dbgs() << "Remove redundant Copy : "); | 
|  | LLVM_DEBUG(MI->print(dbgs())); | 
|  |  | 
|  | MI->eraseFromParent(); | 
|  | Changed = true; | 
|  | LastChange = I; | 
|  | ++NumCopiesRemoved; | 
|  | continue; | 
|  | } | 
|  | } | 
|  |  | 
|  | if (MI->modifiesRegister(TargetReg, TRI)) | 
|  | break; | 
|  | } | 
|  |  | 
|  | if (!Changed) | 
|  | return false; | 
|  |  | 
|  | MachineBasicBlock::iterator CondBr = PredMBB->getFirstTerminator(); | 
|  | assert((CondBr->getOpcode() == RISCV::BEQ || | 
|  | CondBr->getOpcode() == RISCV::BNE) && | 
|  | "Unexpected opcode"); | 
|  | assert(CondBr->getOperand(0).getReg() == TargetReg && "Unexpected register"); | 
|  |  | 
|  | // Otherwise, we have to fixup the use-def chain, starting with the | 
|  | // BEQ/BNE. Conservatively mark as much as we can live. | 
|  | CondBr->clearRegisterKills(TargetReg, TRI); | 
|  |  | 
|  | // Add newly used reg to the block's live-in list if it isn't there already. | 
|  | if (!MBB.isLiveIn(TargetReg)) | 
|  | MBB.addLiveIn(TargetReg); | 
|  |  | 
|  | // Clear any kills of TargetReg between CondBr and the last removed COPY. | 
|  | for (MachineInstr &MMI : make_range(MBB.begin(), LastChange)) | 
|  | MMI.clearRegisterKills(TargetReg, TRI); | 
|  |  | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool RISCVRedundantCopyElimination::runOnMachineFunction(MachineFunction &MF) { | 
|  | if (skipFunction(MF.getFunction())) | 
|  | return false; | 
|  |  | 
|  | TII = MF.getSubtarget().getInstrInfo(); | 
|  | TRI = MF.getSubtarget().getRegisterInfo(); | 
|  | MRI = &MF.getRegInfo(); | 
|  |  | 
|  | bool Changed = false; | 
|  | for (MachineBasicBlock &MBB : MF) | 
|  | Changed |= optimizeBlock(MBB); | 
|  |  | 
|  | return Changed; | 
|  | } | 
|  |  | 
|  | FunctionPass *llvm::createRISCVRedundantCopyEliminationPass() { | 
|  | return new RISCVRedundantCopyElimination(); | 
|  | } |