Skip to content

Commit 0bb9865

Browse files
committedJan 30, 2019
[WebAssembly] Restore stack pointer right after catch instruction
Summary: After the staack is unwound due to a thrown exxception, `__stack_pointer` global can point to an invalid address. So a `global.set` to restore `__stack_pointer` should be inserted right after `catch` instruction. But after r352598 the `global.set` instruction is inserted not right after `catch` but after `block` - `br-on-exn` - `end_block` - `extract_exception` sequence. This CL fixes it. While doing that, we can actually move ReplacePhysRegs pass after LateEHPrepare and merge EHRestoreStackPointer pass into LateEHPrepare, and now placing `global.set` to `__stack_pointer` right after `catch` is much easier. Otherwise it is hard to guarantee that `global.set` is still right after `catch` and not touched with other transformations, in which case we have to do something to hoist it. Reviewers: dschuff Subscribers: mgorny, sbc100, jgravelle-google, sunfish, llvm-commits Differential Revision: https://reviews.llvm.org/D57421 llvm-svn: 352681
1 parent 9ab2310 commit 0bb9865

File tree

6 files changed

+38
-99
lines changed

6 files changed

+38
-99
lines changed
 

‎llvm/lib/Target/WebAssembly/CMakeLists.txt

-1
Original file line numberDiff line numberDiff line change
@@ -21,7 +21,6 @@ add_llvm_target(WebAssemblyCodeGen
2121
WebAssemblyCFGSort.cpp
2222
WebAssemblyDebugValueManager.cpp
2323
WebAssemblyLateEHPrepare.cpp
24-
WebAssemblyEHRestoreStackPointer.cpp
2524
WebAssemblyExceptionInfo.cpp
2625
WebAssemblyExplicitLocals.cpp
2726
WebAssemblyFastISel.cpp

‎llvm/lib/Target/WebAssembly/WebAssembly.h

-2
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,6 @@ FunctionPass *createWebAssemblyArgumentMove();
3838
FunctionPass *createWebAssemblySetP2AlignOperands();
3939

4040
// Late passes.
41-
FunctionPass *createWebAssemblyEHRestoreStackPointer();
4241
FunctionPass *createWebAssemblyReplacePhysRegs();
4342
FunctionPass *createWebAssemblyPrepareForLiveIntervals();
4443
FunctionPass *createWebAssemblyOptimizeLiveIntervals();
@@ -63,7 +62,6 @@ void initializeFixFunctionBitcastsPass(PassRegistry &);
6362
void initializeOptimizeReturnedPass(PassRegistry &);
6463
void initializeWebAssemblyArgumentMovePass(PassRegistry &);
6564
void initializeWebAssemblySetP2AlignOperandsPass(PassRegistry &);
66-
void initializeWebAssemblyEHRestoreStackPointerPass(PassRegistry &);
6765
void initializeWebAssemblyReplacePhysRegsPass(PassRegistry &);
6866
void initializeWebAssemblyPrepareForLiveIntervalsPass(PassRegistry &);
6967
void initializeWebAssemblyOptimizeLiveIntervalsPass(PassRegistry &);

‎llvm/lib/Target/WebAssembly/WebAssemblyEHRestoreStackPointer.cpp

-86
This file was deleted.

‎llvm/lib/Target/WebAssembly/WebAssemblyLateEHPrepare.cpp

+32
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,7 @@ class WebAssemblyLateEHPrepare final : public MachineFunctionPass {
3434
bool replaceFuncletReturns(MachineFunction &MF);
3535
bool addCatches(MachineFunction &MF);
3636
bool addExceptionExtraction(MachineFunction &MF);
37+
bool restoreStackPointer(MachineFunction &MF);
3738

3839
public:
3940
static char ID; // Pass identification, replacement for typeid
@@ -113,6 +114,7 @@ bool WebAssemblyLateEHPrepare::runOnMachineFunction(MachineFunction &MF) {
113114
Changed |= replaceFuncletReturns(MF);
114115
Changed |= addCatches(MF);
115116
Changed |= addExceptionExtraction(MF);
117+
Changed |= restoreStackPointer(MF);
116118
return Changed;
117119
}
118120

@@ -330,3 +332,33 @@ bool WebAssemblyLateEHPrepare::addExceptionExtraction(MachineFunction &MF) {
330332

331333
return true;
332334
}
335+
336+
// After the stack is unwound due to a thrown exception, the __stack_pointer
337+
// global can point to an invalid address. This inserts instructions that
338+
// restore __stack_pointer global.
339+
bool WebAssemblyLateEHPrepare::restoreStackPointer(MachineFunction &MF) {
340+
const auto *FrameLowering = static_cast<const WebAssemblyFrameLowering *>(
341+
MF.getSubtarget().getFrameLowering());
342+
if (!FrameLowering->needsPrologForEH(MF))
343+
return false;
344+
bool Changed = false;
345+
346+
for (auto &MBB : MF) {
347+
if (!MBB.isEHPad())
348+
continue;
349+
Changed = true;
350+
351+
// Insert __stack_pointer restoring instructions at the beginning of each EH
352+
// pad, after the catch instruction. Here it is safe to assume that SP32
353+
// holds the latest value of __stack_pointer, because the only exception for
354+
// this case is when a function uses the red zone, but that only happens
355+
// with leaf functions, and we don't restore __stack_pointer in leaf
356+
// functions anyway.
357+
auto InsertPos = MBB.begin();
358+
if (MBB.begin()->getOpcode() == WebAssembly::CATCH)
359+
InsertPos++;
360+
FrameLowering->writeSPToGlobal(WebAssembly::SP32, MF, MBB, InsertPos,
361+
MBB.begin()->getDebugLoc());
362+
}
363+
return Changed;
364+
}

‎llvm/lib/Target/WebAssembly/WebAssemblyTargetMachine.cpp

+5-9
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,6 @@ extern "C" void LLVMInitializeWebAssemblyTarget() {
5757
initializeOptimizeReturnedPass(PR);
5858
initializeWebAssemblyArgumentMovePass(PR);
5959
initializeWebAssemblySetP2AlignOperandsPass(PR);
60-
initializeWebAssemblyEHRestoreStackPointerPass(PR);
6160
initializeWebAssemblyReplacePhysRegsPass(PR);
6261
initializeWebAssemblyPrepareForLiveIntervalsPass(PR);
6362
initializeWebAssemblyOptimizeLiveIntervalsPass(PR);
@@ -284,14 +283,6 @@ void WebAssemblyPassConfig::addPostRegAlloc() {
284283
void WebAssemblyPassConfig::addPreEmitPass() {
285284
TargetPassConfig::addPreEmitPass();
286285

287-
// Restore __stack_pointer global after an exception is thrown.
288-
addPass(createWebAssemblyEHRestoreStackPointer());
289-
290-
// Now that we have a prologue and epilogue and all frame indices are
291-
// rewritten, eliminate SP and FP. This allows them to be stackified,
292-
// colored, and numbered with the rest of the registers.
293-
addPass(createWebAssemblyReplacePhysRegs());
294-
295286
// Rewrite pseudo call_indirect instructions as real instructions.
296287
// This needs to run before register stackification, because we change the
297288
// order of the arguments.
@@ -304,6 +295,11 @@ void WebAssemblyPassConfig::addPreEmitPass() {
304295
// Every CFG-changing optimizations should come before this.
305296
addPass(createWebAssemblyLateEHPrepare());
306297

298+
// Now that we have a prologue and epilogue and all frame indices are
299+
// rewritten, eliminate SP and FP. This allows them to be stackified,
300+
// colored, and numbered with the rest of the registers.
301+
addPass(createWebAssemblyReplacePhysRegs());
302+
307303
// Preparations and optimizations related to register stackification.
308304
if (getOptLevel() != CodeGenOpt::None) {
309305
// LiveIntervals isn't commonly run this late. Re-establish preconditions.

‎llvm/test/CodeGen/WebAssembly/exception.ll

+1-1
Original file line numberDiff line numberDiff line change
@@ -30,12 +30,12 @@ define void @test_rethrow(i8* %p) {
3030
; CHECK: try
3131
; CHECK: call foo@FUNCTION
3232
; CHECK: catch $[[EXCEPT_REF:[0-9]+]]=
33+
; CHECK: global.set __stack_pointer@GLOBAL
3334
; CHECK: block i32
3435
; CHECK: br_on_exn 0, __cpp_exception@EVENT, $[[EXCEPT_REF]]
3536
; CHECK: rethrow
3637
; CHECK: end_block
3738
; CHECK: extract_exception $[[EXN:[0-9]+]]=
38-
; CHECK: global.set __stack_pointer@GLOBAL
3939
; CHECK-DAG: i32.store __wasm_lpad_context
4040
; CHECK-DAG: i32.store __wasm_lpad_context+4
4141
; CHECK: i32.call $drop=, _Unwind_CallPersonality@FUNCTION, $[[EXN]]

0 commit comments

Comments
 (0)
Please sign in to comment.