wasmer/lib/runtime-core/src/backing.rs

592 lines
22 KiB
Rust
Raw Normal View History

2019-01-08 17:09:47 +00:00
use crate::{
error::{LinkError, LinkResult},
2019-01-11 03:59:57 +00:00
export::{Context, Export},
2019-01-28 19:55:44 +00:00
global::Global,
import::ImportObject,
2019-02-05 07:07:58 +00:00
memory::Memory,
2019-01-13 03:02:19 +00:00
module::{ImportName, ModuleInner},
2019-01-16 18:26:10 +00:00
structures::{BoxedMap, Map, SliceMap, TypedIndex},
2019-01-29 18:16:39 +00:00
table::Table,
2019-01-16 18:26:10 +00:00
types::{
2019-01-29 18:16:39 +00:00
ImportedFuncIndex, ImportedGlobalIndex, ImportedMemoryIndex, ImportedTableIndex,
Initializer, LocalGlobalIndex, LocalMemoryIndex, LocalOrImport, LocalTableIndex, Value,
2019-01-16 18:26:10 +00:00
},
2019-01-08 17:09:47 +00:00
vm,
};
2019-01-29 18:16:39 +00:00
use std::slice;
2019-01-08 17:09:47 +00:00
#[derive(Debug)]
pub struct LocalBacking {
2019-02-05 07:07:58 +00:00
pub(crate) memories: BoxedMap<LocalMemoryIndex, Memory>,
2019-01-29 18:16:39 +00:00
pub(crate) tables: BoxedMap<LocalTableIndex, Table>,
2019-01-28 19:55:44 +00:00
pub(crate) globals: BoxedMap<LocalGlobalIndex, Global>,
2019-01-08 17:09:47 +00:00
pub(crate) vm_memories: BoxedMap<LocalMemoryIndex, *mut vm::LocalMemory>,
2019-01-29 18:16:39 +00:00
pub(crate) vm_tables: BoxedMap<LocalTableIndex, *mut vm::LocalTable>,
2019-01-28 19:55:44 +00:00
pub(crate) vm_globals: BoxedMap<LocalGlobalIndex, *mut vm::LocalGlobal>,
2019-01-08 17:09:47 +00:00
}
// impl LocalBacking {
// pub fn memory(&mut self, local_memory_index: LocalMemoryIndex) -> &mut Memory {
// &mut self.memories[local_memory_index]
// }
2019-01-13 03:02:19 +00:00
// pub fn table(&mut self, local_table_index: LocalTableIndex) -> &mut TableBacking {
// &mut self.tables[local_table_index]
// }
// }
2019-01-13 03:02:19 +00:00
impl LocalBacking {
pub(crate) fn new(module: &ModuleInner, imports: &ImportBacking, vmctx: *mut vm::Ctx) -> Self {
2019-01-08 17:09:47 +00:00
let mut memories = Self::generate_memories(module);
let mut tables = Self::generate_tables(module);
2019-01-28 19:55:44 +00:00
let mut globals = Self::generate_globals(module, imports);
2019-01-08 17:09:47 +00:00
2019-01-16 18:26:10 +00:00
let vm_memories = Self::finalize_memories(module, imports, &mut memories);
let vm_tables = Self::finalize_tables(module, imports, &mut tables, vmctx);
2019-01-28 19:55:44 +00:00
let vm_globals = Self::finalize_globals(&mut globals);
2019-01-08 17:09:47 +00:00
Self {
memories,
tables,
2019-01-28 19:55:44 +00:00
globals,
2019-01-08 17:09:47 +00:00
vm_memories,
vm_tables,
vm_globals,
}
}
2019-02-05 07:07:58 +00:00
fn generate_memories(module: &ModuleInner) -> BoxedMap<LocalMemoryIndex, Memory> {
2019-01-16 18:26:10 +00:00
let mut memories = Map::with_capacity(module.memories.len());
2019-01-08 17:09:47 +00:00
for (_, &desc) in &module.memories {
2019-02-05 07:07:58 +00:00
memories.push(Memory::new(desc).expect("unable to create memory"));
2019-01-08 17:09:47 +00:00
}
2019-01-16 18:26:10 +00:00
memories.into_boxed_map()
2019-01-08 17:09:47 +00:00
}
2019-01-13 03:02:19 +00:00
fn finalize_memories(
module: &ModuleInner,
2019-01-16 18:26:10 +00:00
imports: &ImportBacking,
2019-02-05 07:07:58 +00:00
memories: &mut SliceMap<LocalMemoryIndex, Memory>,
) -> BoxedMap<LocalMemoryIndex, *mut vm::LocalMemory> {
2019-01-16 18:26:10 +00:00
// For each init that has some data...
for init in module
.data_initializers
.iter()
.filter(|init| init.data.len() > 0)
{
2019-01-17 01:59:12 +00:00
let init_base = match init.base {
Initializer::Const(Value::I32(offset)) => offset as u32,
Initializer::Const(_) => panic!("a const initializer must be the i32 type"),
2019-01-28 19:55:44 +00:00
Initializer::GetGlobal(import_global_index) => {
if let Value::I32(x) = imports.globals[import_global_index].get() {
x as u32
2019-01-17 01:59:12 +00:00
} else {
panic!("unsupported global type for initialzer")
}
}
} as usize;
2019-01-16 18:26:10 +00:00
match init.memory_index.local_or_import(module) {
LocalOrImport::Local(local_memory_index) => {
let memory_desc = module.memories[local_memory_index];
2019-01-17 01:59:12 +00:00
let data_top = init_base + init.data.len();
2019-01-29 23:44:15 +00:00
assert!(memory_desc.minimum.bytes().0 >= data_top);
2019-01-28 18:59:05 +00:00
let mem = &memories[local_memory_index];
2019-02-05 07:07:58 +00:00
for (mem_byte, data_byte) in mem
.view(init_base..init_base + init.data.len())
.unwrap()
.iter()
.zip(init.data.iter())
{
mem_byte.set(*data_byte);
2019-02-04 23:07:32 +00:00
}
2019-01-16 18:26:10 +00:00
}
LocalOrImport::Import(imported_memory_index) => {
// Write the initialization data to the memory that
// we think the imported memory is.
2019-01-17 21:09:05 +00:00
unsafe {
let local_memory = &*imports.vm_memories[imported_memory_index];
2019-01-17 21:09:05 +00:00
let memory_slice =
slice::from_raw_parts_mut(local_memory.base, local_memory.bound);
2019-01-17 21:09:05 +00:00
let mem_init_view =
&mut memory_slice[init_base..init_base + init.data.len()];
mem_init_view.copy_from_slice(&init.data);
}
2019-01-16 18:26:10 +00:00
}
}
2019-01-08 17:09:47 +00:00
}
memories
.iter_mut()
2019-02-05 07:07:58 +00:00
.map(|(_, mem)| mem.vm_local_memory())
2019-01-16 18:26:10 +00:00
.collect::<Map<_, _>>()
.into_boxed_map()
2019-01-08 17:09:47 +00:00
}
2019-01-29 18:16:39 +00:00
fn generate_tables(module: &ModuleInner) -> BoxedMap<LocalTableIndex, Table> {
2019-01-16 18:26:10 +00:00
let mut tables = Map::with_capacity(module.tables.len());
2019-01-08 17:09:47 +00:00
2019-01-29 18:16:39 +00:00
for (_, &table_desc) in module.tables.iter() {
let table = Table::new(table_desc).unwrap();
tables.push(table);
2019-01-08 17:09:47 +00:00
}
2019-01-16 18:26:10 +00:00
tables.into_boxed_map()
2019-01-08 17:09:47 +00:00
}
2019-01-18 06:33:46 +00:00
#[allow(clippy::cast_ptr_alignment)]
2019-01-08 17:09:47 +00:00
fn finalize_tables(
2019-01-13 03:02:19 +00:00
module: &ModuleInner,
2019-01-08 17:09:47 +00:00
imports: &ImportBacking,
2019-01-29 18:16:39 +00:00
tables: &mut SliceMap<LocalTableIndex, Table>,
2019-01-11 03:59:57 +00:00
vmctx: *mut vm::Ctx,
2019-01-29 18:16:39 +00:00
) -> BoxedMap<LocalTableIndex, *mut vm::LocalTable> {
2019-01-17 01:59:12 +00:00
for init in &module.elem_initializers {
2019-01-16 18:26:10 +00:00
let init_base = match init.base {
Initializer::Const(Value::I32(offset)) => offset as u32,
Initializer::Const(_) => panic!("a const initializer must be the i32 type"),
2019-01-28 19:55:44 +00:00
Initializer::GetGlobal(import_global_index) => {
if let Value::I32(x) = imports.globals[import_global_index].get() {
x as u32
2019-01-16 18:26:10 +00:00
} else {
panic!("unsupported global type for initialzer")
}
}
} as usize;
match init.table_index.local_or_import(module) {
LocalOrImport::Local(local_table_index) => {
2019-01-29 18:16:39 +00:00
let table = &tables[local_table_index];
2019-01-29 23:44:15 +00:00
if (table.size() as usize) < init_base + init.elements.len() {
let delta = (init_base + init.elements.len()) - table.size() as usize;
2019-01-29 18:16:39 +00:00
// Grow the table if it's too small.
table.grow(delta as u32).expect("couldn't grow table");
2019-01-16 18:26:10 +00:00
}
2019-01-17 21:09:05 +00:00
2019-01-29 18:16:39 +00:00
table.anyfunc_direct_access_mut(|elements| {
for (i, &func_index) in init.elements.iter().enumerate() {
let sig_index = module.func_assoc[func_index];
let sig_id = vm::SigId(sig_index.index() as u32);
let (func, ctx) = match func_index.local_or_import(module) {
LocalOrImport::Local(local_func_index) => (
module
.func_resolver
.get(module, local_func_index)
.unwrap()
.as_ptr()
as *const vm::Func,
vmctx,
),
LocalOrImport::Import(imported_func_index) => {
let vm::ImportedFunc { func, vmctx } =
imports.vm_functions[imported_func_index];
(func, vmctx)
2019-01-17 21:09:05 +00:00
}
2019-01-29 18:16:39 +00:00
};
elements[init_base + i] = vm::Anyfunc { func, ctx, sig_id };
2019-01-16 18:26:10 +00:00
}
2019-01-29 18:16:39 +00:00
});
}
LocalOrImport::Import(import_table_index) => {
let table = &imports.tables[import_table_index];
2019-01-29 23:44:15 +00:00
if (table.size() as usize) < init_base + init.elements.len() {
let delta = (init_base + init.elements.len()) - table.size() as usize;
2019-01-29 18:16:39 +00:00
// Grow the table if it's too small.
table.grow(delta as u32).expect("couldn't grow table");
2019-01-08 17:09:47 +00:00
}
2019-01-29 18:16:39 +00:00
table.anyfunc_direct_access_mut(|elements| {
for (i, &func_index) in init.elements.iter().enumerate() {
let sig_index = module.func_assoc[func_index];
let sig_id = vm::SigId(sig_index.index() as u32);
let (func, ctx) = match func_index.local_or_import(module) {
LocalOrImport::Local(local_func_index) => (
module
.func_resolver
.get(module, local_func_index)
.unwrap()
.as_ptr()
as *const vm::Func,
vmctx,
),
LocalOrImport::Import(imported_func_index) => {
let vm::ImportedFunc { func, vmctx } =
imports.vm_functions[imported_func_index];
(func, vmctx)
}
};
elements[init_base + i] = vm::Anyfunc { func, ctx, sig_id };
}
});
2019-01-08 17:09:47 +00:00
}
}
}
tables
.iter_mut()
2019-01-29 18:16:39 +00:00
.map(|(_, table)| table.vm_local_table())
2019-01-16 18:26:10 +00:00
.collect::<Map<_, _>>()
.into_boxed_map()
2019-01-08 17:09:47 +00:00
}
2019-01-28 19:55:44 +00:00
fn generate_globals(
module: &ModuleInner,
imports: &ImportBacking,
) -> BoxedMap<LocalGlobalIndex, Global> {
2019-01-16 18:26:10 +00:00
let mut globals = Map::with_capacity(module.globals.len());
2019-01-28 19:55:44 +00:00
for (_, global_init) in module.globals.iter() {
let value = match &global_init.init {
Initializer::Const(value) => value.clone(),
Initializer::GetGlobal(import_global_index) => {
imports.globals[*import_global_index].get()
}
};
let global = if global_init.desc.mutable {
Global::new_mutable(value)
} else {
Global::new(value)
};
globals.push(global);
}
2019-01-08 17:09:47 +00:00
2019-01-16 18:26:10 +00:00
globals.into_boxed_map()
2019-01-08 17:09:47 +00:00
}
fn finalize_globals(
2019-01-28 19:55:44 +00:00
globals: &mut SliceMap<LocalGlobalIndex, Global>,
) -> BoxedMap<LocalGlobalIndex, *mut vm::LocalGlobal> {
2019-01-08 17:09:47 +00:00
globals
2019-01-28 19:55:44 +00:00
.iter_mut()
.map(|(_, global)| global.vm_local_global())
.collect::<Map<_, _>>()
.into_boxed_map()
2019-01-08 17:09:47 +00:00
}
}
#[derive(Debug)]
pub struct ImportBacking {
2019-02-05 07:07:58 +00:00
pub(crate) memories: BoxedMap<ImportedMemoryIndex, Memory>,
2019-01-29 18:16:39 +00:00
pub(crate) tables: BoxedMap<ImportedTableIndex, Table>,
2019-01-28 19:55:44 +00:00
pub(crate) globals: BoxedMap<ImportedGlobalIndex, Global>,
pub(crate) vm_functions: BoxedMap<ImportedFuncIndex, vm::ImportedFunc>,
pub(crate) vm_memories: BoxedMap<ImportedMemoryIndex, *mut vm::LocalMemory>,
2019-01-29 18:16:39 +00:00
pub(crate) vm_tables: BoxedMap<ImportedTableIndex, *mut vm::LocalTable>,
2019-01-28 19:55:44 +00:00
pub(crate) vm_globals: BoxedMap<ImportedGlobalIndex, *mut vm::LocalGlobal>,
2019-01-08 17:09:47 +00:00
}
impl ImportBacking {
2019-01-11 03:59:57 +00:00
pub fn new(
2019-01-13 03:02:19 +00:00
module: &ModuleInner,
2019-02-02 23:58:33 +00:00
imports: &ImportObject,
2019-01-11 03:59:57 +00:00
vmctx: *mut vm::Ctx,
) -> LinkResult<Self> {
2019-01-18 22:18:06 +00:00
let mut failed = false;
2019-01-18 21:44:44 +00:00
let mut link_errors = vec![];
let vm_functions = import_functions(module, imports, vmctx).unwrap_or_else(|le| {
2019-01-18 22:18:06 +00:00
failed = true;
2019-01-18 21:44:44 +00:00
link_errors.extend(le);
Map::new().into_boxed_map()
});
2019-01-28 19:55:44 +00:00
let (memories, vm_memories) = import_memories(module, imports).unwrap_or_else(|le| {
2019-01-18 22:18:06 +00:00
failed = true;
2019-01-18 21:44:44 +00:00
link_errors.extend(le);
(Map::new().into_boxed_map(), Map::new().into_boxed_map())
2019-01-18 21:44:44 +00:00
});
2019-01-29 18:16:39 +00:00
let (tables, vm_tables) = import_tables(module, imports).unwrap_or_else(|le| {
2019-01-18 22:18:06 +00:00
failed = true;
2019-01-18 21:44:44 +00:00
link_errors.extend(le);
2019-01-29 18:16:39 +00:00
(Map::new().into_boxed_map(), Map::new().into_boxed_map())
2019-01-18 21:44:44 +00:00
});
2019-01-28 19:55:44 +00:00
let (globals, vm_globals) = import_globals(module, imports).unwrap_or_else(|le| {
2019-01-18 22:18:06 +00:00
failed = true;
2019-01-18 21:44:44 +00:00
link_errors.extend(le);
2019-01-28 19:55:44 +00:00
(Map::new().into_boxed_map(), Map::new().into_boxed_map())
2019-01-18 21:44:44 +00:00
});
2019-01-18 22:18:06 +00:00
if failed {
2019-01-18 21:44:44 +00:00
Err(link_errors)
} else {
Ok(ImportBacking {
memories,
2019-01-29 18:16:39 +00:00
tables,
2019-01-28 19:55:44 +00:00
globals,
vm_functions,
vm_memories,
vm_tables,
vm_globals,
2019-01-18 21:44:44 +00:00
})
}
}
2019-01-17 21:09:05 +00:00
pub fn imported_func(&self, index: ImportedFuncIndex) -> vm::ImportedFunc {
self.vm_functions[index].clone()
2019-01-17 21:09:05 +00:00
}
}
2019-01-16 18:26:10 +00:00
fn import_functions(
2019-01-13 03:02:19 +00:00
module: &ModuleInner,
2019-02-02 23:58:33 +00:00
imports: &ImportObject,
2019-01-12 20:34:23 +00:00
vmctx: *mut vm::Ctx,
) -> LinkResult<BoxedMap<ImportedFuncIndex, vm::ImportedFunc>> {
2019-01-18 21:44:44 +00:00
let mut link_errors = vec![];
2019-01-16 18:26:10 +00:00
let mut functions = Map::with_capacity(module.imported_functions.len());
for (index, ImportName { namespace, name }) in &module.imported_functions {
let sig_index = module.func_assoc[index.convert_up(module)];
2019-01-29 18:16:39 +00:00
let expected_sig = module.sig_registry.lookup_signature(sig_index);
2019-01-16 18:26:10 +00:00
let import = imports
2019-01-13 03:02:19 +00:00
.get_namespace(namespace)
.and_then(|namespace| namespace.get_export(name));
2019-01-16 18:26:10 +00:00
match import {
Some(Export::Function {
func,
2019-01-12 20:34:23 +00:00
ctx,
2019-01-16 18:26:10 +00:00
signature,
2019-01-12 20:34:23 +00:00
}) => {
2019-01-29 18:16:39 +00:00
if expected_sig == signature {
2019-01-16 18:26:10 +00:00
functions.push(vm::ImportedFunc {
func: func.inner(),
vmctx: match ctx {
Context::External(ctx) => ctx,
Context::Internal => vmctx,
},
});
} else {
2019-01-18 21:44:44 +00:00
link_errors.push(LinkError::IncorrectImportSignature {
namespace: namespace.clone(),
name: name.clone(),
expected: expected_sig.clone(),
found: signature.clone(),
2019-01-18 21:44:44 +00:00
});
}
2019-01-12 20:34:23 +00:00
}
Some(export_type) => {
let export_type_name = match export_type {
Export::Function { .. } => "function",
Export::Memory { .. } => "memory",
Export::Table { .. } => "table",
Export::Global { .. } => "global",
}
.to_string();
2019-01-18 21:44:44 +00:00
link_errors.push(LinkError::IncorrectImportType {
namespace: namespace.clone(),
name: name.clone(),
expected: "function".to_string(),
found: export_type_name,
2019-01-18 21:44:44 +00:00
});
}
None => {
link_errors.push(LinkError::ImportNotFound {
namespace: namespace.clone(),
name: name.clone(),
});
2019-01-12 20:34:23 +00:00
}
}
}
2019-01-18 21:44:44 +00:00
if link_errors.len() > 0 {
Err(link_errors)
} else {
Ok(functions.into_boxed_map())
}
2019-01-12 20:34:23 +00:00
}
2019-01-16 18:26:10 +00:00
fn import_memories(
2019-01-13 16:46:04 +00:00
module: &ModuleInner,
2019-02-02 23:58:33 +00:00
imports: &ImportObject,
) -> LinkResult<(
2019-02-05 07:07:58 +00:00
BoxedMap<ImportedMemoryIndex, Memory>,
2019-01-28 19:55:44 +00:00
BoxedMap<ImportedMemoryIndex, *mut vm::LocalMemory>,
)> {
2019-01-18 21:44:44 +00:00
let mut link_errors = vec![];
2019-01-16 18:26:10 +00:00
let mut memories = Map::with_capacity(module.imported_memories.len());
let mut vm_memories = Map::with_capacity(module.imported_memories.len());
2019-01-16 18:26:10 +00:00
for (_index, (ImportName { namespace, name }, expected_memory_desc)) in
&module.imported_memories
{
let memory_import = imports
.get_namespace(&namespace)
.and_then(|namespace| namespace.get_export(&name));
2019-01-16 18:26:10 +00:00
match memory_import {
2019-02-05 07:07:58 +00:00
Some(Export::Memory(memory)) => {
if expected_memory_desc.fits_in_imported(memory.descriptor()) {
memories.push(memory.clone());
2019-02-05 07:07:58 +00:00
vm_memories.push(memory.vm_local_memory());
2019-01-13 21:52:25 +00:00
} else {
2019-01-29 21:04:42 +00:00
link_errors.push(LinkError::IncorrectMemoryDescriptor {
namespace: namespace.clone(),
name: name.clone(),
expected: *expected_memory_desc,
2019-02-05 07:07:58 +00:00
found: memory.descriptor(),
2019-01-18 21:44:44 +00:00
});
2019-01-13 16:46:04 +00:00
}
}
Some(export_type) => {
let export_type_name = match export_type {
Export::Function { .. } => "function",
Export::Memory { .. } => "memory",
Export::Table { .. } => "table",
Export::Global { .. } => "global",
}
.to_string();
2019-01-18 21:44:44 +00:00
link_errors.push(LinkError::IncorrectImportType {
namespace: namespace.clone(),
name: name.clone(),
expected: "memory".to_string(),
found: export_type_name,
2019-01-18 21:44:44 +00:00
});
}
None => {
link_errors.push(LinkError::ImportNotFound {
namespace: namespace.clone(),
name: name.clone(),
});
2019-01-13 21:52:25 +00:00
}
2019-01-13 16:46:04 +00:00
}
2019-01-13 21:52:25 +00:00
}
2019-01-18 21:44:44 +00:00
if link_errors.len() > 0 {
Err(link_errors)
} else {
2019-01-28 19:55:44 +00:00
Ok((memories.into_boxed_map(), vm_memories.into_boxed_map()))
2019-01-18 21:44:44 +00:00
}
2019-01-13 16:46:04 +00:00
}
2019-01-16 18:26:10 +00:00
fn import_tables(
2019-01-13 03:02:19 +00:00
module: &ModuleInner,
2019-02-02 23:58:33 +00:00
imports: &ImportObject,
2019-01-29 18:16:39 +00:00
) -> LinkResult<(
BoxedMap<ImportedTableIndex, Table>,
BoxedMap<ImportedTableIndex, *mut vm::LocalTable>,
)> {
2019-01-18 21:44:44 +00:00
let mut link_errors = vec![];
2019-01-16 18:26:10 +00:00
let mut tables = Map::with_capacity(module.imported_tables.len());
2019-01-29 18:16:39 +00:00
let mut vm_tables = Map::with_capacity(module.imported_tables.len());
2019-01-16 18:26:10 +00:00
for (_index, (ImportName { namespace, name }, expected_table_desc)) in &module.imported_tables {
let table_import = imports
.get_namespace(&namespace)
.and_then(|namespace| namespace.get_export(&name));
2019-01-16 18:26:10 +00:00
match table_import {
2019-01-29 18:16:39 +00:00
Some(Export::Table(mut table)) => {
2019-01-29 21:04:42 +00:00
if expected_table_desc.fits_in_imported(table.descriptor()) {
2019-01-29 18:16:39 +00:00
vm_tables.push(table.vm_local_table());
tables.push(table);
} else {
2019-01-29 21:04:42 +00:00
link_errors.push(LinkError::IncorrectTableDescriptor {
namespace: namespace.clone(),
name: name.clone(),
2019-01-29 18:16:39 +00:00
expected: *expected_table_desc,
2019-01-29 21:04:42 +00:00
found: table.descriptor(),
2019-01-18 21:44:44 +00:00
});
2019-01-08 17:09:47 +00:00
}
}
Some(export_type) => {
let export_type_name = match export_type {
Export::Function { .. } => "function",
Export::Memory { .. } => "memory",
Export::Table { .. } => "table",
Export::Global { .. } => "global",
}
.to_string();
2019-01-18 21:44:44 +00:00
link_errors.push(LinkError::IncorrectImportType {
namespace: namespace.clone(),
name: name.clone(),
expected: "table".to_string(),
found: export_type_name,
2019-01-18 21:44:44 +00:00
});
}
None => {
link_errors.push(LinkError::ImportNotFound {
namespace: namespace.clone(),
name: name.clone(),
});
2019-01-08 17:09:47 +00:00
}
}
}
2019-01-18 21:44:44 +00:00
if link_errors.len() > 0 {
Err(link_errors)
} else {
2019-01-29 18:16:39 +00:00
Ok((tables.into_boxed_map(), vm_tables.into_boxed_map()))
2019-01-18 21:44:44 +00:00
}
}
2019-01-08 17:09:47 +00:00
2019-01-11 03:59:57 +00:00
fn import_globals(
2019-01-13 03:02:19 +00:00
module: &ModuleInner,
2019-02-02 23:58:33 +00:00
imports: &ImportObject,
2019-01-28 19:55:44 +00:00
) -> LinkResult<(
BoxedMap<ImportedGlobalIndex, Global>,
BoxedMap<ImportedGlobalIndex, *mut vm::LocalGlobal>,
)> {
2019-01-18 21:44:44 +00:00
let mut link_errors = vec![];
2019-01-16 18:26:10 +00:00
let mut globals = Map::with_capacity(module.imported_globals.len());
2019-01-28 19:55:44 +00:00
let mut vm_globals = Map::with_capacity(module.imported_globals.len());
2019-01-17 01:59:12 +00:00
for (_, (ImportName { namespace, name }, imported_global_desc)) in &module.imported_globals {
2019-01-13 03:02:19 +00:00
let import = imports
.get_namespace(namespace)
.and_then(|namespace| namespace.get_export(name));
match import {
2019-01-28 19:55:44 +00:00
Some(Export::Global(mut global)) => {
2019-01-29 21:04:42 +00:00
if global.descriptor() == *imported_global_desc {
2019-01-28 19:55:44 +00:00
vm_globals.push(global.vm_local_global());
globals.push(global);
} else {
2019-01-29 21:04:42 +00:00
link_errors.push(LinkError::IncorrectGlobalDescriptor {
namespace: namespace.clone(),
name: name.clone(),
2019-01-28 19:55:44 +00:00
expected: *imported_global_desc,
2019-01-29 21:04:42 +00:00
found: global.descriptor(),
2019-01-18 21:44:44 +00:00
});
2019-01-08 17:09:47 +00:00
}
}
Some(export_type) => {
let export_type_name = match export_type {
Export::Function { .. } => "function",
Export::Memory { .. } => "memory",
Export::Table { .. } => "table",
Export::Global { .. } => "global",
}
.to_string();
2019-01-18 21:44:44 +00:00
link_errors.push(LinkError::IncorrectImportType {
namespace: namespace.clone(),
name: name.clone(),
expected: "global".to_string(),
found: export_type_name,
2019-01-18 21:44:44 +00:00
});
}
None => {
link_errors.push(LinkError::ImportNotFound {
namespace: namespace.clone(),
name: name.clone(),
});
2019-01-08 17:09:47 +00:00
}
}
}
2019-01-18 21:44:44 +00:00
if link_errors.len() > 0 {
Err(link_errors)
} else {
2019-01-28 19:55:44 +00:00
Ok((globals.into_boxed_map(), vm_globals.into_boxed_map()))
2019-01-18 21:44:44 +00:00
}
2019-01-08 17:09:47 +00:00
}