diff options
author | Alon Zakai <azakai@google.com> | 2023-10-24 10:37:28 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-10-24 17:37:28 +0000 |
commit | 92c8a4682367170485295da6744b3a59fbb8d3ac (patch) | |
tree | d8e59f512bf37d49feea2d37f17a0877fc01ce11 /test/lit/passes/precompute-gc.wast | |
parent | cce277fc649d7406f446f52778286192367a35ad (diff) | |
download | binaryen-92c8a4682367170485295da6744b3a59fbb8d3ac.tar.gz binaryen-92c8a4682367170485295da6744b3a59fbb8d3ac.tar.bz2 binaryen-92c8a4682367170485295da6744b3a59fbb8d3ac.zip |
[NFC] LocalGraph: Optimize params with no sets (#6046)
If a local index has no sets, then all gets of that index read from the entry block
(a param, or a zero for a local). This is actually a common case, where a param
has no other set, and so it is worth optimizing, which this PR does by avoiding
any flowing operation at all for that index: we just skip and write the entry block
as the source of information for such gets.
#6042 on precompute-propagate goes from 3 minutes to 2 seconds with this (!).
But that testcase is rather special in that it is a huge function with many, many
gets in it, so the overhead we remove is very noticeable there.
Diffstat (limited to 'test/lit/passes/precompute-gc.wast')
-rw-r--r-- | test/lit/passes/precompute-gc.wast | 45 |
1 files changed, 39 insertions, 6 deletions
diff --git a/test/lit/passes/precompute-gc.wast b/test/lit/passes/precompute-gc.wast index df755dade..b2e9593d5 100644 --- a/test/lit/passes/precompute-gc.wast +++ b/test/lit/passes/precompute-gc.wast @@ -229,7 +229,7 @@ (struct.get $struct 0 (local.get $x)) ) ) - ;; CHECK: (func $ref-comparisons (type $8) (param $x (ref null $struct)) (param $y (ref null $struct)) + ;; CHECK: (func $ref-comparisons (type $9) (param $x (ref null $struct)) (param $y (ref null $struct)) ;; CHECK-NEXT: (local $z (ref null $struct)) ;; CHECK-NEXT: (local $w (ref null $struct)) ;; CHECK-NEXT: (call $log @@ -407,7 +407,7 @@ (local.get $tempresult) ) - ;; CHECK: (func $propagate-different-params (type $9) (param $input1 (ref $empty)) (param $input2 (ref $empty)) (result i32) + ;; CHECK: (func $propagate-different-params (type $10) (param $input1 (ref $empty)) (param $input2 (ref $empty)) (result i32) ;; CHECK-NEXT: (local $tempresult i32) ;; CHECK-NEXT: (local.set $tempresult ;; CHECK-NEXT: (ref.eq @@ -723,7 +723,7 @@ ) ) - ;; CHECK: (func $helper (type $10) (param $0 i32) (result i32) + ;; CHECK: (func $helper (type $11) (param $0 i32) (result i32) ;; CHECK-NEXT: (unreachable) ;; CHECK-NEXT: ) (func $helper (param i32) (result i32) @@ -801,14 +801,14 @@ ) ) - ;; CHECK: (func $receive-f64 (type $11) (param $0 f64) + ;; CHECK: (func $receive-f64 (type $12) (param $0 f64) ;; CHECK-NEXT: (unreachable) ;; CHECK-NEXT: ) (func $receive-f64 (param f64) (unreachable) ) - ;; CHECK: (func $odd-cast-and-get-non-null (type $12) (param $temp (ref $func-return-i32)) + ;; CHECK: (func $odd-cast-and-get-non-null (type $13) (param $temp (ref $func-return-i32)) ;; CHECK-NEXT: (local.set $temp ;; CHECK-NEXT: (ref.cast (ref nofunc) ;; CHECK-NEXT: (ref.func $receive-f64) @@ -836,7 +836,7 @@ ) ) - ;; CHECK: (func $new_block_unreachable (type $13) (result anyref) + ;; CHECK: (func $new_block_unreachable (type $8) (result anyref) ;; CHECK-NEXT: (block ;; (replaces something unreachable we can't emit) ;; CHECK-NEXT: (drop ;; CHECK-NEXT: (block @@ -1033,4 +1033,37 @@ ) ) ) + + ;; CHECK: (func $get-nonnullable-in-unreachable (type $8) (result anyref) + ;; CHECK-NEXT: (local $x (ref any)) + ;; CHECK-NEXT: (local.tee $x + ;; CHECK-NEXT: (unreachable) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (if + ;; CHECK-NEXT: (i32.const 1) + ;; CHECK-NEXT: (unreachable) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (local.get $x) + ;; CHECK-NEXT: ) + (func $get-nonnullable-in-unreachable (result anyref) + (local $x (ref any)) + ;; We cannot read a non-nullable local without setting it first, but it is ok + ;; to do so here because we are in unreachable code. We should also not error + ;; about this get seeming to read the default value from the function entry + ;; (because it does not, as the entry is not reachable from it). Nothing is + ;; expected to be optimized here. + + ;; This unreachable set is needed for the later get to validate. + (local.set $x + (unreachable) + ) + ;; This if is needed so we have an interesting enough CFG that a possible + ;; assertion can be hit about reading the default value from the entry in a + ;; later block. + (if + (i32.const 1) + (unreachable) + ) + (local.get $x) + ) ) |