xref: /src/contrib/llvm-project/llvm/lib/Target/SystemZ/SystemZPostRewrite.cpp (revision 81ad626541db97eb356e2c1d4a20eb2a26a766ab)
1e6d15924SDimitry Andric //==---- SystemZPostRewrite.cpp - Select pseudos after RegAlloc ---*- C++ -*-=//
2e6d15924SDimitry Andric //
3e6d15924SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4e6d15924SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5e6d15924SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6e6d15924SDimitry Andric //
7e6d15924SDimitry Andric //===----------------------------------------------------------------------===//
8e6d15924SDimitry Andric //
9e6d15924SDimitry Andric // This file contains a pass that is run immediately after VirtRegRewriter
10e6d15924SDimitry Andric // but before MachineCopyPropagation. The purpose is to lower pseudos to
11e6d15924SDimitry Andric // target instructions before any later pass might substitute a register for
12e6d15924SDimitry Andric // another.
13e6d15924SDimitry Andric //
14e6d15924SDimitry Andric //===----------------------------------------------------------------------===//
15e6d15924SDimitry Andric 
16e6d15924SDimitry Andric #include "SystemZ.h"
17e6d15924SDimitry Andric #include "SystemZInstrInfo.h"
18e6d15924SDimitry Andric #include "SystemZSubtarget.h"
19e6d15924SDimitry Andric #include "llvm/ADT/Statistic.h"
20145449b1SDimitry Andric #include "llvm/CodeGen/LivePhysRegs.h"
21e6d15924SDimitry Andric #include "llvm/CodeGen/MachineFunctionPass.h"
22e6d15924SDimitry Andric #include "llvm/CodeGen/MachineInstrBuilder.h"
23e6d15924SDimitry Andric using namespace llvm;
24e6d15924SDimitry Andric 
25e6d15924SDimitry Andric #define DEBUG_TYPE "systemz-postrewrite"
26e6d15924SDimitry Andric STATISTIC(MemFoldCopies, "Number of copies inserted before folded mem ops.");
271d5ae102SDimitry Andric STATISTIC(LOCRMuxJumps, "Number of LOCRMux jump-sequences (lower is better)");
28e6d15924SDimitry Andric 
29e6d15924SDimitry Andric namespace {
30e6d15924SDimitry Andric 
31e6d15924SDimitry Andric class SystemZPostRewrite : public MachineFunctionPass {
32e6d15924SDimitry Andric public:
33e6d15924SDimitry Andric   static char ID;
SystemZPostRewrite()34e6d15924SDimitry Andric   SystemZPostRewrite() : MachineFunctionPass(ID) {
35e6d15924SDimitry Andric     initializeSystemZPostRewritePass(*PassRegistry::getPassRegistry());
36e6d15924SDimitry Andric   }
37e6d15924SDimitry Andric 
38e6d15924SDimitry Andric   const SystemZInstrInfo *TII;
39e6d15924SDimitry Andric 
40e6d15924SDimitry Andric   bool runOnMachineFunction(MachineFunction &Fn) override;
41e6d15924SDimitry Andric 
42e6d15924SDimitry Andric private:
431d5ae102SDimitry Andric   void selectLOCRMux(MachineBasicBlock &MBB,
441d5ae102SDimitry Andric                      MachineBasicBlock::iterator MBBI,
451d5ae102SDimitry Andric                      MachineBasicBlock::iterator &NextMBBI,
461d5ae102SDimitry Andric                      unsigned LowOpcode,
471d5ae102SDimitry Andric                      unsigned HighOpcode);
481d5ae102SDimitry Andric   void selectSELRMux(MachineBasicBlock &MBB,
491d5ae102SDimitry Andric                      MachineBasicBlock::iterator MBBI,
501d5ae102SDimitry Andric                      MachineBasicBlock::iterator &NextMBBI,
511d5ae102SDimitry Andric                      unsigned LowOpcode,
521d5ae102SDimitry Andric                      unsigned HighOpcode);
531d5ae102SDimitry Andric   bool expandCondMove(MachineBasicBlock &MBB,
541d5ae102SDimitry Andric                       MachineBasicBlock::iterator MBBI,
551d5ae102SDimitry Andric                       MachineBasicBlock::iterator &NextMBBI);
56e6d15924SDimitry Andric   bool selectMI(MachineBasicBlock &MBB, MachineBasicBlock::iterator MBBI,
57e6d15924SDimitry Andric                 MachineBasicBlock::iterator &NextMBBI);
58e6d15924SDimitry Andric   bool selectMBB(MachineBasicBlock &MBB);
59e6d15924SDimitry Andric };
60e6d15924SDimitry Andric 
61e6d15924SDimitry Andric char SystemZPostRewrite::ID = 0;
62e6d15924SDimitry Andric 
63e6d15924SDimitry Andric } // end anonymous namespace
64e6d15924SDimitry Andric 
65e6d15924SDimitry Andric INITIALIZE_PASS(SystemZPostRewrite, "systemz-post-rewrite",
666f8fc217SDimitry Andric                 "SystemZ Post Rewrite pass", false, false)
67e6d15924SDimitry Andric 
68e6d15924SDimitry Andric /// Returns an instance of the Post Rewrite pass.
createSystemZPostRewritePass(SystemZTargetMachine & TM)69e6d15924SDimitry Andric FunctionPass *llvm::createSystemZPostRewritePass(SystemZTargetMachine &TM) {
70e6d15924SDimitry Andric   return new SystemZPostRewrite();
71e6d15924SDimitry Andric }
72e6d15924SDimitry Andric 
731d5ae102SDimitry Andric // MI is a load-register-on-condition pseudo instruction.  Replace it with
741d5ae102SDimitry Andric // LowOpcode if source and destination are both low GR32s and HighOpcode if
751d5ae102SDimitry Andric // source and destination are both high GR32s. Otherwise, a branch sequence
761d5ae102SDimitry Andric // is created.
selectLOCRMux(MachineBasicBlock & MBB,MachineBasicBlock::iterator MBBI,MachineBasicBlock::iterator & NextMBBI,unsigned LowOpcode,unsigned HighOpcode)771d5ae102SDimitry Andric void SystemZPostRewrite::selectLOCRMux(MachineBasicBlock &MBB,
781d5ae102SDimitry Andric                                        MachineBasicBlock::iterator MBBI,
791d5ae102SDimitry Andric                                        MachineBasicBlock::iterator &NextMBBI,
801d5ae102SDimitry Andric                                        unsigned LowOpcode,
811d5ae102SDimitry Andric                                        unsigned HighOpcode) {
821d5ae102SDimitry Andric   Register DestReg = MBBI->getOperand(0).getReg();
831d5ae102SDimitry Andric   Register SrcReg = MBBI->getOperand(2).getReg();
841d5ae102SDimitry Andric   bool DestIsHigh = SystemZ::isHighReg(DestReg);
851d5ae102SDimitry Andric   bool SrcIsHigh = SystemZ::isHighReg(SrcReg);
861d5ae102SDimitry Andric 
871d5ae102SDimitry Andric   if (!DestIsHigh && !SrcIsHigh)
881d5ae102SDimitry Andric     MBBI->setDesc(TII->get(LowOpcode));
891d5ae102SDimitry Andric   else if (DestIsHigh && SrcIsHigh)
901d5ae102SDimitry Andric     MBBI->setDesc(TII->get(HighOpcode));
911d5ae102SDimitry Andric   else
921d5ae102SDimitry Andric     expandCondMove(MBB, MBBI, NextMBBI);
931d5ae102SDimitry Andric }
941d5ae102SDimitry Andric 
951d5ae102SDimitry Andric // MI is a select pseudo instruction.  Replace it with LowOpcode if source
961d5ae102SDimitry Andric // and destination are all low GR32s and HighOpcode if source and destination
971d5ae102SDimitry Andric // are all high GR32s. Otherwise, a branch sequence is created.
selectSELRMux(MachineBasicBlock & MBB,MachineBasicBlock::iterator MBBI,MachineBasicBlock::iterator & NextMBBI,unsigned LowOpcode,unsigned HighOpcode)981d5ae102SDimitry Andric void SystemZPostRewrite::selectSELRMux(MachineBasicBlock &MBB,
991d5ae102SDimitry Andric                                        MachineBasicBlock::iterator MBBI,
1001d5ae102SDimitry Andric                                        MachineBasicBlock::iterator &NextMBBI,
1011d5ae102SDimitry Andric                                        unsigned LowOpcode,
1021d5ae102SDimitry Andric                                        unsigned HighOpcode) {
1031d5ae102SDimitry Andric   Register DestReg = MBBI->getOperand(0).getReg();
1041d5ae102SDimitry Andric   Register Src1Reg = MBBI->getOperand(1).getReg();
1051d5ae102SDimitry Andric   Register Src2Reg = MBBI->getOperand(2).getReg();
1061d5ae102SDimitry Andric   bool DestIsHigh = SystemZ::isHighReg(DestReg);
1071d5ae102SDimitry Andric   bool Src1IsHigh = SystemZ::isHighReg(Src1Reg);
1081d5ae102SDimitry Andric   bool Src2IsHigh = SystemZ::isHighReg(Src2Reg);
1091d5ae102SDimitry Andric 
1101d5ae102SDimitry Andric   // If sources and destination aren't all high or all low, we may be able to
1111d5ae102SDimitry Andric   // simplify the operation by moving one of the sources to the destination
1121d5ae102SDimitry Andric   // first.  But only if this doesn't clobber the other source.
1131d5ae102SDimitry Andric   if (DestReg != Src1Reg && DestReg != Src2Reg) {
1141d5ae102SDimitry Andric     if (DestIsHigh != Src1IsHigh) {
1151d5ae102SDimitry Andric       BuildMI(*MBBI->getParent(), MBBI, MBBI->getDebugLoc(),
1161d5ae102SDimitry Andric               TII->get(SystemZ::COPY), DestReg)
1171d5ae102SDimitry Andric         .addReg(MBBI->getOperand(1).getReg(), getRegState(MBBI->getOperand(1)));
1181d5ae102SDimitry Andric       MBBI->getOperand(1).setReg(DestReg);
1191d5ae102SDimitry Andric       Src1Reg = DestReg;
1201d5ae102SDimitry Andric       Src1IsHigh = DestIsHigh;
1211d5ae102SDimitry Andric     } else if (DestIsHigh != Src2IsHigh) {
1221d5ae102SDimitry Andric       BuildMI(*MBBI->getParent(), MBBI, MBBI->getDebugLoc(),
1231d5ae102SDimitry Andric               TII->get(SystemZ::COPY), DestReg)
1241d5ae102SDimitry Andric         .addReg(MBBI->getOperand(2).getReg(), getRegState(MBBI->getOperand(2)));
1251d5ae102SDimitry Andric       MBBI->getOperand(2).setReg(DestReg);
1261d5ae102SDimitry Andric       Src2Reg = DestReg;
1271d5ae102SDimitry Andric       Src2IsHigh = DestIsHigh;
1281d5ae102SDimitry Andric     }
1291d5ae102SDimitry Andric   }
1301d5ae102SDimitry Andric 
1311d5ae102SDimitry Andric   // If the destination (now) matches one source, prefer this to be first.
1321d5ae102SDimitry Andric   if (DestReg != Src1Reg && DestReg == Src2Reg) {
1331d5ae102SDimitry Andric     TII->commuteInstruction(*MBBI, false, 1, 2);
1341d5ae102SDimitry Andric     std::swap(Src1Reg, Src2Reg);
1351d5ae102SDimitry Andric     std::swap(Src1IsHigh, Src2IsHigh);
1361d5ae102SDimitry Andric   }
1371d5ae102SDimitry Andric 
1381d5ae102SDimitry Andric   if (!DestIsHigh && !Src1IsHigh && !Src2IsHigh)
1391d5ae102SDimitry Andric     MBBI->setDesc(TII->get(LowOpcode));
1401d5ae102SDimitry Andric   else if (DestIsHigh && Src1IsHigh && Src2IsHigh)
1411d5ae102SDimitry Andric     MBBI->setDesc(TII->get(HighOpcode));
1421d5ae102SDimitry Andric   else
1431d5ae102SDimitry Andric     // Given the simplification above, we must already have a two-operand case.
1441d5ae102SDimitry Andric     expandCondMove(MBB, MBBI, NextMBBI);
1451d5ae102SDimitry Andric }
1461d5ae102SDimitry Andric 
1471d5ae102SDimitry Andric // Replace MBBI by a branch sequence that performs a conditional move of
1481d5ae102SDimitry Andric // operand 2 to the destination register. Operand 1 is expected to be the
1491d5ae102SDimitry Andric // same register as the destination.
expandCondMove(MachineBasicBlock & MBB,MachineBasicBlock::iterator MBBI,MachineBasicBlock::iterator & NextMBBI)1501d5ae102SDimitry Andric bool SystemZPostRewrite::expandCondMove(MachineBasicBlock &MBB,
1511d5ae102SDimitry Andric                                         MachineBasicBlock::iterator MBBI,
1521d5ae102SDimitry Andric                                         MachineBasicBlock::iterator &NextMBBI) {
1531d5ae102SDimitry Andric   MachineFunction &MF = *MBB.getParent();
1541d5ae102SDimitry Andric   const BasicBlock *BB = MBB.getBasicBlock();
1551d5ae102SDimitry Andric   MachineInstr &MI = *MBBI;
1561d5ae102SDimitry Andric   DebugLoc DL = MI.getDebugLoc();
1571d5ae102SDimitry Andric   Register DestReg = MI.getOperand(0).getReg();
1581d5ae102SDimitry Andric   Register SrcReg = MI.getOperand(2).getReg();
1591d5ae102SDimitry Andric   unsigned CCValid = MI.getOperand(3).getImm();
1601d5ae102SDimitry Andric   unsigned CCMask = MI.getOperand(4).getImm();
1611d5ae102SDimitry Andric   assert(DestReg == MI.getOperand(1).getReg() &&
1621d5ae102SDimitry Andric          "Expected destination and first source operand to be the same.");
1631d5ae102SDimitry Andric 
1641d5ae102SDimitry Andric   LivePhysRegs LiveRegs(TII->getRegisterInfo());
1651d5ae102SDimitry Andric   LiveRegs.addLiveOuts(MBB);
1661d5ae102SDimitry Andric   for (auto I = std::prev(MBB.end()); I != MBBI; --I)
1671d5ae102SDimitry Andric     LiveRegs.stepBackward(*I);
1681d5ae102SDimitry Andric 
1691d5ae102SDimitry Andric   // Splice MBB at MI, moving the rest of the block into RestMBB.
1701d5ae102SDimitry Andric   MachineBasicBlock *RestMBB = MF.CreateMachineBasicBlock(BB);
1711d5ae102SDimitry Andric   MF.insert(std::next(MachineFunction::iterator(MBB)), RestMBB);
1721d5ae102SDimitry Andric   RestMBB->splice(RestMBB->begin(), &MBB, MI, MBB.end());
1731d5ae102SDimitry Andric   RestMBB->transferSuccessors(&MBB);
1746f8fc217SDimitry Andric   for (MCPhysReg R : LiveRegs)
1756f8fc217SDimitry Andric     RestMBB->addLiveIn(R);
1761d5ae102SDimitry Andric 
1771d5ae102SDimitry Andric   // Create a new block MoveMBB to hold the move instruction.
1781d5ae102SDimitry Andric   MachineBasicBlock *MoveMBB = MF.CreateMachineBasicBlock(BB);
1791d5ae102SDimitry Andric   MF.insert(std::next(MachineFunction::iterator(MBB)), MoveMBB);
1801d5ae102SDimitry Andric   MoveMBB->addLiveIn(SrcReg);
1816f8fc217SDimitry Andric   for (MCPhysReg R : LiveRegs)
1826f8fc217SDimitry Andric     MoveMBB->addLiveIn(R);
1831d5ae102SDimitry Andric 
1841d5ae102SDimitry Andric   // At the end of MBB, create a conditional branch to RestMBB if the
1851d5ae102SDimitry Andric   // condition is false, otherwise fall through to MoveMBB.
1861d5ae102SDimitry Andric   BuildMI(&MBB, DL, TII->get(SystemZ::BRC))
1871d5ae102SDimitry Andric     .addImm(CCValid).addImm(CCMask ^ CCValid).addMBB(RestMBB);
1881d5ae102SDimitry Andric   MBB.addSuccessor(RestMBB);
1891d5ae102SDimitry Andric   MBB.addSuccessor(MoveMBB);
1901d5ae102SDimitry Andric 
1911d5ae102SDimitry Andric   // In MoveMBB, emit an instruction to move SrcReg into DestReg,
1921d5ae102SDimitry Andric   // then fall through to RestMBB.
1931d5ae102SDimitry Andric   BuildMI(*MoveMBB, MoveMBB->end(), DL, TII->get(SystemZ::COPY), DestReg)
1941d5ae102SDimitry Andric       .addReg(MI.getOperand(2).getReg(), getRegState(MI.getOperand(2)));
1951d5ae102SDimitry Andric   MoveMBB->addSuccessor(RestMBB);
1961d5ae102SDimitry Andric 
1971d5ae102SDimitry Andric   NextMBBI = MBB.end();
1981d5ae102SDimitry Andric   MI.eraseFromParent();
1991d5ae102SDimitry Andric   LOCRMuxJumps++;
2001d5ae102SDimitry Andric   return true;
2011d5ae102SDimitry Andric }
2021d5ae102SDimitry Andric 
203e6d15924SDimitry Andric /// If MBBI references a pseudo instruction that should be selected here,
204e6d15924SDimitry Andric /// do it and return true.  Otherwise return false.
selectMI(MachineBasicBlock & MBB,MachineBasicBlock::iterator MBBI,MachineBasicBlock::iterator & NextMBBI)205e6d15924SDimitry Andric bool SystemZPostRewrite::selectMI(MachineBasicBlock &MBB,
206e6d15924SDimitry Andric                                   MachineBasicBlock::iterator MBBI,
207e6d15924SDimitry Andric                                   MachineBasicBlock::iterator &NextMBBI) {
208e6d15924SDimitry Andric   MachineInstr &MI = *MBBI;
209e6d15924SDimitry Andric   unsigned Opcode = MI.getOpcode();
210e6d15924SDimitry Andric 
211e6d15924SDimitry Andric   // Note: If this could be done during regalloc in foldMemoryOperandImpl()
212e6d15924SDimitry Andric   // while also updating the LiveIntervals, there would be no need for the
213e6d15924SDimitry Andric   // MemFoldPseudo to begin with.
214e6d15924SDimitry Andric   int TargetMemOpcode = SystemZ::getTargetMemOpcode(Opcode);
215e6d15924SDimitry Andric   if (TargetMemOpcode != -1) {
216e6d15924SDimitry Andric     MI.setDesc(TII->get(TargetMemOpcode));
217e6d15924SDimitry Andric     MI.tieOperands(0, 1);
2181d5ae102SDimitry Andric     Register DstReg = MI.getOperand(0).getReg();
219e6d15924SDimitry Andric     MachineOperand &SrcMO = MI.getOperand(1);
220e6d15924SDimitry Andric     if (DstReg != SrcMO.getReg()) {
221e6d15924SDimitry Andric       BuildMI(MBB, &MI, MI.getDebugLoc(), TII->get(SystemZ::COPY), DstReg)
222e6d15924SDimitry Andric         .addReg(SrcMO.getReg());
223e6d15924SDimitry Andric       SrcMO.setReg(DstReg);
224e6d15924SDimitry Andric       MemFoldCopies++;
225e6d15924SDimitry Andric     }
226e6d15924SDimitry Andric     return true;
227e6d15924SDimitry Andric   }
228e6d15924SDimitry Andric 
2291d5ae102SDimitry Andric   switch (Opcode) {
2301d5ae102SDimitry Andric   case SystemZ::LOCRMux:
2311d5ae102SDimitry Andric     selectLOCRMux(MBB, MBBI, NextMBBI, SystemZ::LOCR, SystemZ::LOCFHR);
2321d5ae102SDimitry Andric     return true;
2331d5ae102SDimitry Andric   case SystemZ::SELRMux:
2341d5ae102SDimitry Andric     selectSELRMux(MBB, MBBI, NextMBBI, SystemZ::SELR, SystemZ::SELFHR);
2351d5ae102SDimitry Andric     return true;
2361d5ae102SDimitry Andric   }
2371d5ae102SDimitry Andric 
238e6d15924SDimitry Andric   return false;
239e6d15924SDimitry Andric }
240e6d15924SDimitry Andric 
241e6d15924SDimitry Andric /// Iterate over the instructions in basic block MBB and select any
242e6d15924SDimitry Andric /// pseudo instructions.  Return true if anything was modified.
selectMBB(MachineBasicBlock & MBB)243e6d15924SDimitry Andric bool SystemZPostRewrite::selectMBB(MachineBasicBlock &MBB) {
244e6d15924SDimitry Andric   bool Modified = false;
245e6d15924SDimitry Andric 
246e6d15924SDimitry Andric   MachineBasicBlock::iterator MBBI = MBB.begin(), E = MBB.end();
247e6d15924SDimitry Andric   while (MBBI != E) {
248e6d15924SDimitry Andric     MachineBasicBlock::iterator NMBBI = std::next(MBBI);
249e6d15924SDimitry Andric     Modified |= selectMI(MBB, MBBI, NMBBI);
250e6d15924SDimitry Andric     MBBI = NMBBI;
251e6d15924SDimitry Andric   }
252e6d15924SDimitry Andric 
253e6d15924SDimitry Andric   return Modified;
254e6d15924SDimitry Andric }
255e6d15924SDimitry Andric 
runOnMachineFunction(MachineFunction & MF)256e6d15924SDimitry Andric bool SystemZPostRewrite::runOnMachineFunction(MachineFunction &MF) {
257145449b1SDimitry Andric   TII = MF.getSubtarget<SystemZSubtarget>().getInstrInfo();
258e6d15924SDimitry Andric 
259e6d15924SDimitry Andric   bool Modified = false;
260e6d15924SDimitry Andric   for (auto &MBB : MF)
261e6d15924SDimitry Andric     Modified |= selectMBB(MBB);
262e6d15924SDimitry Andric 
263e6d15924SDimitry Andric   return Modified;
264e6d15924SDimitry Andric }
265e6d15924SDimitry Andric 
266