mirror of
https://github.com/mii443/wasmer.git
synced 2025-08-23 16:59:27 +00:00
123 lines
4.3 KiB
Rust
123 lines
4.3 KiB
Rust
use crate::store::{EngineType, StoreOptions};
|
|
use crate::warning;
|
|
use anyhow::{Context, Result};
|
|
use std::path::{Path, PathBuf};
|
|
use structopt::StructOpt;
|
|
use wasmer_compiler::{ArtifactCreate, UniversalArtifactBuild};
|
|
use wasmer_compiler::{CpuFeature, ModuleEnvironment, Target, Triple};
|
|
use wasmer_types::entity::PrimaryMap;
|
|
use wasmer_types::{CompileError, MemoryIndex, MemoryStyle, TableIndex, TableStyle};
|
|
|
|
#[derive(Debug, StructOpt)]
|
|
/// The options for the `wasmer compile` subcommand
|
|
pub struct Compile {
|
|
/// Input file
|
|
#[structopt(name = "FILE", parse(from_os_str))]
|
|
path: PathBuf,
|
|
|
|
/// Output file
|
|
#[structopt(name = "OUTPUT PATH", short = "o", parse(from_os_str))]
|
|
output: PathBuf,
|
|
|
|
/// Compilation Target triple
|
|
#[structopt(long = "target")]
|
|
target_triple: Option<Triple>,
|
|
|
|
#[structopt(flatten)]
|
|
store: StoreOptions,
|
|
|
|
#[structopt(short = "m", multiple = true, number_of_values = 1)]
|
|
cpu_features: Vec<CpuFeature>,
|
|
}
|
|
|
|
impl Compile {
|
|
/// Runs logic for the `compile` subcommand
|
|
pub fn execute(&self) -> Result<()> {
|
|
self.inner_execute()
|
|
.context(format!("failed to compile `{}`", self.path.display()))
|
|
}
|
|
|
|
pub(crate) fn get_recommend_extension(
|
|
engine_type: &EngineType,
|
|
target_triple: &Triple,
|
|
) -> Result<&'static str> {
|
|
Ok(match engine_type {
|
|
EngineType::Universal => {
|
|
wasmer_compiler::UniversalArtifactBuild::get_default_extension(target_triple)
|
|
}
|
|
})
|
|
}
|
|
|
|
fn inner_execute(&self) -> Result<()> {
|
|
let target = self
|
|
.target_triple
|
|
.as_ref()
|
|
.map(|target_triple| {
|
|
let mut features = self
|
|
.cpu_features
|
|
.clone()
|
|
.into_iter()
|
|
.fold(CpuFeature::set(), |a, b| a | b);
|
|
// Cranelift requires SSE2, so we have this "hack" for now to facilitate
|
|
// usage
|
|
features |= CpuFeature::SSE2;
|
|
Target::new(target_triple.clone(), features)
|
|
})
|
|
.unwrap_or_default();
|
|
let (mut engine, engine_type, compiler_type) =
|
|
self.store.get_engine_for_target(target.clone())?;
|
|
let output_filename = self
|
|
.output
|
|
.file_stem()
|
|
.map(|osstr| osstr.to_string_lossy().to_string())
|
|
.unwrap_or_default();
|
|
let recommended_extension = Self::get_recommend_extension(&engine_type, target.triple())?;
|
|
match self.output.extension() {
|
|
Some(ext) => {
|
|
if ext != recommended_extension {
|
|
warning!("the output file has a wrong extension. We recommend using `{}.{}` for the chosen target", &output_filename, &recommended_extension)
|
|
}
|
|
}
|
|
None => {
|
|
warning!("the output file has no extension. We recommend using `{}.{}` for the chosen target", &output_filename, &recommended_extension)
|
|
}
|
|
}
|
|
let tunables = self.store.get_tunables_for_target(&target)?;
|
|
|
|
println!("Engine: {}", engine_type.to_string());
|
|
println!("Compiler: {}", compiler_type.to_string());
|
|
println!("Target: {}", target.triple());
|
|
|
|
// compile and save the artifact (without using module from api)
|
|
let path: &Path = self.path.as_ref();
|
|
let wasm_bytes = std::fs::read(path)?;
|
|
let environ = ModuleEnvironment::new();
|
|
let translation = environ.translate(&wasm_bytes).map_err(CompileError::Wasm)?;
|
|
let module = translation.module;
|
|
let memory_styles: PrimaryMap<MemoryIndex, MemoryStyle> = module
|
|
.memories
|
|
.values()
|
|
.map(|memory_type| tunables.memory_style(memory_type))
|
|
.collect();
|
|
let table_styles: PrimaryMap<TableIndex, TableStyle> = module
|
|
.tables
|
|
.values()
|
|
.map(|table_type| tunables.table_style(table_type))
|
|
.collect();
|
|
let artifact = UniversalArtifactBuild::new(
|
|
&mut engine,
|
|
&wasm_bytes,
|
|
&target,
|
|
memory_styles,
|
|
table_styles,
|
|
)?;
|
|
artifact.serialize_to_file(self.output.as_ref())?;
|
|
eprintln!(
|
|
"✔ File compiled successfully to `{}`.",
|
|
self.output.display(),
|
|
);
|
|
|
|
Ok(())
|
|
}
|
|
}
|