Path: blob/main/contrib/llvm-project/llvm/lib/Target/AArch64/AArch64CondBrTuning.cpp
35269 views
//===-- AArch64CondBrTuning.cpp --- Conditional branch tuning for AArch64 -===//1//2// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.3// See https://llvm.org/LICENSE.txt for license information.4// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception5//6//===----------------------------------------------------------------------===//7/// \file8/// This file contains a pass that transforms CBZ/CBNZ/TBZ/TBNZ instructions9/// into a conditional branch (B.cond), when the NZCV flags can be set for10/// "free". This is preferred on targets that have more flexibility when11/// scheduling B.cond instructions as compared to CBZ/CBNZ/TBZ/TBNZ (assuming12/// all other variables are equal). This can also reduce register pressure.13///14/// A few examples:15///16/// 1) add w8, w0, w1 -> cmn w0, w1 ; CMN is an alias of ADDS.17/// cbz w8, .LBB_2 -> b.eq .LBB0_218///19/// 2) add w8, w0, w1 -> adds w8, w0, w1 ; w8 has multiple uses.20/// cbz w8, .LBB1_2 -> b.eq .LBB1_221///22/// 3) sub w8, w0, w1 -> subs w8, w0, w1 ; w8 has multiple uses.23/// tbz w8, #31, .LBB6_2 -> b.pl .LBB6_224///25//===----------------------------------------------------------------------===//2627#include "AArch64.h"28#include "AArch64Subtarget.h"29#include "llvm/CodeGen/MachineFunction.h"30#include "llvm/CodeGen/MachineFunctionPass.h"31#include "llvm/CodeGen/MachineInstrBuilder.h"32#include "llvm/CodeGen/MachineRegisterInfo.h"33#include "llvm/CodeGen/Passes.h"34#include "llvm/CodeGen/TargetInstrInfo.h"35#include "llvm/CodeGen/TargetRegisterInfo.h"36#include "llvm/CodeGen/TargetSubtargetInfo.h"37#include "llvm/Support/Debug.h"38#include "llvm/Support/raw_ostream.h"3940using namespace llvm;4142#define DEBUG_TYPE "aarch64-cond-br-tuning"43#define AARCH64_CONDBR_TUNING_NAME "AArch64 Conditional Branch Tuning"4445namespace {46class AArch64CondBrTuning : public MachineFunctionPass {47const AArch64InstrInfo *TII;48const TargetRegisterInfo *TRI;4950MachineRegisterInfo *MRI;5152public:53static char ID;54AArch64CondBrTuning() : MachineFunctionPass(ID) {55initializeAArch64CondBrTuningPass(*PassRegistry::getPassRegistry());56}57void getAnalysisUsage(AnalysisUsage &AU) const override;58bool runOnMachineFunction(MachineFunction &MF) override;59StringRef getPassName() const override { return AARCH64_CONDBR_TUNING_NAME; }6061private:62MachineInstr *getOperandDef(const MachineOperand &MO);63MachineInstr *convertToFlagSetting(MachineInstr &MI, bool IsFlagSetting,64bool Is64Bit);65MachineInstr *convertToCondBr(MachineInstr &MI);66bool tryToTuneBranch(MachineInstr &MI, MachineInstr &DefMI);67};68} // end anonymous namespace6970char AArch64CondBrTuning::ID = 0;7172INITIALIZE_PASS(AArch64CondBrTuning, "aarch64-cond-br-tuning",73AARCH64_CONDBR_TUNING_NAME, false, false)7475void AArch64CondBrTuning::getAnalysisUsage(AnalysisUsage &AU) const {76AU.setPreservesCFG();77MachineFunctionPass::getAnalysisUsage(AU);78}7980MachineInstr *AArch64CondBrTuning::getOperandDef(const MachineOperand &MO) {81if (!MO.getReg().isVirtual())82return nullptr;83return MRI->getUniqueVRegDef(MO.getReg());84}8586MachineInstr *AArch64CondBrTuning::convertToFlagSetting(MachineInstr &MI,87bool IsFlagSetting,88bool Is64Bit) {89// If this is already the flag setting version of the instruction (e.g., SUBS)90// just make sure the implicit-def of NZCV isn't marked dead.91if (IsFlagSetting) {92for (MachineOperand &MO : MI.implicit_operands())93if (MO.isReg() && MO.isDead() && MO.getReg() == AArch64::NZCV)94MO.setIsDead(false);95return &MI;96}97unsigned NewOpc = TII->convertToFlagSettingOpc(MI.getOpcode());98Register NewDestReg = MI.getOperand(0).getReg();99if (MRI->hasOneNonDBGUse(MI.getOperand(0).getReg()))100NewDestReg = Is64Bit ? AArch64::XZR : AArch64::WZR;101102MachineInstrBuilder MIB = BuildMI(*MI.getParent(), MI, MI.getDebugLoc(),103TII->get(NewOpc), NewDestReg);104for (const MachineOperand &MO : llvm::drop_begin(MI.operands()))105MIB.add(MO);106107return MIB;108}109110MachineInstr *AArch64CondBrTuning::convertToCondBr(MachineInstr &MI) {111AArch64CC::CondCode CC;112MachineBasicBlock *TargetMBB = TII->getBranchDestBlock(MI);113switch (MI.getOpcode()) {114default:115llvm_unreachable("Unexpected opcode!");116117case AArch64::CBZW:118case AArch64::CBZX:119CC = AArch64CC::EQ;120break;121case AArch64::CBNZW:122case AArch64::CBNZX:123CC = AArch64CC::NE;124break;125case AArch64::TBZW:126case AArch64::TBZX:127CC = AArch64CC::PL;128break;129case AArch64::TBNZW:130case AArch64::TBNZX:131CC = AArch64CC::MI;132break;133}134return BuildMI(*MI.getParent(), MI, MI.getDebugLoc(), TII->get(AArch64::Bcc))135.addImm(CC)136.addMBB(TargetMBB);137}138139bool AArch64CondBrTuning::tryToTuneBranch(MachineInstr &MI,140MachineInstr &DefMI) {141// We don't want NZCV bits live across blocks.142if (MI.getParent() != DefMI.getParent())143return false;144145bool IsFlagSetting = true;146unsigned MIOpc = MI.getOpcode();147MachineInstr *NewCmp = nullptr, *NewBr = nullptr;148switch (DefMI.getOpcode()) {149default:150return false;151case AArch64::ADDWri:152case AArch64::ADDWrr:153case AArch64::ADDWrs:154case AArch64::ADDWrx:155case AArch64::ANDWri:156case AArch64::ANDWrr:157case AArch64::ANDWrs:158case AArch64::BICWrr:159case AArch64::BICWrs:160case AArch64::SUBWri:161case AArch64::SUBWrr:162case AArch64::SUBWrs:163case AArch64::SUBWrx:164IsFlagSetting = false;165[[fallthrough]];166case AArch64::ADDSWri:167case AArch64::ADDSWrr:168case AArch64::ADDSWrs:169case AArch64::ADDSWrx:170case AArch64::ANDSWri:171case AArch64::ANDSWrr:172case AArch64::ANDSWrs:173case AArch64::BICSWrr:174case AArch64::BICSWrs:175case AArch64::SUBSWri:176case AArch64::SUBSWrr:177case AArch64::SUBSWrs:178case AArch64::SUBSWrx:179switch (MIOpc) {180default:181llvm_unreachable("Unexpected opcode!");182183case AArch64::CBZW:184case AArch64::CBNZW:185case AArch64::TBZW:186case AArch64::TBNZW:187// Check to see if the TBZ/TBNZ is checking the sign bit.188if ((MIOpc == AArch64::TBZW || MIOpc == AArch64::TBNZW) &&189MI.getOperand(1).getImm() != 31)190return false;191192// There must not be any instruction between DefMI and MI that clobbers or193// reads NZCV.194if (isNZCVTouchedInInstructionRange(DefMI, MI, TRI))195return false;196LLVM_DEBUG(dbgs() << " Replacing instructions:\n ");197LLVM_DEBUG(DefMI.print(dbgs()));198LLVM_DEBUG(dbgs() << " ");199LLVM_DEBUG(MI.print(dbgs()));200201NewCmp = convertToFlagSetting(DefMI, IsFlagSetting, /*Is64Bit=*/false);202NewBr = convertToCondBr(MI);203break;204}205break;206207case AArch64::ADDXri:208case AArch64::ADDXrr:209case AArch64::ADDXrs:210case AArch64::ADDXrx:211case AArch64::ANDXri:212case AArch64::ANDXrr:213case AArch64::ANDXrs:214case AArch64::BICXrr:215case AArch64::BICXrs:216case AArch64::SUBXri:217case AArch64::SUBXrr:218case AArch64::SUBXrs:219case AArch64::SUBXrx:220IsFlagSetting = false;221[[fallthrough]];222case AArch64::ADDSXri:223case AArch64::ADDSXrr:224case AArch64::ADDSXrs:225case AArch64::ADDSXrx:226case AArch64::ANDSXri:227case AArch64::ANDSXrr:228case AArch64::ANDSXrs:229case AArch64::BICSXrr:230case AArch64::BICSXrs:231case AArch64::SUBSXri:232case AArch64::SUBSXrr:233case AArch64::SUBSXrs:234case AArch64::SUBSXrx:235switch (MIOpc) {236default:237llvm_unreachable("Unexpected opcode!");238239case AArch64::CBZX:240case AArch64::CBNZX:241case AArch64::TBZX:242case AArch64::TBNZX: {243// Check to see if the TBZ/TBNZ is checking the sign bit.244if ((MIOpc == AArch64::TBZX || MIOpc == AArch64::TBNZX) &&245MI.getOperand(1).getImm() != 63)246return false;247// There must not be any instruction between DefMI and MI that clobbers or248// reads NZCV.249if (isNZCVTouchedInInstructionRange(DefMI, MI, TRI))250return false;251LLVM_DEBUG(dbgs() << " Replacing instructions:\n ");252LLVM_DEBUG(DefMI.print(dbgs()));253LLVM_DEBUG(dbgs() << " ");254LLVM_DEBUG(MI.print(dbgs()));255256NewCmp = convertToFlagSetting(DefMI, IsFlagSetting, /*Is64Bit=*/true);257NewBr = convertToCondBr(MI);258break;259}260}261break;262}263(void)NewCmp; (void)NewBr;264assert(NewCmp && NewBr && "Expected new instructions.");265266LLVM_DEBUG(dbgs() << " with instruction:\n ");267LLVM_DEBUG(NewCmp->print(dbgs()));268LLVM_DEBUG(dbgs() << " ");269LLVM_DEBUG(NewBr->print(dbgs()));270271// If this was a flag setting version of the instruction, we use the original272// instruction by just clearing the dead marked on the implicit-def of NCZV.273// Therefore, we should not erase this instruction.274if (!IsFlagSetting)275DefMI.eraseFromParent();276MI.eraseFromParent();277return true;278}279280bool AArch64CondBrTuning::runOnMachineFunction(MachineFunction &MF) {281if (skipFunction(MF.getFunction()))282return false;283284LLVM_DEBUG(285dbgs() << "********** AArch64 Conditional Branch Tuning **********\n"286<< "********** Function: " << MF.getName() << '\n');287288TII = static_cast<const AArch64InstrInfo *>(MF.getSubtarget().getInstrInfo());289TRI = MF.getSubtarget().getRegisterInfo();290MRI = &MF.getRegInfo();291292bool Changed = false;293for (MachineBasicBlock &MBB : MF) {294bool LocalChange = false;295for (MachineInstr &MI : MBB.terminators()) {296switch (MI.getOpcode()) {297default:298break;299case AArch64::CBZW:300case AArch64::CBZX:301case AArch64::CBNZW:302case AArch64::CBNZX:303case AArch64::TBZW:304case AArch64::TBZX:305case AArch64::TBNZW:306case AArch64::TBNZX:307MachineInstr *DefMI = getOperandDef(MI.getOperand(0));308LocalChange = (DefMI && tryToTuneBranch(MI, *DefMI));309break;310}311// If the optimization was successful, we can't optimize any other312// branches because doing so would clobber the NZCV flags.313if (LocalChange) {314Changed = true;315break;316}317}318}319return Changed;320}321322FunctionPass *llvm::createAArch64CondBrTuning() {323return new AArch64CondBrTuning();324}325326327