diff --git a/llvm/lib/Target/PowerPC/PPCMIPeephole.cpp b/llvm/lib/Target/PowerPC/PPCMIPeephole.cpp --- a/llvm/lib/Target/PowerPC/PPCMIPeephole.cpp +++ b/llvm/lib/Target/PowerPC/PPCMIPeephole.cpp @@ -379,6 +379,7 @@ // Perform peephole optimizations. bool PPCMIPeephole::simplifyCode(void) { bool Simplified = false; + bool TrapOpt = false; MachineInstr* ToErase = nullptr; std::map TOCSaves; const TargetRegisterInfo *TRI = &TII->getRegisterInfo(); @@ -420,6 +421,13 @@ ToErase->eraseFromParent(); ToErase = nullptr; } + // If a conditional trap instruction got optimized to an + // unconditional trap, eliminate all the instructions between + // the trap and the terminator of the MBB. + if (TrapOpt && !MI.isTerminator()) { + ToErase = &MI; + continue; + } // Ignore debug instructions. if (MI.isDebugInstr()) @@ -997,6 +1005,70 @@ ++NumRotatesCollapsed; break; } + // Replace TDI/TWI with the unconditional trap "TRAP" if always trap, + // delete them if it will never trap. + case PPC::TDI: + case PPC::TWI: { + MachineInstr *LiMIA = getVRegDefOrNull(&MI.getOperand(1), MRI); + // will not optimize if no value set + if (!(LiMIA && (LiMIA->getOpcode() == PPC::LI || + LiMIA->getOpcode() == PPC::LI8))) + break; + auto LiImm = MI.getOperand(0).getImm(); + // unconditionally trap for the following cases + if ((LiImm == 31) || + ((LiImm & 0x10) && ((int64_t)LiMIA->getOperand(1).getImm() < + (int64_t)MI.getOperand(2).getImm())) || + ((LiImm & 0x8) && ((int64_t)LiMIA->getOperand(1).getImm() > + (int64_t)MI.getOperand(2).getImm())) || + ((LiImm & 0x2) && ((uint64_t)LiMIA->getOperand(1).getImm() < + (uint64_t)MI.getOperand(2).getImm())) || + ((LiImm & 0x1) && ((uint64_t)LiMIA->getOperand(1).getImm() > + (uint64_t)MI.getOperand(2).getImm())) || + ((LiImm & 0x4) && + (LiMIA->getOperand(1).getImm() == MI.getOperand(2).getImm()))) { + BuildMI(MBB, &MI, MI.getDebugLoc(), TII->get(PPC::TRAP)); + TrapOpt = true; + } + // will not trap for the other cases + ToErase = &MI; + Simplified = true; + break; + } + // Replace TD/TW with the unconditional trap "TRAP" if it always trap, + // delete them if it will never trap. + case PPC::TD: + case PPC::TW: { + MachineInstr *LiMIA = getVRegDefOrNull(&MI.getOperand(1), MRI); + MachineInstr *LiMIB = getVRegDefOrNull(&MI.getOperand(2), MRI); + // will not optimize if no value set + if (!(LiMIA && (LiMIA->getOpcode() == PPC::LI || + LiMIA->getOpcode() == PPC::LI8))) + break; + if (!(LiMIB && (LiMIB->getOpcode() == PPC::LI || + LiMIB->getOpcode() == PPC::LI8))) + break; + auto LiImm = MI.getOperand(0).getImm(); + // unconditionally trap for the following cases + if ((LiImm == 31) || + ((LiImm & 0x10) && ((int64_t)LiMIA->getOperand(1).getImm() < + (int64_t)LiMIB->getOperand(1).getImm())) || + ((LiImm & 0x8) && ((int64_t)LiMIA->getOperand(1).getImm() > + (int64_t)LiMIB->getOperand(1).getImm())) || + ((LiImm & 0x2) && ((uint64_t)LiMIA->getOperand(1).getImm() < + (uint64_t)LiMIB->getOperand(1).getImm())) || + ((LiImm & 0x1) && ((uint64_t)LiMIA->getOperand(1).getImm() > + (uint64_t)LiMIB->getOperand(1).getImm())) || + ((LiImm & 0x4) && (LiMIA->getOperand(1).getImm() == + LiMIB->getOperand(1).getImm()))) { + BuildMI(MBB, &MI, MI.getDebugLoc(), TII->get(PPC::TRAP)); + TrapOpt = true; + } + // will not trap for the other cases + ToErase = &MI; + Simplified = true; + break; + } } } @@ -1006,6 +1078,8 @@ ToErase->eraseFromParent(); ToErase = nullptr; } + // reset TrapOpt to false at the end of basic block + TrapOpt = false; } // Eliminate all the TOC save instructions which are redundant. diff --git a/llvm/test/CodeGen/PowerPC/mi-peepholes-trap-opt.mir b/llvm/test/CodeGen/PowerPC/mi-peepholes-trap-opt.mir new file mode 100644 --- /dev/null +++ b/llvm/test/CodeGen/PowerPC/mi-peepholes-trap-opt.mir @@ -0,0 +1,448 @@ +# RUN: llc -mtriple powerpc64le-unknown-linux-gnu -mcpu=pwr8 -x mir < %s \ +# RUN: -verify-machineinstrs -start-before=ppc-mi-peepholes | FileCheck %s + +--- +name: conditional_trap_opt_reg_implicit_def +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = IMPLICIT_DEF + %1:gprc = IMPLICIT_DEF + %2:g8rc = IMPLICIT_DEF + %3:g8rc = IMPLICIT_DEF + TW 8, %0, %1 + TD 8, %2, %3 + TWI 24, %0, 0 + TDI 24, %2, 0 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_reg_implicit_def + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: twgt 3, 3 + # CHECK-NEXT: tdgt 3, 3 + # CHECK-NEXT: twnei 3, 0 + # CHECK-NEXT: tdnei 3, 0 + # CHECK-NEXT: blr +--- +name: conditional_trap_opt_TW_31 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI 3 + %1:gprc = LI 0 + TW 31, %1, %0 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TW_31 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TW_16 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI 5 + %1:gprc = LI 1 + TW 16, %1, %0 + TW 16, %0, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TW_16 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TW_8 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI -1 + %1:gprc = LI 10 + TW 8, %1, %0 + TW 8, %0, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TW_8 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TW_2 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI -1 + %1:gprc = LI 2 + TW 2, %1, %0 + TW 2, %0, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TW_2 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TW_1 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI -3 + %1:gprc = LI 4 + TW 1, %1, %0 + TW 1, %0, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TW_1 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TW_4 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI 5 + %1:gprc = LI 1 + TW 4, %1, %0 + TW 4, %1, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TW_4 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TWI_31 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI 3 + TWI 31, %0, 0 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TWI_31 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TWI_16 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI 5 + %1:gprc = LI 1 + TWI 16, %1, 5 + TWI 16, %0, 1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TWI_16 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TWI_8 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI -1 + %1:gprc = LI 10 + TWI 8, %1, -1 + TWI 8, %0, 10 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TWI_8 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TWI_2 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI -1 + %1:gprc = LI 2 + TWI 2, %1, -1 + TWI 2, %0, 2 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TWI_2 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TWI_1 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI -3 + %1:gprc = LI 4 + TWI 1, %1, -3 + TWI 1, %0, 4 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TWI_1 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TWI_4 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI 5 + %1:gprc = LI 1 + TWI 4, %1, 5 + TWI 4, %1, 1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TWI_4 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TD_31 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 3 + %1:g8rc = LI8 0 + TD 31, %1, %0 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TD_31 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TD_16 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 5 + %1:g8rc = LI8 1 + TD 16, %1, %0 + TD 16, %0, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TD_16 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TD_8 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 -1 + %1:g8rc = LI8 10 + TD 8, %1, %0 + TD 8, %0, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TD_8 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TD_2 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 -1 + %1:g8rc = LI8 2 + TD 2, %1, %0 + TD 2, %0, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TD_2 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TD_1 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 -3 + %1:g8rc = LI8 4 + TD 1, %1, %0 + TD 1, %0, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TD_1 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TD_4 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 5 + %1:g8rc = LI8 1 + TD 4, %1, %0 + TD 4, %1, %1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TD_4 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TDI_31 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 3 + TDI 31, %0, 0 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TDI_31 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TDI_16 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 5 + %1:g8rc = LI8 1 + TDI 16, %1, 5 + TDI 16, %0, 1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TDI_16 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TDI_8 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 -1 + %1:g8rc = LI8 10 + TDI 8, %1, -1 + TDI 8, %0, 10 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TDI_8 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TDI_2 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 -1 + %1:g8rc = LI8 2 + TDI 2, %1, -1 + TDI 2, %0, 2 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TDI_2 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TDI_1 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 -3 + %1:g8rc = LI8 4 + TDI 1, %1, -3 + TDI 1, %0, 4 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TDI_1 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_TDI_4 +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:g8rc = LI8 5 + %1:g8rc = LI8 1 + TDI 4, %1, 5 + TDI 4, %1, 1 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_TDI_4 + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr + +--- +name: conditional_trap_opt_multiple_traps +alignment: 16 +tracksRegLiveness: true +body: | + bb.0.entry: + %0:gprc = LI 5 + %1:gprc = LI 1 + %2:g8rc = LI8 -1 + %3:g8rc = LI8 0 + TWI 31, %1, 5 + TDI 31, %3, 0 + TW 31, %0, %1 + TD 31, %2, %2 + BLR8 implicit $lr8, implicit $rm +... + # CHECK-LABEL: conditional_trap_opt_multiple_traps + # CHECK: # %bb.0: # %entry + # CHECK-NEXT: trap + # CHECK-NEXT: blr