mod error;
#[cfg(test)]
mod tests;
pub use self::error::InstantiationError;
use super::{element::ElementSegmentKind, export, ConstExpr, InitDataSegment, Module};
use crate::{
core::UntypedVal,
error::ErrorKind,
errors::MemoryError,
func::WasmFuncEntity,
memory::DataSegment,
module::FuncIdx,
value::WithType,
AsContext,
AsContextMut,
ElementSegment,
Error,
Extern,
ExternType,
Func,
Global,
Instance,
InstanceEntity,
InstanceEntityBuilder,
Memory,
Ref,
Table,
Val,
};
impl Module {
pub(crate) fn instantiate<I>(
&self,
mut context: impl AsContextMut,
externals: I,
) -> Result<Instance, Error>
where
I: IntoIterator<Item = Extern, IntoIter: ExactSizeIterator>,
{
let mut context = context.as_context_mut().store;
if !context.can_create_more_instances(1) {
return Err(Error::from(InstantiationError::TooManyInstances));
}
let handle = context.as_context_mut().store.inner.alloc_instance();
let mut builder = InstanceEntity::build(self);
self.extract_imports(&context, &mut builder, externals)?;
self.extract_functions(&mut context, &mut builder, handle);
self.extract_tables(&mut context, &mut builder)?;
self.extract_memories(&mut context, &mut builder)?;
self.extract_globals(&mut context, &mut builder);
self.extract_exports(&mut builder);
self.extract_start_fn(&mut builder);
self.initialize_table_elements(&mut context, &mut builder)?;
self.initialize_memory_data(&mut context, &mut builder)?;
Self::start(&mut context, builder, handle)?;
Ok(handle)
}
fn start(
mut store: impl AsContextMut,
builder: InstanceEntityBuilder,
instance: Instance,
) -> Result<(), Error> {
let opt_start_index = builder.get_start().map(FuncIdx::into_u32);
store
.as_context_mut()
.store
.inner
.initialize_instance(instance, builder.finish());
if let Some(start_index) = opt_start_index {
let start_func = instance
.get_func_by_index(store.as_context_mut(), start_index)
.unwrap_or_else(|| {
panic!("encountered invalid start function after validation: {start_index}")
});
start_func.call(store.as_context_mut(), &[], &mut [])?
}
Ok(())
}
fn extract_imports<I>(
&self,
store: impl AsContext,
builder: &mut InstanceEntityBuilder,
externals: I,
) -> Result<(), InstantiationError>
where
I: IntoIterator<Item = Extern, IntoIter: ExactSizeIterator>,
{
let imports = self.imports();
let externals = externals.into_iter();
if imports.len() != externals.len() {
return Err(InstantiationError::InvalidNumberOfImports {
required: imports.len(),
given: externals.len(),
});
}
for (import, external) in imports.zip(externals) {
match (import.ty(), external) {
(ExternType::Func(expected_signature), Extern::Func(func)) => {
let actual_signature = func.ty(&store);
if &actual_signature != expected_signature {
return Err(InstantiationError::FuncTypeMismatch {
actual: actual_signature,
expected: expected_signature.clone(),
});
}
builder.push_func(func);
}
(ExternType::Table(required), Extern::Table(table)) => {
let imported = table.dynamic_ty(&store);
if !imported.is_subtype_of(required) {
return Err(InstantiationError::TableTypeMismatch {
expected: *required,
actual: imported,
});
}
builder.push_table(table);
}
(ExternType::Memory(required), Extern::Memory(memory)) => {
let imported = memory.dynamic_ty(&store);
if !imported.is_subtype_of(required) {
return Err(InstantiationError::MemoryTypeMismatch {
expected: *required,
actual: imported,
});
}
builder.push_memory(memory);
}
(ExternType::Global(required), Extern::Global(global)) => {
let imported = global.ty(&store);
let required = *required;
if imported != required {
return Err(InstantiationError::GlobalTypeMismatch {
expected: required,
actual: imported,
});
}
builder.push_global(global);
}
(expected_import, actual_extern_val) => {
return Err(InstantiationError::ImportsExternalsMismatch {
expected: expected_import.clone(),
actual: actual_extern_val,
});
}
}
}
Ok(())
}
fn extract_functions(
&self,
mut context: impl AsContextMut,
builder: &mut InstanceEntityBuilder,
handle: Instance,
) {
for (func_type, func_body) in self.internal_funcs() {
let wasm_func = WasmFuncEntity::new(func_type, func_body, handle);
let func = context
.as_context_mut()
.store
.inner
.alloc_func(wasm_func.into());
builder.push_func(func);
}
}
fn extract_tables(
&self,
mut context: impl AsContextMut,
builder: &mut InstanceEntityBuilder,
) -> Result<(), InstantiationError> {
let ctx = context.as_context_mut().store;
if !ctx.can_create_more_tables(self.len_tables()) {
return Err(InstantiationError::TooManyTables);
}
for table_type in self.internal_tables().copied() {
let init = Val::default(table_type.element());
let table =
Table::new(context.as_context_mut(), table_type, init).map_err(|error| {
let error = match error.kind() {
ErrorKind::Table(error) => *error,
error => panic!("unexpected error: {error}"),
};
InstantiationError::FailedToInstantiateTable(error)
})?;
builder.push_table(table);
}
Ok(())
}
fn extract_memories(
&self,
mut context: impl AsContextMut,
builder: &mut InstanceEntityBuilder,
) -> Result<(), InstantiationError> {
let ctx = context.as_context_mut().store;
if !ctx.can_create_more_memories(self.len_memories()) {
return Err(InstantiationError::TooManyMemories);
}
for memory_type in self.internal_memories().copied() {
let memory = Memory::new(context.as_context_mut(), memory_type).map_err(|error| {
let error = match error.kind() {
ErrorKind::Memory(error) => *error,
error => panic!("unexpected error: {error}"),
};
InstantiationError::FailedToInstantiateMemory(error)
})?;
builder.push_memory(memory);
}
Ok(())
}
fn extract_globals(&self, mut context: impl AsContextMut, builder: &mut InstanceEntityBuilder) {
for (global_type, global_init) in self.internal_globals() {
let value_type = global_type.content();
let init_value = Self::eval_init_expr(context.as_context_mut(), builder, global_init);
let mutability = global_type.mutability();
let global = Global::new(
context.as_context_mut(),
init_value.with_type(value_type),
mutability,
);
builder.push_global(global);
}
}
fn eval_init_expr(
context: impl AsContext,
builder: &InstanceEntityBuilder,
init_expr: &ConstExpr,
) -> UntypedVal {
init_expr
.eval_with_context(
|global_index| builder.get_global(global_index).get(&context),
|func_index| <Ref<Func>>::from(builder.get_func(func_index)),
)
.expect("must evaluate to proper value")
}
fn extract_exports(&self, builder: &mut InstanceEntityBuilder) {
for (field, idx) in &self.module_header().exports {
let external = match idx {
export::ExternIdx::Func(func_index) => {
let func_index = func_index.into_u32();
let func = builder.get_func(func_index);
Extern::Func(func)
}
export::ExternIdx::Table(table_index) => {
let table_index = table_index.into_u32();
let table = builder.get_table(table_index);
Extern::Table(table)
}
export::ExternIdx::Memory(memory_index) => {
let memory_index = memory_index.into_u32();
let memory = builder.get_memory(memory_index);
Extern::Memory(memory)
}
export::ExternIdx::Global(global_index) => {
let global_index = global_index.into_u32();
let global = builder.get_global(global_index);
Extern::Global(global)
}
};
builder.push_export(field, external);
}
}
fn extract_start_fn(&self, builder: &mut InstanceEntityBuilder) {
if let Some(start_fn) = self.module_header().start {
builder.set_start(start_fn)
}
}
fn initialize_table_elements(
&self,
mut context: impl AsContextMut,
builder: &mut InstanceEntityBuilder,
) -> Result<(), Error> {
for segment in &self.module_header().element_segments[..] {
let get_global = |index| builder.get_global(index);
let get_func = |index| builder.get_func(index);
let element =
ElementSegment::new(context.as_context_mut(), segment, get_func, get_global);
if let ElementSegmentKind::Active(active) = segment.kind() {
let dst_index = u64::from(Self::eval_init_expr(
context.as_context(),
builder,
active.offset(),
));
let table = builder.get_table(active.table_index().into_u32());
let len_table = table.size(&context);
let len_items = element.size(&context);
dst_index
.checked_add(u64::from(len_items))
.filter(|&max_index| max_index <= len_table)
.ok_or(InstantiationError::ElementSegmentDoesNotFit {
table,
table_index: dst_index,
len: len_items,
})?;
let (table, elem) = context
.as_context_mut()
.store
.inner
.resolve_table_and_element_mut(&table, &element);
table.init(elem.as_ref(), dst_index, 0, len_items, None)?;
elem.drop_items();
}
builder.push_element_segment(element);
}
Ok(())
}
fn initialize_memory_data(
&self,
mut context: impl AsContextMut,
builder: &mut InstanceEntityBuilder,
) -> Result<(), Error> {
for segment in &self.inner.data_segments {
let segment = match segment {
InitDataSegment::Active {
memory_index,
offset,
bytes,
} => {
let memory = builder.get_memory(memory_index.into_u32());
let offset = Self::eval_init_expr(context.as_context(), builder, offset);
let offset = match usize::try_from(u64::from(offset)) {
Ok(offset) => offset,
Err(_) => return Err(Error::from(MemoryError::OutOfBoundsAccess)),
};
memory.write(context.as_context_mut(), offset, bytes)?;
DataSegment::new_active(context.as_context_mut())
}
InitDataSegment::Passive { bytes } => {
DataSegment::new_passive(context.as_context_mut(), bytes)
}
};
builder.push_data_segment(segment);
}
Ok(())
}
}