2019-02-07 00:26:45 +00:00
|
|
|
use crate::relocation::{ExternalRelocation, TrapSink};
|
|
|
|
|
|
|
|
use hashbrown::HashMap;
|
|
|
|
use wasmer_runtime_core::{
|
2019-02-19 17:58:01 +00:00
|
|
|
backend::{
|
|
|
|
sys::Memory,
|
|
|
|
CacheGen,
|
|
|
|
},
|
2019-02-07 00:26:45 +00:00
|
|
|
cache::{Cache, Error},
|
2019-02-19 17:58:01 +00:00
|
|
|
module::{ModuleInfo, ModuleInner},
|
2019-02-07 00:26:45 +00:00
|
|
|
structures::Map,
|
|
|
|
types::{LocalFuncIndex, SigIndex},
|
|
|
|
};
|
2019-02-19 17:58:01 +00:00
|
|
|
use std::{
|
|
|
|
sync::Arc,
|
|
|
|
cell::UnsafeCell,
|
|
|
|
};
|
2019-02-07 00:26:45 +00:00
|
|
|
|
|
|
|
use serde_bench::{deserialize, serialize};
|
|
|
|
|
2019-02-19 17:58:01 +00:00
|
|
|
pub struct CacheGenerator {
|
|
|
|
backend_cache: BackendCache,
|
|
|
|
memory: Arc<Memory>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CacheGenerator {
|
|
|
|
pub fn new(backend_cache: BackendCache, memory: Arc<Memory>) -> Self {
|
|
|
|
Self { backend_cache, memory }
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl CacheGen for CacheGenerator {
|
|
|
|
fn generate_cache(&self, module: &ModuleInner) -> Result<(Box<ModuleInfo>, Box<[u8]>, Arc<Memory>), Error> {
|
|
|
|
let info = Box::new(module.info.clone());
|
2019-02-19 23:36:22 +00:00
|
|
|
Ok((info, self.backend_cache.into_backend_data()?.into_boxed_slice(), Arc::clone(&self.memory)))
|
2019-02-19 17:58:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-02-07 00:26:45 +00:00
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
pub struct TrampolineCache {
|
|
|
|
#[serde(with = "serde_bytes")]
|
|
|
|
pub code: Vec<u8>,
|
|
|
|
pub offsets: HashMap<SigIndex, usize>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
|
|
pub struct BackendCache {
|
|
|
|
pub external_relocs: Map<LocalFuncIndex, Box<[ExternalRelocation]>>,
|
|
|
|
pub offsets: Map<LocalFuncIndex, usize>,
|
2019-02-19 17:58:01 +00:00
|
|
|
pub trap_sink: Arc<TrapSink>,
|
2019-02-07 00:26:45 +00:00
|
|
|
pub trampolines: TrampolineCache,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl BackendCache {
|
|
|
|
pub fn from_cache(cache: Cache) -> Result<(ModuleInfo, Memory, Self), Error> {
|
2019-02-19 23:36:22 +00:00
|
|
|
let (info, backend_data, compiled_code_arc) = cache.consume();
|
2019-02-07 00:26:45 +00:00
|
|
|
|
2019-02-19 23:36:22 +00:00
|
|
|
// If this is the only references to this arc, move the memory out.
|
|
|
|
// else, clone the memory to a new location. This could take a long time,
|
|
|
|
// depending on the throughput of your memcpy implementation.
|
|
|
|
let compiled_code = match Arc::try_unwrap(compiled_code_arc) {
|
|
|
|
Ok(code) => code,
|
|
|
|
Err(arc) => (*arc).clone(),
|
|
|
|
};
|
|
|
|
|
|
|
|
let backend_cache = deserialize(&backend_data)
|
2019-02-07 00:26:45 +00:00
|
|
|
.map_err(|e| Error::DeserializeError(e.to_string()))?;
|
|
|
|
|
|
|
|
Ok((info, compiled_code, backend_cache))
|
|
|
|
}
|
|
|
|
|
2019-02-19 23:36:22 +00:00
|
|
|
pub fn into_backend_data(&self) -> Result<Vec<u8>, Error> {
|
2019-02-07 00:26:45 +00:00
|
|
|
let mut buffer = Vec::new();
|
|
|
|
|
2019-02-19 23:36:22 +00:00
|
|
|
serialize(&mut buffer, self).map_err(|e| Error::SerializeError(e.to_string()))?;
|
2019-02-07 00:26:45 +00:00
|
|
|
|
|
|
|
Ok(buffer)
|
|
|
|
}
|
|
|
|
}
|