xref: /freebsd-src/contrib/llvm-project/llvm/lib/Target/WebAssembly/WebAssemblyLowerBrUnless.cpp (revision 0b57cec536236d46e3dba9bd041533462f33dbb7)
1*0b57cec5SDimitry Andric //===-- WebAssemblyLowerBrUnless.cpp - Lower br_unless --------------------===//
2*0b57cec5SDimitry Andric //
3*0b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4*0b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
5*0b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6*0b57cec5SDimitry Andric //
7*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
8*0b57cec5SDimitry Andric ///
9*0b57cec5SDimitry Andric /// \file
10*0b57cec5SDimitry Andric /// This file lowers br_unless into br_if with an inverted condition.
11*0b57cec5SDimitry Andric ///
12*0b57cec5SDimitry Andric /// br_unless is not currently in the spec, but it's very convenient for LLVM
13*0b57cec5SDimitry Andric /// to use. This pass allows LLVM to use it, for now.
14*0b57cec5SDimitry Andric ///
15*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
16*0b57cec5SDimitry Andric 
17*0b57cec5SDimitry Andric #include "MCTargetDesc/WebAssemblyMCTargetDesc.h"
18*0b57cec5SDimitry Andric #include "WebAssembly.h"
19*0b57cec5SDimitry Andric #include "WebAssemblyMachineFunctionInfo.h"
20*0b57cec5SDimitry Andric #include "WebAssemblySubtarget.h"
21*0b57cec5SDimitry Andric #include "llvm/CodeGen/MachineFunctionPass.h"
22*0b57cec5SDimitry Andric #include "llvm/CodeGen/MachineInstrBuilder.h"
23*0b57cec5SDimitry Andric #include "llvm/Support/Debug.h"
24*0b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h"
25*0b57cec5SDimitry Andric using namespace llvm;
26*0b57cec5SDimitry Andric 
27*0b57cec5SDimitry Andric #define DEBUG_TYPE "wasm-lower-br_unless"
28*0b57cec5SDimitry Andric 
29*0b57cec5SDimitry Andric namespace {
30*0b57cec5SDimitry Andric class WebAssemblyLowerBrUnless final : public MachineFunctionPass {
31*0b57cec5SDimitry Andric   StringRef getPassName() const override {
32*0b57cec5SDimitry Andric     return "WebAssembly Lower br_unless";
33*0b57cec5SDimitry Andric   }
34*0b57cec5SDimitry Andric 
35*0b57cec5SDimitry Andric   void getAnalysisUsage(AnalysisUsage &AU) const override {
36*0b57cec5SDimitry Andric     AU.setPreservesCFG();
37*0b57cec5SDimitry Andric     MachineFunctionPass::getAnalysisUsage(AU);
38*0b57cec5SDimitry Andric   }
39*0b57cec5SDimitry Andric 
40*0b57cec5SDimitry Andric   bool runOnMachineFunction(MachineFunction &MF) override;
41*0b57cec5SDimitry Andric 
42*0b57cec5SDimitry Andric public:
43*0b57cec5SDimitry Andric   static char ID; // Pass identification, replacement for typeid
44*0b57cec5SDimitry Andric   WebAssemblyLowerBrUnless() : MachineFunctionPass(ID) {}
45*0b57cec5SDimitry Andric };
46*0b57cec5SDimitry Andric } // end anonymous namespace
47*0b57cec5SDimitry Andric 
48*0b57cec5SDimitry Andric char WebAssemblyLowerBrUnless::ID = 0;
49*0b57cec5SDimitry Andric INITIALIZE_PASS(WebAssemblyLowerBrUnless, DEBUG_TYPE,
50*0b57cec5SDimitry Andric                 "Lowers br_unless into inverted br_if", false, false)
51*0b57cec5SDimitry Andric 
52*0b57cec5SDimitry Andric FunctionPass *llvm::createWebAssemblyLowerBrUnless() {
53*0b57cec5SDimitry Andric   return new WebAssemblyLowerBrUnless();
54*0b57cec5SDimitry Andric }
55*0b57cec5SDimitry Andric 
56*0b57cec5SDimitry Andric bool WebAssemblyLowerBrUnless::runOnMachineFunction(MachineFunction &MF) {
57*0b57cec5SDimitry Andric   LLVM_DEBUG(dbgs() << "********** Lowering br_unless **********\n"
58*0b57cec5SDimitry Andric                        "********** Function: "
59*0b57cec5SDimitry Andric                     << MF.getName() << '\n');
60*0b57cec5SDimitry Andric 
61*0b57cec5SDimitry Andric   auto &MFI = *MF.getInfo<WebAssemblyFunctionInfo>();
62*0b57cec5SDimitry Andric   const auto &TII = *MF.getSubtarget<WebAssemblySubtarget>().getInstrInfo();
63*0b57cec5SDimitry Andric   auto &MRI = MF.getRegInfo();
64*0b57cec5SDimitry Andric 
65*0b57cec5SDimitry Andric   for (auto &MBB : MF) {
66*0b57cec5SDimitry Andric     for (auto MII = MBB.begin(); MII != MBB.end();) {
67*0b57cec5SDimitry Andric       MachineInstr *MI = &*MII++;
68*0b57cec5SDimitry Andric       if (MI->getOpcode() != WebAssembly::BR_UNLESS)
69*0b57cec5SDimitry Andric         continue;
70*0b57cec5SDimitry Andric 
71*0b57cec5SDimitry Andric       unsigned Cond = MI->getOperand(1).getReg();
72*0b57cec5SDimitry Andric       bool Inverted = false;
73*0b57cec5SDimitry Andric 
74*0b57cec5SDimitry Andric       // Attempt to invert the condition in place.
75*0b57cec5SDimitry Andric       if (MFI.isVRegStackified(Cond)) {
76*0b57cec5SDimitry Andric         assert(MRI.hasOneDef(Cond));
77*0b57cec5SDimitry Andric         MachineInstr *Def = MRI.getVRegDef(Cond);
78*0b57cec5SDimitry Andric         switch (Def->getOpcode()) {
79*0b57cec5SDimitry Andric           using namespace WebAssembly;
80*0b57cec5SDimitry Andric         case EQ_I32:
81*0b57cec5SDimitry Andric           Def->setDesc(TII.get(NE_I32));
82*0b57cec5SDimitry Andric           Inverted = true;
83*0b57cec5SDimitry Andric           break;
84*0b57cec5SDimitry Andric         case NE_I32:
85*0b57cec5SDimitry Andric           Def->setDesc(TII.get(EQ_I32));
86*0b57cec5SDimitry Andric           Inverted = true;
87*0b57cec5SDimitry Andric           break;
88*0b57cec5SDimitry Andric         case GT_S_I32:
89*0b57cec5SDimitry Andric           Def->setDesc(TII.get(LE_S_I32));
90*0b57cec5SDimitry Andric           Inverted = true;
91*0b57cec5SDimitry Andric           break;
92*0b57cec5SDimitry Andric         case GE_S_I32:
93*0b57cec5SDimitry Andric           Def->setDesc(TII.get(LT_S_I32));
94*0b57cec5SDimitry Andric           Inverted = true;
95*0b57cec5SDimitry Andric           break;
96*0b57cec5SDimitry Andric         case LT_S_I32:
97*0b57cec5SDimitry Andric           Def->setDesc(TII.get(GE_S_I32));
98*0b57cec5SDimitry Andric           Inverted = true;
99*0b57cec5SDimitry Andric           break;
100*0b57cec5SDimitry Andric         case LE_S_I32:
101*0b57cec5SDimitry Andric           Def->setDesc(TII.get(GT_S_I32));
102*0b57cec5SDimitry Andric           Inverted = true;
103*0b57cec5SDimitry Andric           break;
104*0b57cec5SDimitry Andric         case GT_U_I32:
105*0b57cec5SDimitry Andric           Def->setDesc(TII.get(LE_U_I32));
106*0b57cec5SDimitry Andric           Inverted = true;
107*0b57cec5SDimitry Andric           break;
108*0b57cec5SDimitry Andric         case GE_U_I32:
109*0b57cec5SDimitry Andric           Def->setDesc(TII.get(LT_U_I32));
110*0b57cec5SDimitry Andric           Inverted = true;
111*0b57cec5SDimitry Andric           break;
112*0b57cec5SDimitry Andric         case LT_U_I32:
113*0b57cec5SDimitry Andric           Def->setDesc(TII.get(GE_U_I32));
114*0b57cec5SDimitry Andric           Inverted = true;
115*0b57cec5SDimitry Andric           break;
116*0b57cec5SDimitry Andric         case LE_U_I32:
117*0b57cec5SDimitry Andric           Def->setDesc(TII.get(GT_U_I32));
118*0b57cec5SDimitry Andric           Inverted = true;
119*0b57cec5SDimitry Andric           break;
120*0b57cec5SDimitry Andric         case EQ_I64:
121*0b57cec5SDimitry Andric           Def->setDesc(TII.get(NE_I64));
122*0b57cec5SDimitry Andric           Inverted = true;
123*0b57cec5SDimitry Andric           break;
124*0b57cec5SDimitry Andric         case NE_I64:
125*0b57cec5SDimitry Andric           Def->setDesc(TII.get(EQ_I64));
126*0b57cec5SDimitry Andric           Inverted = true;
127*0b57cec5SDimitry Andric           break;
128*0b57cec5SDimitry Andric         case GT_S_I64:
129*0b57cec5SDimitry Andric           Def->setDesc(TII.get(LE_S_I64));
130*0b57cec5SDimitry Andric           Inverted = true;
131*0b57cec5SDimitry Andric           break;
132*0b57cec5SDimitry Andric         case GE_S_I64:
133*0b57cec5SDimitry Andric           Def->setDesc(TII.get(LT_S_I64));
134*0b57cec5SDimitry Andric           Inverted = true;
135*0b57cec5SDimitry Andric           break;
136*0b57cec5SDimitry Andric         case LT_S_I64:
137*0b57cec5SDimitry Andric           Def->setDesc(TII.get(GE_S_I64));
138*0b57cec5SDimitry Andric           Inverted = true;
139*0b57cec5SDimitry Andric           break;
140*0b57cec5SDimitry Andric         case LE_S_I64:
141*0b57cec5SDimitry Andric           Def->setDesc(TII.get(GT_S_I64));
142*0b57cec5SDimitry Andric           Inverted = true;
143*0b57cec5SDimitry Andric           break;
144*0b57cec5SDimitry Andric         case GT_U_I64:
145*0b57cec5SDimitry Andric           Def->setDesc(TII.get(LE_U_I64));
146*0b57cec5SDimitry Andric           Inverted = true;
147*0b57cec5SDimitry Andric           break;
148*0b57cec5SDimitry Andric         case GE_U_I64:
149*0b57cec5SDimitry Andric           Def->setDesc(TII.get(LT_U_I64));
150*0b57cec5SDimitry Andric           Inverted = true;
151*0b57cec5SDimitry Andric           break;
152*0b57cec5SDimitry Andric         case LT_U_I64:
153*0b57cec5SDimitry Andric           Def->setDesc(TII.get(GE_U_I64));
154*0b57cec5SDimitry Andric           Inverted = true;
155*0b57cec5SDimitry Andric           break;
156*0b57cec5SDimitry Andric         case LE_U_I64:
157*0b57cec5SDimitry Andric           Def->setDesc(TII.get(GT_U_I64));
158*0b57cec5SDimitry Andric           Inverted = true;
159*0b57cec5SDimitry Andric           break;
160*0b57cec5SDimitry Andric         case EQ_F32:
161*0b57cec5SDimitry Andric           Def->setDesc(TII.get(NE_F32));
162*0b57cec5SDimitry Andric           Inverted = true;
163*0b57cec5SDimitry Andric           break;
164*0b57cec5SDimitry Andric         case NE_F32:
165*0b57cec5SDimitry Andric           Def->setDesc(TII.get(EQ_F32));
166*0b57cec5SDimitry Andric           Inverted = true;
167*0b57cec5SDimitry Andric           break;
168*0b57cec5SDimitry Andric         case EQ_F64:
169*0b57cec5SDimitry Andric           Def->setDesc(TII.get(NE_F64));
170*0b57cec5SDimitry Andric           Inverted = true;
171*0b57cec5SDimitry Andric           break;
172*0b57cec5SDimitry Andric         case NE_F64:
173*0b57cec5SDimitry Andric           Def->setDesc(TII.get(EQ_F64));
174*0b57cec5SDimitry Andric           Inverted = true;
175*0b57cec5SDimitry Andric           break;
176*0b57cec5SDimitry Andric         case EQZ_I32: {
177*0b57cec5SDimitry Andric           // Invert an eqz by replacing it with its operand.
178*0b57cec5SDimitry Andric           Cond = Def->getOperand(1).getReg();
179*0b57cec5SDimitry Andric           Def->eraseFromParent();
180*0b57cec5SDimitry Andric           Inverted = true;
181*0b57cec5SDimitry Andric           break;
182*0b57cec5SDimitry Andric         }
183*0b57cec5SDimitry Andric         default:
184*0b57cec5SDimitry Andric           break;
185*0b57cec5SDimitry Andric         }
186*0b57cec5SDimitry Andric       }
187*0b57cec5SDimitry Andric 
188*0b57cec5SDimitry Andric       // If we weren't able to invert the condition in place. Insert an
189*0b57cec5SDimitry Andric       // instruction to invert it.
190*0b57cec5SDimitry Andric       if (!Inverted) {
191*0b57cec5SDimitry Andric         unsigned Tmp = MRI.createVirtualRegister(&WebAssembly::I32RegClass);
192*0b57cec5SDimitry Andric         BuildMI(MBB, MI, MI->getDebugLoc(), TII.get(WebAssembly::EQZ_I32), Tmp)
193*0b57cec5SDimitry Andric             .addReg(Cond);
194*0b57cec5SDimitry Andric         MFI.stackifyVReg(Tmp);
195*0b57cec5SDimitry Andric         Cond = Tmp;
196*0b57cec5SDimitry Andric         Inverted = true;
197*0b57cec5SDimitry Andric       }
198*0b57cec5SDimitry Andric 
199*0b57cec5SDimitry Andric       // The br_unless condition has now been inverted. Insert a br_if and
200*0b57cec5SDimitry Andric       // delete the br_unless.
201*0b57cec5SDimitry Andric       assert(Inverted);
202*0b57cec5SDimitry Andric       BuildMI(MBB, MI, MI->getDebugLoc(), TII.get(WebAssembly::BR_IF))
203*0b57cec5SDimitry Andric           .add(MI->getOperand(0))
204*0b57cec5SDimitry Andric           .addReg(Cond);
205*0b57cec5SDimitry Andric       MBB.erase(MI);
206*0b57cec5SDimitry Andric     }
207*0b57cec5SDimitry Andric   }
208*0b57cec5SDimitry Andric 
209*0b57cec5SDimitry Andric   return true;
210*0b57cec5SDimitry Andric }
211