182 lines
5.4 KiB
Rust
Raw Normal View History

2019-02-19 09:58:01 -08:00
use crate::cache::{BackendCache, CacheGenerator};
use crate::{resolver::FuncResolverBuilder, signal::Caller, trampoline::Trampolines};
2019-01-16 17:59:12 -08:00
use cranelift_codegen::{ir, isa};
use cranelift_entity::EntityRef;
use cranelift_wasm;
use hashbrown::HashMap;
2019-02-19 09:58:01 -08:00
use std::sync::Arc;
2019-02-22 14:07:03 -08:00
use wasmer_runtime_core::cache::{Artifact, Error as CacheError};
use wasmer_runtime_core::{
2019-03-27 14:01:27 -07:00
backend::{Backend, CompilerConfig},
error::CompileResult,
2019-02-19 15:36:22 -08:00
module::{ModuleInfo, ModuleInner, StringTable},
2019-01-16 17:59:12 -08:00
structures::{Map, TypedIndex},
2019-01-16 10:26:10 -08:00
types::{
2019-01-16 17:59:12 -08:00
FuncIndex, FuncSig, GlobalIndex, LocalFuncIndex, MemoryIndex, SigIndex, TableIndex, Type,
2019-01-16 10:26:10 -08:00
},
};
2019-01-16 17:59:12 -08:00
/// This contains all of the items in a `ModuleInner` except the `func_resolver`.
2019-01-16 10:26:10 -08:00
pub struct Module {
pub info: ModuleInfo,
2019-01-16 17:59:12 -08:00
}
impl Module {
2019-03-27 14:01:27 -07:00
pub fn new(compiler_config: &CompilerConfig) -> Self {
2019-01-16 17:59:12 -08:00
Self {
info: ModuleInfo {
memories: Map::new(),
globals: Map::new(),
tables: Map::new(),
2019-01-18 14:30:15 -08:00
imported_functions: Map::new(),
imported_memories: Map::new(),
imported_tables: Map::new(),
imported_globals: Map::new(),
2019-01-16 17:59:12 -08:00
exports: HashMap::new(),
2019-01-16 17:59:12 -08:00
data_initializers: Vec::new(),
elem_initializers: Vec::new(),
2019-01-16 17:59:12 -08:00
start_func: None,
2019-01-16 17:59:12 -08:00
func_assoc: Map::new(),
signatures: Map::new(),
backend: Backend::Cranelift,
2019-01-16 17:59:12 -08:00
namespace_table: StringTable::new(),
name_table: StringTable::new(),
2019-03-27 14:01:27 -07:00
em_symbol_map: compiler_config.symbol_map.clone(),
custom_sections: HashMap::new(),
2019-01-16 17:59:12 -08:00
},
}
}
pub fn compile(
self,
2019-01-16 17:59:12 -08:00
isa: &isa::TargetIsa,
functions: Map<LocalFuncIndex, ir::Function>,
) -> CompileResult<ModuleInner> {
2019-01-23 00:31:58 -06:00
let (func_resolver_builder, handler_data) =
FuncResolverBuilder::new(isa, functions, &self.info)?;
2019-02-19 09:58:01 -08:00
let trampolines = Arc::new(Trampolines::new(isa, &self.info));
2019-01-18 14:30:15 -08:00
2019-02-20 16:41:41 -08:00
let (func_resolver, backend_cache) = func_resolver_builder.finalize(
&self.info.signatures,
Arc::clone(&trampolines),
handler_data.clone(),
)?;
let cache_gen = Box::new(CacheGenerator::new(
backend_cache,
Arc::clone(&func_resolver.memory),
));
2019-01-18 14:30:15 -08:00
2019-04-12 10:27:14 -07:00
let runnable_module = Caller::new(handler_data, trampolines, func_resolver);
Ok(ModuleInner {
runnable_module: Box::new(runnable_module),
2019-02-20 16:41:41 -08:00
cache_gen,
info: self.info,
})
2019-01-16 17:59:12 -08:00
}
pub fn from_cache(cache: Artifact) -> Result<ModuleInner, CacheError> {
let (info, compiled_code, backend_cache) = BackendCache::from_cache(cache)?;
let (func_resolver_builder, trampolines, handler_data) =
FuncResolverBuilder::new_from_backend_cache(backend_cache, compiled_code, &info)?;
2019-02-20 16:41:41 -08:00
let (func_resolver, backend_cache) = func_resolver_builder
.finalize(
&info.signatures,
Arc::clone(&trampolines),
handler_data.clone(),
)
.map_err(|e| CacheError::Unknown(format!("{:?}", e)))?;
2019-02-20 16:41:41 -08:00
let cache_gen = Box::new(CacheGenerator::new(
backend_cache,
Arc::clone(&func_resolver.memory),
));
2019-04-12 10:27:14 -07:00
let runnable_module = Caller::new(handler_data, trampolines, func_resolver);
Ok(ModuleInner {
runnable_module: Box::new(runnable_module),
2019-02-20 16:41:41 -08:00
cache_gen,
2019-02-19 09:58:01 -08:00
info,
})
}
2019-01-16 17:59:12 -08:00
}
pub struct Converter<T>(pub T);
macro_rules! convert_clif_to_runtime_index {
($clif_index:ident, $runtime_index:ident) => {
impl From<Converter<cranelift_wasm::$clif_index>> for $runtime_index {
fn from(clif_index: Converter<cranelift_wasm::$clif_index>) -> Self {
$runtime_index::new(clif_index.0.index())
}
}
impl From<Converter<$runtime_index>> for cranelift_wasm::$clif_index {
fn from(runtime_index: Converter<$runtime_index>) -> Self {
cranelift_wasm::$clif_index::new(runtime_index.0.index())
}
}
};
($(($clif_index:ident: $runtime_index:ident),)*) => {
$(
convert_clif_to_runtime_index!($clif_index, $runtime_index);
)*
};
}
convert_clif_to_runtime_index![
(FuncIndex: FuncIndex),
(MemoryIndex: MemoryIndex),
(TableIndex: TableIndex),
(GlobalIndex: GlobalIndex),
(SignatureIndex: SigIndex),
];
impl From<Converter<ir::Signature>> for FuncSig {
2019-04-03 00:13:40 -05:00
fn from(signature: Converter<ir::Signature>) -> Self {
2019-01-29 10:16:39 -08:00
FuncSig::new(
signature
2019-01-16 17:59:12 -08:00
.0
.params
.iter()
.map(|param| Converter(param.value_type).into())
2019-01-29 10:16:39 -08:00
.collect::<Vec<_>>(),
signature
2019-01-16 17:59:12 -08:00
.0
.returns
.iter()
.map(|ret| Converter(ret.value_type).into())
2019-01-29 10:16:39 -08:00
.collect::<Vec<_>>(),
)
2019-01-16 17:59:12 -08:00
}
}
2019-01-16 10:26:10 -08:00
2019-01-16 17:59:12 -08:00
impl From<Converter<ir::Type>> for Type {
fn from(ty: Converter<ir::Type>) -> Self {
match ty.0 {
ir::types::I32 => Type::I32,
ir::types::I64 => Type::I64,
ir::types::F32 => Type::F32,
ir::types::F64 => Type::F64,
_ => panic!("unsupported wasm type"),
}
}
}