mirror of
https://github.com/fluencelabs/wasmer
synced 2024-12-14 06:35:40 +00:00
8fe9b7eac2
* Allow a module to have a different signature registry than the process-specific * Add core ability to build compiled code caches * Remove timing printouts * Serialize/Deserialize memories to reduce copies * Work more on api * Relocate local functions relatively before external functions * Fix incorrect definition in test * merge errors caused by merge * Fix emscripten compile * Fix review comments
129 lines
3.3 KiB
Rust
129 lines
3.3 KiB
Rust
#[cfg(feature = "cache")]
|
|
mod cache;
|
|
mod call;
|
|
mod func_env;
|
|
mod libcalls;
|
|
mod module;
|
|
mod module_env;
|
|
mod relocation;
|
|
mod resolver;
|
|
mod trampoline;
|
|
|
|
use cranelift_codegen::{
|
|
isa,
|
|
settings::{self, Configurable},
|
|
};
|
|
use target_lexicon::Triple;
|
|
#[cfg(feature = "cache")]
|
|
use wasmer_runtime_core::{
|
|
backend::sys::Memory,
|
|
cache::{Cache, Error as CacheError},
|
|
module::ModuleInfo,
|
|
};
|
|
use wasmer_runtime_core::{
|
|
backend::{Compiler, Token},
|
|
error::{CompileError, CompileResult},
|
|
module::ModuleInner,
|
|
};
|
|
#[cfg(feature = "cache")]
|
|
#[macro_use]
|
|
extern crate serde_derive;
|
|
#[cfg(feature = "cache")]
|
|
extern crate serde;
|
|
|
|
use wasmparser::{self, WasmDecoder};
|
|
|
|
pub struct CraneliftCompiler {}
|
|
|
|
impl CraneliftCompiler {
|
|
pub fn new() -> Self {
|
|
Self {}
|
|
}
|
|
}
|
|
|
|
impl Compiler for CraneliftCompiler {
|
|
/// Compiles wasm binary to a wasmer module.
|
|
fn compile(&self, wasm: &[u8], _: Token) -> CompileResult<ModuleInner> {
|
|
validate(wasm)?;
|
|
|
|
let isa = get_isa();
|
|
|
|
let mut module = module::Module::empty();
|
|
let module_env = module_env::ModuleEnv::new(&mut module, &*isa);
|
|
|
|
let func_bodies = module_env.translate(wasm)?;
|
|
|
|
module.compile(&*isa, func_bodies)
|
|
}
|
|
|
|
/// Create a wasmer Module from an already-compiled cache.
|
|
#[cfg(feature = "cache")]
|
|
unsafe fn from_cache(&self, cache: Cache, _: Token) -> Result<ModuleInner, CacheError> {
|
|
module::Module::from_cache(cache)
|
|
}
|
|
|
|
#[cfg(feature = "cache")]
|
|
fn compile_to_backend_cache_data(
|
|
&self,
|
|
wasm: &[u8],
|
|
_: Token,
|
|
) -> CompileResult<(Box<ModuleInfo>, Vec<u8>, Memory)> {
|
|
validate(wasm)?;
|
|
|
|
let isa = get_isa();
|
|
|
|
let mut module = module::Module::empty();
|
|
let module_env = module_env::ModuleEnv::new(&mut module, &*isa);
|
|
|
|
let func_bodies = module_env.translate(wasm)?;
|
|
|
|
let (info, backend_cache, compiled_code) = module
|
|
.compile_to_backend_cache(&*isa, func_bodies)
|
|
.map_err(|e| CompileError::InternalError {
|
|
msg: format!("{:?}", e),
|
|
})?;
|
|
|
|
let buffer =
|
|
backend_cache
|
|
.into_backend_data()
|
|
.map_err(|e| CompileError::InternalError {
|
|
msg: format!("{:?}", e),
|
|
})?;
|
|
|
|
Ok((Box::new(info), buffer, compiled_code))
|
|
}
|
|
}
|
|
|
|
fn get_isa() -> Box<isa::TargetIsa> {
|
|
let flags = {
|
|
let mut builder = settings::builder();
|
|
builder.set("opt_level", "best").unwrap();
|
|
|
|
if cfg!(not(test)) {
|
|
builder.set("enable_verifier", "false").unwrap();
|
|
}
|
|
|
|
let flags = settings::Flags::new(builder);
|
|
debug_assert_eq!(flags.opt_level(), settings::OptLevel::Best);
|
|
flags
|
|
};
|
|
isa::lookup(Triple::host()).unwrap().finish(flags)
|
|
}
|
|
|
|
fn validate(bytes: &[u8]) -> CompileResult<()> {
|
|
let mut parser = wasmparser::ValidatingParser::new(bytes, None);
|
|
loop {
|
|
let state = parser.read();
|
|
match *state {
|
|
wasmparser::ParserState::EndWasm => break Ok(()),
|
|
wasmparser::ParserState::Error(err) => Err(CompileError::ValidationError {
|
|
msg: err.message.to_string(),
|
|
})?,
|
|
_ => {}
|
|
}
|
|
}
|
|
}
|
|
|
|
/// The current version of this crate
|
|
pub const VERSION: &str = env!("CARGO_PKG_VERSION");
|