summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--src/passes/CodePushing.cpp4
-rw-r--r--test/lit/passes/code-pushing-gc.wast80
2 files changed, 82 insertions, 2 deletions
diff --git a/src/passes/CodePushing.cpp b/src/passes/CodePushing.cpp
index c71078558..1b253c6ea 100644
--- a/src/passes/CodePushing.cpp
+++ b/src/passes/CodePushing.cpp
@@ -134,14 +134,14 @@ private:
return nullptr;
}
- // Push past conditional control flow.
+ // Try to push past conditional control flow.
// TODO: push into ifs as well
bool isPushPoint(Expression* curr) {
// look through drops
if (auto* drop = curr->dynCast<Drop>()) {
curr = drop->value;
}
- if (curr->is<If>()) {
+ if (curr->is<If>() || curr->is<BrOn>()) {
return true;
}
if (auto* br = curr->dynCast<Break>()) {
diff --git a/test/lit/passes/code-pushing-gc.wast b/test/lit/passes/code-pushing-gc.wast
new file mode 100644
index 000000000..9af0fc019
--- /dev/null
+++ b/test/lit/passes/code-pushing-gc.wast
@@ -0,0 +1,80 @@
+;; NOTE: Assertions have been generated by update_lit_checks.py and should not be edited.
+;; RUN: wasm-opt %s --code-pushing -all -S -o - | filecheck %s
+
+(module
+ ;; CHECK: (func $br_on
+ ;; CHECK-NEXT: (local $x funcref)
+ ;; CHECK-NEXT: (drop
+ ;; CHECK-NEXT: (block $out (result (ref func))
+ ;; CHECK-NEXT: (drop
+ ;; CHECK-NEXT: (br_on_func $out
+ ;; CHECK-NEXT: (ref.func $br_on)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (local.set $x
+ ;; CHECK-NEXT: (ref.func $br_on)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (drop
+ ;; CHECK-NEXT: (local.get $x)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (ref.func $br_on)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ (func $br_on
+ (local $x (ref null func))
+ (drop
+ (block $out (result (ref func))
+ ;; We can push the local.set past the br_on.
+ (local.set $x (ref.func $br_on))
+ (drop
+ (br_on_func $out
+ (ref.func $br_on)
+ )
+ )
+ (drop
+ (local.get $x)
+ )
+ (ref.func $br_on)
+ )
+ )
+ )
+
+ ;; CHECK: (func $br_on_no
+ ;; CHECK-NEXT: (local $x funcref)
+ ;; CHECK-NEXT: (drop
+ ;; CHECK-NEXT: (block $out (result (ref func))
+ ;; CHECK-NEXT: (local.set $x
+ ;; CHECK-NEXT: (ref.func $br_on_no)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (drop
+ ;; CHECK-NEXT: (br_on_func $out
+ ;; CHECK-NEXT: (ref.func $br_on_no)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (ref.func $br_on_no)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: (drop
+ ;; CHECK-NEXT: (local.get $x)
+ ;; CHECK-NEXT: )
+ ;; CHECK-NEXT: )
+ (func $br_on_no
+ (local $x (ref null func))
+ ;; We can't push here since the local.get is outside of the loop.
+ (drop
+ (block $out (result (ref func))
+ (local.set $x (ref.func $br_on_no))
+ (drop
+ (br_on_func $out
+ (ref.func $br_on_no)
+ )
+ )
+ (ref.func $br_on_no)
+ )
+ )
+ (drop
+ (local.get $x)
+ )
+ )
+)