;; Test `return_call_ref` operator (module ;; Auxiliary definitions (type $proc (func)) (type $-i32 (func (result i32))) (type $-i64 (func (result i64))) (type $-f32 (func (result f32))) (type $-f64 (func (result f64))) (type $i32-i32 (func (param i32) (result i32))) (type $i64-i64 (func (param i64) (result i64))) (type $f32-f32 (func (param f32) (result f32))) (type $f64-f64 (func (param f64) (result f64))) (type $f32-i32 (func (param f32 i32) (result i32))) (type $i32-i64 (func (param i32 i64) (result i64))) (type $f64-f32 (func (param f64 f32) (result f32))) (type $i64-f64 (func (param i64 f64) (result f64))) (type $i64i64-i64 (func (param i64 i64) (result i64))) (func $const-i32 (result i32) (i32.const 0x132)) (func $const-i64 (result i64) (i64.const 0x164)) (func $const-f32 (result f32) (f32.const 0xf32)) (func $const-f64 (result f64) (f64.const 0xf64)) (func $id-i32 (param i32) (result i32) (local.get 0)) (func $id-i64 (param i64) (result i64) (local.get 0)) (func $id-f32 (param f32) (result f32) (local.get 0)) (func $id-f64 (param f64) (result f64) (local.get 0)) (func $f32-i32 (param f32 i32) (result i32) (local.get 1)) (func $i32-i64 (param i32 i64) (result i64) (local.get 1)) (func $f64-f32 (param f64 f32) (result f32) (local.get 1)) (func $i64-f64 (param i64 f64) (result f64) (local.get 1)) (global $const-i32 (ref $-i32) (ref.func $const-i32)) (global $const-i64 (ref $-i64) (ref.func $const-i64)) (global $const-f32 (ref $-f32) (ref.func $const-f32)) (global $const-f64 (ref $-f64) (ref.func $const-f64)) (global $id-i32 (ref $i32-i32) (ref.func $id-i32)) (global $id-i64 (ref $i64-i64) (ref.func $id-i64)) (global $id-f32 (ref $f32-f32) (ref.func $id-f32)) (global $id-f64 (ref $f64-f64) (ref.func $id-f64)) (global $f32-i32 (ref $f32-i32) (ref.func $f32-i32)) (global $i32-i64 (ref $i32-i64) (ref.func $i32-i64)) (global $f64-f32 (ref $f64-f32) (ref.func $f64-f32)) (global $i64-f64 (ref $i64-f64) (ref.func $i64-f64)) (elem declare func $const-i32 $const-i64 $const-f32 $const-f64 $id-i32 $id-i64 $id-f32 $id-f64 $f32-i32 $i32-i64 $f64-f32 $i64-f64 ) ;; Typing (func (export "type-i32") (result i32) (return_call_ref $-i32 (global.get $const-i32)) ) (func (export "type-i64") (result i64) (return_call_ref $-i64 (global.get $const-i64)) ) (func (export "type-f32") (result f32) (return_call_ref $-f32 (global.get $const-f32)) ) (func (export "type-f64") (result f64) (return_call_ref $-f64 (global.get $const-f64)) ) (func (export "type-first-i32") (result i32) (return_call_ref $i32-i32 (i32.const 32) (global.get $id-i32)) ) (func (export "type-first-i64") (result i64) (return_call_ref $i64-i64 (i64.const 64) (global.get $id-i64)) ) (func (export "type-first-f32") (result f32) (return_call_ref $f32-f32 (f32.const 1.32) (global.get $id-f32)) ) (func (export "type-first-f64") (result f64) (return_call_ref $f64-f64 (f64.const 1.64) (global.get $id-f64)) ) (func (export "type-second-i32") (result i32) (return_call_ref $f32-i32 (f32.const 32.1) (i32.const 32) (global.get $f32-i32)) ) (func (export "type-second-i64") (result i64) (return_call_ref $i32-i64 (i32.const 32) (i64.const 64) (global.get $i32-i64)) ) (func (export "type-second-f32") (result f32) (return_call_ref $f64-f32 (f64.const 64) (f32.const 32) (global.get $f64-f32)) ) (func (export "type-second-f64") (result f64) (return_call_ref $i64-f64 (i64.const 64) (f64.const 64.1) (global.get $i64-f64)) ) ;; Null (func (export "null") (return_call_ref $proc (ref.null $proc)) ) ;; Recursion (global $fac-acc (ref $i64i64-i64) (ref.func $fac-acc)) (elem declare func $fac-acc) (func $fac-acc (export "fac-acc") (param i64 i64) (result i64) (if (result i64) (i64.eqz (local.get 0)) (then (local.get 1)) (else (return_call_ref $i64i64-i64 (i64.sub (local.get 0) (i64.const 1)) (i64.mul (local.get 0) (local.get 1)) (global.get $fac-acc) ) ) ) ) (global $count (ref $i64-i64) (ref.func $count)) (elem declare func $count) (func $count (export "count") (param i64) (result i64) (if (result i64) (i64.eqz (local.get 0)) (then (local.get 0)) (else (return_call_ref $i64-i64 (i64.sub (local.get 0) (i64.const 1)) (global.get $count) ) ) ) ) (global $even (ref $i64-i64) (ref.func $even)) (global $odd (ref $i64-i64) (ref.func $odd)) (elem declare func $even) (func $even (export "even") (param i64) (result i64) (if (result i64) (i64.eqz (local.get 0)) (then (i64.const 44)) (else (return_call_ref $i64-i64 (i64.sub (local.get 0) (i64.const 1)) (global.get $odd) ) ) ) ) (elem declare func $odd) (func $odd (export "odd") (param i64) (result i64) (if (result i64) (i64.eqz (local.get 0)) (then (i64.const 99)) (else (return_call_ref $i64-i64 (i64.sub (local.get 0) (i64.const 1)) (global.get $even) ) ) ) ) ) (assert_return (invoke "type-i32") (i32.const 0x132)) (assert_return (invoke "type-i64") (i64.const 0x164)) (assert_return (invoke "type-f32") (f32.const 0xf32)) (assert_return (invoke "type-f64") (f64.const 0xf64)) (assert_return (invoke "type-first-i32") (i32.const 32)) (assert_return (invoke "type-first-i64") (i64.const 64)) (assert_return (invoke "type-first-f32") (f32.const 1.32)) (assert_return (invoke "type-first-f64") (f64.const 1.64)) (assert_return (invoke "type-second-i32") (i32.const 32)) (assert_return (invoke "type-second-i64") (i64.const 64)) (assert_return (invoke "type-second-f32") (f32.const 32)) (assert_return (invoke "type-second-f64") (f64.const 64.1)) (assert_trap (invoke "null") "null function reference") (assert_return (invoke "fac-acc" (i64.const 0) (i64.const 1)) (i64.const 1)) (assert_return (invoke "fac-acc" (i64.const 1) (i64.const 1)) (i64.const 1)) (assert_return (invoke "fac-acc" (i64.const 5) (i64.const 1)) (i64.const 120)) (assert_return (invoke "fac-acc" (i64.const 25) (i64.const 1)) (i64.const 7034535277573963776) ) (assert_return (invoke "count" (i64.const 0)) (i64.const 0)) (assert_return (invoke "count" (i64.const 1000)) (i64.const 0)) (assert_return (invoke "count" (i64.const 1_000_000)) (i64.const 0)) (assert_return (invoke "even" (i64.const 0)) (i64.const 44)) (assert_return (invoke "even" (i64.const 1)) (i64.const 99)) (assert_return (invoke "even" (i64.const 100)) (i64.const 44)) (assert_return (invoke "even" (i64.const 77)) (i64.const 99)) (assert_return (invoke "even" (i64.const 1_000_000)) (i64.const 44)) (assert_return (invoke "even" (i64.const 1_000_001)) (i64.const 99)) (assert_return (invoke "odd" (i64.const 0)) (i64.const 99)) (assert_return (invoke "odd" (i64.const 1)) (i64.const 44)) (assert_return (invoke "odd" (i64.const 200)) (i64.const 99)) (assert_return (invoke "odd" (i64.const 77)) (i64.const 44)) (assert_return (invoke "odd" (i64.const 1_000_000)) (i64.const 99)) (assert_return (invoke "odd" (i64.const 999_999)) (i64.const 44)) ;; More typing (module (type $t (func)) (type $t1 (func (result (ref $t)))) (type $t2 (func (result (ref null $t)))) (type $t3 (func (result (ref func)))) (type $t4 (func (result (ref null func)))) (elem declare func $f11 $f22 $f33 $f44) (func $f11 (result (ref $t)) (return_call_ref $t1 (ref.func $f11))) (func $f21 (result (ref null $t)) (return_call_ref $t1 (ref.func $f11))) (func $f22 (result (ref null $t)) (return_call_ref $t2 (ref.func $f22))) (func $f31 (result (ref func)) (return_call_ref $t1 (ref.func $f11))) (func $f33 (result (ref func)) (return_call_ref $t3 (ref.func $f33))) (func $f41 (result (ref null func)) (return_call_ref $t1 (ref.func $f11))) (func $f42 (result (ref null func)) (return_call_ref $t2 (ref.func $f22))) (func $f43 (result (ref null func)) (return_call_ref $t3 (ref.func $f33))) (func $f44 (result (ref null func)) (return_call_ref $t4 (ref.func $f44))) ) (assert_invalid (module (type $t (func)) (type $t2 (func (result (ref null $t)))) (elem declare func $f22) (func $f12 (result (ref $t)) (return_call_ref $t2 (ref.func $f22))) (func $f22 (result (ref null $t)) (return_call_ref $t2 (ref.func $f22))) ) "type mismatch" ) (assert_invalid (module (type $t (func)) (type $t3 (func (result (ref func)))) (elem declare func $f33) (func $f13 (result (ref $t)) (return_call_ref $t3 (ref.func $f33))) (func $f33 (result (ref func)) (return_call_ref $t3 (ref.func $f33))) ) "type mismatch" ) (assert_invalid (module (type $t (func)) (type $t4 (func (result (ref null func)))) (elem declare func $f44) (func $f14 (result (ref $t)) (return_call_ref $t4 (ref.func $f44))) (func $f44 (result (ref null func)) (return_call_ref $t4 (ref.func $f44))) ) "type mismatch" ) (assert_invalid (module (type $t (func)) (type $t3 (func (result (ref func)))) (elem declare func $f33) (func $f23 (result (ref null $t)) (return_call_ref $t3 (ref.func $f33))) (func $f33 (result (ref func)) (return_call_ref $t3 (ref.func $f33))) ) "type mismatch" ) (assert_invalid (module (type $t (func)) (type $t4 (func (result (ref null func)))) (elem declare func $f44) (func $f24 (result (ref null $t)) (return_call_ref $t4 (ref.func $f44))) (func $f44 (result (ref null func)) (return_call_ref $t4 (ref.func $f44))) ) "type mismatch" ) (assert_invalid (module (type $t4 (func (result (ref null func)))) (elem declare func $f44) (func $f34 (result (ref func)) (return_call_ref $t4 (ref.func $f44))) (func $f44 (result (ref null func)) (return_call_ref $t4 (ref.func $f44))) ) "type mismatch" ) ;; Unreachable typing. (module (type $t (func (result i32))) (func (export "unreachable") (result i32) (unreachable) (return_call_ref $t) ) ) (assert_trap (invoke "unreachable") "unreachable") (module (elem declare func $f) (type $t (func (param i32) (result i32))) (func $f (param i32) (result i32) (local.get 0)) (func (export "unreachable") (result i32) (unreachable) (ref.func $f) (return_call_ref $t) ) ) (assert_trap (invoke "unreachable") "unreachable") (module (elem declare func $f) (type $t (func (param i32) (result i32))) (func $f (param i32) (result i32) (local.get 0)) (func (export "unreachable") (result i32) (unreachable) (i32.const 0) (ref.func $f) (return_call_ref $t) (i32.const 0) ) ) (assert_trap (invoke "unreachable") "unreachable") (assert_invalid (module (elem declare func $f) (type $t (func (param i32) (result i32))) (func $f (param i32) (result i32) (local.get 0)) (func (export "unreachable") (result i32) (unreachable) (i64.const 0) (ref.func $f) (return_call_ref $t) ) ) "type mismatch" ) (assert_invalid (module (elem declare func $f) (type $t (func (param i32) (result i32))) (func $f (param i32) (result i32) (local.get 0)) (func (export "unreachable") (result i32) (unreachable) (ref.func $f) (return_call_ref $t) (i64.const 0) ) ) "type mismatch" ) (assert_invalid (module (type $t (func)) (func $f (param $r externref) (return_call_ref $t (local.get $r)) ) ) "type mismatch" )