blob: f1f6048f7653272b761ae7c332a07aca76bbbf34 [file] [log] [blame] [edit]
;; NOTE: Assertions have been generated by update_lit_checks.py --all-items and should not be edited.
;; RUN: foreach %s %t wasm-opt --global-refining -all -S -o - | filecheck %s
;; RUN: foreach %s %t wasm-opt --global-refining --closed-world -all -S -o - | filecheck %s --check-prefix=CLOSD
(module
;; Globals with no assignments aside from their initial values. The first is a
;; null, so we can optimize to a nullfuncref. The second is a ref.func which
;; lets us refine to the specific function type.
;; CHECK: (type $foo_t (func))
;; CLOSD: (type $foo_t (func))
(type $foo_t (func))
;; CHECK: (global $func-null-init (mut nullfuncref) (ref.null nofunc))
;; CLOSD: (global $func-null-init (mut nullfuncref) (ref.null nofunc))
(global $func-null-init (mut funcref) (ref.null $foo_t))
;; CHECK: (global $func-func-init (mut (ref $foo_t)) (ref.func $foo))
;; CLOSD: (global $func-func-init (mut (ref $foo_t)) (ref.func $foo))
(global $func-func-init (mut funcref) (ref.func $foo))
;; CHECK: (func $foo (type $foo_t)
;; CHECK-NEXT: (nop)
;; CHECK-NEXT: )
;; CLOSD: (func $foo (type $foo_t)
;; CLOSD-NEXT: (nop)
;; CLOSD-NEXT: )
(func $foo (type $foo_t))
)
(module
;; Globals with later assignments of null. The global with a function in its
;; init will update the null to allow it to refine.
;; CHECK: (type $foo_t (func))
;; CLOSD: (type $foo_t (func))
(type $foo_t (func))
;; CHECK: (global $func-null-init (mut nullfuncref) (ref.null nofunc))
;; CLOSD: (global $func-null-init (mut nullfuncref) (ref.null nofunc))
(global $func-null-init (mut funcref) (ref.null $foo_t))
;; CHECK: (global $func-func-init (mut (ref null $foo_t)) (ref.func $foo))
;; CLOSD: (global $func-func-init (mut (ref null $foo_t)) (ref.func $foo))
(global $func-func-init (mut funcref) (ref.func $foo))
;; CHECK: (func $foo (type $foo_t)
;; CHECK-NEXT: (global.set $func-null-init
;; CHECK-NEXT: (ref.null nofunc)
;; CHECK-NEXT: )
;; CHECK-NEXT: (global.set $func-func-init
;; CHECK-NEXT: (ref.null nofunc)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CLOSD: (func $foo (type $foo_t)
;; CLOSD-NEXT: (global.set $func-null-init
;; CLOSD-NEXT: (ref.null nofunc)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: (global.set $func-func-init
;; CLOSD-NEXT: (ref.null nofunc)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: )
(func $foo (type $foo_t)
(global.set $func-null-init (ref.null func))
(global.set $func-func-init (ref.null $foo_t))
)
)
(module
;; Globals with later assignments of something non-null. Both can be refined,
;; and the one with a non-null initial value can even become non-nullable.
;; CHECK: (type $0 (func))
;; CHECK: (global $func-null-init (mut (ref null $0)) (ref.null nofunc))
;; CLOSD: (type $0 (func))
;; CLOSD: (global $func-null-init (mut (ref null $0)) (ref.null nofunc))
(global $func-null-init (mut funcref) (ref.null func))
;; CHECK: (global $func-func-init (mut (ref $0)) (ref.func $foo))
;; CLOSD: (global $func-func-init (mut (ref $0)) (ref.func $foo))
(global $func-func-init (mut funcref) (ref.func $foo))
;; CHECK: (elem declare func $foo)
;; CHECK: (func $foo (type $0)
;; CHECK-NEXT: (global.set $func-null-init
;; CHECK-NEXT: (ref.func $foo)
;; CHECK-NEXT: )
;; CHECK-NEXT: (global.set $func-func-init
;; CHECK-NEXT: (ref.func $foo)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CLOSD: (elem declare func $foo)
;; CLOSD: (func $foo (type $0)
;; CLOSD-NEXT: (global.set $func-null-init
;; CLOSD-NEXT: (ref.func $foo)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: (global.set $func-func-init
;; CLOSD-NEXT: (ref.func $foo)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: )
(func $foo
(global.set $func-null-init (ref.func $foo))
(global.set $func-func-init (ref.func $foo))
)
)
(module
;; A global with multiple later assignments. The refined type is more
;; specific than the original, but less than each of the non-null values.
;; CHECK: (type $0 (func))
;; CHECK: (type $struct (struct))
;; CLOSD: (type $0 (func))
;; CLOSD: (type $struct (struct))
(type $struct (struct))
(type $array (array i8))
;; CHECK: (global $global (mut eqref) (ref.null none))
;; CLOSD: (global $global (mut eqref) (ref.null none))
(global $global (mut anyref) (ref.null any))
;; CHECK: (func $foo (type $0)
;; CHECK-NEXT: (global.set $global
;; CHECK-NEXT: (ref.i31
;; CHECK-NEXT: (i32.const 0)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: (global.set $global
;; CHECK-NEXT: (struct.new_default $struct)
;; CHECK-NEXT: )
;; CHECK-NEXT: (global.set $global
;; CHECK-NEXT: (ref.null none)
;; CHECK-NEXT: )
;; CHECK-NEXT: (global.set $global
;; CHECK-NEXT: (ref.null none)
;; CHECK-NEXT: )
;; CHECK-NEXT: (global.set $global
;; CHECK-NEXT: (ref.null none)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CLOSD: (func $foo (type $0)
;; CLOSD-NEXT: (global.set $global
;; CLOSD-NEXT: (ref.i31
;; CLOSD-NEXT: (i32.const 0)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: )
;; CLOSD-NEXT: (global.set $global
;; CLOSD-NEXT: (struct.new_default $struct)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: (global.set $global
;; CLOSD-NEXT: (ref.null none)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: (global.set $global
;; CLOSD-NEXT: (ref.null none)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: (global.set $global
;; CLOSD-NEXT: (ref.null none)
;; CLOSD-NEXT: )
;; CLOSD-NEXT: )
(func $foo
(global.set $global (ref.i31 (i32.const 0)))
(global.set $global (struct.new_default $struct))
(global.set $global (ref.null eq))
;; These nulls will be updated.
(global.set $global (ref.null i31))
(global.set $global (ref.null $array))
)
)
;; We can refine here, but as it is an export we only do so in open world.
(module
;; CHECK: (type $0 (func))
;; CHECK: (global $func-init (mut (ref $0)) (ref.func $foo))
;; CLOSD: (type $0 (func))
;; CLOSD: (global $func-init (mut funcref) (ref.func $foo))
(global $func-init (mut funcref) (ref.func $foo))
;; CHECK: (export "global" (global $func-init))
;; CLOSD: (export "global" (global $func-init))
(export "global" (global $func-init))
;; CHECK: (func $foo (type $0)
;; CHECK-NEXT: (nop)
;; CHECK-NEXT: )
;; CLOSD: (func $foo (type $0)
;; CLOSD-NEXT: (nop)
;; CLOSD-NEXT: )
(func $foo
(nop)
)
)
;; We can refine $a, after which we should update the global.get in the other
;; global, or else we'd error on validation.
;; TODO: we could optimize further here and refine the type of the global $b.
(module
;; CHECK: (type $super (sub (func)))
;; CLOSD: (type $super (sub (func)))
(type $super (sub (func)))
;; CHECK: (type $sub (sub $super (func)))
;; CLOSD: (type $sub (sub $super (func)))
(type $sub (sub $super (func)))
;; CHECK: (global $a (ref $sub) (ref.func $func))
;; CLOSD: (global $a (ref $sub) (ref.func $func))
(global $a (ref $super) (ref.func $func))
;; CHECK: (global $b (ref $super) (global.get $a))
;; CLOSD: (global $b (ref $super) (global.get $a))
(global $b (ref $super) (global.get $a))
;; CHECK: (func $func (type $sub)
;; CHECK-NEXT: (nop)
;; CHECK-NEXT: )
;; CLOSD: (func $func (type $sub)
;; CLOSD-NEXT: (nop)
;; CLOSD-NEXT: )
(func $func (type $sub)
)
)