use crate::{ handle_result, wasm_byte_vec_t, wasm_engine_t, wasm_exporttype_t, wasm_exporttype_vec_t, wasm_importtype_t, wasm_importtype_vec_t, wasm_store_t, wasmtime_error_t, }; use std::ptr; use wasmtime::{Engine, Module}; #[repr(C)] #[derive(Clone)] pub struct wasm_module_t { pub(crate) module: Module, pub(crate) imports: Vec, pub(crate) exports: Vec, } wasmtime_c_api_macros::declare_ref!(wasm_module_t); #[repr(C)] #[derive(Clone)] pub struct wasm_shared_module_t { module: Module, } wasmtime_c_api_macros::declare_own!(wasm_shared_module_t); #[no_mangle] pub extern "C" fn wasm_module_new( store: &wasm_store_t, binary: &wasm_byte_vec_t, ) -> Option> { let mut ret = ptr::null_mut(); let engine = wasm_engine_t { engine: store.store.engine().clone(), }; match wasmtime_module_new(&engine, binary, &mut ret) { Some(_err) => None, None => { assert!(!ret.is_null()); Some(unsafe { Box::from_raw(ret) }) } } } #[no_mangle] pub extern "C" fn wasmtime_module_new( engine: &wasm_engine_t, binary: &wasm_byte_vec_t, ret: &mut *mut wasm_module_t, ) -> Option> { let binary = binary.as_slice(); handle_result(Module::from_binary(&engine.engine, binary), |module| { let imports = module .imports() .map(|i| { wasm_importtype_t::new( i.module().to_owned(), i.name().map(|s| s.to_owned()), i.ty(), ) }) .collect::>(); let exports = module .exports() .map(|e| wasm_exporttype_t::new(e.name().to_owned(), e.ty())) .collect::>(); let module = Box::new(wasm_module_t { module: module, imports, exports, }); *ret = Box::into_raw(module); }) } #[no_mangle] pub extern "C" fn wasm_module_validate(store: &wasm_store_t, binary: &wasm_byte_vec_t) -> bool { wasmtime_module_validate(store, binary).is_none() } #[no_mangle] pub extern "C" fn wasmtime_module_validate( store: &wasm_store_t, binary: &wasm_byte_vec_t, ) -> Option> { let binary = binary.as_slice(); handle_result(Module::validate(store.store.engine(), binary), |()| {}) } #[no_mangle] pub extern "C" fn wasm_module_exports(module: &wasm_module_t, out: &mut wasm_exporttype_vec_t) { let buffer = module .exports .iter() .map(|et| Some(Box::new(et.clone()))) .collect::>(); out.set_buffer(buffer); } #[no_mangle] pub extern "C" fn wasm_module_imports(module: &wasm_module_t, out: &mut wasm_importtype_vec_t) { let buffer = module .imports .iter() .map(|it| Some(Box::new(it.clone()))) .collect::>(); out.set_buffer(buffer); } #[no_mangle] pub extern "C" fn wasm_module_share(module: &wasm_module_t) -> Box { Box::new(wasm_shared_module_t { module: module.module.clone(), }) } #[no_mangle] pub extern "C" fn wasm_module_obtain( store: &wasm_store_t, shared_module: &wasm_shared_module_t, ) -> Option> { let module = shared_module.module.clone(); if !Engine::same(store.store.engine(), module.engine()) { return None; } let imports = module .imports() .map(|i| { wasm_importtype_t::new( i.module().to_owned(), i.name().map(|s| s.to_owned()), i.ty(), ) }) .collect::>(); let exports = module .exports() .map(|e| wasm_exporttype_t::new(e.name().to_owned(), e.ty())) .collect::>(); Some(Box::new(wasm_module_t { module: module, imports, exports, })) } #[no_mangle] pub extern "C" fn wasm_module_serialize(module: &wasm_module_t, ret: &mut wasm_byte_vec_t) { drop(wasmtime_module_serialize(module, ret)); } #[no_mangle] pub extern "C" fn wasm_module_deserialize( store: &wasm_store_t, binary: &wasm_byte_vec_t, ) -> Option> { let mut ret = ptr::null_mut(); let engine = wasm_engine_t { engine: store.store.engine().clone(), }; match wasmtime_module_deserialize(&engine, binary, &mut ret) { Some(_err) => None, None => { assert!(!ret.is_null()); Some(unsafe { Box::from_raw(ret) }) } } } #[no_mangle] pub extern "C" fn wasmtime_module_serialize( module: &wasm_module_t, ret: &mut wasm_byte_vec_t, ) -> Option> { handle_result(module.module.serialize(), |buf| { ret.set_buffer(buf); }) } #[no_mangle] pub extern "C" fn wasmtime_module_deserialize( engine: &wasm_engine_t, binary: &wasm_byte_vec_t, ret: &mut *mut wasm_module_t, ) -> Option> { handle_result( Module::deserialize(&engine.engine, binary.as_slice()), |module| { let imports = module .imports() .map(|i| { wasm_importtype_t::new( i.module().to_owned(), i.name().map(|s| s.to_owned()), i.ty(), ) }) .collect::>(); let exports = module .exports() .map(|e| wasm_exporttype_t::new(e.name().to_owned(), e.ty())) .collect::>(); let module = Box::new(wasm_module_t { module: module, imports, exports, }); *ret = Box::into_raw(module); }, ) }