blob: f3bfa5506e4b31d4ef1548a7bc649578ba107d77 [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
11/// \brief This file lowers br_unless into br_if with an inverted condition.
12///
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
18#include "WebAssembly.h"
Dan Gohman83947562016-01-20 05:54:22 +000019#include "MCTargetDesc/WebAssemblyMCTargetDesc.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;
50FunctionPass *llvm::createWebAssemblyLowerBrUnless() {
51 return new WebAssemblyLowerBrUnless();
52}
53
54bool WebAssemblyLowerBrUnless::runOnMachineFunction(MachineFunction &MF) {
55 DEBUG(dbgs() << "********** Lowering br_unless **********\n"
56 "********** Function: "
57 << MF.getName() << '\n');
58
59 auto &MFI = *MF.getInfo<WebAssemblyFunctionInfo>();
60 const auto &TII = *MF.getSubtarget<WebAssemblySubtarget>().getInstrInfo();
61 auto &MRI = MF.getRegInfo();
62
63 for (auto &MBB : MF) {
Dan Gohman83947562016-01-20 05:54:22 +000064 for (auto MII = MBB.begin(); MII != MBB.end();) {
Dan Gohmanf0b165a2015-12-05 03:03:35 +000065 MachineInstr *MI = &*MII++;
66 if (MI->getOpcode() != WebAssembly::BR_UNLESS)
67 continue;
68
Dan Gohman06b49582016-02-08 21:50:13 +000069 unsigned Cond = MI->getOperand(1).getReg();
Dan Gohmanf0b165a2015-12-05 03:03:35 +000070 bool Inverted = false;
71
72 // Attempt to invert the condition in place.
73 if (MFI.isVRegStackified(Cond)) {
74 assert(MRI.hasOneDef(Cond));
75 MachineInstr *Def = MRI.getVRegDef(Cond);
76 switch (Def->getOpcode()) {
Dan Gohman83947562016-01-20 05:54:22 +000077 using namespace WebAssembly;
Dan Gohmanf0b165a2015-12-05 03:03:35 +000078 case EQ_I32: Def->setDesc(TII.get(NE_I32)); Inverted = true; break;
79 case NE_I32: Def->setDesc(TII.get(EQ_I32)); Inverted = true; break;
80 case GT_S_I32: Def->setDesc(TII.get(LE_S_I32)); Inverted = true; break;
81 case GE_S_I32: Def->setDesc(TII.get(LT_S_I32)); Inverted = true; break;
82 case LT_S_I32: Def->setDesc(TII.get(GE_S_I32)); Inverted = true; break;
83 case LE_S_I32: Def->setDesc(TII.get(GT_S_I32)); Inverted = true; break;
84 case GT_U_I32: Def->setDesc(TII.get(LE_U_I32)); Inverted = true; break;
85 case GE_U_I32: Def->setDesc(TII.get(LT_U_I32)); Inverted = true; break;
86 case LT_U_I32: Def->setDesc(TII.get(GE_U_I32)); Inverted = true; break;
87 case LE_U_I32: Def->setDesc(TII.get(GT_U_I32)); Inverted = true; break;
88 case EQ_I64: Def->setDesc(TII.get(NE_I64)); Inverted = true; break;
89 case NE_I64: Def->setDesc(TII.get(EQ_I64)); Inverted = true; break;
90 case GT_S_I64: Def->setDesc(TII.get(LE_S_I64)); Inverted = true; break;
91 case GE_S_I64: Def->setDesc(TII.get(LT_S_I64)); Inverted = true; break;
92 case LT_S_I64: Def->setDesc(TII.get(GE_S_I64)); Inverted = true; break;
93 case LE_S_I64: Def->setDesc(TII.get(GT_S_I64)); Inverted = true; break;
94 case GT_U_I64: Def->setDesc(TII.get(LE_U_I64)); Inverted = true; break;
95 case GE_U_I64: Def->setDesc(TII.get(LT_U_I64)); Inverted = true; break;
96 case LT_U_I64: Def->setDesc(TII.get(GE_U_I64)); Inverted = true; break;
97 case LE_U_I64: Def->setDesc(TII.get(GT_U_I64)); Inverted = true; break;
98 case EQ_F32: Def->setDesc(TII.get(NE_F32)); Inverted = true; break;
99 case NE_F32: Def->setDesc(TII.get(EQ_F32)); Inverted = true; break;
100 case EQ_F64: Def->setDesc(TII.get(NE_F64)); Inverted = true; break;
101 case NE_F64: Def->setDesc(TII.get(EQ_F64)); Inverted = true; break;
102 default: break;
103 }
104 }
105
106 // If we weren't able to invert the condition in place. Insert an
107 // expression to invert it.
108 if (!Inverted) {
Dan Gohmanf0b165a2015-12-05 03:03:35 +0000109 unsigned Tmp = MRI.createVirtualRegister(&WebAssembly::I32RegClass);
110 MFI.stackifyVReg(Tmp);
Dan Gohman804749c2016-05-16 18:59:34 +0000111 BuildMI(MBB, MI, MI->getDebugLoc(), TII.get(WebAssembly::EQZ_I32), Tmp)
112 .addReg(Cond);
Dan Gohmanf0b165a2015-12-05 03:03:35 +0000113 Cond = Tmp;
114 Inverted = true;
115 }
116
117 // The br_unless condition has now been inverted. Insert a br_if and
118 // delete the br_unless.
119 assert(Inverted);
120 BuildMI(MBB, MI, MI->getDebugLoc(), TII.get(WebAssembly::BR_IF))
Dan Gohman06b49582016-02-08 21:50:13 +0000121 .addOperand(MI->getOperand(0))
122 .addReg(Cond);
Dan Gohmanf0b165a2015-12-05 03:03:35 +0000123 MBB.erase(MI);
124 }
125 }
126
127 return true;
128}