xref: /freebsd/contrib/llvm-project/llvm/lib/Target/SystemZ/SystemZPostRewrite.cpp (revision 81ad626541db97eb356e2c1d4a20eb2a26a766ab)
10b57cec5SDimitry Andric //==---- SystemZPostRewrite.cpp - Select pseudos after RegAlloc ---*- C++ -*-=//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric //
90b57cec5SDimitry Andric // This file contains a pass that is run immediately after VirtRegRewriter
100b57cec5SDimitry Andric // but before MachineCopyPropagation. The purpose is to lower pseudos to
110b57cec5SDimitry Andric // target instructions before any later pass might substitute a register for
120b57cec5SDimitry Andric // another.
130b57cec5SDimitry Andric //
140b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
150b57cec5SDimitry Andric 
160b57cec5SDimitry Andric #include "SystemZ.h"
170b57cec5SDimitry Andric #include "SystemZInstrInfo.h"
180b57cec5SDimitry Andric #include "SystemZSubtarget.h"
190b57cec5SDimitry Andric #include "llvm/ADT/Statistic.h"
20*81ad6265SDimitry Andric #include "llvm/CodeGen/LivePhysRegs.h"
210b57cec5SDimitry Andric #include "llvm/CodeGen/MachineFunctionPass.h"
220b57cec5SDimitry Andric #include "llvm/CodeGen/MachineInstrBuilder.h"
230b57cec5SDimitry Andric using namespace llvm;
240b57cec5SDimitry Andric 
250b57cec5SDimitry Andric #define DEBUG_TYPE "systemz-postrewrite"
260b57cec5SDimitry Andric STATISTIC(MemFoldCopies, "Number of copies inserted before folded mem ops.");
278bcb0991SDimitry Andric STATISTIC(LOCRMuxJumps, "Number of LOCRMux jump-sequences (lower is better)");
280b57cec5SDimitry Andric 
290b57cec5SDimitry Andric namespace {
300b57cec5SDimitry Andric 
310b57cec5SDimitry Andric class SystemZPostRewrite : public MachineFunctionPass {
320b57cec5SDimitry Andric public:
330b57cec5SDimitry Andric   static char ID;
SystemZPostRewrite()340b57cec5SDimitry Andric   SystemZPostRewrite() : MachineFunctionPass(ID) {
350b57cec5SDimitry Andric     initializeSystemZPostRewritePass(*PassRegistry::getPassRegistry());
360b57cec5SDimitry Andric   }
370b57cec5SDimitry Andric 
380b57cec5SDimitry Andric   const SystemZInstrInfo *TII;
390b57cec5SDimitry Andric 
400b57cec5SDimitry Andric   bool runOnMachineFunction(MachineFunction &Fn) override;
410b57cec5SDimitry Andric 
420b57cec5SDimitry Andric private:
438bcb0991SDimitry Andric   void selectLOCRMux(MachineBasicBlock &MBB,
448bcb0991SDimitry Andric                      MachineBasicBlock::iterator MBBI,
458bcb0991SDimitry Andric                      MachineBasicBlock::iterator &NextMBBI,
468bcb0991SDimitry Andric                      unsigned LowOpcode,
478bcb0991SDimitry Andric                      unsigned HighOpcode);
488bcb0991SDimitry Andric   void selectSELRMux(MachineBasicBlock &MBB,
498bcb0991SDimitry Andric                      MachineBasicBlock::iterator MBBI,
508bcb0991SDimitry Andric                      MachineBasicBlock::iterator &NextMBBI,
518bcb0991SDimitry Andric                      unsigned LowOpcode,
528bcb0991SDimitry Andric                      unsigned HighOpcode);
538bcb0991SDimitry Andric   bool expandCondMove(MachineBasicBlock &MBB,
548bcb0991SDimitry Andric                       MachineBasicBlock::iterator MBBI,
558bcb0991SDimitry Andric                       MachineBasicBlock::iterator &NextMBBI);
560b57cec5SDimitry Andric   bool selectMI(MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI,
570b57cec5SDimitry Andric                 MachineBasicBlock::iterator &NextMBBI);
580b57cec5SDimitry Andric   bool selectMBB(MachineBasicBlock &MBB);
590b57cec5SDimitry Andric };
600b57cec5SDimitry Andric 
610b57cec5SDimitry Andric char SystemZPostRewrite::ID = 0;
620b57cec5SDimitry Andric 
630b57cec5SDimitry Andric } // end anonymous namespace
640b57cec5SDimitry Andric 
650b57cec5SDimitry Andric INITIALIZE_PASS(SystemZPostRewrite, "systemz-post-rewrite",
6604eeddc0SDimitry Andric                 "SystemZ Post Rewrite pass", false, false)
670b57cec5SDimitry Andric 
680b57cec5SDimitry Andric /// Returns an instance of the Post Rewrite pass.
createSystemZPostRewritePass(SystemZTargetMachine & TM)690b57cec5SDimitry Andric FunctionPass *llvm::createSystemZPostRewritePass(SystemZTargetMachine &TM) {
700b57cec5SDimitry Andric   return new SystemZPostRewrite();
710b57cec5SDimitry Andric }
720b57cec5SDimitry Andric 
738bcb0991SDimitry Andric // MI is a load-register-on-condition pseudo instruction.  Replace it with
748bcb0991SDimitry Andric // LowOpcode if source and destination are both low GR32s and HighOpcode if
758bcb0991SDimitry Andric // source and destination are both high GR32s. Otherwise, a branch sequence
768bcb0991SDimitry Andric // is created.
selectLOCRMux(MachineBasicBlock & MBB,MachineBasicBlock::iterator MBBI,MachineBasicBlock::iterator & NextMBBI,unsigned LowOpcode,unsigned HighOpcode)778bcb0991SDimitry Andric void SystemZPostRewrite::selectLOCRMux(MachineBasicBlock &MBB,
788bcb0991SDimitry Andric                                        MachineBasicBlock::iterator MBBI,
798bcb0991SDimitry Andric                                        MachineBasicBlock::iterator &NextMBBI,
808bcb0991SDimitry Andric                                        unsigned LowOpcode,
818bcb0991SDimitry Andric                                        unsigned HighOpcode) {
828bcb0991SDimitry Andric   Register DestReg = MBBI->getOperand(0).getReg();
838bcb0991SDimitry Andric   Register SrcReg = MBBI->getOperand(2).getReg();
848bcb0991SDimitry Andric   bool DestIsHigh = SystemZ::isHighReg(DestReg);
858bcb0991SDimitry Andric   bool SrcIsHigh = SystemZ::isHighReg(SrcReg);
868bcb0991SDimitry Andric 
878bcb0991SDimitry Andric   if (!DestIsHigh && !SrcIsHigh)
888bcb0991SDimitry Andric     MBBI->setDesc(TII->get(LowOpcode));
898bcb0991SDimitry Andric   else if (DestIsHigh && SrcIsHigh)
908bcb0991SDimitry Andric     MBBI->setDesc(TII->get(HighOpcode));
918bcb0991SDimitry Andric   else
928bcb0991SDimitry Andric     expandCondMove(MBB, MBBI, NextMBBI);
938bcb0991SDimitry Andric }
948bcb0991SDimitry Andric 
958bcb0991SDimitry Andric // MI is a select pseudo instruction.  Replace it with LowOpcode if source
968bcb0991SDimitry Andric // and destination are all low GR32s and HighOpcode if source and destination
978bcb0991SDimitry Andric // are all high GR32s. Otherwise, a branch sequence is created.
selectSELRMux(MachineBasicBlock & MBB,MachineBasicBlock::iterator MBBI,MachineBasicBlock::iterator & NextMBBI,unsigned LowOpcode,unsigned HighOpcode)988bcb0991SDimitry Andric void SystemZPostRewrite::selectSELRMux(MachineBasicBlock &MBB,
998bcb0991SDimitry Andric                                        MachineBasicBlock::iterator MBBI,
1008bcb0991SDimitry Andric                                        MachineBasicBlock::iterator &NextMBBI,
1018bcb0991SDimitry Andric                                        unsigned LowOpcode,
1028bcb0991SDimitry Andric                                        unsigned HighOpcode) {
1038bcb0991SDimitry Andric   Register DestReg = MBBI->getOperand(0).getReg();
1048bcb0991SDimitry Andric   Register Src1Reg = MBBI->getOperand(1).getReg();
1058bcb0991SDimitry Andric   Register Src2Reg = MBBI->getOperand(2).getReg();
1068bcb0991SDimitry Andric   bool DestIsHigh = SystemZ::isHighReg(DestReg);
1078bcb0991SDimitry Andric   bool Src1IsHigh = SystemZ::isHighReg(Src1Reg);
1088bcb0991SDimitry Andric   bool Src2IsHigh = SystemZ::isHighReg(Src2Reg);
1098bcb0991SDimitry Andric 
1108bcb0991SDimitry Andric   // If sources and destination aren't all high or all low, we may be able to
1118bcb0991SDimitry Andric   // simplify the operation by moving one of the sources to the destination
1128bcb0991SDimitry Andric   // first.  But only if this doesn't clobber the other source.
1138bcb0991SDimitry Andric   if (DestReg != Src1Reg && DestReg != Src2Reg) {
1148bcb0991SDimitry Andric     if (DestIsHigh != Src1IsHigh) {
1158bcb0991SDimitry Andric       BuildMI(*MBBI->getParent(), MBBI, MBBI->getDebugLoc(),
1168bcb0991SDimitry Andric               TII->get(SystemZ::COPY), DestReg)
1178bcb0991SDimitry Andric         .addReg(MBBI->getOperand(1).getReg(), getRegState(MBBI->getOperand(1)));
1188bcb0991SDimitry Andric       MBBI->getOperand(1).setReg(DestReg);
1198bcb0991SDimitry Andric       Src1Reg = DestReg;
1208bcb0991SDimitry Andric       Src1IsHigh = DestIsHigh;
1218bcb0991SDimitry Andric     } else if (DestIsHigh != Src2IsHigh) {
1228bcb0991SDimitry Andric       BuildMI(*MBBI->getParent(), MBBI, MBBI->getDebugLoc(),
1238bcb0991SDimitry Andric               TII->get(SystemZ::COPY), DestReg)
1248bcb0991SDimitry Andric         .addReg(MBBI->getOperand(2).getReg(), getRegState(MBBI->getOperand(2)));
1258bcb0991SDimitry Andric       MBBI->getOperand(2).setReg(DestReg);
1268bcb0991SDimitry Andric       Src2Reg = DestReg;
1278bcb0991SDimitry Andric       Src2IsHigh = DestIsHigh;
1288bcb0991SDimitry Andric     }
1298bcb0991SDimitry Andric   }
1308bcb0991SDimitry Andric 
1318bcb0991SDimitry Andric   // If the destination (now) matches one source, prefer this to be first.
1328bcb0991SDimitry Andric   if (DestReg != Src1Reg && DestReg == Src2Reg) {
1338bcb0991SDimitry Andric     TII->commuteInstruction(*MBBI, false, 1, 2);
1348bcb0991SDimitry Andric     std::swap(Src1Reg, Src2Reg);
1358bcb0991SDimitry Andric     std::swap(Src1IsHigh, Src2IsHigh);
1368bcb0991SDimitry Andric   }
1378bcb0991SDimitry Andric 
1388bcb0991SDimitry Andric   if (!DestIsHigh && !Src1IsHigh && !Src2IsHigh)
1398bcb0991SDimitry Andric     MBBI->setDesc(TII->get(LowOpcode));
1408bcb0991SDimitry Andric   else if (DestIsHigh && Src1IsHigh && Src2IsHigh)
1418bcb0991SDimitry Andric     MBBI->setDesc(TII->get(HighOpcode));
1428bcb0991SDimitry Andric   else
1438bcb0991SDimitry Andric     // Given the simplification above, we must already have a two-operand case.
1448bcb0991SDimitry Andric     expandCondMove(MBB, MBBI, NextMBBI);
1458bcb0991SDimitry Andric }
1468bcb0991SDimitry Andric 
1478bcb0991SDimitry Andric // Replace MBBI by a branch sequence that performs a conditional move of
1488bcb0991SDimitry Andric // operand 2 to the destination register. Operand 1 is expected to be the
1498bcb0991SDimitry Andric // same register as the destination.
expandCondMove(MachineBasicBlock & MBB,MachineBasicBlock::iterator MBBI,MachineBasicBlock::iterator & NextMBBI)1508bcb0991SDimitry Andric bool SystemZPostRewrite::expandCondMove(MachineBasicBlock &MBB,
1518bcb0991SDimitry Andric                                         MachineBasicBlock::iterator MBBI,
1528bcb0991SDimitry Andric                                         MachineBasicBlock::iterator &NextMBBI) {
1538bcb0991SDimitry Andric   MachineFunction &MF = *MBB.getParent();
1548bcb0991SDimitry Andric   const BasicBlock *BB = MBB.getBasicBlock();
1558bcb0991SDimitry Andric   MachineInstr &MI = *MBBI;
1568bcb0991SDimitry Andric   DebugLoc DL = MI.getDebugLoc();
1578bcb0991SDimitry Andric   Register DestReg = MI.getOperand(0).getReg();
1588bcb0991SDimitry Andric   Register SrcReg = MI.getOperand(2).getReg();
1598bcb0991SDimitry Andric   unsigned CCValid = MI.getOperand(3).getImm();
1608bcb0991SDimitry Andric   unsigned CCMask = MI.getOperand(4).getImm();
1618bcb0991SDimitry Andric   assert(DestReg == MI.getOperand(1).getReg() &&
1628bcb0991SDimitry Andric          "Expected destination and first source operand to be the same.");
1638bcb0991SDimitry Andric 
1648bcb0991SDimitry Andric   LivePhysRegs LiveRegs(TII->getRegisterInfo());
1658bcb0991SDimitry Andric   LiveRegs.addLiveOuts(MBB);
1668bcb0991SDimitry Andric   for (auto I = std::prev(MBB.end()); I != MBBI; --I)
1678bcb0991SDimitry Andric     LiveRegs.stepBackward(*I);
1688bcb0991SDimitry Andric 
1698bcb0991SDimitry Andric   // Splice MBB at MI, moving the rest of the block into RestMBB.
1708bcb0991SDimitry Andric   MachineBasicBlock *RestMBB = MF.CreateMachineBasicBlock(BB);
1718bcb0991SDimitry Andric   MF.insert(std::next(MachineFunction::iterator(MBB)), RestMBB);
1728bcb0991SDimitry Andric   RestMBB->splice(RestMBB->begin(), &MBB, MI, MBB.end());
1738bcb0991SDimitry Andric   RestMBB->transferSuccessors(&MBB);
17404eeddc0SDimitry Andric   for (MCPhysReg R : LiveRegs)
17504eeddc0SDimitry Andric     RestMBB->addLiveIn(R);
1768bcb0991SDimitry Andric 
1778bcb0991SDimitry Andric   // Create a new block MoveMBB to hold the move instruction.
1788bcb0991SDimitry Andric   MachineBasicBlock *MoveMBB = MF.CreateMachineBasicBlock(BB);
1798bcb0991SDimitry Andric   MF.insert(std::next(MachineFunction::iterator(MBB)), MoveMBB);
1808bcb0991SDimitry Andric   MoveMBB->addLiveIn(SrcReg);
18104eeddc0SDimitry Andric   for (MCPhysReg R : LiveRegs)
18204eeddc0SDimitry Andric     MoveMBB->addLiveIn(R);
1838bcb0991SDimitry Andric 
1848bcb0991SDimitry Andric   // At the end of MBB, create a conditional branch to RestMBB if the
1858bcb0991SDimitry Andric   // condition is false, otherwise fall through to MoveMBB.
1868bcb0991SDimitry Andric   BuildMI(&MBB, DL, TII->get(SystemZ::BRC))
1878bcb0991SDimitry Andric     .addImm(CCValid).addImm(CCMask ^ CCValid).addMBB(RestMBB);
1888bcb0991SDimitry Andric   MBB.addSuccessor(RestMBB);
1898bcb0991SDimitry Andric   MBB.addSuccessor(MoveMBB);
1908bcb0991SDimitry Andric 
1918bcb0991SDimitry Andric   // In MoveMBB, emit an instruction to move SrcReg into DestReg,
1928bcb0991SDimitry Andric   // then fall through to RestMBB.
1938bcb0991SDimitry Andric   BuildMI(*MoveMBB, MoveMBB->end(), DL, TII->get(SystemZ::COPY), DestReg)
1948bcb0991SDimitry Andric       .addReg(MI.getOperand(2).getReg(), getRegState(MI.getOperand(2)));
1958bcb0991SDimitry Andric   MoveMBB->addSuccessor(RestMBB);
1968bcb0991SDimitry Andric 
1978bcb0991SDimitry Andric   NextMBBI = MBB.end();
1988bcb0991SDimitry Andric   MI.eraseFromParent();
1998bcb0991SDimitry Andric   LOCRMuxJumps++;
2008bcb0991SDimitry Andric   return true;
2018bcb0991SDimitry Andric }
2028bcb0991SDimitry Andric 
2030b57cec5SDimitry Andric /// If MBBI references a pseudo instruction that should be selected here,
2040b57cec5SDimitry Andric /// do it and return true.  Otherwise return false.
selectMI(MachineBasicBlock & MBB,MachineBasicBlock::iterator MBBI,MachineBasicBlock::iterator & NextMBBI)2050b57cec5SDimitry Andric bool SystemZPostRewrite::selectMI(MachineBasicBlock &MBB,
2060b57cec5SDimitry Andric                                   MachineBasicBlock::iterator MBBI,
2070b57cec5SDimitry Andric                                   MachineBasicBlock::iterator &NextMBBI) {
2080b57cec5SDimitry Andric   MachineInstr &MI = *MBBI;
2090b57cec5SDimitry Andric   unsigned Opcode = MI.getOpcode();
2100b57cec5SDimitry Andric 
2110b57cec5SDimitry Andric   // Note: If this could be done during regalloc in foldMemoryOperandImpl()
2120b57cec5SDimitry Andric   // while also updating the LiveIntervals, there would be no need for the
2130b57cec5SDimitry Andric   // MemFoldPseudo to begin with.
2140b57cec5SDimitry Andric   int TargetMemOpcode = SystemZ::getTargetMemOpcode(Opcode);
2150b57cec5SDimitry Andric   if (TargetMemOpcode != -1) {
2160b57cec5SDimitry Andric     MI.setDesc(TII->get(TargetMemOpcode));
2170b57cec5SDimitry Andric     MI.tieOperands(0, 1);
2188bcb0991SDimitry Andric     Register DstReg = MI.getOperand(0).getReg();
2190b57cec5SDimitry Andric     MachineOperand &SrcMO = MI.getOperand(1);
2200b57cec5SDimitry Andric     if (DstReg != SrcMO.getReg()) {
2210b57cec5SDimitry Andric       BuildMI(MBB, &MI, MI.getDebugLoc(), TII->get(SystemZ::COPY), DstReg)
2220b57cec5SDimitry Andric         .addReg(SrcMO.getReg());
2230b57cec5SDimitry Andric       SrcMO.setReg(DstReg);
2240b57cec5SDimitry Andric       MemFoldCopies++;
2250b57cec5SDimitry Andric     }
2260b57cec5SDimitry Andric     return true;
2270b57cec5SDimitry Andric   }
2280b57cec5SDimitry Andric 
2298bcb0991SDimitry Andric   switch (Opcode) {
2308bcb0991SDimitry Andric   case SystemZ::LOCRMux:
2318bcb0991SDimitry Andric     selectLOCRMux(MBB, MBBI, NextMBBI, SystemZ::LOCR, SystemZ::LOCFHR);
2328bcb0991SDimitry Andric     return true;
2338bcb0991SDimitry Andric   case SystemZ::SELRMux:
2348bcb0991SDimitry Andric     selectSELRMux(MBB, MBBI, NextMBBI, SystemZ::SELR, SystemZ::SELFHR);
2358bcb0991SDimitry Andric     return true;
2368bcb0991SDimitry Andric   }
2378bcb0991SDimitry Andric 
2380b57cec5SDimitry Andric   return false;
2390b57cec5SDimitry Andric }
2400b57cec5SDimitry Andric 
2410b57cec5SDimitry Andric /// Iterate over the instructions in basic block MBB and select any
2420b57cec5SDimitry Andric /// pseudo instructions.  Return true if anything was modified.
selectMBB(MachineBasicBlock & MBB)2430b57cec5SDimitry Andric bool SystemZPostRewrite::selectMBB(MachineBasicBlock &MBB) {
2440b57cec5SDimitry Andric   bool Modified = false;
2450b57cec5SDimitry Andric 
2460b57cec5SDimitry Andric   MachineBasicBlock::iterator MBBI = MBB.begin(), E = MBB.end();
2470b57cec5SDimitry Andric   while (MBBI != E) {
2480b57cec5SDimitry Andric     MachineBasicBlock::iterator NMBBI = std::next(MBBI);
2490b57cec5SDimitry Andric     Modified |= selectMI(MBB, MBBI, NMBBI);
2500b57cec5SDimitry Andric     MBBI = NMBBI;
2510b57cec5SDimitry Andric   }
2520b57cec5SDimitry Andric 
2530b57cec5SDimitry Andric   return Modified;
2540b57cec5SDimitry Andric }
2550b57cec5SDimitry Andric 
runOnMachineFunction(MachineFunction & MF)2560b57cec5SDimitry Andric bool SystemZPostRewrite::runOnMachineFunction(MachineFunction &MF) {
257*81ad6265SDimitry Andric   TII = MF.getSubtarget<SystemZSubtarget>().getInstrInfo();
2580b57cec5SDimitry Andric 
2590b57cec5SDimitry Andric   bool Modified = false;
2600b57cec5SDimitry Andric   for (auto &MBB : MF)
2610b57cec5SDimitry Andric     Modified |= selectMBB(MBB);
2620b57cec5SDimitry Andric 
2630b57cec5SDimitry Andric   return Modified;
2640b57cec5SDimitry Andric }
2650b57cec5SDimitry Andric 
266