diff options
author | Alon Zakai <azakai@google.com> | 2023-04-10 15:36:11 -0700 |
---|---|---|
committer | GitHub <noreply@github.com> | 2023-04-10 15:36:11 -0700 |
commit | de8c2b5bbfb3410a7d526a5b0c3eb40f9db75a71 (patch) | |
tree | 64d03eb10cf1f921f04e04683bd234c2fc53e072 /test/lit/passes/gufa-refs.wast | |
parent | 35a237db1b96b37cfa7638d2f8d22aa46a626683 (diff) | |
download | binaryen-de8c2b5bbfb3410a7d526a5b0c3eb40f9db75a71.tar.gz binaryen-de8c2b5bbfb3410a7d526a5b0c3eb40f9db75a71.tar.bz2 binaryen-de8c2b5bbfb3410a7d526a5b0c3eb40f9db75a71.zip |
[GUFA] Fix packed field filtering (#5652)
Technically we need to filter both before and after combining, that is,
if a location's contents will be filtered by F() then if the new contents
are x and old contents y then we need to end up with
F(F(x) U F(y)). That is, filtering before is necessary to ensure the union
of content does not end up unnecessarily large, and the filtering
after is necessary to ensure the final result is properly filtered to fit.
(If our representation were perfect then this would not be needed, but
it is not, as the union of two exact types can end up as a very large
cone, for example.)
For efficiency we have been filtering afterwards. But that is not enough
for packed fields, it turns out, where we must filter before. If we don't,
then if inputs 0 and 0x100 arrive to an i8 field then combining them
we get "unknown integer" (which is then filtered by 0xff, but it's too
late). By filtering before, the actual values are both 0 and we end up
with that as the only possible value.
It turns out that filtering before is enough for such fields, so do only
that.
Diffstat (limited to 'test/lit/passes/gufa-refs.wast')
-rw-r--r-- | test/lit/passes/gufa-refs.wast | 64 |
1 files changed, 64 insertions, 0 deletions
diff --git a/test/lit/passes/gufa-refs.wast b/test/lit/passes/gufa-refs.wast index fe3eb70fc..a1652c7d3 100644 --- a/test/lit/passes/gufa-refs.wast +++ b/test/lit/passes/gufa-refs.wast @@ -5525,3 +5525,67 @@ ) ) ) + +;; Packed field combination. +(module + ;; CHECK: (type $A (struct (field i8))) + (type $A (struct (field i8))) + (type $B (struct (field i8))) + + ;; CHECK: (type $none_=>_none (func)) + + ;; CHECK: (func $A (type $none_=>_none) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (struct.get_u $A 0 + ;; CHECK-NEXT: (struct.new $A + ;; CHECK-NEXT: (i32.const 305419896) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (struct.get_u $A 0 + ;; CHECK-NEXT: (struct.new_default $A) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $A + ;; We write two values to $A, which are different, so we cannot infer. + (drop + (struct.get_u $A 0 + (struct.new $A + (i32.const 0x12345678) + ) + ) + ) + (drop + (struct.get_u $A 0 + (struct.new_default $A) + ) + ) + ) + + ;; CHECK: (func $B (type $none_=>_none) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (i32.const 0) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: (drop + ;; CHECK-NEXT: (i32.const 0) + ;; CHECK-NEXT: ) + ;; CHECK-NEXT: ) + (func $B + ;; We write two values to $B, which *seem* different, but given the field is + ;; packed they are both actually 0, so we can optimize here. + (drop + (struct.get_u $B 0 + (struct.new $B + (i32.const 0x12345600) ;; only this changed compared to func $A + ) + ) + ) + (drop + (struct.get_u $B 0 + (struct.new_default $B) + ) + ) + ) +) |