use anyhow::Result;
use cranelift_codegen::isa;
use cranelift_codegen::settings::{self, Configurable, SetError};
use std::fmt;
use std::sync::Arc;
use wasmtime_environ::{CacheStore, CompilerBuilder, Setting, SettingKind};
struct Builder {
flags: settings::Builder,
isa_flags: isa::Builder,
linkopts: LinkOptions,
cache_store: Option<Arc<dyn CacheStore>>,
}
#[derive(Clone, Default)]
pub struct LinkOptions {
pub padding_between_functions: usize,
pub force_jump_veneers: bool,
}
pub fn builder() -> Box<dyn CompilerBuilder> {
let mut flags = settings::builder();
flags
.enable("avoid_div_traps")
.expect("should be valid flag");
flags
.set("enable_probestack", "false")
.expect("should be valid flag");
Box::new(Builder {
flags,
isa_flags: cranelift_native::builder().expect("host machine is not a supported target"),
linkopts: LinkOptions::default(),
cache_store: None,
})
}
impl CompilerBuilder for Builder {
fn triple(&self) -> &target_lexicon::Triple {
self.isa_flags.triple()
}
fn target(&mut self, target: target_lexicon::Triple) -> Result<()> {
self.isa_flags = isa::lookup(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(());
}
if let Err(err) = self.flags.set(name, value) {
match err {
SetError::BadName(_) => {
self.isa_flags.set(name, value)?;
}
_ => return Err(err.into()),
}
}
Ok(())
}
fn enable(&mut self, name: &str) -> Result<()> {
if let Err(err) = self.flags.enable(name) {
match err {
SetError::BadName(_) => {
self.isa_flags.enable(name)?;
}
_ => return Err(err.into()),
}
}
Ok(())
}
fn build(&self) -> Result<Box<dyn wasmtime_environ::Compiler>> {
let isa = self
.isa_flags
.clone()
.finish(settings::Flags::new(self.flags.clone()))?;
Ok(Box::new(crate::compiler::Compiler::new(
isa,
self.cache_store.clone(),
self.linkopts.clone(),
)))
}
fn settings(&self) -> Vec<Setting> {
self.isa_flags
.iter()
.map(|s| Setting {
description: s.description,
name: s.name,
values: s.values,
kind: match s.kind {
settings::SettingKind::Preset => SettingKind::Preset,
settings::SettingKind::Enum => SettingKind::Enum,
settings::SettingKind::Num => SettingKind::Num,
settings::SettingKind::Bool => SettingKind::Bool,
},
})
.collect()
}
fn enable_incremental_compilation(
&mut self,
cache_store: Arc<dyn wasmtime_environ::CacheStore>,
) {
self.cache_store = Some(cache_store);
}
}
impl fmt::Debug for Builder {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
f.debug_struct("Builder")
.field(
"flags",
&settings::Flags::new(self.flags.clone()).to_string(),
)
.finish()
}
}