;; NOTE: Assertions have been generated by update_lit_checks.py --all-items and should not be edited. ;; RUN: foreach %s %t wasm-opt -all -S -o - | filecheck %s ;; Tests for validation of non-nullable locals. (module ;; CHECK: (type $0 (func)) ;; CHECK: (type $1 (func (param (ref func)))) ;; CHECK: (type $2 (func (param funcref) (result i32))) ;; CHECK: (elem declare func $helper) ;; CHECK: (func $no-uses (type $0) ;; CHECK-NEXT: (local $x (ref func)) ;; CHECK-NEXT: (nop) ;; CHECK-NEXT: ) (func $no-uses ;; A local with no uses validates. (local $x (ref func)) ) ;; CHECK: (func $func-scope (type $0) ;; CHECK-NEXT: (local $x (ref func)) ;; CHECK-NEXT: (local.set $x ;; CHECK-NEXT: (ref.func $helper) ;; CHECK-NEXT: ) ;; CHECK-NEXT: (drop ;; CHECK-NEXT: (local.get $x) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) (func $func-scope ;; a set in the func scope helps a get validate there. (local $x (ref func)) (local.set $x (ref.func $helper) ) (drop (local.get $x) ) ) ;; CHECK: (func $inner-scope (type $0) ;; CHECK-NEXT: (local $x (ref func)) ;; CHECK-NEXT: (block $b ;; CHECK-NEXT: (local.set $x ;; CHECK-NEXT: (ref.func $helper) ;; CHECK-NEXT: ) ;; CHECK-NEXT: (drop ;; CHECK-NEXT: (local.get $x) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) (func $inner-scope ;; a set in an inner scope helps a get validate there. (local $x (ref func)) (block $b (local.set $x (ref.func $helper) ) (drop (local.get $x) ) ) ) ;; CHECK: (func $func-to-inner (type $0) ;; CHECK-NEXT: (local $x (ref func)) ;; CHECK-NEXT: (local.set $x ;; CHECK-NEXT: (ref.func $helper) ;; CHECK-NEXT: ) ;; CHECK-NEXT: (block $b ;; CHECK-NEXT: (drop ;; CHECK-NEXT: (local.get $x) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) (func $func-to-inner ;; a set in an outer scope helps a get validate. (local $x (ref func)) (local.set $x (ref.func $helper) ) (block $b (drop (local.get $x) ) ) ) ;; CHECK: (func $inner-to-func (type $0) ;; CHECK-NEXT: (local $x funcref) ;; CHECK-NEXT: (block $b ;; CHECK-NEXT: (local.set $x ;; CHECK-NEXT: (ref.func $helper) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) ;; CHECK-NEXT: (drop ;; CHECK-NEXT: (local.get $x) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) (func $inner-to-func ;; a set in an inner scope does *not* help a get validate, but the type is ;; nullable so that's ok. (local $x (ref null func)) (block $b (local.set $x (ref.func $helper) ) ) (drop (local.get $x) ) ) ;; CHECK: (func $if-condition (type $0) ;; CHECK-NEXT: (local $x (ref func)) ;; CHECK-NEXT: (if ;; CHECK-NEXT: (call $helper2 ;; CHECK-NEXT: (local.tee $x ;; CHECK-NEXT: (ref.func $helper) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) ;; CHECK-NEXT: (then ;; CHECK-NEXT: (drop ;; CHECK-NEXT: (local.get $x) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) ;; CHECK-NEXT: (else ;; CHECK-NEXT: (drop ;; CHECK-NEXT: (local.get $x) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) (func $if-condition (local $x (ref func)) (if (call $helper2 ;; Tee in the condition is good enough for the arms. (local.tee $x (ref.func $helper) ) ) (then (drop (local.get $x) ) ) (else (drop (local.get $x) ) ) ) ) ;; CHECK: (func $get-without-set-but-param (type $1) (param $x (ref func)) ;; CHECK-NEXT: (drop ;; CHECK-NEXT: (local.get $x) ;; CHECK-NEXT: ) ;; CHECK-NEXT: ) (func $get-without-set-but-param ;; As a parameter, this is ok to get without a set. (param $x (ref func)) (drop (local.get $x) ) ) ;; CHECK: (func $helper (type $0) ;; CHECK-NEXT: (nop) ;; CHECK-NEXT: ) (func $helper) ;; CHECK: (func $helper2 (type $2) (param $0 funcref) (result i32) ;; CHECK-NEXT: (unreachable) ;; CHECK-NEXT: ) (func $helper2 (param funcref) (result i32) (unreachable) ) )