diff options
author | Alon Zakai <azakai@google.com> | 2024-03-18 14:18:09 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2024-03-18 14:18:09 -0700 |
commit | d8086c63a9e3e6bbd1bdc5d7e0843af8433cc4c8 (patch) | |
tree | b46989cd5f1e397b6eda83ca7741a1a5d2716728 /test/lit/passes/dae_tnh.wast | |
parent | c166ca015860b337e9ce07a5e02cb707964056ba (diff) | |
download | binaryen-d8086c63a9e3e6bbd1bdc5d7e0843af8433cc4c8.tar.gz binaryen-d8086c63a9e3e6bbd1bdc5d7e0843af8433cc4c8.tar.bz2 binaryen-d8086c63a9e3e6bbd1bdc5d7e0843af8433cc4c8.zip |
DeadArgumentElimination/SignaturePruning: Prune params even if called with effects (#6395)
Before this PR, when we saw a param was unused we sometimes could not remove it.
For example, if there was one call like this:
(call $target
(call $other)
)
That nested call has effects, so we can't just remove it from the outer call - we'd need to
move it first. That motion was hard to integrate which was why it was left out, but it
turns out that is sometimes very important. E.g. in Java it is common to have such calls
that send the this parameter as the result of another call; not being able to remove such
params meant we kept those nested calls alive, creating empty structs just to have
something to send there.
To fix this, this builds on top of #6394 which makes it easier to move all children out of
a parent, leaving only nested things that can be easily moved around and removed. In
more detail, DeadArgumentElimination/SignaturePruning track whether we run into effects that
prevent removing a field. If we do, then we queue an operation to move the children
out, which we do using a new utility ParamUtils::localizeCallsTo. The pass then does
another iteration after that operation.
Alternatively we could try to move things around immediately, but that is quite hard:
those passes already track a lot of state. It is simpler to do the fixup in an entirely
separate utility. That does come at the cost of the utility doing another pass on the
module and the pass itself running another iteration, but this situation is not the most
common.
Diffstat (limited to 'test/lit/passes/dae_tnh.wast')
-rw-r--r-- | test/lit/passes/dae_tnh.wast | 52 |
1 files changed, 41 insertions, 11 deletions
diff --git a/test/lit/passes/dae_tnh.wast b/test/lit/passes/dae_tnh.wast index 33e6eb09b..6d75aed30 100644 --- a/test/lit/passes/dae_tnh.wast +++ b/test/lit/passes/dae_tnh.wast @@ -39,14 +39,18 @@ ;; CHECK: (type $1 (func (param i32))) ;; CHECK: (func $caller (type $0) - ;; CHECK-NEXT: (call $target - ;; CHECK-NEXT: (unreachable) - ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (unreachable) ;; CHECK-NEXT: ) (func $caller - ;; Removing this parameter would require the type of the call to change from - ;; unreachable to none. We don't handle such complexity and ignore such - ;; cases. + ;; Removing this parameter would make the type of the call change from + ;; unreachable to none. But the call itself is in unreachable code, so we + ;; will replace it with an unreachable (and then, once the call is gone, the + ;; target can be better optimized; however, no other calls remain here, so + ;; the pass does nothing as it considers it dead at that point). + ;; + ;; This test verifies we do the proper thing even in TNH mode, as in TNH + ;; mode |unreachable| seems to have no effects, but for validation reasons + ;; we must still replace the call here. (call $target (unreachable) ) @@ -59,13 +63,40 @@ ) ) -;; As above, but use a return_call. We can optimize that, since return_calls -;; have type unreachable anyhow, and the optimization would not change the type. +;; As above but the called target has a result. +(module + ;; CHECK: (type $0 (func (result i32))) + + ;; CHECK: (type $1 (func (param i32) (result i32))) + + ;; CHECK: (func $caller (type $0) (result i32) + ;; CHECK-NEXT: (unreachable) + ;; CHECK-NEXT: ) + (func $caller (result i32) + ;; Again, the call is replaced by an unreachable. + (call $target + (unreachable) + ) + ) + + ;; CHECK: (func $target (type $1) (param $0 i32) (result i32) + ;; CHECK-NEXT: (i32.const 42) + ;; CHECK-NEXT: ) + (func $target (param i32) (result i32) + (i32.const 42) + ) +) + +;; As above, but use a return_call. We can optimize that too (return_calls have +;; type unreachable anyhow, and the optimization would not change the type, so +;; it is even simpler). (module ;; CHECK: (type $0 (func)) + ;; CHECK: (type $1 (func (param i32))) + ;; CHECK: (func $caller (type $0) - ;; CHECK-NEXT: (return_call $target) + ;; CHECK-NEXT: (unreachable) ;; CHECK-NEXT: ) (func $caller (return_call $target @@ -73,8 +104,7 @@ ) ) - ;; CHECK: (func $target (type $0) - ;; CHECK-NEXT: (local $0 i32) + ;; CHECK: (func $target (type $1) (param $0 i32) ;; CHECK-NEXT: (nop) ;; CHECK-NEXT: ) (func $target (param i32) |