summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/passes/Inlining.cpp10
-rw-r--r--test/lit/passes/inlining_enable-tail-call.wast62
2 files changed, 72 insertions, 0 deletions
diff --git a/src/passes/Inlining.cpp b/src/passes/Inlining.cpp
index 36610a224..21c06e528 100644
--- a/src/passes/Inlining.cpp
+++ b/src/passes/Inlining.cpp
@@ -247,6 +247,7 @@ struct Updater : public PostWalker<Updater> {
Module* module;
std::map<Index, Index> localMapping;
Name returnName;
+ bool isReturn;
Builder* builder;
void visitReturn(Return* curr) {
replaceCurrent(builder->makeBreak(returnName, curr->value));
@@ -257,6 +258,14 @@ struct Updater : public PostWalker<Updater> {
// not cause unbounded stack growth because inlining and return calling both
// avoid creating a new stack frame.
template<typename T> void handleReturnCall(T* curr, HeapType targetType) {
+ if (isReturn) {
+ // If the inlined callsite was already a return_call, then we can keep
+ // return_calls in the inlined function rather than downgrading them.
+ // That is, if A->B and B->C and both those calls are return_calls
+ // then after inlining A->B we want to now have A->C be a
+ // return_call.
+ return;
+ }
curr->isReturn = false;
curr->type = targetType.getSignature().results;
if (curr->type.isConcrete()) {
@@ -329,6 +338,7 @@ doInlining(Module* module, Function* into, const InliningAction& action) {
Updater updater;
updater.module = module;
updater.returnName = block->name;
+ updater.isReturn = call->isReturn;
updater.builder = &builder;
// Set up a locals mapping
for (Index i = 0; i < from->getNumLocals(); i++) {
diff --git a/test/lit/passes/inlining_enable-tail-call.wast b/test/lit/passes/inlining_enable-tail-call.wast
index 646d2cf0d..0c460b552 100644
--- a/test/lit/passes/inlining_enable-tail-call.wast
+++ b/test/lit/passes/inlining_enable-tail-call.wast
@@ -705,3 +705,65 @@
(unreachable)
)
)
+
+(module
+ ;; CHECK: (type $i32_=>_i32 (func (param i32) (result i32)))
+
+ ;; CHECK: (export "is_even" (func $is_even))
+ (export "is_even" (func $is_even))
+ ;; CHECK: (func $is_even (param $i i32) (result i32)
+ ;; CHECK-NEXT: (local $1 i32)
+ ;; CHECK-NEXT: (if (result i32)
+ ;; CHECK-NEXT: (i32.eqz
+ ;; CHECK-NEXT: (local.get $i)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (i32.const 1)
+ ;; CHECK-NEXT: (return
+ ;; CHECK-NEXT: (block $__inlined_func$is_odd (result i32)
+ ;; CHECK-NEXT: (local.set $1
+ ;; CHECK-NEXT: (i32.sub
+ ;; CHECK-NEXT: (local.get $i)
+ ;; CHECK-NEXT: (i32.const 1)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (if (result i32)
+ ;; CHECK-NEXT: (i32.eqz
+ ;; CHECK-NEXT: (local.get $1)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (i32.const 0)
+ ;; CHECK-NEXT: (return_call $is_even
+ ;; CHECK-NEXT: (i32.sub
+ ;; CHECK-NEXT: (local.get $1)
+ ;; CHECK-NEXT: (i32.const 1)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ (func $is_even (param $i i32) (result i32)
+ (if (result i32)
+ (i32.eqz (local.get $i))
+ (i32.const 1)
+ (return_call $is_odd
+ (i32.sub
+ (local.get $i)
+ (i32.const 1)
+ )
+ )
+ )
+ )
+ (func $is_odd (param $i i32) (result i32)
+ (if (result i32)
+ (i32.eqz (local.get $i))
+ (i32.const 0)
+ (return_call $is_even
+ (i32.sub
+ (local.get $i)
+ (i32.const 1)
+ )
+ )
+ )
+ )
+)