blob: 674361080a5cc8ee5633508ccc91639a3c4c3df8 [file] [log] [blame] [edit]
;; NOTE: Assertions have been generated by update_lit_checks.py and should not be edited.
;; RUN: wasm-opt %s --simplify-locals -all -S -o - | filecheck %s
(module
;; CHECK: (func $test-nn (type $0)
;; CHECK-NEXT: (local $nn anyref)
;; CHECK-NEXT: (nop)
;; CHECK-NEXT: (try
;; CHECK-NEXT: (do
;; CHECK-NEXT: (local.set $nn
;; CHECK-NEXT: (ref.as_non_null
;; CHECK-NEXT: (ref.null none)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: (catch_all
;; CHECK-NEXT: (drop
;; CHECK-NEXT: (ref.as_non_null
;; CHECK-NEXT: (local.get $nn)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
(func $test-nn
(local $nn (ref any))
;; We can sink this set into the try, but the spec does not allow it to
;; remain non-nullable. Even though we are not changing dominance (we are
;; not changing it, because there is nothing that can throw in the try's
;; body that can reach the catch_all before the local.set that we move
;; there). See
;; https://github.com/WebAssembly/function-references/issues/44#issuecomment-1083146887
(local.set $nn
(ref.as_non_null
(ref.null any)
)
)
(try
(do
(drop
(local.get $nn)
)
)
(catch_all
(drop
(local.get $nn)
)
)
)
)
;; CHECK: (func $test-nn-tuple (type $0)
;; CHECK-NEXT: (local $nn (tuple i32 anyref i64))
;; CHECK-NEXT: (nop)
;; CHECK-NEXT: (try
;; CHECK-NEXT: (do
;; CHECK-NEXT: (local.set $nn
;; CHECK-NEXT: (tuple.make 3
;; CHECK-NEXT: (i32.const 0)
;; CHECK-NEXT: (ref.as_non_null
;; CHECK-NEXT: (ref.null none)
;; CHECK-NEXT: )
;; CHECK-NEXT: (i64.const 0)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: (catch_all
;; CHECK-NEXT: (tuple.drop 3
;; CHECK-NEXT: (tuple.make 3
;; CHECK-NEXT: (tuple.extract 3 0
;; CHECK-NEXT: (local.get $nn)
;; CHECK-NEXT: )
;; CHECK-NEXT: (ref.as_non_null
;; CHECK-NEXT: (tuple.extract 3 1
;; CHECK-NEXT: (local.get $nn)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: (tuple.extract 3 2
;; CHECK-NEXT: (local.get $nn)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
(func $test-nn-tuple
;; Same as above, but now the local is a tuple containing a non-nullable element
(local $nn (tuple i32 (ref any) i64))
(local.set $nn
(tuple.make 3
(i32.const 0)
(ref.as_non_null
(ref.null any)
)
(i64.const 0)
)
)
(try
(do
(tuple.drop 3
(local.get $nn)
)
)
(catch_all
(tuple.drop 3
(local.get $nn)
)
)
)
)
;; CHECK: (func $test-nullable (type $0)
;; CHECK-NEXT: (local $nullable anyref)
;; CHECK-NEXT: (nop)
;; CHECK-NEXT: (try
;; CHECK-NEXT: (do
;; CHECK-NEXT: (local.set $nullable
;; CHECK-NEXT: (ref.as_non_null
;; CHECK-NEXT: (ref.null none)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: (catch_all
;; CHECK-NEXT: (drop
;; CHECK-NEXT: (local.get $nullable)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
(func $test-nullable
;; As above, but now the local is nullable. Here we can optimize the set
;; into the try, with no other necessary changes.
(local $nullable (ref null any))
(local.set $nullable
(ref.as_non_null
(ref.null any)
)
)
(try
(do
(drop
(local.get $nullable)
)
)
(catch_all
(drop
(local.get $nullable)
)
)
)
)
;; CHECK: (func $if-return-tuple-nn (type $0)
;; CHECK-NEXT: (local $temp (tuple (ref func) nullref))
;; CHECK-NEXT: (if
;; CHECK-NEXT: (i32.const 0)
;; CHECK-NEXT: (then
;; CHECK-NEXT: (nop)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
(func $if-return-tuple-nn
(local $temp (tuple (ref func) (ref null none)))
;; We should not emit a return value for this if, as the tuple has a non-
;; nullable element, so it is nondefaultable.
;;
;; Instead, we can remove the local.set entirely, as it has no gets.
(if
(i32.const 0)
(then
(local.set $temp
(tuple.make 2
(ref.func $if-return-tuple-nn)
(ref.null none)
)
)
)
)
)
)