diff --git a/src/core/error.rs b/src/core/error.rs index be6685f8..4abd3298 100644 --- a/src/core/error.rs +++ b/src/core/error.rs @@ -1,12 +1,13 @@ use crate::core::indices::GlobalIdx; use crate::validation_stack::LabelKind; +use crate::RefType; use core::fmt::{Display, Formatter}; use core::str::Utf8Error; use crate::core::reader::section_header::SectionTy; use crate::core::reader::types::ValType; -use super::indices::{DataIdx, MemIdx}; +use super::indices::{DataIdx, ElemIdx, FuncIdx, MemIdx, TableIdx, TypeIdx}; #[derive(Debug, PartialEq, Eq, Clone)] pub enum RuntimeError { @@ -17,6 +18,10 @@ pub enum RuntimeError { // https://github.com/wasmi-labs/wasmi/blob/37d1449524a322817c55026eb21eb97dd693b9ce/crates/core/src/trap.rs#L265C5-L265C27 BadConversionToInteger, MemoryAccessOutOfBounds, + TableAccessOutOfBounds, + ElementAccessOutOfBounds, + UninitializedElement, + SignatureMismatch, } #[derive(Debug, PartialEq, Eq, Clone)] @@ -51,11 +56,21 @@ pub enum Error { GlobalIsConst, RuntimeError(RuntimeError), FoundLabel(LabelKind), + FoundUnspecifiedValTypes, MemoryIsNotDefined(MemIdx), // mem.align, wanted alignment ErroneousAlignment(u32, u32), NoDataSegments, DataSegmentNotFound(DataIdx), + UnknownTable, + TableIsNotDefined(TableIdx), + ElementIsNotDefined(ElemIdx), + DifferentRefTypes(RefType, RefType), + ExpectedARefType(ValType), + WrongRefTypeForInteropValue(RefType, RefType), + FunctionIsNotDefined(FuncIdx), + ReferencingAnUnreferencedFunction(FuncIdx), + FunctionTypeIsNotDefined(TypeIdx), } impl Display for Error { @@ -131,6 +146,7 @@ impl Display for Error { Error::FoundLabel(lk) => f.write_fmt(format_args!( "Expecting a ValType, a Label was found: {lk:?}" )), + Error::FoundUnspecifiedValTypes => f.write_str("Found UnspecifiedValTypes"), Error::ExpectedAnOperand => f.write_str("Expected a ValType"), // Error => f.write_str("Expected an operand (ValType) on the stack") Error::MemoryIsNotDefined(memidx) => f.write_fmt(format_args!( "C.mems[{}] is NOT defined when it should be", @@ -146,6 +162,39 @@ impl Display for Error { Error::DataSegmentNotFound(data_idx) => { f.write_fmt(format_args!("Data Segment {} not found", data_idx)) } + Error::UnknownTable => f.write_str("Unknown Table"), + Error::TableIsNotDefined(table_idx) => f.write_fmt(format_args!( + "C.tables[{}] is NOT defined when it should be", + table_idx + )), + Error::ElementIsNotDefined(elem_idx) => f.write_fmt(format_args!( + "C.elems[{}] is NOT defined when it should be", + elem_idx + )), + Error::DifferentRefTypes(rref1, rref2) => f.write_fmt(format_args!( + "RefType {} is NOT equal to RefType {}", + rref1, rref2 + )), + Error::ExpectedARefType(found_valtype) => f.write_fmt(format_args!( + "Expected a RefType, found a {:?} instead", + found_valtype + )), + Error::WrongRefTypeForInteropValue(ref_given, ref_wanted) => f.write_fmt(format_args!( + "Wrong RefType for InteropValue: Given {} - Needed {}", + ref_given, ref_wanted + )), + Error::FunctionIsNotDefined(func_idx) => f.write_fmt(format_args!( + "C.functions[{}] is NOT defined when it should be", + func_idx + )), + Error::ReferencingAnUnreferencedFunction(func_idx) => f.write_fmt(format_args!( + "Referenced a function ({}) that was not referenced in validation", + func_idx + )), + Error::FunctionTypeIsNotDefined(func_ty_idx) => f.write_fmt(format_args!( + "C.fn_types[{}] is NOT defined when it should be", + func_ty_idx + )), } } } @@ -159,6 +208,10 @@ impl Display for RuntimeError { RuntimeError::StackSmash => f.write_str("Stack smashed"), RuntimeError::BadConversionToInteger => f.write_str("Bad conversion to integer"), RuntimeError::MemoryAccessOutOfBounds => f.write_str("Memory access out of bounds"), + RuntimeError::TableAccessOutOfBounds => f.write_str("Table access out of bounds"), + RuntimeError::ElementAccessOutOfBounds => f.write_str("Element access out of bounds"), + RuntimeError::UninitializedElement => f.write_str("Uninitialized element"), + RuntimeError::SignatureMismatch => f.write_str("Indirect call signature mismatch"), } } } diff --git a/src/core/reader/types/data.rs b/src/core/reader/types/data.rs index 8451f717..00106bfb 100644 --- a/src/core/reader/types/data.rs +++ b/src/core/reader/types/data.rs @@ -36,7 +36,7 @@ impl WasmReadable for DataSegment { 0 => { // active { memory 0, offset e } trace!("Data section: active"); - let offset = { read_constant_instructions(wasm, None, None)? }; + let offset = { read_constant_instructions(wasm, None, None, None)? }; let byte_vec = wasm.read_vec(|el| el.read_u8())?; @@ -81,7 +81,8 @@ impl WasmReadable for DataSegment { 0 => { // active { memory 0, offset e } trace!("Data section: active"); - let offset = { read_constant_instructions(wasm, None, None).unwrap_validated() }; + let offset = + { read_constant_instructions(wasm, None, None, None).unwrap_validated() }; let byte_vec = wasm .read_vec(|el| Ok(el.read_u8().unwrap_validated())) diff --git a/src/core/reader/types/element.rs b/src/core/reader/types/element.rs new file mode 100644 index 00000000..7a3cec8f --- /dev/null +++ b/src/core/reader/types/element.rs @@ -0,0 +1,165 @@ +use core::fmt::Debug; + +use alloc::vec::Vec; + +use crate::{ + core::reader::span::Span, read_constant_expression::read_constant_instructions, Error, Result, +}; + +use super::RefType; + +#[derive(Clone)] +pub struct ElemType { + pub init: ElemItems, + pub mode: ElemMode, +} + +impl Debug for ElemType { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + write!( + f, + "ElemType {{\n\tinit: {:?},\n\tmode: {:?},\n\t#ty: {}\n}}", + self.init, + self.mode, + self.init.ty() + ) + } +} + +impl ElemType { + pub fn ty(&self) -> RefType { + self.init.ty() + } + pub fn to_ref_type(&self) -> RefType { + match self.init { + ElemItems::Exprs(rref, _) => rref, + ElemItems::RefFuncs(_) => RefType::FuncRef, + } + } + + pub fn read_from_wasm( + wasm: &mut crate::core::reader::WasmReader, + functions: &[usize], + referenced_functions: &mut alloc::collections::btree_set::BTreeSet, + tables_length: usize, + ) -> Result> { + use crate::core::reader::types::element::*; + use crate::RefType; + // https://webassembly.github.io/spec/core/binary/modules.html#element-section + + wasm.read_vec(|wasm| { + let ty = wasm.read_var_u32().unwrap(); + // https://webassembly.github.io/spec/core/syntax/modules.html#element-segments + // https://webassembly.github.io/spec/core/binary/modules.html#element-section + // We can treat the ttype as a 3bit integer + // If it's not 3 bits I am not sure what to do + // bit 0 => diff between passive|declartive and active segment + // bit 1 => presence of an explicit table index for an active segment + // bit 2 => use of element type and element expressions instead of element kind and element indices + assert!(ty <= 0b111, "Element section is not encoded correctly. The type of this element is over 7 (0b111)"); + // decide if we should + let elem_mode = if ty & 0b001 == 0b001 { + if ty & 0b010 == 0b010 { + ElemMode::Declarative + } else { + ElemMode::Passive + } + } else { + let table_idx = if ty & 0b010 == 0b010 { + wasm.read_var_u32()? + } else { + 0 + }; + if tables_length <= table_idx as usize { + return Err(Error::UnknownTable); + } + let expr = read_constant_instructions(wasm, None, None, Some(&functions))?; + + ElemMode::Active(ActiveElem { + table: table_idx, + offset: expr, + }) + }; + let use_of_el_ty_and_el_exprs = ty & 0b100 == 0b100; + + let reftype_or_elemkind: Option = match if ty & 0b011 != 0 { + if use_of_el_ty_and_el_exprs { + Some(wasm.read_u8()?) + } else { + let read = wasm.read_u8()?; + match read { + 0x00 => None, + _ => todo!("Only FuncRefs are allowed"), + } + } + } else { + None + } { + None => None, + Some(ty) => Some(RefType::from_byte(ty)?), + }; + + match reftype_or_elemkind { + Some(rty) => trace!("REFTYPE: {}", rty), + None => { + trace!("REFTYPE NONE!") + } + }; + + let items: ElemItems = if use_of_el_ty_and_el_exprs { + ElemItems::Exprs( + reftype_or_elemkind.unwrap_or(RefType::FuncRef), + wasm.read_vec(|w| read_constant_instructions(w, None, None, Some(&functions)))?, + ) + } else { + assert!(reftype_or_elemkind.is_none()); + ElemItems::RefFuncs(wasm.read_vec(|w| { + let offset = w.read_var_u32()?; + referenced_functions.insert(offset); + Ok(offset) + })?) + }; + + let el = ElemType { + init: items, + mode: elem_mode, + }; + + Ok(el) + }) + } +} + +/// Here we can't implement WasmReadable because we also want a mutable +/// reference to a BTreeSet (`referenced_functions`) +/// +/// This comes in handy later on when we are validating the actual code of +/// the functions so that we can make sure we are not referencing invalid functions + +#[derive(Debug, Clone)] +pub enum ElemItems { + RefFuncs(Vec), + Exprs(RefType, Vec), +} + +impl ElemItems { + pub fn ty(&self) -> RefType { + match self { + Self::RefFuncs(_) => RefType::FuncRef, + Self::Exprs(rty, _) => *rty, + } + } +} + +#[derive(Debug, Clone)] +pub enum ElemMode { + Passive, + Active(ActiveElem), + Declarative, +} + +#[derive(Debug, Clone)] +pub struct ActiveElem { + pub table: u32, + pub offset: Span, +} diff --git a/src/core/reader/types/import.rs b/src/core/reader/types/import.rs index 966ae947..7e260765 100644 --- a/src/core/reader/types/import.rs +++ b/src/core/reader/types/import.rs @@ -6,6 +6,8 @@ use crate::core::reader::{WasmReadable, WasmReader}; use crate::execution::assert_validated::UnwrapValidatedExt; use crate::{unreachable_validated, Error, Result}; +use super::TableType; + #[derive(Debug)] pub struct Import { #[allow(warnings)] @@ -47,7 +49,7 @@ pub enum ImportDesc { #[allow(dead_code)] Func(TypeIdx), #[allow(dead_code)] - Table(()), + Table(TableType), // TODO TableType #[allow(dead_code)] Mem(()), @@ -60,7 +62,8 @@ impl WasmReadable for ImportDesc { fn read(wasm: &mut WasmReader) -> Result { let desc = match wasm.read_u8()? { 0x00 => Self::Func(wasm.read_var_u32()? as TypeIdx), - 0x01 => todo!("read TableType"), + // https://webassembly.github.io/spec/core/binary/types.html#table-types + 0x01 => Self::Table(TableType::read(wasm)?), 0x02 => todo!("read MemType"), 0x03 => todo!("read GlobalType"), other => return Err(Error::InvalidImportDesc(other)), @@ -72,7 +75,7 @@ impl WasmReadable for ImportDesc { fn read_unvalidated(wasm: &mut WasmReader) -> Self { match wasm.read_u8().unwrap_validated() { 0x00 => Self::Func(wasm.read_var_u32().unwrap_validated() as TypeIdx), - 0x01 => todo!("read TableType"), + 0x01 => Self::Table(TableType::read_unvalidated(wasm)), 0x02 => todo!("read MemType"), 0x03 => todo!("read GlobalType"), _ => unreachable_validated!(), diff --git a/src/core/reader/types/mod.rs b/src/core/reader/types/mod.rs index 6a3a5303..2c241810 100644 --- a/src/core/reader/types/mod.rs +++ b/src/core/reader/types/mod.rs @@ -3,14 +3,16 @@ //! See: use alloc::vec::Vec; -use core::fmt::{Debug, Formatter}; +use core::fmt::{Debug, Display, Formatter}; use crate::core::reader::{WasmReadable, WasmReader}; use crate::execution::assert_validated::UnwrapValidatedExt; +use crate::value::{ExternAddr, FuncAddr, Ref}; use crate::Result; use crate::{unreachable_validated, Error}; pub mod data; +pub mod element; pub mod export; pub mod function_code_header; pub mod global; @@ -85,6 +87,40 @@ pub enum RefType { ExternRef, } +impl Display for RefType { + fn fmt(&self, f: &mut Formatter<'_>) -> core::fmt::Result { + write!( + f, + "{}", + match self { + Self::ExternRef => "ExternRef", + Self::FuncRef => "FuncRef", + } + ) + } +} + +impl RefType { + /// TODO: we have to make sure they are NOT null Refs, but still, they are not valid ones as we cast them from RefTypes which don't hold addresses per-se + pub fn to_null_ref(&self) -> Ref { + match self { + RefType::ExternRef => Ref::Extern(ExternAddr::null()), + RefType::FuncRef => Ref::Func(FuncAddr::null()), + } + } +} + +impl RefType { + // } + pub fn from_byte(byte: u8) -> Result { + match byte { + 0x70 => Ok(RefType::FuncRef), + 0x6F => Ok(RefType::ExternRef), + _ => Err(Error::InvalidRefType), + } + } +} + impl WasmReadable for RefType { fn read(wasm: &mut WasmReader) -> Result { let ty = match wasm.peek_u8()? { @@ -129,22 +165,31 @@ impl ValType { impl WasmReadable for ValType { fn read(wasm: &mut WasmReader) -> Result { - let numtype = NumType::read(wasm).map(ValType::NumType); - let vectype = VecType::read(wasm).map(|_ty| ValType::VecType); - let reftype = RefType::read(wasm).map(ValType::RefType); - - numtype - .or(vectype) - .or(reftype) - .map_err(|_| Error::InvalidValType) + if let Ok(numtype) = NumType::read(wasm).map(ValType::NumType) { + return Ok(numtype); + }; + if let Ok(vectype) = VecType::read(wasm).map(|_ty| ValType::VecType) { + return Ok(vectype); + }; + if let Ok(reftype) = RefType::read(wasm).map(ValType::RefType) { + return Ok(reftype); + } + + Err(Error::InvalidValType) } fn read_unvalidated(wasm: &mut WasmReader) -> Self { - let numtype = NumType::read(wasm).map(ValType::NumType); - let vectype = VecType::read(wasm).map(|_ty| ValType::VecType); - let reftype = RefType::read(wasm).map(ValType::RefType); + if let Ok(numtype) = NumType::read(wasm).map(ValType::NumType) { + return numtype; + }; + if let Ok(vectype) = VecType::read(wasm).map(|_ty| ValType::VecType) { + return vectype; + }; + if let Ok(reftype) = RefType::read(wasm).map(ValType::RefType) { + return reftype; + } - numtype.or(vectype).or(reftype).unwrap_validated() + unreachable!() } } @@ -278,17 +323,25 @@ pub struct TableType { pub lim: Limits, } +// https://webassembly.github.io/spec/core/syntax/types.html#limits impl WasmReadable for TableType { fn read(wasm: &mut WasmReader) -> Result { let et = RefType::read(wasm)?; - let lim = Limits::read(wasm)?; + let mut lim = Limits::read(wasm)?; + if lim.max.is_none() { + lim.max = Some(u32::MAX) + }; + let table_type = Self { et, lim }; + trace!("Table: {:?}", table_type); Ok(Self { et, lim }) } fn read_unvalidated(wasm: &mut WasmReader) -> Self { let et = RefType::read_unvalidated(wasm); - let lim = Limits::read_unvalidated(wasm); - + let mut lim = Limits::read_unvalidated(wasm); + if lim.max.is_none() { + lim.max = Some(u32::MAX) + }; Self { et, lim } } } diff --git a/src/core/reader/types/opcode.rs b/src/core/reader/types/opcode.rs index 7f0f5aac..aa0e1a26 100644 --- a/src/core/reader/types/opcode.rs +++ b/src/core/reader/types/opcode.rs @@ -4,11 +4,14 @@ pub const END: u8 = 0x0B; pub const RETURN: u8 = 0x0F; pub const CALL: u8 = 0x10; pub const DROP: u8 = 0x1A; +pub const CALL_INDIRECT: u8 = 0x11; pub const LOCAL_GET: u8 = 0x20; pub const LOCAL_SET: u8 = 0x21; pub const LOCAL_TEE: u8 = 0x22; pub const GLOBAL_GET: u8 = 0x23; pub const GLOBAL_SET: u8 = 0x24; +pub const TABLE_GET: u8 = 0x25; +pub const TABLE_SET: u8 = 0x26; pub const I32_LOAD: u8 = 0x28; pub const I64_LOAD: u8 = 0x29; pub const F32_LOAD: u8 = 0x2A; @@ -162,6 +165,7 @@ pub const I64_REINTERPRET_F64: u8 = 0xBD; pub const F32_REINTERPRET_I32: u8 = 0xBE; pub const F64_REINTERPRET_I64: u8 = 0xBF; pub const REF_NULL: u8 = 0xD0; +pub const REF_IS_NULL: u8 = 0xD1; pub const REF_FUNC: u8 = 0xD2; pub const FC_EXTENSIONS: u8 = 0xFC; @@ -178,4 +182,10 @@ pub mod fc_extensions { pub const DATA_DROP: u8 = 0x09; pub const MEMORY_COPY: u8 = 0x0A; pub const MEMORY_FILL: u8 = 0x0B; + pub const TABLE_INIT: u8 = 0x0C; + pub const ELEM_DROP: u8 = 0x0D; + pub const TABLE_COPY: u8 = 0x0E; + pub const TABLE_GROW: u8 = 0x0F; + pub const TABLE_SIZE: u8 = 0x10; + pub const TABLE_FILL: u8 = 0x11; } diff --git a/src/execution/const_interpreter_loop.rs b/src/execution/const_interpreter_loop.rs index 135b2e96..d2d5c161 100644 --- a/src/execution/const_interpreter_loop.rs +++ b/src/execution/const_interpreter_loop.rs @@ -1,8 +1,13 @@ use crate::{ - assert_validated::UnwrapValidatedExt, core::reader::WasmReader, value_stack::Stack, NumType, - ValType, + assert_validated::UnwrapValidatedExt, + core::reader::WasmReader, + value::{FuncAddr, Ref}, + value_stack::Stack, + NumType, ValType, Value, }; +use super::store::FuncInst; + /// Execute a previosly-validated constant expression. These type of expressions are used for initializing global /// variables. /// @@ -28,6 +33,7 @@ pub(crate) fn run_const( mut wasm: WasmReader, stack: &mut Stack, _imported_globals: (), /*todo!*/ + funcs: &[FuncInst], ) { use crate::core::reader::types::opcode::*; loop { @@ -95,6 +101,17 @@ pub(crate) fn run_const( trace!("Constant instruction: i64.mul [{v1} {v2}] -> [{res}]"); stack.push_value(res.into()); } + REF_FUNC => { + // not unwrap validated because, guess what? not validated + let func_idx = wasm.read_var_u32().unwrap() as usize; + if func_idx >= funcs.len() { + panic!( + "Out of bounds ref.func ({func_idx}) access (max: {})", + funcs.len() + ); + } + stack.push_value(Value::Ref(Ref::Func(FuncAddr::new(Some(func_idx))))); + } other => { panic!("Unknown constant instruction {other:#x}, validation allowed an unimplemented instruction."); } diff --git a/src/execution/interpreter_loop.rs b/src/execution/interpreter_loop.rs index 2b85886f..20f84e6e 100644 --- a/src/execution/interpreter_loop.rs +++ b/src/execution/interpreter_loop.rs @@ -16,7 +16,7 @@ use alloc::vec::Vec; use crate::{ assert_validated::UnwrapValidatedExt, core::{ - indices::{DataIdx, FuncIdx, GlobalIdx, LocalIdx}, + indices::{DataIdx, FuncIdx, GlobalIdx, LocalIdx, TableIdx, TypeIdx}, reader::{ types::{memarg::MemArg, FuncType}, WasmReadable, WasmReader, @@ -24,9 +24,9 @@ use crate::{ }, locals::Locals, store::{DataInst, Store}, - value, + value::{self, FuncAddr, Ref}, value_stack::Stack, - Limits, NumType, RuntimeError, ValType, Value, + Limits, NumType, RefType, RuntimeError, ValType, Value, }; #[cfg(feature = "hooks")] @@ -116,6 +116,51 @@ pub(super) fn run( wasm.move_start_to(func_to_call_inst.code_expr) .unwrap_validated(); } + CALL_INDIRECT => { + let type_idx = wasm.read_var_u32().unwrap_validated() as TypeIdx; + let table_idx = wasm.read_var_u32().unwrap_validated() as TableIdx; + + let tab = store.tables.get(table_idx).unwrap_validated(); + let func_ty = types.get(type_idx).unwrap_validated(); + + let i: u32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); + + let r = tab + .elem + .get(i as usize) + .ok_or(RuntimeError::TableAccessOutOfBounds) + .and_then(|r| { + if r.is_null() { + trace!("table_idx ({table_idx}) --- element index in table ({i})"); + Err(RuntimeError::UninitializedElement) + } else { + Ok(r) + } + })?; + + let func_addr = match *r { + Ref::Func(func_addr) => func_addr.addr, + Ref::Extern(_) => unreachable!(), + }; + + let func_to_call_inst = store.funcs.get(func_addr).unwrap_validated(); + + let func_ty_actual_index = func_to_call_inst.ty; + + if type_idx != func_ty_actual_index { + return Err(RuntimeError::SignatureMismatch); + } + + let params = stack.pop_tail_iter(func_ty.params.valtypes.len()); + let remaining_locals = func_to_call_inst.locals.iter().cloned(); + + trace!("Instruction: call_indirect [{func_addr:?}]"); + let locals = Locals::new(params, remaining_locals); + stack.push_stackframe(func_addr, func_ty, locals, wasm.pc); + + wasm.move_start_to(func_to_call_inst.code_expr) + .unwrap_validated(); + } DROP => { stack.drop_value(); } @@ -138,6 +183,45 @@ pub(super) fn run( global.value = stack.pop_value(global.global.ty.ty) } + TABLE_GET => { + let table_idx = wasm.read_var_u32().unwrap_validated() as TableIdx; + + let tab = store.tables.get(table_idx).unwrap_validated(); + + let i: i32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); + + let val = tab + .elem + .get(i as usize) + .ok_or(RuntimeError::TableAccessOutOfBounds)?; + + stack.push_value((*val).into()); + trace!( + "Instruction: table.get '{}' [{}] -> [{}]", + table_idx, + i, + val + ); + } + TABLE_SET => { + let table_idx = wasm.read_var_u32().unwrap_validated() as TableIdx; + + let tab = &mut store.tables[table_idx]; + + let val: Ref = stack.pop_value(ValType::RefType(tab.ty.et)).into(); + let i: i32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); + + tab.elem + .get_mut(i as usize) + .ok_or(RuntimeError::TableAccessOutOfBounds) + .map(|r| *r = val)?; + trace!( + "Instruction: table.set '{}' [{} {}] -> []", + table_idx, + i, + val + ) + } I32_LOAD => { let memarg = MemArg::read_unvalidated(&mut wasm); let relative_address: u32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); @@ -1894,6 +1978,28 @@ pub(super) fn run( trace!("Instruction: f64.reinterpret_i64 [{v1}] -> [{res:.17}]"); stack.push_value(res.into()); } + REF_NULL => { + let reftype = RefType::read_unvalidated(&mut wasm); + + stack.push_value(Value::Ref(reftype.to_null_ref())); + trace!("Instruction: ref.null '{}' -> [{}]", reftype, reftype); + } + REF_IS_NULL => { + let rref = stack.pop_unknown_ref(); + let is_null = match rref { + Ref::Extern(rref) => rref.is_null, + Ref::Func(rref) => rref.is_null, + }; + + let res = if is_null { 1 } else { 0 }; + trace!("Instruction: ref.is_null [{}] -> [{}]", rref, res); + stack.push_value(Value::I32(res)); + } + // https://webassembly.github.io/spec/core/exec/instructions.html#xref-syntax-instructions-syntax-instr-ref-mathsf-ref-func-x + REF_FUNC => { + let func_idx = wasm.read_var_u32().unwrap_validated() as FuncIdx; + stack.push_value(Value::Ref(Ref::Func(FuncAddr::new(Some(func_idx))))); + } FC_EXTENSIONS => { // Should we call instruction hook here as well? Multibyte instruction let second_instr_byte = wasm.read_u8().unwrap_validated(); @@ -2161,6 +2267,190 @@ pub(super) fn run( trace!("Instruction: memory.fill"); } + // https://webassembly.github.io/spec/core/exec/instructions.html#xref-syntax-instructions-syntax-instr-table-mathsf-table-init-x-y + // https://webassembly.github.io/spec/core/binary/instructions.html#table-instructions + // in binary format it seems that elemidx is first ??????? + // this is ONLY for passive elements + TABLE_INIT => { + let elem_idx = wasm.read_var_u32().unwrap_validated() as usize; + let table_idx = wasm.read_var_u32().unwrap_validated() as usize; + + let n: i32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); // size + let s: i32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); // offset + let d: i32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); // dst + + let tab_len = store.tables.get(table_idx).unwrap_validated().len(); + let tab = store.tables.get_mut(table_idx).unwrap_validated(); + + let elem_len = if store.passive_elem_indexes.contains(&elem_idx) { + store.elements.get(elem_idx).unwrap_validated().len() + } else { + 0 + }; + + trace!( + "Instruction: table.init '{}' '{}' [{} {} {}] -> []", + elem_idx, + table_idx, + d, + s, + n + ); + + let final_src_offset = (s as usize) + .checked_add(n as usize) + .filter(|&res| res <= elem_len) + .ok_or(RuntimeError::TableAccessOutOfBounds)?; + + (d as usize) + .checked_add(n as usize) + .filter(|&res| res <= tab_len) + .ok_or(RuntimeError::TableAccessOutOfBounds)?; + + let elem = store.elements.get(elem_idx).unwrap_validated(); + + let dest = &mut tab.elem[d as usize..]; + let src = &elem.elem[s as usize..final_src_offset]; + dest[..src.len()].copy_from_slice(src); + } + ELEM_DROP => { + let elem_idx = wasm.read_var_u32().unwrap_validated() as usize; + + // WARN: i'm not sure if this is okay or not + store.elements.get_mut(elem_idx).unwrap_validated().elem = vec![]; + } + // https://webassembly.github.io/spec/core/exec/instructions.html#xref-syntax-instructions-syntax-instr-table-mathsf-table-copy-x-y + TABLE_COPY => { + let table_x_idx = wasm.read_var_u32().unwrap_validated() as usize; + let table_y_idx = wasm.read_var_u32().unwrap_validated() as usize; + + let tab_x_elem_len = store.tables[table_x_idx].elem.len(); + let tab_y_elem_len = store.tables[table_y_idx].elem.len(); + + let n: u32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); // size + let s: u32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); // source + let d: u32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); // destination + + let src_res = match s.checked_add(n) { + Some(res) => { + if res > tab_y_elem_len as u32 { + return Err(RuntimeError::TableAccessOutOfBounds); + } else { + res as usize + } + } + _ => return Err(RuntimeError::TableAccessOutOfBounds), + }; + + let dst_res = match d.checked_add(n) { + Some(res) => { + if res > tab_x_elem_len as u32 { + return Err(RuntimeError::TableAccessOutOfBounds); + } else { + res as usize + } + } + _ => return Err(RuntimeError::TableAccessOutOfBounds), + }; + + let dst = table_x_idx; + let src = table_y_idx; + + if table_x_idx == table_y_idx { + store.tables[table_x_idx] + .elem + .copy_within(s as usize..src_res, d as usize); // } + } else { + use core::cmp::Ordering::*; + let (src_table, dst_table) = match dst.cmp(&src) { + Greater => { + let (left, right) = store.tables.split_at_mut(dst); + (&left[src], &mut right[0]) + } + Less => { + let (left, right) = store.tables.split_at_mut(src); + (&right[0], &mut left[dst]) + } + Equal => unreachable!(), + }; + dst_table.elem[d as usize..dst_res] + .copy_from_slice(&src_table.elem[s as usize..src_res]); + } + + trace!( + "Instruction: table.copy '{}' '{}' [{} {} {}] -> []", + table_x_idx, + table_y_idx, + d, + s, + n + ); + } + TABLE_GROW => { + let table_idx = wasm.read_var_u32().unwrap_validated() as usize; + + let tab = store.tables.get_mut(table_idx).unwrap_validated(); + + let sz = tab.elem.len() as u32; + + let n: u32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); + let val = stack.pop_unknown_ref(); + + let max = tab.ty.lim.max.unwrap(); + + let final_size = sz.checked_add(n); + + match final_size { + Some(final_size) => { + if final_size > max { + stack.push_value(Value::I32(u32::MAX)) + } else { + tab.elem.extend(vec![val; n as usize]); + + stack.push_value(Value::I32(sz)); + } + } + _ => stack.push_value(Value::I32(u32::MAX)), + } + } + TABLE_SIZE => { + let table_idx = wasm.read_var_u32().unwrap_validated() as usize; + + let tab = store.tables.get(table_idx).unwrap_validated(); + + let sz = tab.elem.len() as u32; + + stack.push_value(Value::I32(sz)); + + trace!("Instruction: table.size '{}' [] -> [{}]", table_idx, sz); + } + TABLE_FILL => { + let table_idx = wasm.read_var_u32().unwrap_validated() as usize; + + let tab = store.tables.get_mut(table_idx).unwrap_validated(); + let ty = tab.ty.et; + + let n: u32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); // len + let val: Ref = stack.pop_value(ValType::RefType(ty)).into(); + let i: u32 = stack.pop_value(ValType::NumType(NumType::I32)).into(); // dst + + let end = (i as usize) + .checked_add(n as usize) + .ok_or(RuntimeError::TableAccessOutOfBounds)?; + + tab.elem + .get_mut(i as usize..end) + .ok_or(RuntimeError::TableAccessOutOfBounds)? + .fill(val); + + trace!( + "Instruction table.fill '{}' [{} {} {}] -> []", + table_idx, + i, + val, + n + ) + } _ => unreachable!(), } } diff --git a/src/execution/mod.rs b/src/execution/mod.rs index b4a79fdc..da3c286f 100644 --- a/src/execution/mod.rs +++ b/src/execution/mod.rs @@ -1,11 +1,13 @@ use alloc::string::ToString; +use alloc::vec; use alloc::vec::Vec; use const_interpreter_loop::run_const; use function_ref::FunctionRef; use interpreter_loop::run; use locals::Locals; -use store::DataInst; +use store::{DataInst, ElemInst, TableInst}; +use value::{ExternAddr, FuncAddr, Ref}; use value_stack::Stack; use crate::core::reader::types::export::{Export, ExportDesc}; @@ -342,6 +344,260 @@ where .collect() }; + // https://webassembly.github.io/spec/core/exec/modules.html#tables + let mut tables: Vec = validation_info + .tables + .iter() + .map(|ty| TableInst::new(*ty)) + .collect(); + + let mut passive_elem_indexes: Vec = vec![]; + // https://webassembly.github.io/spec/core/syntax/modules.html#element-segments + let elements: Vec = validation_info + .elements + .iter() + .enumerate() + .filter_map(|(i, el)| { + use crate::core::reader::types::element::*; + trace!("{:#?}", el); + match el.mode.clone() { + ElemMode::Passive => { + passive_elem_indexes.push(i); + // can be copied at runtime + Some(match el.ty() { + crate::RefType::FuncRef => { + ElemInst { + ty: el.ty(), + elem: match &el.init { + ElemItems::Exprs(_, sub_programs) => { + sub_programs.iter().map(|sub_program| { + let value = { + let mut wasm = WasmReader::new(validation_info.wasm); + wasm.move_start_to(*sub_program).unwrap_validated(); + let mut stack = Stack::new(); + // TODO: fully implement run_const + run_const(wasm, &mut stack, (), &function_instances); + let value = stack.peek_unknown_value(); + if value.is_none() { + panic!("No value on the stack for element segment offset"); + } + value.unwrap() + }; + + let offset: u32 = match value { + // Value::I32(val) => val, + // Value::I64(val) => { + // if val > u32::MAX as u64 { + // panic!("i64 value for data segment offset is out of reach") + // } + // val as u32 + // } + // INFO: no need to implement all of them, it's either i32 or i64, otherwise offset is WRONG + // INFO2: wait, we might need globals handling, now that I think about it, but make it return an u32 anyways + Value::Ref(rref) => { + match rref { + Ref::Func(func_addr) => func_addr.addr as u32, + Ref::Extern(_) => unreachable!() + } + } + _ => unreachable!() + }; + + Ref::Func(FuncAddr::new(Some(offset as usize))) + }).collect::>() + }, + ElemItems::RefFuncs(func_idxs) => func_idxs.iter().map(|func_idx| {Ref::Func(FuncAddr::new(Some(*func_idx as usize)))}).collect::>() + } + } + } + crate::RefType::ExternRef => { + ElemInst { + ty: el.ty(), + elem: match &el.init { + ElemItems::Exprs(_, sub_programs) => { + sub_programs.iter().map(|sub_program| { + let value = { + let mut wasm = WasmReader::new(validation_info.wasm); + wasm.move_start_to(*sub_program).unwrap_validated(); + let mut stack = Stack::new(); + // TODO: fully implement run_const + run_const(wasm, &mut stack, (), &function_instances); + let value = stack.peek_unknown_value(); + if value.is_none() { + panic!("No value on the stack for element segment offset"); + } + value.unwrap() + }; + + let offset: u32 = match value { + // Value::I32(val) => val, + // Value::I64(val) => { + // if val > u32::MAX as u64 { + // panic!("i64 value for data segment offset is out of reach") + // } + // val as u32 + // } + // INFO: no need to implement all of them, it's either i32 or i64, otherwise offset is WRONG + // INFO2: wait, we might need globals handling, now that I think about it, but make it return an u32 anyways + Value::Ref(rref) => { + match rref { + Ref::Func(_) => unreachable!(), + Ref::Extern(extern_addr) => extern_addr.addr as u32 + } + } + _ => { + unreachable!() + }, + }; + + Ref::Extern(ExternAddr::new(Some(offset as usize))) + }).collect::>() + }, + ElemItems::RefFuncs(_) => panic!("RefFuncs allowed only for Functions!") + } + } + }, + }) + }, + ElemMode::Active(active_elem) => { + // copies itself right now, when instantiating + let table_idx = active_elem.table as usize; + assert!(tables.len() > table_idx); + // if tables.len() <= table_idx { + // return Err(Error::TableIsNotDefined(table_idx)); + // } + + let value = { + let mut wasm = WasmReader::new(validation_info.wasm); + wasm.move_start_to(active_elem.offset).unwrap_validated(); + let mut stack = Stack::new(); + // TODO: fully implement run_const + run_const(wasm, &mut stack, (), &[]); + let value = stack.peek_unknown_value(); + if value.is_none() { + panic!("No value on the stack for element segment offset"); + } + value.unwrap() + }; + + // TODO: this shouldn't be a simple value, should it? I mean it can't be, but it can also be any type of ValType + // TODO: also, do we need to forcefully make it i32? + let offset: u32 = match value { + Value::I32(val) => val, + Value::I64(val) => { + if val > u32::MAX as u64 { + panic!("i64 value for data segment offset is out of reach") + } + val as u32 + } + // INFO: no need to implement all of them, it's either i32 or i64, otherwise offset is WRONG + // INFO2: wait, we might need globals handling, now that I think about it, but make it return an u32 anyways + _ => unreachable!(), + }; + let offset: usize = offset as usize; + + let el = match el.ty() { + crate::RefType::FuncRef => { + ElemInst { + ty: el.ty(), + elem: match &el.init { + // ElemItems::Exprs(_, _) => unreachable!(), + ElemItems::Exprs(_, exprs) => { + (*exprs).iter().map(|expr| { + let value = { + let mut wasm = WasmReader::new(validation_info.wasm); + wasm.move_start_to(*expr).unwrap_validated(); + let mut stack = Stack::new(); + // TODO: fully implement run_const + run_const(wasm, &mut stack, (), &[]); + let value = stack.peek_unknown_value(); + if value.is_none() { + panic!("No value on the stack for element segment offset"); + } + value.unwrap() + }; + + // TODO: this shouldn't be a simple value, should it? I mean it can't be, but it can also be any type of ValType + // TODO: also, do we need to forcefully make it i32? + let offset: u32 = match value { + Value::I32(val) => val, + Value::I64(val) => { + if val > u32::MAX as u64 { + panic!("i64 value for data segment offset is out of reach") + } + val as u32 + } + // INFO: no need to implement all of them, it's either i32 or i64, otherwise offset is WRONG + // INFO2: wait, we might need globals handling, now that I think about it, but make it return an u32 anyways + _ => unreachable!(), + }; + let offset: usize = offset as usize; + Ref::Func(FuncAddr::new(Some(offset))) + }).collect::>() + }, + ElemItems::RefFuncs(func_idxs) => func_idxs.iter().map(|func_idx| {Ref::Func(FuncAddr::new(Some(*func_idx as usize)))}).collect::>() + } + } + } + crate::RefType::ExternRef => { + ElemInst { + ty: el.ty(), + elem: match &el.init { + ElemItems::Exprs(_, exprs) => { + (*exprs).iter().map(|expr| { + let value = { + let mut wasm = WasmReader::new(validation_info.wasm); + wasm.move_start_to(*expr).unwrap_validated(); + let mut stack = Stack::new(); + // TODO: fully implement run_const + run_const(wasm, &mut stack, (), &[]); + let value = stack.peek_unknown_value(); + if value.is_none() { + panic!("No value on the stack for element segment offset"); + } + value.unwrap() + }; + + // TODO: this shouldn't be a simple value, should it? I mean it can't be, but it can also be any type of ValType + // TODO: also, do we need to forcefully make it i32? + let offset: u32 = match value { + Value::I32(val) => val, + Value::I64(val) => { + if val > u32::MAX as u64 { + panic!("i64 value for data segment offset is out of reach") + } + val as u32 + } + // INFO: no need to implement all of them, it's either i32 or i64, otherwise offset is WRONG + // INFO2: wait, we might need globals handling, now that I think about it, but make it return an u32 anyways + _ => unreachable!(), + }; + let offset: usize = offset as usize; + Ref::Extern(ExternAddr::new(Some(offset))) + }).collect::>() + }, + ElemItems::RefFuncs(_) => unreachable!() + } + } + }, + }; + let table = &mut tables[table_idx]; + + assert!(table.len() >= (offset + el.len())); + + el.elem.iter().enumerate().for_each(|(i, rref)| { + table.elem[i + offset] = *rref; + }); + + Some(el) + } + ElemMode::Declarative => { + None + } + } + }) + .collect(); + let mut memory_instances: Vec = validation_info .memories .iter() @@ -364,7 +620,7 @@ where let mut wasm = WasmReader::new(validation_info.wasm); wasm.move_start_to(active_data.offset).unwrap_validated(); let mut stack = Stack::new(); - run_const(wasm, &mut stack, ()); + run_const(wasm, &mut stack, (), &[]); let value = stack.peek_unknown_value(); if value.is_none() { panic!("No value on the stack for data segment offset"); @@ -415,7 +671,7 @@ where wasm.move_start_to(global.init_expr).unwrap_validated(); // We shouldn't need to clear the stack. If validation is correct, it will remain empty after execution. - run_const(wasm, &mut stack, ()); + run_const(wasm, &mut stack, (), &[]); let value = stack.pop_value(global.ty.ty); GlobalInst { @@ -431,6 +687,9 @@ where mems: memory_instances, globals: global_instances, data: data_sections, + tables, + elements, + passive_elem_indexes, } } } diff --git a/src/execution/store.rs b/src/execution/store.rs index a43025c2..4c2d1b49 100644 --- a/src/execution/store.rs +++ b/src/execution/store.rs @@ -2,6 +2,8 @@ use alloc::vec; use alloc::vec::Vec; use core::iter; +use crate::RefType; + use crate::core::indices::TypeIdx; use crate::core::reader::span::Span; use crate::core::reader::types::global::Global; @@ -18,6 +20,25 @@ pub struct Store { pub mems: Vec, pub globals: Vec, pub data: Vec, + pub tables: Vec, + pub elements: Vec, + pub passive_elem_indexes: Vec, +} + +#[derive(Clone, Debug)] +/// https://webassembly.github.io/spec/core/exec/runtime.html#element-instances +pub struct ElemInst { + pub ty: RefType, + pub elem: Vec, +} + +impl ElemInst { + pub fn len(&self) -> usize { + self.elem.len() + } + pub fn is_empty(&self) -> bool { + self.elem.is_empty() + } } pub struct FuncInst { @@ -26,12 +47,29 @@ pub struct FuncInst { pub code_expr: Span, } -#[allow(dead_code)] +#[derive(Debug)] pub struct TableInst { pub ty: TableType, pub elem: Vec, } +impl TableInst { + pub fn len(&self) -> usize { + self.elem.len() + } + + pub fn is_empty(&self) -> bool { + self.elem.is_empty() + } + + pub fn new(ty: TableType) -> Self { + Self { + ty, + elem: vec![Ref::default_from_ref_type(ty.et); ty.lim.min as usize], + } + } +} + pub struct MemInst { #[allow(warnings)] pub ty: MemType, diff --git a/src/execution/value.rs b/src/execution/value.rs index fadecdbd..a99dbe00 100644 --- a/src/execution/value.rs +++ b/src/execution/value.rs @@ -6,7 +6,7 @@ use core::ops::{Add, Div, Mul, Sub}; use crate::core::reader::types::{NumType, ValType}; use crate::execution::assert_validated::UnwrapValidatedExt; -use crate::unreachable_validated; +use crate::{unreachable_validated, Error, RefType, Result}; #[derive(Clone, Debug, Copy, PartialOrd)] pub struct F32(pub f32); @@ -261,14 +261,141 @@ pub enum Value { F64(F64), // F64, // V128, + Ref(Ref), } -#[derive(Clone, Debug, PartialEq)] -#[allow(dead_code)] +#[derive(Clone, Copy, Debug, PartialEq)] pub enum Ref { - Null, - // Func, - // Extern, + Func(FuncAddr), + Extern(ExternAddr), +} + +impl Ref { + pub fn default_from_ref_type(rref: RefType) -> Self { + match rref { + RefType::ExternRef => Self::Extern(ExternAddr::default()), + RefType::FuncRef => Self::Func(FuncAddr::default()), + } + } + + pub fn is_null(&self) -> bool { + match self { + Self::Extern(extern_addr) => extern_addr.is_null, + Self::Func(func_addr) => func_addr.is_null, + } + } + + pub fn is_specific_func(&self, func_id: u32) -> bool { + match self { + Self::Func(func_addr) => !func_addr.is_null && func_addr.addr == func_id as usize, + _ => unimplemented!(), + } + } +} + +impl Display for Ref { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + match self { + Ref::Func(func_addr) => write!(f, "FuncRef({})", func_addr), + Ref::Extern(extern_addr) => write!(f, "ExternRef({})", extern_addr), + } + } +} + +#[derive(Clone, Copy, PartialEq)] +pub struct FuncAddr { + pub is_null: bool, + // it is the idx of the function in the current module + pub addr: usize, +} + +impl Debug for FuncAddr { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + match self.is_null { + false => write!(f, "FuncAddr {{\n\taddr: {}\n}}", self.addr), + true => write!(f, "FuncAddr {{ NULL }}"), + } + } +} + +impl FuncAddr { + pub fn new(addr: Option) -> Self { + match addr { + None => Self::null(), + Some(u) => Self { + addr: u, + is_null: false, + }, + } + } + pub fn null() -> Self { + Self { + addr: 0, + is_null: true, + } + } +} + +impl Default for FuncAddr { + fn default() -> Self { + Self::null() + } +} + +impl Display for FuncAddr { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + if self.is_null { + write!(f, "FuncAddr {{ NULL }}") + } else { + write!(f, "FuncAddr {{ addr: {} }}", self.addr) + } + } +} + +#[derive(Clone, Copy, Debug, PartialEq)] +pub struct ExternAddr { + pub is_null: bool, + pub addr: usize, +} + +impl ExternAddr { + pub fn new(addr: Option) -> Self { + match addr { + None => Self::null(), + Some(u) => Self { + addr: u, + is_null: false, + }, + } + } + pub fn null() -> Self { + Self { + addr: 0, + is_null: true, + } + } +} + +impl Default for ExternAddr { + fn default() -> Self { + Self::null() + } +} + +impl Display for ExternAddr { + fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result { + if self.is_null { + write!(f, "ExternAddr {{ NULL }}") + } else { + write!(f, "ExternAddr {{ addr: () }}") // No actual addr, so printing () + } + } +} + +#[derive(Clone, Copy, Debug, PartialEq)] +pub enum RefValueTy { + Func, + Extern, } impl Value { @@ -278,6 +405,8 @@ impl Value { ValType::NumType(NumType::I64) => Self::I64(0), ValType::NumType(NumType::F32) => Self::F32(F32(0.0)), ValType::NumType(NumType::F64) => Self::F64(F64(0.0_f64)), + ValType::RefType(RefType::ExternRef) => Self::Ref(Ref::Extern(ExternAddr::null())), + ValType::RefType(RefType::FuncRef) => Self::Ref(Ref::Func(FuncAddr::null())), other => { todo!("cannot determine type for {other:?} because this value is not supported yet") } @@ -290,6 +419,10 @@ impl Value { Value::I64(_) => ValType::NumType(NumType::I64), Value::F32(_) => ValType::NumType(NumType::F32), Value::F64(_) => ValType::NumType(NumType::F64), + Value::Ref(rref) => match rref { + Ref::Extern(_) => ValType::RefType(RefType::ExternRef), + Ref::Func(_) => ValType::RefType(RefType::FuncRef), + }, } } } @@ -452,6 +585,44 @@ impl InteropValue for f64 { } } +#[derive(PartialEq, Debug, Copy, Clone)] +pub struct FuncRefForInteropValue { + rref: Ref, +} + +impl FuncRefForInteropValue { + pub fn new(rref: Ref) -> Result { + match rref { + Ref::Extern(_) => Err(Error::WrongRefTypeForInteropValue( + RefType::ExternRef, + RefType::FuncRef, + )), + Ref::Func(_) => Ok(Self { rref }), + } + } + + pub fn get_ref(&self) -> Ref { + self.rref + } +} + +impl InteropValue for FuncRefForInteropValue { + const TY: ValType = ValType::RefType(RefType::FuncRef); + + #[allow(warnings)] + fn into_value(self) -> Value { + Value::Ref(self.rref) + } + + #[allow(warnings)] + fn from_value(value: Value) -> Self { + match value { + Value::Ref(rref) => unsafe { FuncRefForInteropValue::new(rref).unwrap_unchecked() }, + _ => unreachable_validated!(), + } + } +} + impl InteropValueList for () { const TYS: &'static [ValType] = &[]; @@ -576,3 +747,18 @@ impl_value_conversion!(u64); impl_value_conversion!(i64); impl_value_conversion!(F32); impl_value_conversion!(F64); + +impl From for Value { + fn from(value: Ref) -> Self { + Self::Ref(value) + } +} + +impl From for Ref { + fn from(value: Value) -> Self { + match value { + Value::Ref(rref) => rref, + _ => unreachable!(), + } + } +} diff --git a/src/execution/value_stack.rs b/src/execution/value_stack.rs index 31030863..7bedb230 100644 --- a/src/execution/value_stack.rs +++ b/src/execution/value_stack.rs @@ -7,6 +7,8 @@ use crate::execution::value::Value; use crate::locals::Locals; use crate::unreachable_validated; +use super::value::Ref; + /// The stack at runtime containing /// 1. Values /// 2. Labels @@ -46,6 +48,31 @@ impl Stack { self.values.pop().unwrap_validated(); } + /// Pop a reference of unknown type from the value stack + pub fn pop_unknown_ref(&mut self) -> Ref { + // If there is at least one stack frame, we shall not pop values past the current + // stackframe. However, there is one legitimate reason to pop when there is **no** current + // stackframe: after the outermost function returns, to extract the final return values of + // this interpreter invocation. + debug_assert!( + if !self.frames.is_empty() { + self.values.len() > self.current_stackframe().value_stack_base_idx + } else { + true + }, + "can not pop values past the current stackframe" + ); + + let popped = self.values.pop().unwrap_validated(); + match popped.to_ty() { + ValType::RefType(_) => match popped { + Value::Ref(rref) => rref, + _ => unreachable!(), + }, + _ => unreachable_validated!(), + } + } + /// Pop a value of the given [ValType] from the value stack pub fn pop_value(&mut self, ty: ValType) -> Value { // If there is at least one stack frame, we shall not pop values past the current diff --git a/src/validation/code.rs b/src/validation/code.rs index 078e8057..7552429c 100644 --- a/src/validation/code.rs +++ b/src/validation/code.rs @@ -1,16 +1,22 @@ +use alloc::collections::btree_set::BTreeSet; +use alloc::vec; use alloc::vec::Vec; use core::iter; -use crate::core::indices::{DataIdx, FuncIdx, GlobalIdx, LocalIdx, MemIdx}; +use crate::core::indices::{ + DataIdx, ElemIdx, FuncIdx, GlobalIdx, LocalIdx, MemIdx, TableIdx, TypeIdx, +}; use crate::core::reader::section_header::{SectionHeader, SectionTy}; use crate::core::reader::span::Span; +use crate::core::reader::types::element::ElemType; use crate::core::reader::types::global::Global; use crate::core::reader::types::memarg::MemArg; -use crate::core::reader::types::{FuncType, MemType, NumType, ValType}; +use crate::core::reader::types::{FuncType, MemType, NumType, TableType, ValType}; use crate::core::reader::{WasmReadable, WasmReader}; use crate::validation_stack::ValidationStack; -use crate::{Error, Result}; +use crate::{Error, RefType, Result}; +#[allow(clippy::too_many_arguments)] pub fn validate_code_section( wasm: &mut WasmReader, section_header: SectionHeader, @@ -19,6 +25,9 @@ pub fn validate_code_section( globals: &[Global], memories: &[MemType], data_count: &Option, + tables: &[TableType], + elements: &[ElemType], + referenced_functions: &BTreeSet, ) -> Result> { assert_eq!(section_header.ty, SectionTy::Code); @@ -48,6 +57,9 @@ pub fn validate_code_section( type_idx_of_fn, memories, data_count, + tables, + elements, + referenced_functions, )?; // Check if there were unread trailing instructions after the last END @@ -96,6 +108,9 @@ fn read_instructions( type_idx_of_fn: &[usize], memories: &[MemType], data_count: &Option, + tables: &[TableType], + elements: &[ElemType], + referenced_functions: &BTreeSet, ) -> Result<()> { // TODO we must terminate only if both we saw the final `end` and when we consumed all of the code span loop { @@ -181,6 +196,37 @@ fn read_instructions( stack.push_valtype(*typ); } } + CALL_INDIRECT => { + let type_idx = wasm.read_var_u32()? as TypeIdx; + + let table_idx = wasm.read_var_u32()? as TableIdx; + + if tables.len() <= table_idx { + return Err(Error::TableIsNotDefined(table_idx)); + } + + let tab = &tables[table_idx]; + + if tab.et != RefType::FuncRef { + return Err(Error::WrongRefTypeForInteropValue(tab.et, RefType::FuncRef)); + } + + if type_idx >= fn_types.len() { + return Err(Error::FunctionTypeIsNotDefined(type_idx)); + } + + let func_ty = &fn_types[type_idx]; + + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + + for typ in func_ty.params.valtypes.iter().rev() { + stack.assert_pop_val_type(*typ)?; + } + + for typ in func_ty.returns.valtypes.iter() { + stack.push_valtype(*typ); + } + } // unreachable: [t1*] -> [t2*] UNREACHABLE => { stack.make_unspecified(); @@ -228,6 +274,30 @@ fn read_instructions( stack.assert_pop_val_type(global.ty.ty)?; } + TABLE_GET => { + let table_idx = wasm.read_var_u32()? as TableIdx; + + if tables.len() <= table_idx { + return Err(Error::TableIsNotDefined(table_idx)); + } + + let t = tables.get(table_idx).unwrap().et; + + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + stack.push_valtype(ValType::RefType(t)); + } + TABLE_SET => { + let table_idx = wasm.read_var_u32()? as TableIdx; + + if tables.len() <= table_idx { + return Err(Error::TableIsNotDefined(table_idx)); + } + + let t = tables.get(table_idx).unwrap().et; + + stack.assert_pop_ref_type(Some(t))?; + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + } I32_LOAD => { if memories.is_empty() { return Err(Error::MemoryIsNotDefined(0)); @@ -679,6 +749,36 @@ fn read_instructions( stack.push_valtype(ValType::NumType(NumType::F64)); } + REF_NULL => { + let reftype = RefType::read(wasm)?; + // at validation-time we don't really care if it's null or not + stack.push_valtype(ValType::RefType(reftype)); + } + + REF_IS_NULL => { + stack.assert_pop_ref_type(None)?; + stack.push_valtype(ValType::NumType(NumType::I32)); + } + + // TODO finish this + // https://webassembly.github.io/spec/core/valid/instructions.html#xref-syntax-instructions-syntax-instr-ref-mathsf-ref-func-x + REF_FUNC => { + // We will be making use of fn_types to check for length of possible functions + // Is this okay? + // I don't know + let funcs: Vec<()> = vec![(); fn_types.len()]; + let func_idx = wasm.read_var_u32()? as FuncIdx; + if func_idx >= funcs.len() { + return Err(Error::FunctionIsNotDefined(func_idx)); + } + + if !referenced_functions.contains(&(func_idx as u32)) { + return Err(Error::ReferencingAnUnreferencedFunction(func_idx)); + } + + stack.push_valtype(ValType::RefType(RefType::FuncRef)); + } + FC_EXTENSIONS => { let Ok(second_instr_byte) = wasm.read_u8() else { // TODO only do this if EOF @@ -766,6 +866,96 @@ fn read_instructions( stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; } + TABLE_INIT => { + let elem_idx = wasm.read_var_u32()? as ElemIdx; + let table_idx = wasm.read_var_u32()? as TableIdx; + + if tables.len() <= table_idx { + return Err(Error::TableIsNotDefined(table_idx)); + } + + let t1 = tables[table_idx].et; + + if elements.len() <= elem_idx { + return Err(Error::ElementIsNotDefined(elem_idx)); + } + + let t2 = elements[elem_idx].to_ref_type(); + + if t1 != t2 { + return Err(Error::DifferentRefTypes(t1, t2)); + } + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + // INFO: wasmtime checks for this value to be an index in the tables array, interesting + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + } + ELEM_DROP => { + let elem_idx = wasm.read_var_u32()? as ElemIdx; + + if elements.len() <= elem_idx { + return Err(Error::ElementIsNotDefined(elem_idx)); + } + } + TABLE_COPY => { + let table_x_idx = wasm.read_var_u32()? as TableIdx; + let table_y_idx = wasm.read_var_u32()? as TableIdx; + + if tables.len() <= table_x_idx { + return Err(Error::TableIsNotDefined(table_x_idx)); + } + + if tables.len() <= table_y_idx { + return Err(Error::TableIsNotDefined(table_y_idx)); + } + + let t1 = tables[table_x_idx].et; + let t2 = tables[table_y_idx].et; + + if t1 != t2 { + return Err(Error::DifferentRefTypes(t1, t2)); + } + + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + } + TABLE_GROW => { + let table_idx = wasm.read_var_u32()? as TableIdx; + + if tables.len() <= table_idx { + return Err(Error::TableIsNotDefined(table_idx)); + } + + let t = tables[table_idx].et; + + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + stack.assert_pop_ref_type(Some(t))?; + + stack.push_valtype(ValType::NumType(NumType::I32)); + } + TABLE_SIZE => { + let table_idx = wasm.read_var_u32()? as TableIdx; + + if tables.len() <= table_idx { + return Err(Error::TableIsNotDefined(table_idx)); + } + + stack.push_valtype(ValType::NumType(NumType::I32)); + } + TABLE_FILL => { + let table_idx = wasm.read_var_u32()? as TableIdx; + + if tables.len() <= table_idx { + return Err(Error::TableIsNotDefined(table_idx)); + } + + let t = tables[table_idx].et; + + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + stack.assert_pop_ref_type(Some(t))?; + stack.assert_pop_val_type(ValType::NumType(NumType::I32))?; + } _ => { return Err(Error::InvalidMultiByteInstr( first_instr_byte, diff --git a/src/validation/globals.rs b/src/validation/globals.rs index 2667809f..00012ef4 100644 --- a/src/validation/globals.rs +++ b/src/validation/globals.rs @@ -24,6 +24,8 @@ pub(super) fn validate_global_section( wasm, Some(ty.ty), Some(&[/* todo!(imported globals tpyes) */]), + // we can't refer to any functions + None, )?; Ok(Global { ty, init_expr }) diff --git a/src/validation/mod.rs b/src/validation/mod.rs index 96c925b2..024f4a19 100644 --- a/src/validation/mod.rs +++ b/src/validation/mod.rs @@ -1,9 +1,11 @@ +use alloc::collections::btree_set; use alloc::vec::Vec; use crate::core::indices::{FuncIdx, TypeIdx}; use crate::core::reader::section_header::{SectionHeader, SectionTy}; use crate::core::reader::span::Span; use crate::core::reader::types::data::DataSegment; +use crate::core::reader::types::element::ElemType; use crate::core::reader::types::export::Export; use crate::core::reader::types::global::Global; use crate::core::reader::types::import::Import; @@ -24,7 +26,6 @@ pub struct ValidationInfo<'bytecode> { #[allow(dead_code)] pub(crate) imports: Vec, pub(crate) functions: Vec, - #[allow(dead_code)] pub(crate) tables: Vec, pub(crate) memories: Vec, pub(crate) globals: Vec, @@ -34,6 +35,7 @@ pub struct ValidationInfo<'bytecode> { pub(crate) data: Vec, /// The start function which is automatically executed during instantiation pub(crate) start: Option, + pub(crate) elements: Vec, } pub fn validate(wasm: &[u8]) -> Result { @@ -120,9 +122,12 @@ pub fn validate(wasm: &[u8]) -> Result { while (skip_section(&mut wasm, &mut header)?).is_some() {} - let _: Option<()> = handle_section(&mut wasm, &mut header, SectionTy::Element, |_, _| { - todo!("element section not yet supported") - })?; + let mut referenced_functions = btree_set::BTreeSet::new(); + let elements: Vec = + handle_section(&mut wasm, &mut header, SectionTy::Element, |wasm, _| { + ElemType::read_from_wasm(wasm, &functions, &mut referenced_functions, tables.len()) + })? + .unwrap_or_default(); while (skip_section(&mut wasm, &mut header)?).is_some() {} @@ -149,6 +154,9 @@ pub fn validate(wasm: &[u8]) -> Result { &globals, &memories, &data_count, + &tables, + &elements, + &referenced_functions, ) })? .unwrap_or_default(); @@ -187,6 +195,7 @@ pub fn validate(wasm: &[u8]) -> Result { func_blocks, data: data_section, start, + elements, }) } diff --git a/src/validation/read_constant_expression.rs b/src/validation/read_constant_expression.rs index 93f59654..a0e7f0f1 100644 --- a/src/validation/read_constant_expression.rs +++ b/src/validation/read_constant_expression.rs @@ -85,6 +85,7 @@ pub fn read_constant_instructions( wasm: &mut WasmReader, this_global_valtype: Option, _globals_ty: Option<&[GlobalType]>, + funcs: Option<&[usize]>, ) -> Result { let start_pc = wasm.pc; @@ -131,7 +132,22 @@ pub fn read_constant_instructions( todo!(); } REF_FUNC => { - wasm.read_var_u32().unwrap(); + let func_idx = wasm.read_var_u32().unwrap() as usize; + match funcs { + Some(funcs) => { + if func_idx >= funcs.len() { + panic!( + "Out of bounds ref.func ({func_idx}) access (max: {})", + funcs.len() + ); + } + } + None => { + panic!("Out of bounds ref.func ({func_idx}) access (max: 0)"); + } + } + + stack.push_valtype(ValType::RefType(crate::RefType::FuncRef)); } _ => return Err(Error::InvalidInstr(first_instr_byte)), } diff --git a/src/validation/validation_stack.rs b/src/validation/validation_stack.rs index 7a327ddd..27890814 100644 --- a/src/validation/validation_stack.rs +++ b/src/validation/validation_stack.rs @@ -7,7 +7,7 @@ use super::Result; use alloc::vec::Vec; -use crate::{Error, ValType}; +use crate::{Error, RefType, ValType}; #[derive(Debug, PartialEq, Eq)] pub(super) struct ValidationStack { @@ -74,6 +74,27 @@ impl ValidationStack { .ok_or(Error::InvalidValidationStackValType(None)) } + pub(super) fn assert_pop_ref_type(&mut self, expected_ty: Option) -> Result<()> { + let val = self.pop()?; + match val { + ValidationStackEntry::Val(v) => match v { + ValType::RefType(ref_type) => match expected_ty { + None => Ok(()), + Some(expected_ty) => { + if expected_ty == ref_type { + Ok(()) + } else { + Err(Error::DifferentRefTypes(ref_type, expected_ty)) + } + } + }, + _ => Err(Error::ExpectedARefType(v)), + }, + ValidationStackEntry::UnspecifiedValTypes => Err(Error::FoundUnspecifiedValTypes), + ValidationStackEntry::Label(li) => Err(Error::FoundLabel(li.kind)), + } + } + /// Assert the top-most [`ValidationStackEntry`] is a specific [`ValType`], after popping it from the [`ValidationStack`] /// /// # Returns diff --git a/tests/table.rs b/tests/table.rs new file mode 100644 index 00000000..c51bc821 --- /dev/null +++ b/tests/table.rs @@ -0,0 +1,257 @@ +/* +# This file incorporates code from the WebAssembly testsuite, originally +# available at https://github.com/WebAssembly/testsuite. +# +# The original code is licensed under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +*/ +use wasm::value::{FuncRefForInteropValue, Ref}; +use wasm::Error as GeneralError; +use wasm::{validate, RuntimeInstance}; + +macro_rules! get_func { + ($instance:ident, $func_name:expr) => { + &$instance.get_function_by_name("", $func_name).unwrap() + }; +} + +#[test_log::test] +fn table_basic() { + let w = r#" + (module (table 0 funcref)) + (module (table 1 funcref)) + (module (table 0 0 funcref)) + (module (table 0 1 funcref)) + (module (table 1 256 funcref)) + (module (table 0 65536 funcref)) + (module (table 0 0xffff_ffff funcref)) +"# + .split("\n") + .map(|el| el.trim()) + .filter(|el| !el.is_empty()) + .collect::>(); + + w.iter().for_each(|wat| { + let wasm_bytes = wat::parse_str(wat).unwrap(); + let validation_info = validate(&wasm_bytes).expect("validation failed"); + RuntimeInstance::new(&validation_info).expect("instantiation failed"); + }); +} + +#[test_log::test] +fn table_basic_2() { + let w = r#" + (module (table 0 funcref) (table 0 funcref)) + (module (table (import "spectest" "table") 0 funcref) (table 0 funcref)) +"# + .split("\n") + .map(|el| el.trim()) + .filter(|el| !el.is_empty()) + .collect::>(); + + w.iter().for_each(|wat| { + let wasm_bytes = wat::parse_str(wat).unwrap(); + let validation_info = validate(&wasm_bytes).expect("validation failed"); + RuntimeInstance::new(&validation_info).expect("instantiation failed"); + }); +} + +#[test_log::test] +fn unknown_table() { + let w = r#" + (module (elem (i32.const 0))) + (module (elem (i32.const 0) $f) (func $f)) +"# + .split("\n") + .map(|el| el.trim()) + .filter(|el| !el.is_empty()) + .collect::>(); + + w.iter().for_each(|wat| { + let wasm_bytes = wat::parse_str(wat).unwrap(); + let validation_info = validate(&wasm_bytes); + assert!(validation_info.err().unwrap() == GeneralError::UnknownTable); + }); +} + +#[test_log::test] +fn table_size_minimum_must_not_be_greater_than_maximum() { + let w = r#" + (module (table 1 0 funcref)) + (module (table 0xffff_ffff 0 funcref)) +"# + .split("\n") + .map(|el| el.trim()) + .filter(|el| !el.is_empty()) + .collect::>(); + + w.iter().for_each(|wat| { + let wasm_bytes = wat::parse_str(wat).unwrap(); + let validation_info = validate(&wasm_bytes); + assert!(validation_info.err().unwrap() == GeneralError::InvalidLimit); + }); +} + +#[test_log::test] +fn table_elem_test() { + let w = r#" + (module + (table 2 funcref) + (elem (i32.const 0) $f1 $f3) + (func $f1 (result i32) + i32.const 42) + (func $f2 (result i32) + i32.const 13) + (func $f3 (result i64) + i64.const 13) + (func $f4 (result i32) + i32.const 13) + )"#; + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let instance = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + let table = &instance.store.tables[0]; + assert!(table.len() == 2); + let wanted: [usize; 2] = [0, 2]; + table + .elem + .iter() + .enumerate() + .for_each(|(i, rref)| match *rref { + wasm::value::Ref::Extern(_) => panic!(), + wasm::value::Ref::Func(func_addr) => { + assert!(!func_addr.is_null); + assert!(wanted[i] == func_addr.addr) + } + }); + // assert!(instance.store.tables) +} + +#[test_log::test] +fn table_get_set_test() { + let w = r#" +(module + (table $t3 3 funcref) + (elem (table $t3) (i32.const 1) func $dummy) + (elem func $dummypassive) + (func $dummypassive) + (func $dummy) + (func (export "init") + (table.set $t3 (i32.const 2) (table.get $t3 (i32.const 1))) + ) + (func $f3 (export "get-funcref") (param $i i32) (result funcref) + (table.get $t3 (local.get $i)) + ) +) + "#; + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let get_funcref = get_func!(i, "get-funcref"); + let init = get_func!(i, "init"); + + // assert the function at index 1 is a FuncRef and is NOT null + { + let funcref = i + .invoke::(get_funcref, 1) + .unwrap(); + + let rref = funcref.get_ref(); + + match rref { + Ref::Func(funcaddr) => { + assert!(!funcaddr.is_null) + } + _ => panic!("Expected a FuncRef"), + } + } + + // assert the function at index 2 is a FuncRef and is null + { + let funcref = i + .invoke::(get_funcref, 2) + .unwrap(); + + let rref = funcref.get_ref(); + + match rref { + Ref::Func(funcaddr) => { + assert!(funcaddr.is_null) + } + _ => panic!("Expected a FuncRef"), + } + } + + // set the function at index 2 the same as the one at index 1 + i.invoke::<(), ()>(init, ()).unwrap(); + // assert the function at index 2 is a FuncRef and is NOT null + { + let funcref = i + .invoke::(get_funcref, 2) + .unwrap(); + + let rref = funcref.get_ref(); + + match rref { + Ref::Func(funcaddr) => { + assert!(!funcaddr.is_null) + } + _ => panic!("Expected a FuncRef"), + } + } +} + +// (assert_malformed +// (module quote "(table 0x1_0000_0000 funcref)") +// "i32 constant out of range" +// ) +// (assert_malformed +// (module quote "(table 0x1_0000_0000 0x1_0000_0000 funcref)") +// "i32 constant out of range" +// ) +// (assert_malformed +// (module quote "(table 0 0x1_0000_0000 funcref)") +// "i32 constant out of range" +// ) + +// ;; Duplicate table identifiers + +// #[test_log::test] +// fn duplicate_table() { +// let w = r#" +// (module quote "(table $foo 1 funcref)" "(table $foo 1 funcref)") +// "# +// .split("\n") +// .map(|el| el.trim()) +// .filter(|el| !el.is_empty()) +// .collect::>(); + +// w.iter().for_each(|wat| { +// let wasm_bytes = wat::parse_str(wat).unwrap(); +// let validation_info = validate(&wasm_bytes); +// // assert!(validation_info.err().unwrap() == GeneralError::InvalidLimit); +// }); +// } + +// (assert_malformed (module quote +// "(table $foo 1 funcref)" +// "(table $foo 1 funcref)") +// "duplicate table") +// (assert_malformed (module quote +// "(import \"\" \"\" (table $foo 1 funcref))" +// "(table $foo 1 funcref)") +// "duplicate table") +// (assert_malformed (module quote +// "(import \"\" \"\" (table $foo 1 funcref))" +// "(import \"\" \"\" (table $foo 1 funcref))") +// "duplicate table") diff --git a/tests/table_fill.rs b/tests/table_fill.rs new file mode 100644 index 00000000..9e615a7d --- /dev/null +++ b/tests/table_fill.rs @@ -0,0 +1,313 @@ +/* +# This file incorporates code from the WebAssembly testsuite, originally +# available at https://github.com/WebAssembly/testsuite. +# +# The original code is licensed under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +*/ +use wasm::value::{FuncAddr, FuncRefForInteropValue, Ref}; +use wasm::{validate, RuntimeError, RuntimeInstance}; + +macro_rules! get_func { + ($instance:ident, $func_name:expr) => { + &$instance.get_function_by_name("", $func_name).unwrap() + }; +} + +macro_rules! is_specific_func { + ($self:expr, $func_id:expr) => { + match $self { + Ref::Func(func_addr) => !func_addr.is_null && func_addr.addr == $func_id as usize, + _ => unimplemented!(), + } + }; +} + +#[test_log::test] +fn table_fill_test() { + let w = r#" + (module + (table $t 10 funcref) + + (func (export "fill") (param $i i32) (param $r funcref) (param $n i32) + (table.fill $t (local.get $i) (local.get $r) (local.get $n)) + ) + + (func (export "fill-abbrev") (param $i i32) (param $r funcref) (param $n i32) + (table.fill $t (local.get $i) (local.get $r) (local.get $n)) + ) + + (func (export "get") (param $i i32) (result funcref) + (table.get $t (local.get $i)) + ) + ) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let get = get_func!(i, "get"); + let fill = get_func!(i, "fill"); + let fill_abbrev = get_func!(i, "fill-abbrev"); + + assert!(i + .invoke::(get, 1) + .unwrap() + .get_ref() + .is_null()); + assert!(i + .invoke::(get, 2) + .unwrap() + .get_ref() + .is_null()); + assert!(i + .invoke::(get, 3) + .unwrap() + .get_ref() + .is_null()); + assert!(i + .invoke::(get, 4) + .unwrap() + .get_ref() + .is_null()); + assert!(i + .invoke::(get, 5) + .unwrap() + .get_ref() + .is_null()); + + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill, + ( + 2, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(1)))).unwrap(), + 3, + ), + ) + .unwrap(); + + assert!(i + .invoke::(get, 1) + .unwrap() + .get_ref() + .is_null()); + assert!(is_specific_func!( + i.invoke::(get, 2) + .unwrap() + .get_ref(), + 1 + )); + assert!(is_specific_func!( + i.invoke::(get, 3) + .unwrap() + .get_ref(), + 1 + )); + assert!(is_specific_func!( + i.invoke::(get, 4) + .unwrap() + .get_ref(), + 1 + )); + assert!(i + .invoke::(get, 5) + .unwrap() + .get_ref() + .is_null()); + + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill, + ( + 4, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap(), + 2, + ), + ) + .unwrap(); + + assert!(is_specific_func!( + i.invoke::(get, 3) + .unwrap() + .get_ref(), + 1 + )); + assert!(is_specific_func!( + i.invoke::(get, 4) + .unwrap() + .get_ref(), + 2 + )); + assert!(is_specific_func!( + i.invoke::(get, 5) + .unwrap() + .get_ref(), + 2 + )); + assert!(i + .invoke::(get, 6) + .unwrap() + .get_ref() + .is_null()); + + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill, + ( + 4, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(3)))).unwrap(), + 0, + ), + ) + .unwrap(); + + assert!(is_specific_func!( + i.invoke::(get, 3) + .unwrap() + .get_ref(), + 1 + )); + assert!(is_specific_func!( + i.invoke::(get, 4) + .unwrap() + .get_ref(), + 2 + )); + assert!(is_specific_func!( + i.invoke::(get, 5) + .unwrap() + .get_ref(), + 2 + )); + + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill, + ( + 8, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(4)))).unwrap(), + 2, + ), + ) + .unwrap(); + + assert!(i + .invoke::(get, 7) + .unwrap() + .get_ref() + .is_null()); + assert!(is_specific_func!( + i.invoke::(get, 8) + .unwrap() + .get_ref(), + 4 + )); + assert!(is_specific_func!( + i.invoke::(get, 9) + .unwrap() + .get_ref(), + 4 + )); + + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill_abbrev, + ( + 9, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::null())).unwrap(), + 1, + ), + ) + .unwrap(); + assert!(is_specific_func!( + i.invoke::(get, 8) + .unwrap() + .get_ref(), + 4 + )); + assert!(i + .invoke::(get, 9) + .unwrap() + .get_ref() + .is_null()); + + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill, + ( + 10, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(5)))).unwrap(), + 0, + ), + ) + .unwrap(); + assert!(i + .invoke::(get, 9) + .unwrap() + .get_ref() + .is_null()); + + assert!( + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill, + ( + 8, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(6)))).unwrap(), + 3 + ) + ) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + + assert!(i + .invoke::(get, 7) + .unwrap() + .get_ref() + .is_null()); + assert!(is_specific_func!( + i.invoke::(get, 8) + .unwrap() + .get_ref(), + 4 + )); + assert!(i + .invoke::(get, 9) + .unwrap() + .get_ref() + .is_null()); + + assert!( + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill, + ( + 11, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::null())).unwrap(), + 0 + ) + ) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + + assert!( + i.invoke::<(i32, FuncRefForInteropValue, i32), ()>( + fill, + ( + 11, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::null())).unwrap(), + 10 + ) + ) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); +} diff --git a/tests/table_get.rs b/tests/table_get.rs new file mode 100644 index 00000000..6ffe7b63 --- /dev/null +++ b/tests/table_get.rs @@ -0,0 +1,122 @@ +/* +# This file incorporates code from the WebAssembly testsuite, originally +# available at https://github.com/WebAssembly/testsuite. +# +# The original code is licensed under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +*/ + +use wasm::{ + validate, + value::{FuncAddr, FuncRefForInteropValue, Ref}, + RuntimeError, RuntimeInstance, +}; + +macro_rules! get_func { + ($instance:ident, $func_name:expr) => { + &$instance.get_function_by_name("", $func_name).unwrap() + }; +} + +macro_rules! assert_result { + ($instance:expr, $func:expr, $arg:expr, $result:expr) => { + assert_eq!($result, $instance.invoke($func, $arg).unwrap()); + }; +} + +macro_rules! assert_error { + ($instance:expr, $func:expr, $arg:expr, $ret_type:ty, $invoke_param_type:ty, $invoke_return_type:ty, $err_type:expr) => { + let val: $ret_type = + $instance.invoke::<$invoke_param_type, $invoke_return_type>($func, $arg); + assert!(val.is_err()); + assert!(val.unwrap_err() == $err_type); + }; +} + +#[test_log::test] +fn table_funcref_test() { + let w = r#" +(module + (table $t2 2 funcref) + (table $t3 3 funcref) + (elem (table $t3) (i32.const 1) func $dummy) + (func $dummy) + (func (export "init") (param $r funcref) + (table.set $t2 (i32.const 1) (local.get $r)) + (table.set $t3 (i32.const 2) (table.get $t3 (i32.const 1))) + ) + (func (export "get-funcref") (param $i i32) (result funcref) + (table.get (local.get $i)) + ) + (func $f3 (export "get-funcref-2") (param $i i32) (result funcref) + (table.get $t3 (local.get $i)) + ) + (func (export "is_null-funcref") (param $i i32) (result i32) + (ref.is_null (call $f3 (local.get $i))) + ) +) + "#; + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let init = get_func!(i, "init"); + let get_funcref = get_func!(i, "get-funcref"); + let get_funcref_2 = get_func!(i, "get-funcref-2"); + let is_null_funcref = get_func!(i, "is_null-funcref"); + + let func_ref: Ref = Ref::Func(FuncAddr::new(Some(1))); + i.invoke::(init, FuncRefForInteropValue::new(func_ref).unwrap()) + .unwrap(); + + assert_result!( + i, + get_funcref, + 0, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::null())).unwrap() + ); + assert_result!( + i, + get_funcref, + 1, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(1)))).unwrap() + ); + assert_result!( + i, + get_funcref_2, + 0, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::null())).unwrap() + ); + assert_result!(i, is_null_funcref, 1, 0); + assert_result!(i, is_null_funcref, 2, 0); + + assert_error!(i, get_funcref, 2, Result, i32, FuncRefForInteropValue, RuntimeError::TableAccessOutOfBounds); + assert_error!(i, get_funcref_2, 3, Result, i32, FuncRefForInteropValue, RuntimeError::TableAccessOutOfBounds); + assert_error!(i, get_funcref, -1, Result, i32, FuncRefForInteropValue, RuntimeError::TableAccessOutOfBounds); + assert_error!(i, get_funcref_2, -1, Result, i32, FuncRefForInteropValue, RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_type_error_test() { + let invalid_modules = vec![ + r#"(module (table $t 10 funcref) (func $type-index-empty-vs-i32 (result funcref) (table.get $t)))"#, + r#"(module (table $t 10 funcref) (func $type-index-f32-vs-i32 (result funcref) (table.get $t (f32.const 1))))"#, + r#"(module (table $t 10 funcref) (func $type-result-funcref-vs-empty (table.get $t (i32.const 0))))"#, + r#"(module (table $t 10 funcref) (func $type-result-funcref-vs-funcref (result externref) (table.get $t (i32.const 1))))"#, + r#"(module (table $t1 1 funcref) (table $t2 1 externref) (func $type-result-externref-vs-funcref-multi (result funcref) (table.get $t2 (i32.const 0))))"#, + ]; + + for module in invalid_modules { + let wasm_bytes = wat::parse_str(module).unwrap(); + assert!(validate(&wasm_bytes).is_err()); + } +} diff --git a/tests/table_grow.rs b/tests/table_grow.rs new file mode 100644 index 00000000..6df4121d --- /dev/null +++ b/tests/table_grow.rs @@ -0,0 +1,422 @@ +/* +# This file incorporates code from the WebAssembly testsuite, originally +# available at https://github.com/WebAssembly/testsuite. +# +# The original code is licensed under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +*/ +use wasm::value::{FuncAddr, FuncRefForInteropValue, Ref}; +use wasm::{validate, RuntimeError, RuntimeInstance}; + +macro_rules! get_func { + ($instance:ident, $func_name:expr) => { + &$instance.get_function_by_name("", $func_name).unwrap() + }; +} + +macro_rules! assert_result { + ($instance:expr, $func:expr, $arg:expr, $result:expr) => { + assert_eq!($result, $instance.invoke($func, $arg).unwrap()); + }; +} + +#[test_log::test] +fn table_grow_test() { + let w = r#" + (module + (table $t 0 funcref) + (func (export "get") (param $i i32) (result funcref) (table.get $t (local.get $i))) + (func (export "set") (param $i i32) (param $r funcref) (table.set $t (local.get $i) (local.get $r))) + (func (export "grow") (param $sz i32) (param $init funcref) (result i32) + (table.grow $t (local.get $init) (local.get $sz)) + ) + (func (export "grow-abbrev") (param $sz i32) (param $init funcref) (result i32) + (table.grow (local.get $init) (local.get $sz)) + ) + (func (export "size") (result i32) (table.size $t)) + ) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let get = get_func!(i, "get"); + let set = get_func!(i, "set"); + let grow = get_func!(i, "grow"); + let grow_abbrev = get_func!(i, "grow-abbrev"); + let size = get_func!(i, "size"); + + assert!(i.invoke::<(), i32>(size, ()).unwrap() == 0); + assert!( + i.invoke::<(i32, FuncRefForInteropValue), ()>( + set, + ( + 0, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap() + ) + ) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + assert!( + i.invoke::(get, 0) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + + assert!( + i.invoke::<(i32, FuncRefForInteropValue), i32>( + grow, + ( + 1, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::null())).unwrap() + ) + ) + .unwrap() + == 0 + ); + assert!(i.invoke::<(), i32>(size, ()).unwrap() == 1); + assert!(i + .invoke::(get, 0) + .unwrap() + .get_ref() + .is_null()); + assert!(i + .invoke::<(i32, FuncRefForInteropValue), ()>( + set, + ( + 0, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap() + ) + ) + .is_ok()); + assert!( + i.invoke::(get, 0).unwrap() + == FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap() + ); + assert!( + i.invoke::<(i32, FuncRefForInteropValue), ()>( + set, + ( + 1, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap() + ) + ) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + assert!( + i.invoke::(get, 1) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + + assert!( + i.invoke::<(i32, FuncRefForInteropValue), i32>( + grow_abbrev, + ( + 4, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(3)))).unwrap() + ) + ) + .unwrap() + == 1 + ); + assert!(i.invoke::<(), i32>(size, ()).unwrap() == 5); + assert!( + i.invoke::(get, 0).unwrap() + == FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap() + ); + assert!(i + .invoke::<(i32, FuncRefForInteropValue), ()>( + set, + ( + 0, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap() + ) + ) + .is_ok()); + assert!( + i.invoke::(get, 0).unwrap() + == FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap() + ); + assert!( + i.invoke::(get, 1).unwrap() + == FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(3)))).unwrap() + ); + assert!( + i.invoke::(get, 4).unwrap() + == FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(3)))).unwrap() + ); + assert!(i + .invoke::<(i32, FuncRefForInteropValue), ()>( + set, + ( + 4, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(4)))).unwrap() + ) + ) + .is_ok()); + assert!( + i.invoke::(get, 4).unwrap() + == FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(4)))).unwrap() + ); + assert!( + i.invoke::<(i32, FuncRefForInteropValue), ()>( + set, + ( + 5, + FuncRefForInteropValue::new(Ref::Func(FuncAddr::new(Some(2)))).unwrap() + ) + ) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + assert!( + i.invoke::(get, 5) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); +} + +// ... existing code ... + +#[test_log::test] +fn table_grow_outside_i32_range() { + let w = r#" + (module + (table $t 0x10 funcref) + (elem declare func $f) + (func $f (export "grow") (result i32) + (table.grow $t (ref.func $f) (i32.const 0xffff_fff0)) + ) + ) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let grow = get_func!(i, "grow"); + assert_eq!(i.invoke::<(), i32>(grow, ()).unwrap(), -1); +} + +#[test_log::test] +fn table_grow_unlimited() { + let w = r#" + (module + (table $t 0 funcref) + (func (export "grow") (param i32) (result i32) + (table.grow $t (ref.null func) (local.get 0)) + ) + ) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let grow = get_func!(i, "grow"); + assert_result!(i, grow, 0, 0); + assert_result!(i, grow, 1, 0); + assert_result!(i, grow, 0, 1); + assert_result!(i, grow, 2, 1); + assert_result!(i, grow, 800, 3); +} + +#[test_log::test] +fn table_grow_with_max() { + let w = r#" + (module + (table $t 0 10 funcref) + (func (export "grow") (param i32) (result i32) + (table.grow $t (ref.null func) (local.get 0)) + ) + ) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let grow = get_func!(i, "grow"); + assert_result!(i, grow, 0, 0); + assert_result!(i, grow, 1, 0); + assert_result!(i, grow, 1, 1); + assert_result!(i, grow, 2, 2); + assert_result!(i, grow, 6, 4); + assert_result!(i, grow, 0, 10); + assert_result!(i, grow, 1, -1); + assert_result!(i, grow, 0x10000, -1); +} + +#[ignore = "control flow not yet implemented"] +#[test_log::test] +fn table_grow_check_null() { + let w = r#" + (module + (table $t 10 funcref) + (func (export "grow") (param i32) (result i32) + (table.grow $t (ref.null func) (local.get 0)) + ) + (elem declare func 1) + (func (export "check-table-null") (param i32 i32) (result funcref) + (local funcref) + (local.set 2 (ref.func 1)) + (block + (loop + (local.set 2 (table.get $t (local.get 0))) + (br_if 1 (i32.eqz (ref.is_null (local.get 2)))) + (br_if 1 (i32.ge_u (local.get 0) (local.get 1))) + (local.set 0 (i32.add (local.get 0) (i32.const 1))) + (br_if 0 (i32.le_u (local.get 0) (local.get 1))) + ) + ) + (local.get 2) + ) + ) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let grow = get_func!(i, "grow"); + let check_table_null = get_func!(i, "check-table-null"); + + assert_eq!( + i.invoke::<(i32, i32), FuncRefForInteropValue>(check_table_null, (0, 9)) + .unwrap(), + FuncRefForInteropValue::new(Ref::Func(FuncAddr::null())).unwrap() + ); + assert_result!(i, grow, 10, 10); + assert_eq!( + i.invoke::<(i32, i32), FuncRefForInteropValue>(check_table_null, (0, 19)) + .unwrap(), + FuncRefForInteropValue::new(Ref::Func(FuncAddr::null())).unwrap() + ); +} + +#[test_log::test] +fn table_grow_with_exported_table_test() { + // First module - Target with exported table + let target_wat = r#" + (module + (table (export "table") 1 funcref) + (func (export "grow") (result i32) + (table.grow (ref.null func) (i32.const 1)) + ) + ) + "#; + + let wasm_bytes = wat::parse_str(target_wat).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut target_instance = + RuntimeInstance::new(&validation_info).expect("target instantiation failed"); + + let grow = get_func!(target_instance, "grow"); + assert_result!(target_instance, grow, (), 1); +} + +// #[test_log::test] +// fn table_grow_import_table_from_first_module() { +// let import1_wat = r#" +// (module +// (table (export "table") (import "grown-table" "table") 2 funcref) +// (func (export "grow") (result i32) +// (table.grow (ref.null func) (i32.const 1)) +// ) +// ) +// "#; + +// let wasm_bytes = wat::parse_str(import1_wat).unwrap(); +// let validation_info = validate(&wasm_bytes).unwrap(); +// let mut import1_instance = RuntimeInstance::new(&validation_info).expect("import1 instantiation failed"); + +// let grow = get_func!(import1_instance, "grow"); +// assert_result!(import1_instance, grow, (), 2); +// } + +// #[ignore = "table exports not yet implemented"] +// #[test_log::test] +// fn table_grow_import_table_and_check_size() { +// let import2_wat = r#" +// (module +// (import "grown-imported-table" "table" (table 3 funcref)) +// (func (export "size") (result i32) +// (table.size) +// ) +// ) +// "#; + +// let wasm_bytes = wat::parse_str(import2_wat).unwrap(); +// let validation_info = validate(&wasm_bytes).unwrap(); +// let mut import2_instance = RuntimeInstance::new(&validation_info).expect("import2 instantiation failed"); + +// let size = get_func!(import2_instance, "size"); +// assert_result!(import2_instance, size, (), 3); +// } + +#[ignore = "table grow type errors"] +#[test_log::test] +fn table_grow_type_errors() { + // Test cases for type errors + let invalid_cases = [ + ( + r#" + (module + (table $t 0 funcref) + (func $type-init-size-empty-vs-i32-funcref (result i32) + (table.grow $t) + ) + ) + "#, + "type mismatch", + ), + ( + r#" + (module + (table $t 0 funcref) + (func $type-size-empty-vs-i32 (result i32) + (table.grow $t (ref.null func)) + ) + ) + "#, + "type mismatch", + ), + ( + r#" + (module + (table $t 0 funcref) + (func $type-init-empty-vs-funcref (result i32) + (table.grow $t (i32.const 1)) + ) + ) + "#, + "type mismatch", + ), + // Add more invalid cases as needed + ]; + + for (wat, expected_error) in invalid_cases.iter() { + let result = wat::parse_str(wat); + assert!(result.is_err()); + assert!(result.err().unwrap().to_string().contains(expected_error)); + } +} diff --git a/tests/table_init.rs b/tests/table_init.rs new file mode 100644 index 00000000..1e38bbf2 --- /dev/null +++ b/tests/table_init.rs @@ -0,0 +1,2679 @@ +/* +# This file incorporates code from the WebAssembly testsuite, originally +# available at https://github.com/WebAssembly/testsuite. +# +# The original code is licensed under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +*/ + +use wasm::{validate, Error, RuntimeError, RuntimeInstance}; + +macro_rules! get_func { + ($instance:ident, $func_name:expr) => { + &$instance.get_function_by_name("", $func_name).unwrap() + }; +} + +#[test_log::test] +fn table_init_1_test() { + let w = r#" + (module + (type (func (result i32))) + (func (export "ef0") (result i32) (i32.const 0)) + (func (export "ef1") (result i32) (i32.const 1)) + (func (export "ef2") (result i32) (i32.const 2)) + (func (export "ef3") (result i32) (i32.const 3)) + (func (export "ef4") (result i32) (i32.const 4)) + (table $t0 30 30 funcref) + (table $t1 30 30 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 5)) ;; index 5 + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) ;; index 9 + (func (export "test") + (table.init $t0 1 (i32.const 7) (i32.const 0) (i32.const 4))) + (func (export "check") (param i32) (result i32) + (call_indirect $t0 (type 0) (local.get 0))) + ) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + let check = get_func!(i, "check"); + + i.invoke::<(), ()>(test, ()).unwrap(); + + assert!(i.invoke::(check, 0).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 1).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(3, i.invoke(check, 2).unwrap()); + assert_eq!(1, i.invoke(check, 3).unwrap()); + assert_eq!(4, i.invoke(check, 4).unwrap()); + assert_eq!(1, i.invoke(check, 5).unwrap()); + assert!(i.invoke::(check, 6).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(2, i.invoke(check, 7).unwrap()); + assert_eq!(7, i.invoke(check, 8).unwrap()); + assert_eq!(1, i.invoke(check, 9).unwrap()); + assert_eq!(8, i.invoke(check, 10).unwrap()); + assert!(i.invoke::(check, 11).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 12).unwrap()); + assert_eq!(5, i.invoke(check, 13).unwrap()); + assert_eq!(2, i.invoke(check, 14).unwrap()); + assert_eq!(3, i.invoke(check, 15).unwrap()); + assert_eq!(6, i.invoke(check, 16).unwrap()); + assert!(i.invoke::(check, 17).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 18).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 19).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 20).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 21).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 22).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 23).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 24).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 25).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 26).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 27).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 28).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 29).err().unwrap() == RuntimeError::UninitializedElement); +} + +#[test_log::test] +fn table_init_2_test() { + let w = r#" +(module + (type (func (result i32))) ;; type #0 + (func (export "ef0") (result i32) (i32.const 0)) + (func (export "ef1") (result i32) (i32.const 1)) + (func (export "ef2") (result i32) (i32.const 2)) + (func (export "ef3") (result i32) (i32.const 3)) + (func (export "ef4") (result i32) (i32.const 4)) + (table $t0 30 30 funcref) + (table $t1 30 30 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8) + ) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6) + ) + (func (result i32) (i32.const 5)) ;; index 5 + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) ;; index 9 + (func (export "test") + (table.init $t0 3 (i32.const 15) (i32.const 1) (i32.const 3)) + ) + (func (export "check") (param i32) (result i32) + (call_indirect $t0 (type 0) (local.get 0)) + ) +) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + let check = get_func!(i, "check"); + + i.invoke::<(), ()>(test, ()).unwrap(); + + assert!(i.invoke::(check, 0).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 1).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(3, i.invoke(check, 2).unwrap()); + assert_eq!(1, i.invoke(check, 3).unwrap()); + assert_eq!(4, i.invoke(check, 4).unwrap()); + assert_eq!(1, i.invoke(check, 5).unwrap()); + assert!(i.invoke::(check, 6).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 7).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 8).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 9).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 10).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 11).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 12).unwrap()); + assert_eq!(5, i.invoke(check, 13).unwrap()); + assert_eq!(2, i.invoke(check, 14).unwrap()); + assert_eq!(9, i.invoke(check, 15).unwrap()); + assert_eq!(2, i.invoke(check, 16).unwrap()); + assert_eq!(7, i.invoke(check, 17).unwrap()); + assert!(i.invoke::(check, 18).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 19).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 20).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 21).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 22).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 23).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 24).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 25).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 26).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 27).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 28).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 29).err().unwrap() == RuntimeError::UninitializedElement); +} + +#[test_log::test] +fn table_init_3_test() { + let w = r#" +(module + (type (func (result i32))) ;; type #0 + (func (export "ef0") (result i32) (i32.const 0)) ;; index 0 + (func (export "ef1") (result i32) (i32.const 1)) + (func (export "ef2") (result i32) (i32.const 2)) + (func (export "ef3") (result i32) (i32.const 3)) + (func (export "ef4") (result i32) (i32.const 4)) ;; index 4 + (table $t0 30 30 funcref) + (table $t1 30 30 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 5)) ;; index 5 + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) ;; index 9 + (func (export "test") + (table.init $t0 1 (i32.const 7) (i32.const 0) (i32.const 4)) + (elem.drop 1) + (table.init $t0 3 (i32.const 15) (i32.const 1) (i32.const 3)) + (elem.drop 3) + (table.copy $t0 0 (i32.const 20) (i32.const 15) (i32.const 5)) + (table.copy $t0 0 (i32.const 21) (i32.const 29) (i32.const 1)) + (table.copy $t0 0 (i32.const 24) (i32.const 10) (i32.const 1)) + (table.copy $t0 0 (i32.const 13) (i32.const 11) (i32.const 4)) + (table.copy $t0 0 (i32.const 19) (i32.const 20) (i32.const 5))) + (func (export "check") (param i32) (result i32) + (call_indirect $t0 (type 0) (local.get 0))) +) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + let check = get_func!(i, "check"); + + i.invoke::<(), ()>(test, ()).unwrap(); + + assert!(i.invoke::(check, 0).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 1).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(3, i.invoke(check, 2).unwrap()); + assert_eq!(1, i.invoke(check, 3).unwrap()); + assert_eq!(4, i.invoke(check, 4).unwrap()); + assert_eq!(1, i.invoke(check, 5).unwrap()); + assert!(i.invoke::(check, 6).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(2, i.invoke(check, 7).unwrap()); + assert_eq!(7, i.invoke(check, 8).unwrap()); + assert_eq!(1, i.invoke(check, 9).unwrap()); + assert_eq!(8, i.invoke(check, 10).unwrap()); + assert!(i.invoke::(check, 11).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 12).unwrap()); + assert!(i.invoke::(check, 13).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 14).unwrap()); + assert_eq!(5, i.invoke(check, 15).unwrap()); + assert_eq!(2, i.invoke(check, 16).unwrap()); + assert_eq!(7, i.invoke(check, 17).unwrap()); + assert!(i.invoke::(check, 18).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(9, i.invoke(check, 19).unwrap()); + assert!(i.invoke::(check, 20).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 21).unwrap()); + assert!(i.invoke::(check, 22).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(8, i.invoke(check, 23).unwrap()); + assert_eq!(8, i.invoke(check, 24).unwrap()); + assert!(i.invoke::(check, 25).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 26).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 27).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 28).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 29).err().unwrap() == RuntimeError::UninitializedElement); +} + +#[test_log::test] +fn table_init_4_test() { + let w = r#" +(module + (type (func (result i32))) ;; type #0 + (func (export "ef0") (result i32) (i32.const 0)) ;; index 0 + (func (export "ef1") (result i32) (i32.const 1)) + (func (export "ef2") (result i32) (i32.const 2)) + (func (export "ef3") (result i32) (i32.const 3)) + (func (export "ef4") (result i32) (i32.const 4)) ;; index 4 + (table $t0 30 30 funcref) + (table $t1 30 30 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 5)) ;; index 5 + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) ;; index 9 + (func (export "test") + (table.init $t1 3 (i32.const 15) (i32.const 1) (i32.const 3))) + (func (export "check") (param i32) (result i32) + (call_indirect $t1 (type 0) (local.get 0))) +) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + let check = get_func!(i, "check"); + + i.invoke::<(), ()>(test, ()).unwrap(); + + println!("{:#?}", i.store.tables[1]); + + assert!(i.invoke::(check, 0).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 1).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(3, i.invoke(check, 2).unwrap()); + assert_eq!(1, i.invoke(check, 3).unwrap()); + assert_eq!(4, i.invoke(check, 4).unwrap()); + assert_eq!(1, i.invoke(check, 5).unwrap()); + assert!(i.invoke::(check, 6).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 7).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 8).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 9).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 10).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 11).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 12).unwrap()); + assert_eq!(5, i.invoke(check, 13).unwrap()); + assert_eq!(2, i.invoke(check, 14).unwrap()); + assert_eq!(9, i.invoke(check, 15).unwrap()); + assert_eq!(2, i.invoke(check, 16).unwrap()); + assert_eq!(7, i.invoke(check, 17).unwrap()); + assert!(i.invoke::(check, 18).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 19).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 20).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 21).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 22).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 23).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 24).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 25).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 26).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 27).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 28).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 29).err().unwrap() == RuntimeError::UninitializedElement); +} + +#[test_log::test] +fn table_init_5_test() { + let w = r#" +(module + (type (func (result i32))) ;; type #0 + (func (export "ef0") (result i32) (i32.const 0)) ;; index 0 + (func (export "ef1") (result i32) (i32.const 1)) + (func (export "ef2") (result i32) (i32.const 2)) + (func (export "ef3") (result i32) (i32.const 3)) + (func (export "ef4") (result i32) (i32.const 4)) ;; index 4 + (table $t0 30 30 funcref) + (table $t1 30 30 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 5)) ;; index 5 + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) ;; index 9 + (func (export "test") + (table.init $t1 1 (i32.const 7) (i32.const 0) (i32.const 4)) + (elem.drop 1) + (table.init $t1 3 (i32.const 15) (i32.const 1) (i32.const 3)) + (elem.drop 3) + (table.copy $t1 1 (i32.const 20) (i32.const 15) (i32.const 5)) + (table.copy $t1 1 (i32.const 21) (i32.const 29) (i32.const 1)) + (table.copy $t1 1 (i32.const 24) (i32.const 10) (i32.const 1)) + (table.copy $t1 1 (i32.const 13) (i32.const 11) (i32.const 4)) + (table.copy $t1 1 (i32.const 19) (i32.const 20) (i32.const 5))) + (func (export "check") (param i32) (result i32) + (call_indirect $t1 (type 0) (local.get 0))) +) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + let check = get_func!(i, "check"); + + i.invoke::<(), ()>(test, ()).unwrap(); + + assert!(i.invoke::(check, 0).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 1).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(3, i.invoke(check, 2).unwrap()); + assert_eq!(1, i.invoke(check, 3).unwrap()); + assert_eq!(4, i.invoke(check, 4).unwrap()); + assert_eq!(1, i.invoke(check, 5).unwrap()); + assert!(i.invoke::(check, 6).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(2, i.invoke(check, 7).unwrap()); + assert_eq!(7, i.invoke(check, 8).unwrap()); + assert_eq!(1, i.invoke(check, 9).unwrap()); + assert_eq!(8, i.invoke(check, 10).unwrap()); + assert!(i.invoke::(check, 11).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 12).unwrap()); + assert!(i.invoke::(check, 13).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 14).unwrap()); + assert_eq!(5, i.invoke(check, 15).unwrap()); + assert_eq!(2, i.invoke(check, 16).unwrap()); + assert_eq!(7, i.invoke(check, 17).unwrap()); + assert!(i.invoke::(check, 18).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(9, i.invoke(check, 19).unwrap()); + assert!(i.invoke::(check, 20).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(7, i.invoke(check, 21).unwrap()); + assert!(i.invoke::(check, 22).err().unwrap() == RuntimeError::UninitializedElement); + assert_eq!(8, i.invoke(check, 23).unwrap()); + assert_eq!(8, i.invoke(check, 24).unwrap()); + assert!(i.invoke::(check, 25).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 26).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 27).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 28).err().unwrap() == RuntimeError::UninitializedElement); + assert!(i.invoke::(check, 29).err().unwrap() == RuntimeError::UninitializedElement); +} + +#[test_log::test] +fn table_init_6_test() { + let w = r#" +(module + (func (export "test") +(elem.drop 0))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(validate(&wasm_bytes).err().unwrap() == Error::ElementIsNotDefined(0)); +} + +#[test_log::test] +fn table_init_7_test() { + let w = r#" + (module + (func (export "test") + (table.init 0 (i32.const 12) (i32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(validate(&wasm_bytes).err().unwrap() == Error::TableIsNotDefined(0)); +} + +#[test_log::test] +fn table_init_8_test() { + let w = r#" + (module + (elem funcref (ref.func 0)) + (func (result i32) (i32.const 0)) + (func (export "test") + (elem.drop 4))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(validate(&wasm_bytes).err().unwrap() == Error::ElementIsNotDefined(4)); +} + +#[test_log::test] +fn table_init_9_test() { + let w = r#" + (module + (elem funcref (ref.func 0)) + (func (result i32) (i32.const 0)) + (func (export "test") + (table.init 4 (i32.const 12) (i32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(validate(&wasm_bytes).err().unwrap() == Error::TableIsNotDefined(0)); +} + +#[test_log::test] +fn table_init_10_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (elem.drop 2) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_11_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init 2 (i32.const 12) (i32.const 1) (i32.const 1)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_12_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init 1 (i32.const 12) (i32.const 1) (i32.const 1)) + (table.init 1 (i32.const 21) (i32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_13_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (elem.drop 1) + (elem.drop 1))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_14_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (elem.drop 1) + (table.init 1 (i32.const 12) (i32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_15_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init 1 (i32.const 12) (i32.const 0) (i32.const 5)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_16_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init 1 (i32.const 12) (i32.const 2) (i32.const 3)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_17_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t0 1 (i32.const 28) (i32.const 1) (i32.const 3)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_18_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t0 1 (i32.const 12) (i32.const 4) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_19_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t0 1 (i32.const 12) (i32.const 5) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_20_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t0 1 (i32.const 30) (i32.const 2) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_21_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t0 1 (i32.const 31) (i32.const 2) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_22_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t0 1 (i32.const 30) (i32.const 4) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_23_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t0) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t0) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t0 1 (i32.const 31) (i32.const 5) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_24_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t1 1 (i32.const 26) (i32.const 1) (i32.const 3)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_25_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t1 1 (i32.const 12) (i32.const 4) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_26_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t1 1 (i32.const 12) (i32.const 5) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_27_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t1 1 (i32.const 28) (i32.const 2) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_28_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t1 1 (i32.const 29) (i32.const 2) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_29_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t1 1 (i32.const 28) (i32.const 4) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + i.invoke::<(), ()>(test, ()).unwrap(); +} + +#[test_log::test] +fn table_init_30_test() { + let w = r#" +(module + (table $t0 30 30 funcref) + (table $t1 28 28 funcref) + (elem (table $t1) (i32.const 2) func 3 1 4 1) + (elem funcref + (ref.func 2) (ref.func 7) (ref.func 1) (ref.func 8)) + (elem (table $t1) (i32.const 12) func 7 5 2 3 6) + (elem funcref + (ref.func 5) (ref.func 9) (ref.func 2) (ref.func 7) (ref.func 6)) + (func (result i32) (i32.const 0)) + (func (result i32) (i32.const 1)) + (func (result i32) (i32.const 2)) + (func (result i32) (i32.const 3)) + (func (result i32) (i32.const 4)) + (func (result i32) (i32.const 5)) + (func (result i32) (i32.const 6)) + (func (result i32) (i32.const 7)) + (func (result i32) (i32.const 8)) + (func (result i32) (i32.const 9)) + (func (export "test") + (table.init $t1 1 (i32.const 29) (i32.const 5) (i32.const 0)) + )) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + let test = get_func!(i, "test"); + + assert!(i.invoke::<(), ()>(test, ()).err().unwrap() == RuntimeError::TableAccessOutOfBounds); +} + +#[test_log::test] +fn table_init_31_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (i32.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_32_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (i32.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_33_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (i32.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_34_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (f32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_35_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (f32.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_36_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (f32.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_37_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (f32.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_38_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (i64.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_39_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (i64.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_40_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (i64.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_41_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (i64.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_42_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (f64.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_43_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (f64.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_44_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (f64.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_45_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i32.const 1) (f64.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_46_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (i32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_47_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (i32.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_48_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (i32.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_49_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (i32.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_50_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (f32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_51_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (f32.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_52_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (f32.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_53_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (f32.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_54_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (i64.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_55_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (i64.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_56_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (i64.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_57_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (i64.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_58_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (f64.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_59_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (f64.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_60_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (f64.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_61_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f32.const 1) (f64.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_62_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (i32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_63_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (i32.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_64_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (i32.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_65_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (i32.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_66_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (f32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_67_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (f32.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_68_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (f32.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_69_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (f32.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_70_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (i64.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_71_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (i64.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_72_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (i64.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_73_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (i64.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_74_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (f64.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_75_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (f64.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_76_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (f64.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_77_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (i64.const 1) (f64.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_78_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (i32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_79_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (i32.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_80_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (i32.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_81_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (i32.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_82_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (f32.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_83_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (f32.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_84_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (f32.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_85_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (f32.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_86_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (i64.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_87_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (i64.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_88_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (i64.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_89_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (i64.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_90_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (f64.const 1) (i32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_91_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (f64.const 1) (f32.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_92_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (f64.const 1) (i64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_93_test() { + let w = r#" + (module + (table 10 funcref) + (elem funcref (ref.func $f0) (ref.func $f0) (ref.func $f0)) + (func $f0) + (func (export "test") + (table.init 0 (f64.const 1) (f64.const 1) (f64.const 1)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + assert!(matches!( + validate(&wasm_bytes).err().unwrap(), + Error::InvalidValidationStackValType(_) + )); +} + +#[test_log::test] +fn table_init_94_test() { + let w = r#" +(module + (type (func (result i32))) + (table 32 64 funcref) + (elem funcref + (ref.func $f0) (ref.func $f1) (ref.func $f2) (ref.func $f3) + (ref.func $f4) (ref.func $f5) (ref.func $f6) (ref.func $f7) + (ref.func $f8) (ref.func $f9) (ref.func $f10) (ref.func $f11) + (ref.func $f12) (ref.func $f13) (ref.func $f14) (ref.func $f15)) + (func $f0 (export "f0") (result i32) (i32.const 0)) + (func $f1 (export "f1") (result i32) (i32.const 1)) + (func $f2 (export "f2") (result i32) (i32.const 2)) + (func $f3 (export "f3") (result i32) (i32.const 3)) + (func $f4 (export "f4") (result i32) (i32.const 4)) + (func $f5 (export "f5") (result i32) (i32.const 5)) + (func $f6 (export "f6") (result i32) (i32.const 6)) + (func $f7 (export "f7") (result i32) (i32.const 7)) + (func $f8 (export "f8") (result i32) (i32.const 8)) + (func $f9 (export "f9") (result i32) (i32.const 9)) + (func $f10 (export "f10") (result i32) (i32.const 10)) + (func $f11 (export "f11") (result i32) (i32.const 11)) + (func $f12 (export "f12") (result i32) (i32.const 12)) + (func $f13 (export "f13") (result i32) (i32.const 13)) + (func $f14 (export "f14") (result i32) (i32.const 14)) + (func $f15 (export "f15") (result i32) (i32.const 15)) + (func (export "test") (param $n i32) (result i32) + (call_indirect (type 0) (local.get $n))) + (func (export "run") (param $offs i32) (param $len i32) + (table.init 0 (local.get $offs) (i32.const 0) (local.get $len)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut inst = RuntimeInstance::new(&validation_info).unwrap(); + + let run = get_func!(inst, "run"); + let test = get_func!(inst, "test"); + assert!( + inst.invoke::<(i32, i32), ()>(run, (24, 16)).err().unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + for i in 0..32 { + assert!( + inst.invoke::(test, i).err().unwrap() == RuntimeError::UninitializedElement + ); + } +} + +#[test_log::test] +fn table_init_95_test() { + let w = r#" +(module + (type (func (result i32))) + (table 32 64 funcref) + (elem funcref + (ref.func $f0) (ref.func $f1) (ref.func $f2) (ref.func $f3) + (ref.func $f4) (ref.func $f5) (ref.func $f6) (ref.func $f7) + (ref.func $f8) (ref.func $f9) (ref.func $f10) (ref.func $f11) + (ref.func $f12) (ref.func $f13) (ref.func $f14) (ref.func $f15)) + (func $f0 (export "f0") (result i32) (i32.const 0)) + (func $f1 (export "f1") (result i32) (i32.const 1)) + (func $f2 (export "f2") (result i32) (i32.const 2)) + (func $f3 (export "f3") (result i32) (i32.const 3)) + (func $f4 (export "f4") (result i32) (i32.const 4)) + (func $f5 (export "f5") (result i32) (i32.const 5)) + (func $f6 (export "f6") (result i32) (i32.const 6)) + (func $f7 (export "f7") (result i32) (i32.const 7)) + (func $f8 (export "f8") (result i32) (i32.const 8)) + (func $f9 (export "f9") (result i32) (i32.const 9)) + (func $f10 (export "f10") (result i32) (i32.const 10)) + (func $f11 (export "f11") (result i32) (i32.const 11)) + (func $f12 (export "f12") (result i32) (i32.const 12)) + (func $f13 (export "f13") (result i32) (i32.const 13)) + (func $f14 (export "f14") (result i32) (i32.const 14)) + (func $f15 (export "f15") (result i32) (i32.const 15)) + (func (export "test") (param $n i32) (result i32) + (call_indirect (type 0) (local.get $n))) + (func (export "run") (param $offs i32) (param $len i32) + (table.init 0 (local.get $offs) (i32.const 0) (local.get $len)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut inst = RuntimeInstance::new(&validation_info).unwrap(); + + let run = get_func!(inst, "run"); + let test = get_func!(inst, "test"); + assert!( + inst.invoke::<(i32, i32), ()>(run, (25, 16)).err().unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + for i in 0..32 { + assert!( + inst.invoke::(test, i).err().unwrap() == RuntimeError::UninitializedElement + ); + } +} + +#[test_log::test] +fn table_init_96_test() { + let w = r#" +(module + (type (func (result i32))) + (table 160 320 funcref) + (elem funcref + (ref.func $f0) (ref.func $f1) (ref.func $f2) (ref.func $f3) + (ref.func $f4) (ref.func $f5) (ref.func $f6) (ref.func $f7) + (ref.func $f8) (ref.func $f9) (ref.func $f10) (ref.func $f11) + (ref.func $f12) (ref.func $f13) (ref.func $f14) (ref.func $f15)) + (func $f0 (export "f0") (result i32) (i32.const 0)) + (func $f1 (export "f1") (result i32) (i32.const 1)) + (func $f2 (export "f2") (result i32) (i32.const 2)) + (func $f3 (export "f3") (result i32) (i32.const 3)) + (func $f4 (export "f4") (result i32) (i32.const 4)) + (func $f5 (export "f5") (result i32) (i32.const 5)) + (func $f6 (export "f6") (result i32) (i32.const 6)) + (func $f7 (export "f7") (result i32) (i32.const 7)) + (func $f8 (export "f8") (result i32) (i32.const 8)) + (func $f9 (export "f9") (result i32) (i32.const 9)) + (func $f10 (export "f10") (result i32) (i32.const 10)) + (func $f11 (export "f11") (result i32) (i32.const 11)) + (func $f12 (export "f12") (result i32) (i32.const 12)) + (func $f13 (export "f13") (result i32) (i32.const 13)) + (func $f14 (export "f14") (result i32) (i32.const 14)) + (func $f15 (export "f15") (result i32) (i32.const 15)) + (func (export "test") (param $n i32) (result i32) + (call_indirect (type 0) (local.get $n))) + (func (export "run") (param $offs i32) (param $len i32) + (table.init 0 (local.get $offs) (i32.const 0) (local.get $len)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut inst = RuntimeInstance::new(&validation_info).unwrap(); + + let run = get_func!(inst, "run"); + let test = get_func!(inst, "test"); + assert!( + inst.invoke::<(i32, i32), ()>(run, (96, 32)).err().unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + for i in 0..160 { + assert!( + inst.invoke::(test, i).err().unwrap() == RuntimeError::UninitializedElement + ); + } +} + +#[test_log::test] +fn table_init_97_test() { + let w = r#" +(module + (type (func (result i32))) + (table 160 320 funcref) + (elem funcref + (ref.func $f0) (ref.func $f1) (ref.func $f2) (ref.func $f3) + (ref.func $f4) (ref.func $f5) (ref.func $f6) (ref.func $f7) + (ref.func $f8) (ref.func $f9) (ref.func $f10) (ref.func $f11) + (ref.func $f12) (ref.func $f13) (ref.func $f14) (ref.func $f15)) + (func $f0 (export "f0") (result i32) (i32.const 0)) + (func $f1 (export "f1") (result i32) (i32.const 1)) + (func $f2 (export "f2") (result i32) (i32.const 2)) + (func $f3 (export "f3") (result i32) (i32.const 3)) + (func $f4 (export "f4") (result i32) (i32.const 4)) + (func $f5 (export "f5") (result i32) (i32.const 5)) + (func $f6 (export "f6") (result i32) (i32.const 6)) + (func $f7 (export "f7") (result i32) (i32.const 7)) + (func $f8 (export "f8") (result i32) (i32.const 8)) + (func $f9 (export "f9") (result i32) (i32.const 9)) + (func $f10 (export "f10") (result i32) (i32.const 10)) + (func $f11 (export "f11") (result i32) (i32.const 11)) + (func $f12 (export "f12") (result i32) (i32.const 12)) + (func $f13 (export "f13") (result i32) (i32.const 13)) + (func $f14 (export "f14") (result i32) (i32.const 14)) + (func $f15 (export "f15") (result i32) (i32.const 15)) + (func (export "test") (param $n i32) (result i32) + (call_indirect (type 0) (local.get $n))) + (func (export "run") (param $offs i32) (param $len i32) + (table.init 0 (local.get $offs) (i32.const 0) (local.get $len)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut inst = RuntimeInstance::new(&validation_info).unwrap(); + + let run = get_func!(inst, "run"); + let test = get_func!(inst, "test"); + assert!( + inst.invoke::<(i32, i32), ()>(run, (97, 31)).err().unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + for i in 0..160 { + assert!( + inst.invoke::(test, i).err().unwrap() == RuntimeError::UninitializedElement + ); + } +} + +#[test_log::test] +fn table_init_98_test() { + let w = r#" +(module + (type (func (result i32))) + (table 64 64 funcref) + (elem funcref + (ref.func $f0) (ref.func $f1) (ref.func $f2) (ref.func $f3) + (ref.func $f4) (ref.func $f5) (ref.func $f6) (ref.func $f7) + (ref.func $f8) (ref.func $f9) (ref.func $f10) (ref.func $f11) + (ref.func $f12) (ref.func $f13) (ref.func $f14) (ref.func $f15)) + (func $f0 (export "f0") (result i32) (i32.const 0)) + (func $f1 (export "f1") (result i32) (i32.const 1)) + (func $f2 (export "f2") (result i32) (i32.const 2)) + (func $f3 (export "f3") (result i32) (i32.const 3)) + (func $f4 (export "f4") (result i32) (i32.const 4)) + (func $f5 (export "f5") (result i32) (i32.const 5)) + (func $f6 (export "f6") (result i32) (i32.const 6)) + (func $f7 (export "f7") (result i32) (i32.const 7)) + (func $f8 (export "f8") (result i32) (i32.const 8)) + (func $f9 (export "f9") (result i32) (i32.const 9)) + (func $f10 (export "f10") (result i32) (i32.const 10)) + (func $f11 (export "f11") (result i32) (i32.const 11)) + (func $f12 (export "f12") (result i32) (i32.const 12)) + (func $f13 (export "f13") (result i32) (i32.const 13)) + (func $f14 (export "f14") (result i32) (i32.const 14)) + (func $f15 (export "f15") (result i32) (i32.const 15)) + (func (export "test") (param $n i32) (result i32) + (call_indirect (type 0) (local.get $n))) + (func (export "run") (param $offs i32) (param $len i32) + (table.init 0 (local.get $offs) (i32.const 0) (local.get $len)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut inst = RuntimeInstance::new(&validation_info).unwrap(); + + let run = get_func!(inst, "run"); + let test = get_func!(inst, "test"); + assert!( + inst.invoke::<(i32, u32), ()>(run, (48, 4294967280_u32)) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + for i in 0..64 { + assert!( + inst.invoke::(test, i).err().unwrap() == RuntimeError::UninitializedElement + ); + } +} + +#[test_log::test] +fn table_init_99_test() { + let w = r#" +(module + (type (func (result i32))) + (table 16 16 funcref) + (elem funcref + (ref.func $f0) (ref.func $f1) (ref.func $f2) (ref.func $f3) + (ref.func $f4) (ref.func $f5) (ref.func $f6) (ref.func $f7) + (ref.func $f8) (ref.func $f9) (ref.func $f10) (ref.func $f11) + (ref.func $f12) (ref.func $f13) (ref.func $f14) (ref.func $f15)) + (func $f0 (export "f0") (result i32) (i32.const 0)) + (func $f1 (export "f1") (result i32) (i32.const 1)) + (func $f2 (export "f2") (result i32) (i32.const 2)) + (func $f3 (export "f3") (result i32) (i32.const 3)) + (func $f4 (export "f4") (result i32) (i32.const 4)) + (func $f5 (export "f5") (result i32) (i32.const 5)) + (func $f6 (export "f6") (result i32) (i32.const 6)) + (func $f7 (export "f7") (result i32) (i32.const 7)) + (func $f8 (export "f8") (result i32) (i32.const 8)) + (func $f9 (export "f9") (result i32) (i32.const 9)) + (func $f10 (export "f10") (result i32) (i32.const 10)) + (func $f11 (export "f11") (result i32) (i32.const 11)) + (func $f12 (export "f12") (result i32) (i32.const 12)) + (func $f13 (export "f13") (result i32) (i32.const 13)) + (func $f14 (export "f14") (result i32) (i32.const 14)) + (func $f15 (export "f15") (result i32) (i32.const 15)) + (func (export "test") (param $n i32) (result i32) + (call_indirect (type 0) (local.get $n))) + (func (export "run") (param $offs i32) (param $len i32) + (table.init 0 (local.get $offs) (i32.const 8) (local.get $len)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut inst = RuntimeInstance::new(&validation_info).unwrap(); + + let run = get_func!(inst, "run"); + let test = get_func!(inst, "test"); + assert!( + inst.invoke::<(i32, i32), ()>(run, (0, 4294967292_u32 as i32)) + .err() + .unwrap() + == RuntimeError::TableAccessOutOfBounds + ); + for i in 0..16 { + assert!( + inst.invoke::(test, i).err().unwrap() == RuntimeError::UninitializedElement + ); + } +} + +#[test_log::test] +fn table_init_100_test() { + let w = r#" +(module + (table 1 funcref) + ;; 65 elem segments. 64 is the smallest positive number that is encoded + ;; differently as a signed LEB. + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) (elem funcref) (elem funcref) (elem funcref) + (elem funcref) + (func (table.init 64 (i32.const 0) (i32.const 0) (i32.const 0)))) + "#; + + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + RuntimeInstance::new(&validation_info).unwrap(); +} diff --git a/tests/table_size.rs b/tests/table_size.rs new file mode 100644 index 00000000..b529effa --- /dev/null +++ b/tests/table_size.rs @@ -0,0 +1,135 @@ +/* +# This file incorporates code from the WebAssembly testsuite, originally +# available at https://github.com/WebAssembly/testsuite. +# +# The original code is licensed under the Apache License, Version 2.0 +# (the "License"); you may not use this file except in compliance +# with the License. You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +*/ + +use wasm::{validate, RuntimeInstance}; + +macro_rules! get_func { + ($instance:ident, $func_name:expr) => { + &$instance.get_function_by_name("", $func_name).unwrap() + }; +} + +macro_rules! assert_result { + ($instance:expr, $func:expr, $arg:expr, $result:expr) => { + assert_eq!($result, $instance.invoke($func, $arg).unwrap()); + }; +} + +#[test_log::test] +fn table_size_test() { + let w = r#" +(module + (table $t0 0 funcref) + (table $t1 1 funcref) + (table $t2 0 2 funcref) + (table $t3 3 8 funcref) + + (func (export "size-t0") (result i32) table.size) + (func (export "size-t1") (result i32) (table.size $t1)) + (func (export "size-t2") (result i32) (table.size $t2)) + (func (export "size-t3") (result i32) (table.size $t3)) + + (func (export "grow-t0") (param $sz i32) + (drop (table.grow $t0 (ref.null func) (local.get $sz))) + ) + (func (export "grow-t1") (param $sz i32) + (drop (table.grow $t1 (ref.null func) (local.get $sz))) + ) + (func (export "grow-t2") (param $sz i32) + (drop (table.grow $t2 (ref.null func) (local.get $sz))) + ) + (func (export "grow-t3") (param $sz i32) + (drop (table.grow $t3 (ref.null func) (local.get $sz))) + ) +) + "#; + let wasm_bytes = wat::parse_str(w).unwrap(); + let validation_info = validate(&wasm_bytes).unwrap(); + let mut i = RuntimeInstance::new(&validation_info).expect("instantiation failed"); + + // let get_funcref = get_func!(i, "get-funcref"); + // let init = get_func!(i, "init"); + let size_t0 = get_func!(i, "size-t0"); + let size_t1 = get_func!(i, "size-t1"); + let size_t2 = get_func!(i, "size-t2"); + let size_t3 = get_func!(i, "size-t3"); + let grow_t0 = get_func!(i, "grow-t0"); + let grow_t1 = get_func!(i, "grow-t1"); + let grow_t2 = get_func!(i, "grow-t2"); + let grow_t3 = get_func!(i, "grow-t3"); + + assert_result!(i, size_t0, (), 0); + assert_result!(i, grow_t0, 1, ()); + assert_result!(i, size_t0, (), 1); + assert_result!(i, grow_t0, 4, ()); + assert_result!(i, size_t0, (), 5); + assert_result!(i, grow_t0, 0, ()); + assert_result!(i, size_t0, (), 5); + + assert_result!(i, size_t1, (), 1); + assert_result!(i, grow_t1, 1, ()); + assert_result!(i, size_t1, (), 2); + assert_result!(i, grow_t1, 4, ()); + assert_result!(i, size_t1, (), 6); + assert_result!(i, grow_t1, 0, ()); + assert_result!(i, size_t1, (), 6); + + assert_result!(i, size_t2, (), 0); + assert_result!(i, grow_t2, 3, ()); + assert_result!(i, size_t2, (), 0); + assert_result!(i, grow_t2, 1, ()); + assert_result!(i, size_t2, (), 1); + assert_result!(i, grow_t2, 0, ()); + assert_result!(i, size_t2, (), 1); + assert_result!(i, grow_t2, 4, ()); + assert_result!(i, size_t2, (), 1); + assert_result!(i, grow_t2, 1, ()); + assert_result!(i, size_t2, (), 2); + + assert_result!(i, size_t3, (), 3); + assert_result!(i, grow_t3, 1, ()); + assert_result!(i, size_t3, (), 4); + assert_result!(i, grow_t3, 3, ()); + assert_result!(i, size_t3, (), 7); + assert_result!(i, grow_t3, 0, ()); + assert_result!(i, size_t3, (), 7); + assert_result!(i, grow_t3, 2, ()); + assert_result!(i, size_t3, (), 7); + assert_result!(i, grow_t3, 1, ()); + assert_result!(i, size_t3, (), 8); +} + +// ;; Type errors + +// (assert_invalid +// (module +// (table $t 1 externref) +// (func $type-result-i32-vs-empty +// (table.size $t) +// ) +// ) +// "type mismatch" +// ) +// (assert_invalid +// (module +// (table $t 1 externref) +// (func $type-result-i32-vs-f32 (result f32) +// (table.size $t) +// ) +// ) +// "type mismatch" +// )