use anyhow::Result;
use cranelift_codegen::{
isa::{self, OwnedTargetIsa},
CodegenResult,
};
use std::fmt;
use std::path;
use std::sync::Arc;
use target_lexicon::Triple;
use wasmtime_cranelift_shared::isa_builder::IsaBuilder;
use wasmtime_environ::{CacheStore, CompilerBuilder, Setting, Tunables};
struct Builder {
tunables: Option<Tunables>,
inner: IsaBuilder<CodegenResult<OwnedTargetIsa>>,
linkopts: LinkOptions,
cache_store: Option<Arc<dyn CacheStore>>,
clif_dir: Option<path::PathBuf>,
wmemcheck: bool,
}
#[derive(Clone, Default)]
pub struct LinkOptions {
pub padding_between_functions: usize,
pub force_jump_veneers: bool,
}
pub fn builder(triple: Option<Triple>) -> Result<Box<dyn CompilerBuilder>> {
Ok(Box::new(Builder {
tunables: None,
inner: IsaBuilder::new(triple, |triple| isa::lookup(triple).map_err(|e| e.into()))?,
linkopts: LinkOptions::default(),
cache_store: None,
clif_dir: None,
wmemcheck: false,
}))
}
impl CompilerBuilder for Builder {
fn triple(&self) -> &target_lexicon::Triple {
self.inner.triple()
}
fn clif_dir(&mut self, path: &path::Path) -> Result<()> {
self.clif_dir = Some(path.to_path_buf());
Ok(())
}
fn target(&mut self, target: target_lexicon::Triple) -> Result<()> {
self.inner.target(target)?;
Ok(())
}
fn set(&mut self, name: &str, value: &str) -> Result<()> {
if name == "wasmtime_linkopt_padding_between_functions" {
self.linkopts.padding_between_functions = value.parse()?;
return Ok(());
}
if name == "wasmtime_linkopt_force_jump_veneer" {
self.linkopts.force_jump_veneers = value.parse()?;
return Ok(());
}
self.inner.set(name, value)
}
fn enable(&mut self, name: &str) -> Result<()> {
self.inner.enable(name)
}
fn set_tunables(&mut self, tunables: Tunables) -> Result<()> {
self.tunables = Some(tunables);
Ok(())
}
fn build(&self) -> Result<Box<dyn wasmtime_environ::Compiler>> {
let isa = self.inner.build()?;
Ok(Box::new(crate::compiler::Compiler::new(
self.tunables
.as_ref()
.expect("set_tunables not called")
.clone(),
isa,
self.cache_store.clone(),
self.linkopts.clone(),
self.clif_dir.clone(),
self.wmemcheck,
)))
}
fn settings(&self) -> Vec<Setting> {
self.inner.settings()
}
fn enable_incremental_compilation(
&mut self,
cache_store: Arc<dyn wasmtime_environ::CacheStore>,
) -> Result<()> {
self.cache_store = Some(cache_store);
Ok(())
}
fn wmemcheck(&mut self, enable: bool) {
self.wmemcheck = enable;
}
}
impl fmt::Debug for Builder {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_struct("Builder")
.field("shared_flags", &self.inner.shared_flags().to_string())
.finish()
}
}