1 //=== WebAssemblyLateEHPrepare.cpp - WebAssembly Exception Preparation -===//
3 // 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
7 //===----------------------------------------------------------------------===//
10 /// \brief Does various transformations for exception handling.
12 //===----------------------------------------------------------------------===//
14 #include "MCTargetDesc/WebAssemblyMCTargetDesc.h"
15 #include "Utils/WebAssemblyUtilities.h"
16 #include "WebAssembly.h"
17 #include "WebAssemblySubtarget.h"
18 #include "llvm/ADT/SmallPtrSet.h"
19 #include "llvm/CodeGen/MachineInstrBuilder.h"
20 #include "llvm/CodeGen/WasmEHFuncInfo.h"
21 #include "llvm/MC/MCAsmInfo.h"
22 #include "llvm/Support/Debug.h"
23 #include "llvm/Target/TargetMachine.h"
26 #define DEBUG_TYPE "wasm-late-eh-prepare"
29 class WebAssemblyLateEHPrepare final
: public MachineFunctionPass
{
30 StringRef
getPassName() const override
{
31 return "WebAssembly Late Prepare Exception";
34 bool runOnMachineFunction(MachineFunction
&MF
) override
;
35 bool removeUnreachableEHPads(MachineFunction
&MF
);
36 void recordCatchRetBBs(MachineFunction
&MF
);
37 bool hoistCatches(MachineFunction
&MF
);
38 bool addCatchAlls(MachineFunction
&MF
);
39 bool replaceFuncletReturns(MachineFunction
&MF
);
40 bool removeUnnecessaryUnreachables(MachineFunction
&MF
);
41 bool restoreStackPointer(MachineFunction
&MF
);
43 MachineBasicBlock
*getMatchingEHPad(MachineInstr
*MI
);
44 SmallPtrSet
<MachineBasicBlock
*, 8> CatchRetBBs
;
47 static char ID
; // Pass identification, replacement for typeid
48 WebAssemblyLateEHPrepare() : MachineFunctionPass(ID
) {}
50 } // end anonymous namespace
52 char WebAssemblyLateEHPrepare::ID
= 0;
53 INITIALIZE_PASS(WebAssemblyLateEHPrepare
, DEBUG_TYPE
,
54 "WebAssembly Late Exception Preparation", false, false)
56 FunctionPass
*llvm::createWebAssemblyLateEHPrepare() {
57 return new WebAssemblyLateEHPrepare();
60 // Returns the nearest EH pad that dominates this instruction. This does not use
61 // dominator analysis; it just does BFS on its predecessors until arriving at an
62 // EH pad. This assumes valid EH scopes so the first EH pad it arrives in all
63 // possible search paths should be the same.
64 // Returns nullptr in case it does not find any EH pad in the search, or finds
65 // multiple different EH pads.
67 WebAssemblyLateEHPrepare::getMatchingEHPad(MachineInstr
*MI
) {
68 MachineFunction
*MF
= MI
->getParent()->getParent();
69 SmallVector
<MachineBasicBlock
*, 2> WL
;
70 SmallPtrSet
<MachineBasicBlock
*, 2> Visited
;
71 WL
.push_back(MI
->getParent());
72 MachineBasicBlock
*EHPad
= nullptr;
74 MachineBasicBlock
*MBB
= WL
.pop_back_val();
75 if (Visited
.count(MBB
))
79 if (EHPad
&& EHPad
!= MBB
)
84 if (MBB
== &MF
->front())
86 for (auto *Pred
: MBB
->predecessors())
87 if (!CatchRetBBs
.count(Pred
)) // We don't go into child scopes
93 // Erase the specified BBs if the BB does not have any remaining predecessors,
94 // and also all its dead children.
95 template <typename Container
>
96 static void eraseDeadBBsAndChildren(const Container
&MBBs
) {
97 SmallVector
<MachineBasicBlock
*, 8> WL(MBBs
.begin(), MBBs
.end());
98 SmallPtrSet
<MachineBasicBlock
*, 8> Deleted
;
100 MachineBasicBlock
*MBB
= WL
.pop_back_val();
101 if (Deleted
.count(MBB
) || !MBB
->pred_empty())
103 SmallVector
<MachineBasicBlock
*, 4> Succs(MBB
->successors());
104 WL
.append(MBB
->succ_begin(), MBB
->succ_end());
105 for (auto *Succ
: Succs
)
106 MBB
->removeSuccessor(Succ
);
107 // To prevent deleting the same BB multiple times, which can happen when
108 // 'MBBs' contain both a parent and a child
110 MBB
->eraseFromParent();
114 bool WebAssemblyLateEHPrepare::runOnMachineFunction(MachineFunction
&MF
) {
115 LLVM_DEBUG(dbgs() << "********** Late EH Prepare **********\n"
116 "********** Function: "
117 << MF
.getName() << '\n');
119 if (MF
.getTarget().getMCAsmInfo()->getExceptionHandlingType() !=
120 ExceptionHandling::Wasm
)
123 bool Changed
= false;
124 if (MF
.getFunction().hasPersonalityFn()) {
125 Changed
|= removeUnreachableEHPads(MF
);
126 recordCatchRetBBs(MF
);
127 Changed
|= hoistCatches(MF
);
128 Changed
|= addCatchAlls(MF
);
129 Changed
|= replaceFuncletReturns(MF
);
131 Changed
|= removeUnnecessaryUnreachables(MF
);
132 if (MF
.getFunction().hasPersonalityFn())
133 Changed
|= restoreStackPointer(MF
);
137 // Remove unreachable EH pads and its children. If they remain, CFG
138 // stackification can be tricky.
139 bool WebAssemblyLateEHPrepare::removeUnreachableEHPads(MachineFunction
&MF
) {
140 SmallVector
<MachineBasicBlock
*, 4> ToDelete
;
142 if (MBB
.isEHPad() && MBB
.pred_empty())
143 ToDelete
.push_back(&MBB
);
144 eraseDeadBBsAndChildren(ToDelete
);
145 return !ToDelete
.empty();
148 // Record which BB ends with catchret instruction, because this will be replaced
149 // with 'br's later. This set of catchret BBs is necessary in 'getMatchingEHPad'
151 void WebAssemblyLateEHPrepare::recordCatchRetBBs(MachineFunction
&MF
) {
153 for (auto &MBB
: MF
) {
154 auto Pos
= MBB
.getFirstTerminator();
155 if (Pos
== MBB
.end())
157 MachineInstr
*TI
= &*Pos
;
158 if (TI
->getOpcode() == WebAssembly::CATCHRET
)
159 CatchRetBBs
.insert(&MBB
);
163 // Hoist catch instructions to the beginning of their matching EH pad BBs in
165 // (1) catch instruction is not the first instruction in EH pad.
167 // some_other_instruction
170 // (2) catch instruction is in a non-EH pad BB. For example,
175 bool WebAssemblyLateEHPrepare::hoistCatches(MachineFunction
&MF
) {
176 bool Changed
= false;
177 SmallVector
<MachineInstr
*, 16> Catches
;
180 if (WebAssembly::isCatch(MI
.getOpcode()))
181 Catches
.push_back(&MI
);
183 for (auto *Catch
: Catches
) {
184 MachineBasicBlock
*EHPad
= getMatchingEHPad(Catch
);
185 assert(EHPad
&& "No matching EH pad for catch");
186 auto InsertPos
= EHPad
->begin();
187 // Skip EH_LABELs in the beginning of an EH pad if present. We don't use
188 // these labels at the moment, but other targets also seem to have an
189 // EH_LABEL instruction in the beginning of an EH pad.
190 while (InsertPos
!= EHPad
->end() && InsertPos
->isEHLabel())
192 if (InsertPos
== Catch
)
195 EHPad
->insert(InsertPos
, Catch
->removeFromParent());
200 // Add catch_all to beginning of cleanup pads.
201 bool WebAssemblyLateEHPrepare::addCatchAlls(MachineFunction
&MF
) {
202 bool Changed
= false;
203 const auto &TII
= *MF
.getSubtarget
<WebAssemblySubtarget
>().getInstrInfo();
205 for (auto &MBB
: MF
) {
208 auto InsertPos
= MBB
.begin();
209 // Skip EH_LABELs in the beginning of an EH pad if present.
210 while (InsertPos
!= MBB
.end() && InsertPos
->isEHLabel())
212 // This runs after hoistCatches(), so we assume that if there is a catch,
213 // that should be the first non-EH-label instruction in an EH pad.
214 if (InsertPos
== MBB
.end() ||
215 !WebAssembly::isCatch(InsertPos
->getOpcode())) {
217 BuildMI(MBB
, InsertPos
,
218 InsertPos
== MBB
.end() ? DebugLoc() : InsertPos
->getDebugLoc(),
219 TII
.get(WebAssembly::CATCH_ALL
));
225 // Replace pseudo-instructions catchret and cleanupret with br and rethrow
227 bool WebAssemblyLateEHPrepare::replaceFuncletReturns(MachineFunction
&MF
) {
228 bool Changed
= false;
229 const auto &TII
= *MF
.getSubtarget
<WebAssemblySubtarget
>().getInstrInfo();
231 for (auto &MBB
: MF
) {
232 auto Pos
= MBB
.getFirstTerminator();
233 if (Pos
== MBB
.end())
235 MachineInstr
*TI
= &*Pos
;
237 switch (TI
->getOpcode()) {
238 case WebAssembly::CATCHRET
: {
239 // Replace a catchret with a branch
240 MachineBasicBlock
*TBB
= TI
->getOperand(0).getMBB();
241 if (!MBB
.isLayoutSuccessor(TBB
))
242 BuildMI(MBB
, TI
, TI
->getDebugLoc(), TII
.get(WebAssembly::BR
))
244 TI
->eraseFromParent();
248 case WebAssembly::CLEANUPRET
: {
249 // Replace a cleanupret with a rethrow. For C++ support, currently
250 // rethrow's immediate argument is always 0 (= the latest exception).
251 BuildMI(MBB
, TI
, TI
->getDebugLoc(), TII
.get(WebAssembly::RETHROW
))
253 TI
->eraseFromParent();
262 // Remove unnecessary unreachables after a throw or rethrow.
263 bool WebAssemblyLateEHPrepare::removeUnnecessaryUnreachables(
264 MachineFunction
&MF
) {
265 bool Changed
= false;
266 for (auto &MBB
: MF
) {
267 for (auto &MI
: MBB
) {
268 if (MI
.getOpcode() != WebAssembly::THROW
&&
269 MI
.getOpcode() != WebAssembly::RETHROW
)
273 // The instruction after the throw should be an unreachable or a branch to
274 // another BB that should eventually lead to an unreachable. Delete it
275 // because throw itself is a terminator, and also delete successors if
277 MBB
.erase(std::next(MI
.getIterator()), MBB
.end());
278 SmallVector
<MachineBasicBlock
*, 8> Succs(MBB
.successors());
279 for (auto *Succ
: Succs
)
280 if (!Succ
->isEHPad())
281 MBB
.removeSuccessor(Succ
);
282 eraseDeadBBsAndChildren(Succs
);
289 // After the stack is unwound due to a thrown exception, the __stack_pointer
290 // global can point to an invalid address. This inserts instructions that
291 // restore __stack_pointer global.
292 bool WebAssemblyLateEHPrepare::restoreStackPointer(MachineFunction
&MF
) {
293 const auto *FrameLowering
= static_cast<const WebAssemblyFrameLowering
*>(
294 MF
.getSubtarget().getFrameLowering());
295 if (!FrameLowering
->needsPrologForEH(MF
))
297 bool Changed
= false;
299 for (auto &MBB
: MF
) {
304 // Insert __stack_pointer restoring instructions at the beginning of each EH
305 // pad, after the catch instruction. Here it is safe to assume that SP32
306 // holds the latest value of __stack_pointer, because the only exception for
307 // this case is when a function uses the red zone, but that only happens
308 // with leaf functions, and we don't restore __stack_pointer in leaf
310 auto InsertPos
= MBB
.begin();
311 // Skip EH_LABELs in the beginning of an EH pad if present.
312 while (InsertPos
!= MBB
.end() && InsertPos
->isEHLabel())
314 assert(InsertPos
!= MBB
.end() &&
315 WebAssembly::isCatch(InsertPos
->getOpcode()) &&
316 "catch/catch_all should be present in every EH pad at this point");
317 ++InsertPos
; // Skip the catch instruction
318 FrameLowering
->writeSPToGlobal(FrameLowering
->getSPReg(MF
), MF
, MBB
,
319 InsertPos
, MBB
.begin()->getDebugLoc());