blob: 8e034ef3efdc39ae73158fbca73fde0f2917d946 [file] [edit]
;; RUN: foreach %s %t wasm-opt --closed-world --nominal --signature-refining --gto --roundtrip -all -S -o - | filecheck %s
;; RUN: foreach %s %t wasm-opt --closed-world --signature-refining --gto --remove-unused-types --roundtrip -all -S -o - | filecheck %s --check-prefix ISOREC
;; Check that type $A is not included in the final binary after the signature
;; refining optimization. For isorecursive types, this requires an additional
;; --remove-unused-types pass after signature refining.
(module
;; The type $A should not be emitted at all (see below).
;; CHECK-NOT: (type $A
;; ISOREC-NOT: (type $A
(type $A (struct (field (mut (ref null $A)))))
;; CHECK: (type $ref|none|_=>_none (func (param (ref none))))
;; CHECK: (type $funcref_i32_=>_none (func (param funcref i32)))
;; CHECK: (func $struct.get (type $ref|none|_=>_none) (param $0 (ref none))
;; CHECK-NEXT: (drop
;; CHECK-NEXT: (local.get $0)
;; CHECK-NEXT: )
;; CHECK-NEXT: (unreachable)
;; CHECK-NEXT: )
(func $struct.get (param $0 (ref $A))
;; This function is always called with a null, so the parameter type will be
;; refined to that. After doing so, the struct.get will be reading from a
;; null type, and we should avoid erroring on that in --gto which scans all
;; the heap types and updates them. Likewise, we should not error during
;; roundtrip which also scans heap types.
(drop
(struct.get $A 0
(local.get $0)
)
)
)
;; CHECK: (func $caller (type $funcref_i32_=>_none) (param $0 funcref) (param $1 i32)
;; CHECK-NEXT: (call $struct.get
;; CHECK-NEXT: (ref.as_non_null
;; CHECK-NEXT: (ref.null none)
;; CHECK-NEXT: )
;; CHECK-NEXT: )
;; CHECK-NEXT: )
(func $caller (param $0 funcref) (param $1 i32)
(call $struct.get
(ref.as_non_null
(ref.null none)
)
)
)
)