blob: c9a3527d3fbd69fcb268e6980a5cdae1eea515d3 [file] [log] [blame]
Dan Gohmanf0b165a2015-12-05 03:03:35 +00001//===-- WebAssemblyLowerBrUnless.cpp - Lower br_unless --------------------===//
2//
3// The LLVM Compiler Infrastructure
4//
5// This file is distributed under the University of Illinois Open Source
6// License. See LICENSE.TXT for details.
7//
8//===----------------------------------------------------------------------===//
9///
10/// \file
Adrian Prantl5f8f34e42018-05-01 15:54:18 +000011/// This file lowers br_unless into br_if with an inverted condition.
Dan Gohmanf0b165a2015-12-05 03:03:35 +000012///
13/// br_unless is not currently in the spec, but it's very convenient for LLVM
14/// to use. This pass allows LLVM to use it, for now.
15///
16//===----------------------------------------------------------------------===//
17
Dan Gohman83947562016-01-20 05:54:22 +000018#include "MCTargetDesc/WebAssemblyMCTargetDesc.h"
Chandler Carruth6bda14b2017-06-06 11:49:48 +000019#include "WebAssembly.h"
Dan Gohmanf0b165a2015-12-05 03:03:35 +000020#include "WebAssemblyMachineFunctionInfo.h"
21#include "WebAssemblySubtarget.h"
Dan Gohmanf0b165a2015-12-05 03:03:35 +000022#include "llvm/CodeGen/MachineFunctionPass.h"
23#include "llvm/CodeGen/MachineInstrBuilder.h"
24#include "llvm/Support/Debug.h"
25#include "llvm/Support/raw_ostream.h"
26using namespace llvm;
27
28#define DEBUG_TYPE "wasm-lower-br_unless"
29
30namespace {
31class WebAssemblyLowerBrUnless final : public MachineFunctionPass {
Mehdi Amini117296c2016-10-01 02:56:57 +000032 StringRef getPassName() const override {
Dan Gohmanf0b165a2015-12-05 03:03:35 +000033 return "WebAssembly Lower br_unless";
34 }
35
36 void getAnalysisUsage(AnalysisUsage &AU) const override {
37 AU.setPreservesCFG();
38 MachineFunctionPass::getAnalysisUsage(AU);
39 }
40
41 bool runOnMachineFunction(MachineFunction &MF) override;
42
43public:
44 static char ID; // Pass identification, replacement for typeid
45 WebAssemblyLowerBrUnless() : MachineFunctionPass(ID) {}
46};
47} // end anonymous namespace
48
49char WebAssemblyLowerBrUnless::ID = 0;
Jacob Gravelle40926452018-03-30 20:36:58 +000050INITIALIZE_PASS(WebAssemblyLowerBrUnless, DEBUG_TYPE,
51 "Lowers br_unless into inverted br_if", false, false)
52
Dan Gohmanf0b165a2015-12-05 03:03:35 +000053FunctionPass *llvm::createWebAssemblyLowerBrUnless() {
54 return new WebAssemblyLowerBrUnless();
55}
56
57bool WebAssemblyLowerBrUnless::runOnMachineFunction(MachineFunction &MF) {
Nicola Zaghend34e60c2018-05-14 12:53:11 +000058 LLVM_DEBUG(dbgs() << "********** Lowering br_unless **********\n"
59 "********** Function: "
60 << MF.getName() << '\n');
Dan Gohmanf0b165a2015-12-05 03:03:35 +000061
62 auto &MFI = *MF.getInfo<WebAssemblyFunctionInfo>();
63 const auto &TII = *MF.getSubtarget<WebAssemblySubtarget>().getInstrInfo();
64 auto &MRI = MF.getRegInfo();
65
66 for (auto &MBB : MF) {
Dan Gohman83947562016-01-20 05:54:22 +000067 for (auto MII = MBB.begin(); MII != MBB.end();) {
Dan Gohmanf0b165a2015-12-05 03:03:35 +000068 MachineInstr *MI = &*MII++;
69 if (MI->getOpcode() != WebAssembly::BR_UNLESS)
70 continue;
71
Dan Gohman06b49582016-02-08 21:50:13 +000072 unsigned Cond = MI->getOperand(1).getReg();
Dan Gohmanf0b165a2015-12-05 03:03:35 +000073 bool Inverted = false;
74
75 // Attempt to invert the condition in place.
76 if (MFI.isVRegStackified(Cond)) {
77 assert(MRI.hasOneDef(Cond));
78 MachineInstr *Def = MRI.getVRegDef(Cond);
79 switch (Def->getOpcode()) {
Dan Gohman83947562016-01-20 05:54:22 +000080 using namespace WebAssembly;
Heejin Ahnf208f632018-09-05 01:27:38 +000081 case EQ_I32:
82 Def->setDesc(TII.get(NE_I32));
83 Inverted = true;
84 break;
85 case NE_I32:
86 Def->setDesc(TII.get(EQ_I32));
87 Inverted = true;
88 break;
89 case GT_S_I32:
90 Def->setDesc(TII.get(LE_S_I32));
91 Inverted = true;
92 break;
93 case GE_S_I32:
94 Def->setDesc(TII.get(LT_S_I32));
95 Inverted = true;
96 break;
97 case LT_S_I32:
98 Def->setDesc(TII.get(GE_S_I32));
99 Inverted = true;
100 break;
101 case LE_S_I32:
102 Def->setDesc(TII.get(GT_S_I32));
103 Inverted = true;
104 break;
105 case GT_U_I32:
106 Def->setDesc(TII.get(LE_U_I32));
107 Inverted = true;
108 break;
109 case GE_U_I32:
110 Def->setDesc(TII.get(LT_U_I32));
111 Inverted = true;
112 break;
113 case LT_U_I32:
114 Def->setDesc(TII.get(GE_U_I32));
115 Inverted = true;
116 break;
117 case LE_U_I32:
118 Def->setDesc(TII.get(GT_U_I32));
119 Inverted = true;
120 break;
121 case EQ_I64:
122 Def->setDesc(TII.get(NE_I64));
123 Inverted = true;
124 break;
125 case NE_I64:
126 Def->setDesc(TII.get(EQ_I64));
127 Inverted = true;
128 break;
129 case GT_S_I64:
130 Def->setDesc(TII.get(LE_S_I64));
131 Inverted = true;
132 break;
133 case GE_S_I64:
134 Def->setDesc(TII.get(LT_S_I64));
135 Inverted = true;
136 break;
137 case LT_S_I64:
138 Def->setDesc(TII.get(GE_S_I64));
139 Inverted = true;
140 break;
141 case LE_S_I64:
142 Def->setDesc(TII.get(GT_S_I64));
143 Inverted = true;
144 break;
145 case GT_U_I64:
146 Def->setDesc(TII.get(LE_U_I64));
147 Inverted = true;
148 break;
149 case GE_U_I64:
150 Def->setDesc(TII.get(LT_U_I64));
151 Inverted = true;
152 break;
153 case LT_U_I64:
154 Def->setDesc(TII.get(GE_U_I64));
155 Inverted = true;
156 break;
157 case LE_U_I64:
158 Def->setDesc(TII.get(GT_U_I64));
159 Inverted = true;
160 break;
161 case EQ_F32:
162 Def->setDesc(TII.get(NE_F32));
163 Inverted = true;
164 break;
165 case NE_F32:
166 Def->setDesc(TII.get(EQ_F32));
167 Inverted = true;
168 break;
169 case EQ_F64:
170 Def->setDesc(TII.get(NE_F64));
171 Inverted = true;
172 break;
173 case NE_F64:
174 Def->setDesc(TII.get(EQ_F64));
175 Inverted = true;
176 break;
Dan Gohman580c1022017-11-29 20:20:11 +0000177 case EQZ_I32: {
178 // Invert an eqz by replacing it with its operand.
179 Cond = Def->getOperand(1).getReg();
180 Def->eraseFromParent();
181 Inverted = true;
182 break;
183 }
Heejin Ahnf208f632018-09-05 01:27:38 +0000184 default:
185 break;
Dan Gohmanf0b165a2015-12-05 03:03:35 +0000186 }
187 }
188
189 // If we weren't able to invert the condition in place. Insert an
Dan Gohmane0405332016-10-03 22:43:53 +0000190 // instruction to invert it.
Dan Gohmanf0b165a2015-12-05 03:03:35 +0000191 if (!Inverted) {
Dan Gohmanf0b165a2015-12-05 03:03:35 +0000192 unsigned Tmp = MRI.createVirtualRegister(&WebAssembly::I32RegClass);
Dan Gohman804749c2016-05-16 18:59:34 +0000193 BuildMI(MBB, MI, MI->getDebugLoc(), TII.get(WebAssembly::EQZ_I32), Tmp)
194 .addReg(Cond);
Dan Gohman4fc4e422016-10-24 19:49:43 +0000195 MFI.stackifyVReg(Tmp);
Dan Gohmanf0b165a2015-12-05 03:03:35 +0000196 Cond = Tmp;
197 Inverted = true;
198 }
199
200 // The br_unless condition has now been inverted. Insert a br_if and
201 // delete the br_unless.
202 assert(Inverted);
203 BuildMI(MBB, MI, MI->getDebugLoc(), TII.get(WebAssembly::BR_IF))
Diana Picus116bbab2017-01-13 09:58:52 +0000204 .add(MI->getOperand(0))
Dan Gohman06b49582016-02-08 21:50:13 +0000205 .addReg(Cond);
Dan Gohmanf0b165a2015-12-05 03:03:35 +0000206 MBB.erase(MI);
207 }
208 }
209
210 return true;
211}