use std::cmp::Reverse;
use crate::progress::ProgressContext;
use crate::types::module::CompileModuleInfo;
use crate::{
FunctionBodyData, ModuleTranslationState,
lib::std::{boxed::Box, sync::Arc},
translator::ModuleMiddleware,
types::function::Compilation,
};
use crossbeam_channel::unbounded;
use enumset::EnumSet;
use itertools::Itertools;
use wasmer_types::{
CompilationProgressCallback, Features, LocalFunctionIndex,
entity::PrimaryMap,
error::CompileError,
target::{CpuFeature, Target, UserCompilerOptimizations},
};
#[cfg(feature = "translator")]
use wasmparser::{Validator, WasmFeatures};
pub trait CompilerConfig {
fn enable_pic(&mut self) {
}
fn enable_verifier(&mut self) {
}
fn enable_perfmap(&mut self) {
}
fn enable_non_volatile_memops(&mut self) {}
fn enable_readonly_funcref_table(&mut self) {}
fn canonicalize_nans(&mut self, _enable: bool) {
}
fn compiler(self: Box<Self>) -> Box<dyn Compiler>;
fn default_features_for_target(&self, target: &Target) -> Features {
self.supported_features_for_target(target)
}
fn supported_features_for_target(&self, _target: &Target) -> Features {
Features::default()
}
fn push_middleware(&mut self, middleware: Arc<dyn ModuleMiddleware>);
}
impl<T> From<T> for Box<dyn CompilerConfig + 'static>
where
T: CompilerConfig + 'static,
{
fn from(other: T) -> Self {
Box::new(other)
}
}
pub trait Compiler: Send + std::fmt::Debug {
fn name(&self) -> &str;
fn deterministic_id(&self) -> String;
fn with_opts(
&mut self,
suggested_compiler_opts: &UserCompilerOptimizations,
) -> Result<(), CompileError> {
_ = suggested_compiler_opts;
Ok(())
}
#[cfg(feature = "translator")]
fn validate_module(&self, features: &Features, data: &[u8]) -> Result<(), CompileError> {
let mut wasm_features = WasmFeatures::empty();
wasm_features.set(WasmFeatures::BULK_MEMORY, features.bulk_memory);
wasm_features.set(WasmFeatures::THREADS, features.threads);
wasm_features.set(WasmFeatures::REFERENCE_TYPES, features.reference_types);
wasm_features.set(WasmFeatures::MULTI_VALUE, features.multi_value);
wasm_features.set(WasmFeatures::SIMD, features.simd);
wasm_features.set(WasmFeatures::TAIL_CALL, features.tail_call);
wasm_features.set(WasmFeatures::MULTI_MEMORY, features.multi_memory);
wasm_features.set(WasmFeatures::MEMORY64, features.memory64);
wasm_features.set(WasmFeatures::EXCEPTIONS, features.exceptions);
wasm_features.set(WasmFeatures::EXTENDED_CONST, features.extended_const);
wasm_features.set(WasmFeatures::RELAXED_SIMD, features.relaxed_simd);
wasm_features.set(WasmFeatures::WIDE_ARITHMETIC, features.wide_arithmetic);
wasm_features.set(WasmFeatures::TAIL_CALL, features.tail_call);
wasm_features.set(WasmFeatures::MUTABLE_GLOBAL, true);
wasm_features.set(WasmFeatures::SATURATING_FLOAT_TO_INT, true);
wasm_features.set(WasmFeatures::FLOATS, true);
wasm_features.set(WasmFeatures::SIGN_EXTENSION, true);
wasm_features.set(WasmFeatures::GC_TYPES, true);
let mut validator = Validator::new_with_features(wasm_features);
validator
.validate_all(data)
.map_err(|e| CompileError::Validate(format!("{e}")))?;
Ok(())
}
fn compile_module(
&self,
target: &Target,
module: &CompileModuleInfo,
module_translation: &ModuleTranslationState,
function_body_inputs: PrimaryMap<LocalFunctionIndex, FunctionBodyData<'_>>,
progress_callback: Option<&CompilationProgressCallback>,
) -> Result<Compilation, CompileError>;
fn get_middlewares(&self) -> &[Arc<dyn ModuleMiddleware>];
fn enable_readonly_funcref_table(&self) -> bool {
false
}
fn get_cpu_features_used(&self, cpu_features: &EnumSet<CpuFeature>) -> EnumSet<CpuFeature> {
*cpu_features
}
fn get_perfmap_enabled(&self) -> bool {
false
}
}
pub struct FunctionBucket<'a> {
functions: Vec<(LocalFunctionIndex, &'a FunctionBodyData<'a>)>,
pub size: usize,
}
impl<'a> FunctionBucket<'a> {
pub fn new() -> Self {
Self {
functions: Vec::new(),
size: 0,
}
}
}
pub fn build_function_buckets<'a>(
function_body_inputs: &'a PrimaryMap<LocalFunctionIndex, FunctionBodyData<'a>>,
bucket_threshold_size: u64,
) -> Vec<FunctionBucket<'a>> {
let mut function_bodies = function_body_inputs
.iter()
.sorted_by_key(|(id, body)| Reverse((body.data.len(), id.as_u32())))
.collect_vec();
let mut buckets = Vec::new();
while !function_bodies.is_empty() {
let mut next_function_body = Vec::with_capacity(function_bodies.len());
let mut bucket = FunctionBucket::new();
for (fn_index, fn_body) in function_bodies.into_iter() {
if bucket.size + fn_body.data.len() <= bucket_threshold_size as usize
|| bucket.size == 0
{
bucket.size += fn_body.data.len();
bucket.functions.push((fn_index, fn_body));
} else {
next_function_body.push((fn_index, fn_body));
}
}
function_bodies = next_function_body;
buckets.push(bucket);
}
buckets
}
pub trait CompiledFunction {}
pub trait FuncTranslator {}
#[allow(clippy::too_many_arguments)]
pub fn translate_function_buckets<'a, C, T, F, G>(
pool: &rayon::ThreadPool,
func_translator_builder: F,
translate_fn: G,
progress: Option<ProgressContext>,
buckets: &[FunctionBucket<'a>],
) -> Result<Vec<C>, CompileError>
where
T: FuncTranslator,
C: CompiledFunction + Send + Sync,
F: Fn() -> T + Send + Sync + Copy,
G: Fn(&mut T, &LocalFunctionIndex, &FunctionBodyData) -> Result<C, CompileError>
+ Send
+ Sync
+ Copy,
{
let progress = progress.as_ref();
let functions = pool.install(|| {
let (bucket_tx, bucket_rx) = unbounded::<&FunctionBucket<'a>>();
for bucket in buckets {
bucket_tx.send(bucket).map_err(|e| {
CompileError::Resource(format!("cannot allocate crossbeam channel item: {e}"))
})?;
}
drop(bucket_tx);
let (result_tx, result_rx) =
unbounded::<Result<Vec<(LocalFunctionIndex, C)>, CompileError>>();
pool.scope(|s| {
let worker_count = pool.current_num_threads().max(1);
for _ in 0..worker_count {
let bucket_rx = bucket_rx.clone();
let result_tx = result_tx.clone();
s.spawn(move |_| {
let mut func_translator = func_translator_builder();
while let Ok(bucket) = bucket_rx.recv() {
let bucket_result = (|| {
let mut translated_functions = Vec::new();
for (i, input) in bucket.functions.iter() {
let translated = translate_fn(&mut func_translator, i, input)?;
if let Some(progress) = progress {
progress.notify_steps(input.data.len() as u64)?;
}
translated_functions.push((*i, translated));
}
Ok(translated_functions)
})();
if result_tx.send(bucket_result).is_err() {
break;
}
}
});
}
});
drop(result_tx);
let mut functions = Vec::with_capacity(buckets.iter().map(|b| b.functions.len()).sum());
for _ in 0..buckets.len() {
match result_rx.recv().map_err(|e| {
CompileError::Resource(format!("cannot allocate crossbeam channel item: {e}"))
})? {
Ok(bucket_functions) => functions.extend(bucket_functions),
Err(err) => return Err(err),
}
}
Ok(functions)
})?;
Ok(functions
.into_iter()
.sorted_by_key(|x| x.0)
.map(|(_, body)| body)
.collect_vec())
}
pub const WASM_LARGE_FUNCTION_THRESHOLD: u64 = 100_000;
pub const WASM_TRAMPOLINE_ESTIMATED_BODY_SIZE: u64 = 1_000;