blob: 75d04252cbe9929fb7271fb92828f31cb2b97008 [file] [log] [blame]
Heejin Ahn4934f762018-06-25 01:07:11 +00001//=== WebAssemblyLateEHPrepare.cpp - WebAssembly Exception Preparation -===//
2//
Chandler Carruth2946cd72019-01-19 08:50:56 +00003// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4// See https://llvm.org/LICENSE.txt for license information.
5// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
Heejin Ahn4934f762018-06-25 01:07:11 +00006//
7//===----------------------------------------------------------------------===//
8///
9/// \file
10/// \brief Does various transformations for exception handling.
11///
12//===----------------------------------------------------------------------===//
13
14#include "MCTargetDesc/WebAssemblyMCTargetDesc.h"
15#include "WebAssembly.h"
16#include "WebAssemblySubtarget.h"
17#include "WebAssemblyUtilities.h"
Heejin Ahnd6f48782019-01-30 03:21:57 +000018#include "llvm/ADT/SmallSet.h"
Heejin Ahn4934f762018-06-25 01:07:11 +000019#include "llvm/CodeGen/MachineInstrBuilder.h"
20#include "llvm/CodeGen/WasmEHFuncInfo.h"
21#include "llvm/MC/MCAsmInfo.h"
Reid Kleckner904cd3e2019-10-19 01:31:09 +000022#include "llvm/Support/Debug.h"
Heejin Ahn4934f762018-06-25 01:07:11 +000023using namespace llvm;
24
Heejin Ahn8b49b6b2019-03-13 00:37:31 +000025#define DEBUG_TYPE "wasm-late-eh-prepare"
Heejin Ahn4934f762018-06-25 01:07:11 +000026
27namespace {
28class WebAssemblyLateEHPrepare final : public MachineFunctionPass {
29 StringRef getPassName() const override {
Heejin Ahnd6f48782019-01-30 03:21:57 +000030 return "WebAssembly Late Prepare Exception";
Heejin Ahn4934f762018-06-25 01:07:11 +000031 }
32
33 bool runOnMachineFunction(MachineFunction &MF) override;
Heejin Ahnd6f48782019-01-30 03:21:57 +000034 bool addCatches(MachineFunction &MF);
Heejin Ahnb47a18c2019-03-16 04:46:05 +000035 bool replaceFuncletReturns(MachineFunction &MF);
36 bool removeUnnecessaryUnreachables(MachineFunction &MF);
Heejin Ahnd6f48782019-01-30 03:21:57 +000037 bool addExceptionExtraction(MachineFunction &MF);
Heejin Ahn0bb98652019-01-30 22:44:45 +000038 bool restoreStackPointer(MachineFunction &MF);
Heejin Ahn4934f762018-06-25 01:07:11 +000039
40public:
41 static char ID; // Pass identification, replacement for typeid
42 WebAssemblyLateEHPrepare() : MachineFunctionPass(ID) {}
43};
44} // end anonymous namespace
45
46char WebAssemblyLateEHPrepare::ID = 0;
47INITIALIZE_PASS(WebAssemblyLateEHPrepare, DEBUG_TYPE,
Heejin Ahna93e7262018-08-17 00:12:04 +000048 "WebAssembly Late Exception Preparation", false, false)
Heejin Ahn4934f762018-06-25 01:07:11 +000049
50FunctionPass *llvm::createWebAssemblyLateEHPrepare() {
51 return new WebAssemblyLateEHPrepare();
52}
53
54// Returns the nearest EH pad that dominates this instruction. This does not use
55// dominator analysis; it just does BFS on its predecessors until arriving at an
56// EH pad. This assumes valid EH scopes so the first EH pad it arrives in all
57// possible search paths should be the same.
58// Returns nullptr in case it does not find any EH pad in the search, or finds
59// multiple different EH pads.
Heejin Ahn095796a2018-11-16 00:47:18 +000060static MachineBasicBlock *getMatchingEHPad(MachineInstr *MI) {
Heejin Ahn4934f762018-06-25 01:07:11 +000061 MachineFunction *MF = MI->getParent()->getParent();
62 SmallVector<MachineBasicBlock *, 2> WL;
63 SmallPtrSet<MachineBasicBlock *, 2> Visited;
64 WL.push_back(MI->getParent());
65 MachineBasicBlock *EHPad = nullptr;
66 while (!WL.empty()) {
67 MachineBasicBlock *MBB = WL.pop_back_val();
68 if (Visited.count(MBB))
69 continue;
70 Visited.insert(MBB);
71 if (MBB->isEHPad()) {
72 if (EHPad && EHPad != MBB)
73 return nullptr;
74 EHPad = MBB;
75 continue;
76 }
77 if (MBB == &MF->front())
78 return nullptr;
79 WL.append(MBB->pred_begin(), MBB->pred_end());
80 }
81 return EHPad;
82}
83
Heejin Ahn095796a2018-11-16 00:47:18 +000084// Erase the specified BBs if the BB does not have any remaining predecessors,
85// and also all its dead children.
Benjamin Kramer83996e42018-08-08 10:13:19 +000086template <typename Container>
Heejin Ahn095796a2018-11-16 00:47:18 +000087static void eraseDeadBBsAndChildren(const Container &MBBs) {
Benjamin Kramer83996e42018-08-08 10:13:19 +000088 SmallVector<MachineBasicBlock *, 8> WL(MBBs.begin(), MBBs.end());
Heejin Ahn4934f762018-06-25 01:07:11 +000089 while (!WL.empty()) {
90 MachineBasicBlock *MBB = WL.pop_back_val();
Heejin Ahn095796a2018-11-16 00:47:18 +000091 if (!MBB->pred_empty())
92 continue;
Heejin Ahnb68d5912018-10-04 21:03:35 +000093 SmallVector<MachineBasicBlock *, 4> Succs(MBB->succ_begin(),
94 MBB->succ_end());
95 WL.append(MBB->succ_begin(), MBB->succ_end());
96 for (auto *Succ : Succs)
Heejin Ahn4934f762018-06-25 01:07:11 +000097 MBB->removeSuccessor(Succ);
Heejin Ahn4934f762018-06-25 01:07:11 +000098 MBB->eraseFromParent();
99 }
100}
101
102bool WebAssemblyLateEHPrepare::runOnMachineFunction(MachineFunction &MF) {
Heejin Ahn569f0902019-01-09 23:05:21 +0000103 LLVM_DEBUG(dbgs() << "********** Late EH Prepare **********\n"
104 "********** Function: "
105 << MF.getName() << '\n');
106
Heejin Ahn4934f762018-06-25 01:07:11 +0000107 if (MF.getTarget().getMCAsmInfo()->getExceptionHandlingType() !=
108 ExceptionHandling::Wasm)
109 return false;
110
111 bool Changed = false;
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000112 if (MF.getFunction().hasPersonalityFn()) {
113 Changed |= addCatches(MF);
114 Changed |= replaceFuncletReturns(MF);
115 }
Heejin Ahn095796a2018-11-16 00:47:18 +0000116 Changed |= removeUnnecessaryUnreachables(MF);
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000117 if (MF.getFunction().hasPersonalityFn()) {
118 Changed |= addExceptionExtraction(MF);
119 Changed |= restoreStackPointer(MF);
120 }
Heejin Ahn4934f762018-06-25 01:07:11 +0000121 return Changed;
122}
123
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000124// Add catch instruction to beginning of catchpads and cleanuppads.
125bool WebAssemblyLateEHPrepare::addCatches(MachineFunction &MF) {
Heejin Ahn095796a2018-11-16 00:47:18 +0000126 bool Changed = false;
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000127 const auto &TII = *MF.getSubtarget<WebAssemblySubtarget>().getInstrInfo();
128 MachineRegisterInfo &MRI = MF.getRegInfo();
Heejin Ahn095796a2018-11-16 00:47:18 +0000129 for (auto &MBB : MF) {
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000130 if (MBB.isEHPad()) {
Heejin Ahn095796a2018-11-16 00:47:18 +0000131 Changed = true;
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000132 auto InsertPos = MBB.begin();
133 if (InsertPos->isEHLabel()) // EH pad starts with an EH label
134 ++InsertPos;
Daniel Sanders05c145d2019-08-12 22:40:45 +0000135 Register DstReg = MRI.createVirtualRegister(&WebAssembly::EXNREFRegClass);
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000136 BuildMI(MBB, InsertPos, MBB.begin()->getDebugLoc(),
137 TII.get(WebAssembly::CATCH), DstReg);
Heejin Ahn095796a2018-11-16 00:47:18 +0000138 }
139 }
Heejin Ahn095796a2018-11-16 00:47:18 +0000140 return Changed;
141}
142
Heejin Ahn4934f762018-06-25 01:07:11 +0000143bool WebAssemblyLateEHPrepare::replaceFuncletReturns(MachineFunction &MF) {
144 bool Changed = false;
145 const auto &TII = *MF.getSubtarget<WebAssemblySubtarget>().getInstrInfo();
Heejin Ahn4934f762018-06-25 01:07:11 +0000146
147 for (auto &MBB : MF) {
148 auto Pos = MBB.getFirstTerminator();
149 if (Pos == MBB.end())
150 continue;
151 MachineInstr *TI = &*Pos;
152
153 switch (TI->getOpcode()) {
154 case WebAssembly::CATCHRET: {
155 // Replace a catchret with a branch
156 MachineBasicBlock *TBB = TI->getOperand(0).getMBB();
157 if (!MBB.isLayoutSuccessor(TBB))
158 BuildMI(MBB, TI, TI->getDebugLoc(), TII.get(WebAssembly::BR))
159 .addMBB(TBB);
160 TI->eraseFromParent();
161 Changed = true;
162 break;
163 }
Heejin Ahn66ce4192019-03-16 05:38:57 +0000164 case WebAssembly::CLEANUPRET:
165 case WebAssembly::RETHROW_IN_CATCH: {
166 // Replace a cleanupret/rethrow_in_catch with a rethrow
167 auto *EHPad = getMatchingEHPad(TI);
168 auto CatchPos = EHPad->begin();
169 if (CatchPos->isEHLabel()) // EH pad starts with an EH label
170 ++CatchPos;
171 MachineInstr *Catch = &*CatchPos;
Daniel Sanders05c145d2019-08-12 22:40:45 +0000172 Register ExnReg = Catch->getOperand(0).getReg();
Heejin Ahn66ce4192019-03-16 05:38:57 +0000173 BuildMI(MBB, TI, TI->getDebugLoc(), TII.get(WebAssembly::RETHROW))
174 .addReg(ExnReg);
Heejin Ahn4934f762018-06-25 01:07:11 +0000175 TI->eraseFromParent();
176 Changed = true;
177 break;
178 }
179 }
180 }
181 return Changed;
182}
183
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000184bool WebAssemblyLateEHPrepare::removeUnnecessaryUnreachables(
185 MachineFunction &MF) {
Heejin Ahn4934f762018-06-25 01:07:11 +0000186 bool Changed = false;
Heejin Ahn4934f762018-06-25 01:07:11 +0000187 for (auto &MBB : MF) {
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000188 for (auto &MI : MBB) {
189 if (MI.getOpcode() != WebAssembly::THROW &&
190 MI.getOpcode() != WebAssembly::RETHROW)
191 continue;
Heejin Ahn4934f762018-06-25 01:07:11 +0000192 Changed = true;
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000193
194 // The instruction after the throw should be an unreachable or a branch to
195 // another BB that should eventually lead to an unreachable. Delete it
196 // because throw itself is a terminator, and also delete successors if
197 // any.
198 MBB.erase(std::next(MI.getIterator()), MBB.end());
199 SmallVector<MachineBasicBlock *, 8> Succs(MBB.succ_begin(),
200 MBB.succ_end());
201 for (auto *Succ : Succs)
Heejin Ahn66ce4192019-03-16 05:38:57 +0000202 if (!Succ->isEHPad())
203 MBB.removeSuccessor(Succ);
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000204 eraseDeadBBsAndChildren(Succs);
Heejin Ahn4934f762018-06-25 01:07:11 +0000205 }
206 }
Heejin Ahnb47a18c2019-03-16 04:46:05 +0000207
Heejin Ahn4934f762018-06-25 01:07:11 +0000208 return Changed;
209}
210
Heejin Ahnd6f48782019-01-30 03:21:57 +0000211// Wasm uses 'br_on_exn' instruction to check the tag of an exception. It takes
Heejin Ahn9f96a582019-07-15 22:49:25 +0000212// exnref type object returned by 'catch', and branches to the destination if it
213// matches a given tag. We currently use __cpp_exception symbol to represent the
214// tag for all C++ exceptions.
Heejin Ahnd6f48782019-01-30 03:21:57 +0000215//
216// block $l (result i32)
217// ...
Heejin Ahn9f96a582019-07-15 22:49:25 +0000218// ;; exnref $e is on the stack at this point
Heejin Ahnd6f48782019-01-30 03:21:57 +0000219// br_on_exn $l $e ;; branch to $l with $e's arguments
220// ...
221// end
222// ;; Here we expect the extracted values are on top of the wasm value stack
223// ... Handle exception using values ...
224//
Heejin Ahn9f96a582019-07-15 22:49:25 +0000225// br_on_exn takes an exnref object and branches if it matches the given tag.
226// There can be multiple br_on_exn instructions if we want to match for another
227// tag, but for now we only test for __cpp_exception tag, and if it does not
228// match, i.e., it is a foreign exception, we rethrow it.
Heejin Ahnd6f48782019-01-30 03:21:57 +0000229//
230// In the destination BB that's the target of br_on_exn, extracted exception
231// values (in C++'s case a single i32, which represents an exception pointer)
232// are placed on top of the wasm stack. Because we can't model wasm stack in
233// LLVM instruction, we use 'extract_exception' pseudo instruction to retrieve
234// it. The pseudo instruction will be deleted later.
235bool WebAssemblyLateEHPrepare::addExceptionExtraction(MachineFunction &MF) {
Heejin Ahn4934f762018-06-25 01:07:11 +0000236 const auto &TII = *MF.getSubtarget<WebAssemblySubtarget>().getInstrInfo();
Heejin Ahn831efe02019-08-11 06:24:07 +0000237 MachineRegisterInfo &MRI = MF.getRegInfo();
Heejin Ahn4934f762018-06-25 01:07:11 +0000238 auto *EHInfo = MF.getWasmEHFuncInfo();
Heejin Ahnd6f48782019-01-30 03:21:57 +0000239 SmallVector<MachineInstr *, 16> ExtractInstrs;
Heejin Ahnd2a56ac2019-02-26 04:08:49 +0000240 SmallVector<MachineInstr *, 8> ToDelete;
Heejin Ahnd6f48782019-01-30 03:21:57 +0000241 for (auto &MBB : MF) {
Heejin Ahn4934f762018-06-25 01:07:11 +0000242 for (auto &MI : MBB) {
Heejin Ahnd6f48782019-01-30 03:21:57 +0000243 if (MI.getOpcode() == WebAssembly::EXTRACT_EXCEPTION_I32) {
244 if (MI.getOperand(0).isDead())
Heejin Ahnd2a56ac2019-02-26 04:08:49 +0000245 ToDelete.push_back(&MI);
Heejin Ahnd6f48782019-01-30 03:21:57 +0000246 else
247 ExtractInstrs.push_back(&MI);
248 }
Heejin Ahn4934f762018-06-25 01:07:11 +0000249 }
Heejin Ahnd6f48782019-01-30 03:21:57 +0000250 }
Heejin Ahnd2a56ac2019-02-26 04:08:49 +0000251 bool Changed = !ToDelete.empty() || !ExtractInstrs.empty();
252 for (auto *MI : ToDelete)
253 MI->eraseFromParent();
Heejin Ahnd6f48782019-01-30 03:21:57 +0000254 if (ExtractInstrs.empty())
Heejin Ahnd2a56ac2019-02-26 04:08:49 +0000255 return Changed;
Heejin Ahn4934f762018-06-25 01:07:11 +0000256
Heejin Ahnd6f48782019-01-30 03:21:57 +0000257 // Find terminate pads.
258 SmallSet<MachineBasicBlock *, 8> TerminatePads;
259 for (auto &MBB : MF) {
260 for (auto &MI : MBB) {
Heejin Ahn4934f762018-06-25 01:07:11 +0000261 if (MI.isCall()) {
262 const MachineOperand &CalleeOp = MI.getOperand(0);
263 if (CalleeOp.isGlobal() && CalleeOp.getGlobal()->getName() ==
264 WebAssembly::ClangCallTerminateFn)
Heejin Ahnd6f48782019-01-30 03:21:57 +0000265 TerminatePads.insert(getMatchingEHPad(&MI));
Heejin Ahn4934f762018-06-25 01:07:11 +0000266 }
Heejin Ahnd6f48782019-01-30 03:21:57 +0000267 }
268 }
Heejin Ahn4934f762018-06-25 01:07:11 +0000269
Heejin Ahnd6f48782019-01-30 03:21:57 +0000270 for (auto *Extract : ExtractInstrs) {
271 MachineBasicBlock *EHPad = getMatchingEHPad(Extract);
272 assert(EHPad && "No matching EH pad for extract_exception");
Heejin Ahn8b49b6b2019-03-13 00:37:31 +0000273 auto CatchPos = EHPad->begin();
274 if (CatchPos->isEHLabel()) // EH pad starts with an EH label
275 ++CatchPos;
276 MachineInstr *Catch = &*CatchPos;
277
Heejin Ahnd6f48782019-01-30 03:21:57 +0000278 if (Catch->getNextNode() != Extract)
279 EHPad->insert(Catch->getNextNode(), Extract->removeFromParent());
280
281 // - Before:
282 // ehpad:
Heejin Ahn9f96a582019-07-15 22:49:25 +0000283 // %exnref:exnref = catch
Heejin Ahnd6f48782019-01-30 03:21:57 +0000284 // %exn:i32 = extract_exception
285 // ... use exn ...
286 //
287 // - After:
288 // ehpad:
Heejin Ahn9f96a582019-07-15 22:49:25 +0000289 // %exnref:exnref = catch
Heejin Ahnd6f48782019-01-30 03:21:57 +0000290 // br_on_exn %thenbb, $__cpp_exception, %exnref
291 // br %elsebb
292 // elsebb:
293 // rethrow
294 // thenbb:
295 // %exn:i32 = extract_exception
296 // ... use exn ...
Daniel Sanders05c145d2019-08-12 22:40:45 +0000297 Register ExnReg = Catch->getOperand(0).getReg();
Heejin Ahnd6f48782019-01-30 03:21:57 +0000298 auto *ThenMBB = MF.CreateMachineBasicBlock();
299 auto *ElseMBB = MF.CreateMachineBasicBlock();
300 MF.insert(std::next(MachineFunction::iterator(EHPad)), ElseMBB);
301 MF.insert(std::next(MachineFunction::iterator(ElseMBB)), ThenMBB);
302 ThenMBB->splice(ThenMBB->end(), EHPad, Extract, EHPad->end());
303 ThenMBB->transferSuccessors(EHPad);
304 EHPad->addSuccessor(ThenMBB);
305 EHPad->addSuccessor(ElseMBB);
306
307 DebugLoc DL = Extract->getDebugLoc();
308 const char *CPPExnSymbol = MF.createExternalSymbolName("__cpp_exception");
309 BuildMI(EHPad, DL, TII.get(WebAssembly::BR_ON_EXN))
310 .addMBB(ThenMBB)
Sam Cleggef4c66c2019-04-03 00:17:29 +0000311 .addExternalSymbol(CPPExnSymbol)
Heejin Ahnc7397612019-03-05 11:11:34 +0000312 .addReg(ExnReg);
Heejin Ahnd6f48782019-01-30 03:21:57 +0000313 BuildMI(EHPad, DL, TII.get(WebAssembly::BR)).addMBB(ElseMBB);
314
315 // When this is a terminate pad with __clang_call_terminate() call, we don't
316 // rethrow it anymore and call __clang_call_terminate() with a nullptr
317 // argument, which will call std::terminate().
318 //
319 // - Before:
320 // ehpad:
Heejin Ahn9f96a582019-07-15 22:49:25 +0000321 // %exnref:exnref = catch
Heejin Ahnd6f48782019-01-30 03:21:57 +0000322 // %exn:i32 = extract_exception
323 // call @__clang_call_terminate(%exn)
324 // unreachable
325 //
326 // - After:
327 // ehpad:
Heejin Ahn9f96a582019-07-15 22:49:25 +0000328 // %exnref:exnref = catch
Heejin Ahnd6f48782019-01-30 03:21:57 +0000329 // br_on_exn %thenbb, $__cpp_exception, %exnref
330 // br %elsebb
331 // elsebb:
332 // call @__clang_call_terminate(0)
333 // unreachable
334 // thenbb:
335 // %exn:i32 = extract_exception
336 // call @__clang_call_terminate(%exn)
337 // unreachable
338 if (TerminatePads.count(EHPad)) {
339 Function *ClangCallTerminateFn =
340 MF.getFunction().getParent()->getFunction(
341 WebAssembly::ClangCallTerminateFn);
342 assert(ClangCallTerminateFn &&
343 "There is no __clang_call_terminate() function");
Heejin Ahn64517a62019-08-13 17:35:44 +0000344 Register Reg = MRI.createVirtualRegister(&WebAssembly::I32RegClass);
Heejin Ahn831efe02019-08-11 06:24:07 +0000345 BuildMI(ElseMBB, DL, TII.get(WebAssembly::CONST_I32), Reg).addImm(0);
Heejin Ahnd6f48782019-01-30 03:21:57 +0000346 BuildMI(ElseMBB, DL, TII.get(WebAssembly::CALL_VOID))
347 .addGlobalAddress(ClangCallTerminateFn)
Heejin Ahn831efe02019-08-11 06:24:07 +0000348 .addReg(Reg);
Heejin Ahnd6f48782019-01-30 03:21:57 +0000349 BuildMI(ElseMBB, DL, TII.get(WebAssembly::UNREACHABLE));
350
351 } else {
Heejin Ahn66ce4192019-03-16 05:38:57 +0000352 BuildMI(ElseMBB, DL, TII.get(WebAssembly::RETHROW)).addReg(ExnReg);
Heejin Ahnd6f48782019-01-30 03:21:57 +0000353 if (EHInfo->hasEHPadUnwindDest(EHPad))
Heejin Ahn195a62e2019-03-03 22:35:56 +0000354 ElseMBB->addSuccessor(EHInfo->getEHPadUnwindDest(EHPad));
Heejin Ahnd6f48782019-01-30 03:21:57 +0000355 }
Heejin Ahn4934f762018-06-25 01:07:11 +0000356 }
Heejin Ahn4934f762018-06-25 01:07:11 +0000357
Heejin Ahn4934f762018-06-25 01:07:11 +0000358 return true;
359}
Heejin Ahn0bb98652019-01-30 22:44:45 +0000360
361// After the stack is unwound due to a thrown exception, the __stack_pointer
362// global can point to an invalid address. This inserts instructions that
363// restore __stack_pointer global.
364bool WebAssemblyLateEHPrepare::restoreStackPointer(MachineFunction &MF) {
365 const auto *FrameLowering = static_cast<const WebAssemblyFrameLowering *>(
366 MF.getSubtarget().getFrameLowering());
367 if (!FrameLowering->needsPrologForEH(MF))
368 return false;
369 bool Changed = false;
370
371 for (auto &MBB : MF) {
372 if (!MBB.isEHPad())
373 continue;
374 Changed = true;
375
376 // Insert __stack_pointer restoring instructions at the beginning of each EH
377 // pad, after the catch instruction. Here it is safe to assume that SP32
378 // holds the latest value of __stack_pointer, because the only exception for
379 // this case is when a function uses the red zone, but that only happens
380 // with leaf functions, and we don't restore __stack_pointer in leaf
381 // functions anyway.
382 auto InsertPos = MBB.begin();
Heejin Ahn8b49b6b2019-03-13 00:37:31 +0000383 if (InsertPos->isEHLabel()) // EH pad starts with an EH label
384 ++InsertPos;
385 if (InsertPos->getOpcode() == WebAssembly::CATCH)
386 ++InsertPos;
Heejin Ahn0bb98652019-01-30 22:44:45 +0000387 FrameLowering->writeSPToGlobal(WebAssembly::SP32, MF, MBB, InsertPos,
388 MBB.begin()->getDebugLoc());
389 }
390 return Changed;
391}