use super::bindings::wasm_module_t;
use crate::bindings::wasm_byte_vec_new;
use crate::bindings::wasm_byte_vec_new_empty;
use crate::bindings::wasm_byte_vec_t;
use crate::bindings::wasm_module_delete;
use crate::bindings::wasm_module_new;
use crate::bindings::wasm_store_new;
use crate::bindings::wasm_store_t;
use crate::errors::InstantiationError;
use crate::errors::RuntimeError;
use crate::imports::Imports;
use crate::store::AsStoreMut;
use crate::store::AsStoreRef;
use crate::vm::VMInstance;
use crate::Extern;
use crate::IntoBytes;
use crate::{AsEngineRef, ExportType, ImportType};
use bytes::Bytes;
use std::path::Path;
use std::sync::Arc;
use tracing::{debug, warn};
use wasmer_types::{
CompileError, DeserializeError, ExportsIterator, ExternType, FunctionType, GlobalType,
ImportsIterator, MemoryType, ModuleInfo, Mutability, Pages, SerializeError, TableType, Type,
};
pub(crate) struct ModuleHandle {
pub(crate) inner: *mut wasm_module_t,
pub(crate) store: std::sync::Mutex<crate::store::Store>,
}
impl PartialEq for ModuleHandle {
fn eq(&self, other: &Self) -> bool {
self.inner == other.inner
}
}
impl Eq for ModuleHandle {}
impl ModuleHandle {
fn new(engine: &impl AsEngineRef, binary: &[u8]) -> Result<Self, CompileError> {
#[cfg(feature = "wamr")]
let bytes = wasm_byte_vec_t {
size: binary.len(),
data: binary.as_ptr() as _,
num_elems: binary.len(),
size_of_elem: 1,
lock: std::ptr::null_mut(),
};
#[cfg(any(feature = "wasmi", feature = "v8"))]
let bytes = wasm_byte_vec_t {
size: binary.len(),
data: binary.as_ptr() as _,
};
let store = crate::store::Store::new(engine.as_engine_ref().engine().clone());
let inner = unsafe { wasm_module_new(store.inner.store.inner, &bytes as *const _) };
let store = std::sync::Mutex::new(store);
if inner.is_null() {
return Err(CompileError::Validate(format!("module is null")));
}
Ok(ModuleHandle { inner, store })
}
}
impl Drop for ModuleHandle {
fn drop(&mut self) {
unsafe { wasm_module_delete(self.inner) }
}
}
#[derive(Clone, PartialEq, Eq)]
pub struct Module {
pub(crate) handle: Arc<ModuleHandle>,
name: Option<String>,
raw_bytes: Option<Bytes>,
info: ModuleInfo,
}
unsafe impl Send for Module {}
unsafe impl Sync for Module {}
impl Module {
pub(crate) fn from_binary(
_engine: &impl AsEngineRef,
binary: &[u8],
) -> Result<Self, CompileError> {
unsafe { Self::from_binary_unchecked(_engine, binary) }
}
pub(crate) unsafe fn from_binary_unchecked(
engine: &impl AsEngineRef,
binary: &[u8],
) -> Result<Self, CompileError> {
let mut binary = binary.to_vec();
let binary = binary.into_bytes();
let module = ModuleHandle::new(engine, &binary)?;
let info = crate::module_info_polyfill::translate_module(&binary[..])
.unwrap()
.info;
Ok(Self {
handle: Arc::new(module),
name: info.name.clone(),
raw_bytes: Some(binary.into_bytes()),
info,
})
}
pub fn validate(engine: &impl AsEngineRef, binary: &[u8]) -> Result<(), CompileError> {
let engine = engine.as_engine_ref();
unimplemented!();
}
pub fn name(&self) -> Option<&str> {
self.name.as_ref().map(|s| s.as_ref())
}
pub fn serialize(&self) -> Result<Bytes, SerializeError> {
return self.raw_bytes.clone().ok_or(SerializeError::Generic(
"Not able to serialize module".to_string(),
));
}
pub unsafe fn deserialize_unchecked(
engine: &impl AsEngineRef,
bytes: impl IntoBytes,
) -> Result<Self, DeserializeError> {
Self::deserialize(engine, bytes)
}
pub unsafe fn deserialize(
engine: &impl AsEngineRef,
bytes: impl IntoBytes,
) -> Result<Self, DeserializeError> {
return Self::from_binary(engine, &bytes.into_bytes())
.map_err(|e| DeserializeError::Compiler(e));
}
pub unsafe fn deserialize_from_file_unchecked(
engine: &impl AsEngineRef,
path: impl AsRef<Path>,
) -> Result<Self, DeserializeError> {
let bytes = std::fs::read(path.as_ref())?;
Self::deserialize_unchecked(engine, bytes)
}
pub unsafe fn deserialize_from_file(
engine: &impl AsEngineRef,
path: impl AsRef<Path>,
) -> Result<Self, DeserializeError> {
let bytes = std::fs::read(path.as_ref())?;
Self::deserialize(engine, bytes)
}
pub fn set_name(&mut self, name: &str) -> bool {
self.name = Some(name.to_string());
true
}
pub fn imports<'a>(&'a self) -> ImportsIterator<impl Iterator<Item = ImportType> + 'a> {
self.info().imports()
}
pub fn exports<'a>(&'a self) -> ExportsIterator<impl Iterator<Item = ExportType> + 'a> {
self.info().exports()
}
pub fn custom_sections<'a>(&'a self, name: &'a str) -> impl Iterator<Item = Box<[u8]>> + 'a {
self.info().custom_sections(name)
}
pub(crate) fn info(&self) -> &ModuleInfo {
&self.info
}
}