///! Basic tests for instructions/constructions, missing in wabt tests use std::sync::Arc; use std::cell::RefCell; use std::collections::HashMap; use builder::module; use elements::{ExportEntry, Internal, ImportEntry, External, GlobalEntry, GlobalType, InitExpr, ValueType, Opcodes, Opcode, FunctionType, TableType, MemoryType}; use interpreter::{Error, UserError, ProgramInstance}; use interpreter::memory::MemoryInstance; use interpreter::value::{RuntimeValue, TryInto}; use interpreter::variable::{VariableInstance, ExternalVariableValue, VariableType}; use interpreter::host::{HostModuleBuilder, HostModule}; use interpreter::store::{Store, MemoryId}; use super::utils::program_with_default_env; #[test] fn import_function() { let module1 = module() .with_export(ExportEntry::new("external_func".into(), Internal::Function(0))) .function() .signature().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::I32Const(3), Opcode::End, ])).build() .build() .build(); let module2 = module() .with_import(ImportEntry::new("external_module".into(), "external_func".into(), External::Function(0))) .function() .signature().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::Call(0), Opcode::I32Const(7), Opcode::I32Add, Opcode::End, ])).build() .build() .build(); let mut program = ProgramInstance::new(); program.add_module("external_module", module1, &mut ()).unwrap(); program.add_module("main", module2, &mut ()).unwrap(); assert_eq!(program.invoke_index("external_module", 0, vec![], &mut ()).unwrap().unwrap(), RuntimeValue::I32(3)); assert_eq!(program.invoke_index("main", 1, vec![], &mut ()).unwrap().unwrap(), RuntimeValue::I32(10)); } #[test] fn wrong_import() { let side_module = module() .with_export(ExportEntry::new("cool_func".into(), Internal::Function(0))) .function() .signature().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::I32Const(3), Opcode::End, ])).build() .build() .build(); let module = module() .with_import(ImportEntry::new("side_module".into(), "not_cool_func".into(), External::Function(0))) .function() .signature().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::Call(0), Opcode::I32Const(7), Opcode::I32Add, Opcode::End, ])).build() .build() .build(); let mut program = ProgramInstance::new(); let _side_module_instance = program.add_module("side_module", side_module, &mut ()).unwrap(); assert!(program.add_module("main", module, &mut ()).is_err()); } #[test] fn global_get_set() { let module = module() .with_global(GlobalEntry::new(GlobalType::new(ValueType::I32, true), InitExpr::new(vec![Opcode::I32Const(42), Opcode::End]))) .function() .signature().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::GetGlobal(0), Opcode::I32Const(8), Opcode::I32Add, Opcode::SetGlobal(0), Opcode::GetGlobal(0), Opcode::End, ])).build() .build() .build(); let mut program = ProgramInstance::new(); program.add_module("main", module, &mut ()).unwrap(); assert_eq!(program.invoke_index("main", 0, vec![], &mut ()).unwrap().unwrap(), RuntimeValue::I32(50)); assert_eq!(program.invoke_index("main", 0, vec![], &mut ()).unwrap().unwrap(), RuntimeValue::I32(58)); } // custom user error #[derive(Debug, Clone, PartialEq)] struct UserErrorWithCode { error_code: i32, } impl ::std::fmt::Display for UserErrorWithCode { fn fmt(&self, f: &mut ::std::fmt::Formatter) -> Result<(), ::std::fmt::Error> { write!(f, "{}", self.error_code) } } impl UserError for UserErrorWithCode {} // TODO: Rename to state // user function executor struct FunctionExecutor { pub memory: MemoryId, pub values: Vec, } fn build_env_module() -> HostModule { let mut builder = HostModuleBuilder::::new(); builder.with_func2("add", |store: &mut Store, state: &mut FunctionExecutor, arg: i32, unused: i32| { let memory_value = state.memory.resolve(store).get(0, 1).unwrap()[0]; let fn_argument_unused = unused as u8; let fn_argument = arg as u8; assert_eq!(fn_argument_unused, 0); let sum = memory_value + fn_argument; state.memory.resolve(store).set(0, &vec![sum]).unwrap(); state.values.push(sum as i32); Ok(Some(sum as i32)) }); builder.with_func2("sub", |store: &mut Store, state: &mut FunctionExecutor, arg: i32, unused: i32| { let memory_value = state.memory.resolve(store).get(0, 1).unwrap()[0]; let fn_argument_unused = unused as u8; let fn_argument = arg as u8; assert_eq!(fn_argument_unused, 0); let diff = memory_value - fn_argument; state.memory.resolve(store).set(0, &vec![diff]).unwrap(); state.values.push(diff as i32); Ok(Some(diff as i32)) }); builder.with_func0("err", |_: &mut Store, _: &mut FunctionExecutor| -> Result, Error> { Err(Error::User(Box::new(UserErrorWithCode { error_code: 777 }))) }); builder.with_memory("memory", MemoryType::new(256, None)); builder.build() } #[test] fn native_env_function() { let mut program = program_with_default_env(); let env_host_module = build_env_module(); let env_module = program.add_host_module("env", env_host_module).unwrap(); let env_memory = env_module.export_by_name(program.store(), "memory").unwrap().as_memory().unwrap(); let mut state = FunctionExecutor { memory: env_memory, values: Vec::new(), }; { let module = module() .with_import(ImportEntry::new("env".into(), "add".into(), External::Function(0))) .with_import(ImportEntry::new("env".into(), "sub".into(), External::Function(0))) .function() .signature().param().i32().param().i32().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::GetLocal(0), Opcode::GetLocal(1), Opcode::Call(0), Opcode::End, ])).build() .build() .function() .signature().param().i32().param().i32().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::GetLocal(0), Opcode::GetLocal(1), Opcode::Call(1), Opcode::End, ])).build() .build() .build(); // load module program.add_module("main", module, &mut state).unwrap(); { assert_eq!( program.invoke_index("main", 2, vec![RuntimeValue::I32(7), RuntimeValue::I32(0)], &mut state) .unwrap() .unwrap(), RuntimeValue::I32(7) ); assert_eq!( program.invoke_index("main", 2, vec![RuntimeValue::I32(50), RuntimeValue::I32(0)], &mut state) .unwrap() .unwrap(), RuntimeValue::I32(57) ); assert_eq!( program.invoke_index("main", 3, vec![RuntimeValue::I32(15), RuntimeValue::I32(0)], &mut state) .unwrap() .unwrap(), RuntimeValue::I32(42) ); } } assert_eq!(state.memory.resolve(program.store()).get(0, 1).unwrap()[0], 42); assert_eq!(state.values, vec![7, 57, 42]); } #[test] fn native_env_global() { struct State; let module_constructor = |host_module: HostModule| { let mut program = ProgramInstance::new(); program.add_host_module("env", host_module)?; let module = module() .with_import(ImportEntry::new("env".into(), "ext_global".into(), External::Global(GlobalType::new(ValueType::I32, false)))) .function() .signature().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::GetGlobal(0), Opcode::End, ])).build() .build() .build(); program.add_module("main", module, &mut State)?; program.invoke_index("main", 0, vec![], &mut State) }; // try to add module, exporting non-existant env' variable => error { let mut host_module_builder = HostModuleBuilder::::new(); assert!(module_constructor(host_module_builder.build()).is_err()); } // now add simple variable natively => ok { let mut host_module_builder = HostModuleBuilder::::new(); host_module_builder.with_global("ext_global", GlobalType::new(ValueType::I32, false), RuntimeValue::I32(777)); assert_eq!(module_constructor(host_module_builder.build()).unwrap().unwrap(), RuntimeValue::I32(777)); } } #[test] fn native_custom_error() { let mut program = program_with_default_env(); let env_host_module = build_env_module(); let env_module = program.add_host_module("env", env_host_module).unwrap(); let env_memory = env_module.export_by_name(program.store(), "memory").unwrap().as_memory().unwrap(); let mut state = FunctionExecutor { memory: env_memory, values: Vec::new(), }; let module = module() .with_import(ImportEntry::new("env".into(), "err".into(), External::Function(0))) .function() .signature().param().i32().param().i32().return_type().i32().build() .body().with_opcodes(Opcodes::new(vec![ Opcode::GetLocal(0), Opcode::GetLocal(1), Opcode::Call(0), Opcode::End, ])).build() .build() .build(); let module_instance = program.add_module("main", module, &mut state).unwrap(); let user_error = match program.invoke_index( "main", 0, vec![RuntimeValue::I32(7), RuntimeValue::I32(0)], &mut state ) { Err(Error::User(user_error)) => user_error, result => panic!("Unexpected result {:?}", result), }; assert_eq!(user_error.downcast_ref::().unwrap(), &UserErrorWithCode { error_code: 777 }); } #[test] fn memory_import_limits_initial() { let core_module = module() .memory().with_min(10).build() .with_export(ExportEntry::new("memory".into(), Internal::Memory(0))) .build(); let mut program = ProgramInstance::new(); program.add_module("core", core_module, &mut ()).unwrap(); let test_cases = vec![ (9, false), (10, false), (11, true), ]; for test_case in test_cases { let (import_initial, is_error) = test_case; let client_module = module() .with_import(ImportEntry::new("core".into(), "memory".into(), External::Memory(MemoryType::new(import_initial, None)))) .build(); match program.add_module("client", client_module, &mut ()).map(|_| ()) { Ok(_) if !is_error => (), Err(Error::Initialization(_)) if is_error => (), x @ _ => panic!("unexpected result for test_case {:?}: {:?}", test_case, x), } } } #[test] fn memory_import_limits_maximum() { #[derive(Debug, Clone, Copy, PartialEq)] enum MaximumError { ValueMismatch, Ok }; let test_cases = vec![ (Some(100), None, MaximumError::Ok), (Some(100), Some(98), MaximumError::ValueMismatch), (Some(100), Some(100), MaximumError::Ok), (Some(100), Some(101), MaximumError::Ok), (None, None, MaximumError::Ok), ]; let mut program = ProgramInstance::new(); for test_case in test_cases { let (core_maximum, client_maximum, expected_err) = test_case; let core_module = module() .memory().with_min(10).with_max(core_maximum).build() .with_export(ExportEntry::new("memory".into(), Internal::Memory(0))) .build(); let client_module = module() .with_import(ImportEntry::new("core".into(), "memory".into(), External::Memory(MemoryType::new(10, client_maximum)))) .build(); program.add_module("core", core_module, &mut ()).unwrap(); match program.add_module("client", client_module, &mut ()).map(|_| ()) { Err(Error::Initialization(actual_err)) => match expected_err { MaximumError::ValueMismatch if actual_err == format!("trying to import with limits l1.max={:?} and l2.max={:?}", core_maximum, client_maximum) => (), _ => panic!("unexpected validation error for test_case {:?}: {}", test_case, actual_err), }, Ok(_) if expected_err == MaximumError::Ok => (), x @ _ => panic!("unexpected result for test_case {:?}: {:?}", test_case, x), } } } #[test] fn table_import_limits_initial() { let core_module = module() .table().with_min(10).build() .with_export(ExportEntry::new("table".into(), Internal::Table(0))) .build(); let mut program = ProgramInstance::new(); program.add_module("core", core_module, &mut ()).unwrap(); let test_cases = vec![ (9, false), (10, false), (11, true), ]; for test_case in test_cases { let (import_initial, is_error) = test_case; let client_module = module() .with_import(ImportEntry::new("core".into(), "table".into(), External::Table(TableType::new(import_initial, None)))) .build(); match program.add_module("client", client_module, &mut ()).map(|_| ()) { Ok(_) if !is_error => (), Err(Error::Initialization(ref actual_error)) if is_error && actual_error == &format!("trying to import with limits l1.initial=10 and l2.initial={}", import_initial) => (), x @ _ => panic!("unexpected result for test_case {:?}: {:?}", test_case, x), } } } #[test] fn table_import_limits_maximum() { #[derive(Debug, Clone, Copy, PartialEq)] enum MaximumError { ValueMismatch, Ok }; let test_cases = vec![ (Some(100), None, MaximumError::Ok), (Some(100), Some(98), MaximumError::ValueMismatch), (Some(100), Some(100), MaximumError::Ok), (Some(100), Some(101), MaximumError::Ok), (None, None, MaximumError::Ok), ]; let mut program = ProgramInstance::new(); for test_case in test_cases { let (core_maximum, client_maximum, expected_err) = test_case; let core_module = module() .table().with_min(10).with_max(core_maximum).build() .with_export(ExportEntry::new("table".into(), Internal::Table(0))) .build(); let client_module = module() .with_import(ImportEntry::new("core".into(), "table".into(), External::Table(TableType::new(10, client_maximum)))) .build(); program.add_module("core", core_module, &mut ()).unwrap(); match program.add_module("client", client_module, &mut ()).map(|_| ()) { Err(Error::Initialization(actual_err)) => match expected_err { MaximumError::ValueMismatch if actual_err == format!("trying to import with limits l1.max={:?} and l2.max={:?}", core_maximum, client_maximum) => (), _ => panic!("unexpected validation error for test_case {:?}: {}", test_case, actual_err), }, Ok(_) if expected_err == MaximumError::Ok => (), x @ _ => panic!("unexpected result for test_case {:?}: {:?}", test_case, x), } } }