The implementation is pretty straightforward. Wasm atomic instructions fall
into 5 groups
* atomic read-modify-write
* atomic compare-and-swap
* atomic loads
* atomic stores
* fences
and the implementation mirrors that structure, at both the CLIF and AArch64
levels.
At the CLIF level, there are five new instructions, one for each group. Some
comments about these:
* for those that take addresses (all except fences), the address is contained
entirely in a single `Value`; there is no offset field as there is with
normal loads and stores. Wasm atomics require alignment checks, and
removing the offset makes implementation of those checks a bit simpler.
* atomic loads and stores get their own instructions, rather than reusing the
existing load and store instructions, for two reasons:
- per above comment, makes alignment checking simpler
- reuse of existing loads and stores would require extension of `MemFlags`
to indicate atomicity, which sounds semantically unclean. For example,
then *any* instruction carrying `MemFlags` could be marked as atomic, even
in cases where it is meaningless or ambiguous.
* I tried to specify, in comments, the behaviour of these instructions as
tightly as I could. Unfortunately there is no way (per my limited CLIF
knowledge) to enforce the constraint that they may only be used on I8, I16,
I32 and I64 types, and in particular not on floating point or vector types.
The translation from Wasm to CLIF, in `code_translator.rs` is unremarkable.
At the AArch64 level, there are also five new instructions, one for each
group. All of them except `::Fence` contain multiple real machine
instructions. Atomic r-m-w and atomic c-a-s are emitted as the usual
load-linked store-conditional loops, guarded at both ends by memory fences.
Atomic loads and stores are emitted as a load preceded by a fence, and a store
followed by a fence, respectively. The amount of fencing may be overkill, but
it reflects exactly what the SM Wasm baseline compiler for AArch64 does.
One reason to implement r-m-w and c-a-s as a single insn which is expanded
only at emission time is that we must be very careful what instructions we
allow in between the load-linked and store-conditional. In particular, we
cannot allow *any* extra memory transactions in there, since -- particularly
on low-end hardware -- that might cause the transaction to fail, hence
deadlocking the generated code. That implies that we can't present the LL/SC
loop to the register allocator as its constituent instructions, since it might
insert spills anywhere. Hence we must present it as a single indivisible
unit, as we do here. It also has the benefit of reducing the total amount of
work the RA has to do.
The only other notable feature of the r-m-w and c-a-s translations into
AArch64 code, is that they both need a scratch register internally. Rather
than faking one up by claiming, in `get_regs` that it modifies an extra
scratch register, and having to have a dummy initialisation of it, these new
instructions (`::LLSC` and `::CAS`) simply use fixed registers in the range
x24-x28. We rely on the RA's ability to coalesce V<-->R copies to make the
cost of the resulting extra copies zero or almost zero. x24-x28 are chosen so
as to be call-clobbered, hence their use is less likely to interfere with long
live ranges that span calls.
One subtlety regarding the use of completely fixed input and output registers
is that we must be careful how the surrounding copy from/to of the arg/result
registers is done. In particular, it is not safe to simply emit copies in
some arbitrary order if one of the arg registers is a real reg. For that
reason, the arguments are first moved into virtual regs if they are not
already there, using a new method `<LowerCtx for Lower>::ensure_in_vreg`.
Again, we rely on coalescing to turn them into no-ops in the common case.
There is also a ridealong fix for the AArch64 lowering case for
`Opcode::Trapif | Opcode::Trapff`, which removes a bug in which two trap insns
in a row were generated.
In the patch as submitted there are 6 "FIXME JRS" comments, which mark things
which I believe to be correct, but for which I would appreciate a second
opinion. Unless otherwise directed, I will remove them for the final commit
but leave the associated code/comments unchanged.
777 lines
23 KiB
Rust
777 lines
23 KiB
Rust
//! "Dummy" implementations of `ModuleEnvironment` and `FuncEnvironment` for testing
|
|
//! wasm translation. For complete implementations of `ModuleEnvironment` and
|
|
//! `FuncEnvironment`, see [wasmtime-environ] in [Wasmtime].
|
|
//!
|
|
//! [wasmtime-environ]: https://crates.io/crates/wasmtime-environ
|
|
//! [Wasmtime]: https://github.com/bytecodealliance/wasmtime
|
|
|
|
use crate::environ::{
|
|
FuncEnvironment, GlobalVariable, ModuleEnvironment, ReturnMode, TargetEnvironment,
|
|
WasmFuncType, WasmResult,
|
|
};
|
|
use crate::func_translator::FuncTranslator;
|
|
use crate::state::ModuleTranslationState;
|
|
use crate::translation_utils::{
|
|
DataIndex, DefinedFuncIndex, ElemIndex, FuncIndex, Global, GlobalIndex, Memory, MemoryIndex,
|
|
SignatureIndex, Table, TableIndex,
|
|
};
|
|
use core::convert::TryFrom;
|
|
use cranelift_codegen::cursor::FuncCursor;
|
|
use cranelift_codegen::ir::immediates::{Offset32, Uimm64};
|
|
use cranelift_codegen::ir::types::*;
|
|
use cranelift_codegen::ir::{self, InstBuilder};
|
|
use cranelift_codegen::isa::TargetFrontendConfig;
|
|
use cranelift_entity::{EntityRef, PrimaryMap, SecondaryMap};
|
|
use cranelift_frontend::FunctionBuilder;
|
|
use std::boxed::Box;
|
|
use std::string::String;
|
|
use std::vec::Vec;
|
|
|
|
/// Compute a `ir::ExternalName` for a given wasm function index.
|
|
fn get_func_name(func_index: FuncIndex) -> ir::ExternalName {
|
|
ir::ExternalName::user(0, func_index.as_u32())
|
|
}
|
|
|
|
/// A collection of names under which a given entity is exported.
|
|
pub struct Exportable<T> {
|
|
/// A wasm entity.
|
|
pub entity: T,
|
|
|
|
/// Names under which the entity is exported.
|
|
pub export_names: Vec<String>,
|
|
}
|
|
|
|
impl<T> Exportable<T> {
|
|
pub fn new(entity: T) -> Self {
|
|
Self {
|
|
entity,
|
|
export_names: Vec::new(),
|
|
}
|
|
}
|
|
}
|
|
|
|
/// The main state belonging to a `DummyEnvironment`. This is split out from
|
|
/// `DummyEnvironment` to allow it to be borrowed separately from the
|
|
/// `FuncTranslator` field.
|
|
pub struct DummyModuleInfo {
|
|
/// Target description relevant to frontends producing Cranelift IR.
|
|
config: TargetFrontendConfig,
|
|
|
|
/// Signatures as provided by `declare_signature`.
|
|
pub signatures: PrimaryMap<SignatureIndex, ir::Signature>,
|
|
|
|
/// Module and field names of imported functions as provided by `declare_func_import`.
|
|
pub imported_funcs: Vec<(String, String)>,
|
|
|
|
/// Module and field names of imported globals as provided by `declare_global_import`.
|
|
pub imported_globals: Vec<(String, String)>,
|
|
|
|
/// Module and field names of imported tables as provided by `declare_table_import`.
|
|
pub imported_tables: Vec<(String, String)>,
|
|
|
|
/// Module and field names of imported memories as provided by `declare_memory_import`.
|
|
pub imported_memories: Vec<(String, String)>,
|
|
|
|
/// Functions, imported and local.
|
|
pub functions: PrimaryMap<FuncIndex, Exportable<SignatureIndex>>,
|
|
|
|
/// Function bodies.
|
|
pub function_bodies: PrimaryMap<DefinedFuncIndex, ir::Function>,
|
|
|
|
/// Tables as provided by `declare_table`.
|
|
pub tables: PrimaryMap<TableIndex, Exportable<Table>>,
|
|
|
|
/// Memories as provided by `declare_memory`.
|
|
pub memories: PrimaryMap<MemoryIndex, Exportable<Memory>>,
|
|
|
|
/// Globals as provided by `declare_global`.
|
|
pub globals: PrimaryMap<GlobalIndex, Exportable<Global>>,
|
|
|
|
/// The start function.
|
|
pub start_func: Option<FuncIndex>,
|
|
}
|
|
|
|
impl DummyModuleInfo {
|
|
/// Creates a new `DummyModuleInfo` instance.
|
|
pub fn new(config: TargetFrontendConfig) -> Self {
|
|
Self {
|
|
config,
|
|
signatures: PrimaryMap::new(),
|
|
imported_funcs: Vec::new(),
|
|
imported_globals: Vec::new(),
|
|
imported_tables: Vec::new(),
|
|
imported_memories: Vec::new(),
|
|
functions: PrimaryMap::new(),
|
|
function_bodies: PrimaryMap::new(),
|
|
tables: PrimaryMap::new(),
|
|
memories: PrimaryMap::new(),
|
|
globals: PrimaryMap::new(),
|
|
start_func: None,
|
|
}
|
|
}
|
|
}
|
|
|
|
/// This `ModuleEnvironment` implementation is a "naïve" one, doing essentially nothing and
|
|
/// emitting placeholders when forced to. Don't try to execute code translated for this
|
|
/// environment, essentially here for translation debug purposes.
|
|
pub struct DummyEnvironment {
|
|
/// Module information.
|
|
pub info: DummyModuleInfo,
|
|
|
|
/// Function translation.
|
|
trans: FuncTranslator,
|
|
|
|
/// Vector of wasm bytecode size for each function.
|
|
pub func_bytecode_sizes: Vec<usize>,
|
|
|
|
/// How to return from functions.
|
|
return_mode: ReturnMode,
|
|
|
|
/// Instructs to collect debug data during translation.
|
|
debug_info: bool,
|
|
|
|
/// Name of the module from the wasm file.
|
|
pub module_name: Option<String>,
|
|
|
|
/// Function names.
|
|
function_names: SecondaryMap<FuncIndex, String>,
|
|
}
|
|
|
|
impl DummyEnvironment {
|
|
/// Creates a new `DummyEnvironment` instance.
|
|
pub fn new(config: TargetFrontendConfig, return_mode: ReturnMode, debug_info: bool) -> Self {
|
|
Self {
|
|
info: DummyModuleInfo::new(config),
|
|
trans: FuncTranslator::new(),
|
|
func_bytecode_sizes: Vec::new(),
|
|
return_mode,
|
|
debug_info,
|
|
module_name: None,
|
|
function_names: SecondaryMap::new(),
|
|
}
|
|
}
|
|
|
|
/// Return a `DummyFuncEnvironment` for translating functions within this
|
|
/// `DummyEnvironment`.
|
|
pub fn func_env(&self) -> DummyFuncEnvironment {
|
|
DummyFuncEnvironment::new(&self.info, self.return_mode)
|
|
}
|
|
|
|
fn get_func_type(&self, func_index: FuncIndex) -> SignatureIndex {
|
|
self.info.functions[func_index].entity
|
|
}
|
|
|
|
/// Return the number of imported functions within this `DummyEnvironment`.
|
|
pub fn get_num_func_imports(&self) -> usize {
|
|
self.info.imported_funcs.len()
|
|
}
|
|
|
|
/// Return the name of the function, if a name for the function with
|
|
/// the corresponding index exists.
|
|
pub fn get_func_name(&self, func_index: FuncIndex) -> Option<&str> {
|
|
self.function_names.get(func_index).map(String::as_ref)
|
|
}
|
|
}
|
|
|
|
/// The `FuncEnvironment` implementation for use by the `DummyEnvironment`.
|
|
pub struct DummyFuncEnvironment<'dummy_environment> {
|
|
pub mod_info: &'dummy_environment DummyModuleInfo,
|
|
|
|
return_mode: ReturnMode,
|
|
}
|
|
|
|
impl<'dummy_environment> DummyFuncEnvironment<'dummy_environment> {
|
|
pub fn new(mod_info: &'dummy_environment DummyModuleInfo, return_mode: ReturnMode) -> Self {
|
|
Self {
|
|
mod_info,
|
|
return_mode,
|
|
}
|
|
}
|
|
|
|
// Create a signature for `sigidx` amended with a `vmctx` argument after the standard wasm
|
|
// arguments.
|
|
fn vmctx_sig(&self, sigidx: SignatureIndex) -> ir::Signature {
|
|
let mut sig = self.mod_info.signatures[sigidx].clone();
|
|
sig.params.push(ir::AbiParam::special(
|
|
self.pointer_type(),
|
|
ir::ArgumentPurpose::VMContext,
|
|
));
|
|
sig
|
|
}
|
|
|
|
fn reference_type(&self) -> ir::Type {
|
|
match self.pointer_type() {
|
|
ir::types::I32 => ir::types::R32,
|
|
ir::types::I64 => ir::types::R64,
|
|
_ => panic!("unsupported pointer type"),
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'dummy_environment> TargetEnvironment for DummyFuncEnvironment<'dummy_environment> {
|
|
fn target_config(&self) -> TargetFrontendConfig {
|
|
self.mod_info.config
|
|
}
|
|
}
|
|
|
|
impl<'dummy_environment> FuncEnvironment for DummyFuncEnvironment<'dummy_environment> {
|
|
fn return_mode(&self) -> ReturnMode {
|
|
self.return_mode
|
|
}
|
|
|
|
fn make_global(
|
|
&mut self,
|
|
func: &mut ir::Function,
|
|
index: GlobalIndex,
|
|
) -> WasmResult<GlobalVariable> {
|
|
// Just create a dummy `vmctx` global.
|
|
let offset = i32::try_from((index.index() * 8) + 8).unwrap().into();
|
|
let vmctx = func.create_global_value(ir::GlobalValueData::VMContext {});
|
|
Ok(GlobalVariable::Memory {
|
|
gv: vmctx,
|
|
offset,
|
|
ty: self.mod_info.globals[index].entity.ty,
|
|
})
|
|
}
|
|
|
|
fn make_heap(&mut self, func: &mut ir::Function, _index: MemoryIndex) -> WasmResult<ir::Heap> {
|
|
// Create a static heap whose base address is stored at `vmctx+0`.
|
|
let addr = func.create_global_value(ir::GlobalValueData::VMContext);
|
|
let gv = func.create_global_value(ir::GlobalValueData::Load {
|
|
base: addr,
|
|
offset: Offset32::new(0),
|
|
global_type: self.pointer_type(),
|
|
readonly: true,
|
|
});
|
|
|
|
Ok(func.create_heap(ir::HeapData {
|
|
base: gv,
|
|
min_size: 0.into(),
|
|
offset_guard_size: 0x8000_0000.into(),
|
|
style: ir::HeapStyle::Static {
|
|
bound: 0x1_0000_0000.into(),
|
|
},
|
|
index_type: I32,
|
|
}))
|
|
}
|
|
|
|
fn make_table(&mut self, func: &mut ir::Function, _index: TableIndex) -> WasmResult<ir::Table> {
|
|
// Create a table whose base address is stored at `vmctx+0`.
|
|
let vmctx = func.create_global_value(ir::GlobalValueData::VMContext);
|
|
let base_gv = func.create_global_value(ir::GlobalValueData::Load {
|
|
base: vmctx,
|
|
offset: Offset32::new(0),
|
|
global_type: self.pointer_type(),
|
|
readonly: true, // when tables in wasm become "growable", revisit whether this can be readonly or not.
|
|
});
|
|
let bound_gv = func.create_global_value(ir::GlobalValueData::Load {
|
|
base: vmctx,
|
|
offset: Offset32::new(0),
|
|
global_type: I32,
|
|
readonly: true,
|
|
});
|
|
|
|
Ok(func.create_table(ir::TableData {
|
|
base_gv,
|
|
min_size: Uimm64::new(0),
|
|
bound_gv,
|
|
element_size: Uimm64::from(u64::from(self.pointer_bytes()) * 2),
|
|
index_type: I32,
|
|
}))
|
|
}
|
|
|
|
fn make_indirect_sig(
|
|
&mut self,
|
|
func: &mut ir::Function,
|
|
index: SignatureIndex,
|
|
) -> WasmResult<ir::SigRef> {
|
|
// A real implementation would probably change the calling convention and add `vmctx` and
|
|
// signature index arguments.
|
|
Ok(func.import_signature(self.vmctx_sig(index)))
|
|
}
|
|
|
|
fn make_direct_func(
|
|
&mut self,
|
|
func: &mut ir::Function,
|
|
index: FuncIndex,
|
|
) -> WasmResult<ir::FuncRef> {
|
|
let sigidx = self.mod_info.functions[index].entity;
|
|
// A real implementation would probably add a `vmctx` argument.
|
|
// And maybe attempt some signature de-duplication.
|
|
let signature = func.import_signature(self.vmctx_sig(sigidx));
|
|
let name = get_func_name(index);
|
|
Ok(func.import_function(ir::ExtFuncData {
|
|
name,
|
|
signature,
|
|
colocated: false,
|
|
}))
|
|
}
|
|
|
|
fn translate_call_indirect(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_table_index: TableIndex,
|
|
_table: ir::Table,
|
|
_sig_index: SignatureIndex,
|
|
sig_ref: ir::SigRef,
|
|
callee: ir::Value,
|
|
call_args: &[ir::Value],
|
|
) -> WasmResult<ir::Inst> {
|
|
// Pass the current function's vmctx parameter on to the callee.
|
|
let vmctx = pos
|
|
.func
|
|
.special_param(ir::ArgumentPurpose::VMContext)
|
|
.expect("Missing vmctx parameter");
|
|
|
|
// The `callee` value is an index into a table of function pointers.
|
|
// Apparently, that table is stored at absolute address 0 in this dummy environment.
|
|
// TODO: Generate bounds checking code.
|
|
let ptr = self.pointer_type();
|
|
let callee_offset = if ptr == I32 {
|
|
pos.ins().imul_imm(callee, 4)
|
|
} else {
|
|
let ext = pos.ins().uextend(I64, callee);
|
|
pos.ins().imul_imm(ext, 4)
|
|
};
|
|
let mflags = ir::MemFlags::trusted();
|
|
let func_ptr = pos.ins().load(ptr, mflags, callee_offset, 0);
|
|
|
|
// Build a value list for the indirect call instruction containing the callee, call_args,
|
|
// and the vmctx parameter.
|
|
let mut args = ir::ValueList::default();
|
|
args.push(func_ptr, &mut pos.func.dfg.value_lists);
|
|
args.extend(call_args.iter().cloned(), &mut pos.func.dfg.value_lists);
|
|
args.push(vmctx, &mut pos.func.dfg.value_lists);
|
|
|
|
Ok(pos
|
|
.ins()
|
|
.CallIndirect(ir::Opcode::CallIndirect, INVALID, sig_ref, args)
|
|
.0)
|
|
}
|
|
|
|
fn translate_call(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_callee_index: FuncIndex,
|
|
callee: ir::FuncRef,
|
|
call_args: &[ir::Value],
|
|
) -> WasmResult<ir::Inst> {
|
|
// Pass the current function's vmctx parameter on to the callee.
|
|
let vmctx = pos
|
|
.func
|
|
.special_param(ir::ArgumentPurpose::VMContext)
|
|
.expect("Missing vmctx parameter");
|
|
|
|
// Build a value list for the call instruction containing the call_args and the vmctx
|
|
// parameter.
|
|
let mut args = ir::ValueList::default();
|
|
args.extend(call_args.iter().cloned(), &mut pos.func.dfg.value_lists);
|
|
args.push(vmctx, &mut pos.func.dfg.value_lists);
|
|
|
|
Ok(pos.ins().Call(ir::Opcode::Call, INVALID, callee, args).0)
|
|
}
|
|
|
|
fn translate_memory_grow(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_index: MemoryIndex,
|
|
_heap: ir::Heap,
|
|
_val: ir::Value,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(pos.ins().iconst(I32, -1))
|
|
}
|
|
|
|
fn translate_memory_size(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_index: MemoryIndex,
|
|
_heap: ir::Heap,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(pos.ins().iconst(I32, -1))
|
|
}
|
|
|
|
fn translate_memory_copy(
|
|
&mut self,
|
|
_pos: FuncCursor,
|
|
_index: MemoryIndex,
|
|
_heap: ir::Heap,
|
|
_dst: ir::Value,
|
|
_src: ir::Value,
|
|
_len: ir::Value,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_memory_fill(
|
|
&mut self,
|
|
_pos: FuncCursor,
|
|
_index: MemoryIndex,
|
|
_heap: ir::Heap,
|
|
_dst: ir::Value,
|
|
_val: ir::Value,
|
|
_len: ir::Value,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_memory_init(
|
|
&mut self,
|
|
_pos: FuncCursor,
|
|
_index: MemoryIndex,
|
|
_heap: ir::Heap,
|
|
_seg_index: u32,
|
|
_dst: ir::Value,
|
|
_src: ir::Value,
|
|
_len: ir::Value,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_data_drop(&mut self, _pos: FuncCursor, _seg_index: u32) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_table_size(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_index: TableIndex,
|
|
_table: ir::Table,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(pos.ins().iconst(I32, -1))
|
|
}
|
|
|
|
fn translate_table_grow(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_table_index: TableIndex,
|
|
_table: ir::Table,
|
|
_delta: ir::Value,
|
|
_init_value: ir::Value,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(pos.ins().iconst(I32, -1))
|
|
}
|
|
|
|
fn translate_table_get(
|
|
&mut self,
|
|
builder: &mut FunctionBuilder,
|
|
_table_index: TableIndex,
|
|
_table: ir::Table,
|
|
_index: ir::Value,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(builder.ins().null(self.reference_type()))
|
|
}
|
|
|
|
fn translate_table_set(
|
|
&mut self,
|
|
_builder: &mut FunctionBuilder,
|
|
_table_index: TableIndex,
|
|
_table: ir::Table,
|
|
_value: ir::Value,
|
|
_index: ir::Value,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_table_copy(
|
|
&mut self,
|
|
_pos: FuncCursor,
|
|
_dst_index: TableIndex,
|
|
_dst_table: ir::Table,
|
|
_src_index: TableIndex,
|
|
_src_table: ir::Table,
|
|
_dst: ir::Value,
|
|
_src: ir::Value,
|
|
_len: ir::Value,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_table_fill(
|
|
&mut self,
|
|
_pos: FuncCursor,
|
|
_table_index: TableIndex,
|
|
_dst: ir::Value,
|
|
_val: ir::Value,
|
|
_len: ir::Value,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_table_init(
|
|
&mut self,
|
|
_pos: FuncCursor,
|
|
_seg_index: u32,
|
|
_table_index: TableIndex,
|
|
_table: ir::Table,
|
|
_dst: ir::Value,
|
|
_src: ir::Value,
|
|
_len: ir::Value,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_elem_drop(&mut self, _pos: FuncCursor, _seg_index: u32) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_ref_func(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_func_index: FuncIndex,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(pos.ins().null(self.reference_type()))
|
|
}
|
|
|
|
fn translate_custom_global_get(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_global_index: GlobalIndex,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(pos.ins().iconst(I32, -1))
|
|
}
|
|
|
|
fn translate_custom_global_set(
|
|
&mut self,
|
|
_pos: FuncCursor,
|
|
_global_index: GlobalIndex,
|
|
_val: ir::Value,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn translate_atomic_wait(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_index: MemoryIndex,
|
|
_heap: ir::Heap,
|
|
_addr: ir::Value,
|
|
_expected: ir::Value,
|
|
_timeout: ir::Value,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(pos.ins().iconst(I32, -1))
|
|
}
|
|
|
|
fn translate_atomic_notify(
|
|
&mut self,
|
|
mut pos: FuncCursor,
|
|
_index: MemoryIndex,
|
|
_heap: ir::Heap,
|
|
_addr: ir::Value,
|
|
_count: ir::Value,
|
|
) -> WasmResult<ir::Value> {
|
|
Ok(pos.ins().iconst(I32, 0))
|
|
}
|
|
}
|
|
|
|
impl TargetEnvironment for DummyEnvironment {
|
|
fn target_config(&self) -> TargetFrontendConfig {
|
|
self.info.config
|
|
}
|
|
}
|
|
|
|
impl<'data> ModuleEnvironment<'data> for DummyEnvironment {
|
|
fn declare_signature(&mut self, _wasm: WasmFuncType, sig: ir::Signature) -> WasmResult<()> {
|
|
self.info.signatures.push(sig);
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_func_import(
|
|
&mut self,
|
|
sig_index: SignatureIndex,
|
|
module: &'data str,
|
|
field: &'data str,
|
|
) -> WasmResult<()> {
|
|
assert_eq!(
|
|
self.info.functions.len(),
|
|
self.info.imported_funcs.len(),
|
|
"Imported functions must be declared first"
|
|
);
|
|
self.info.functions.push(Exportable::new(sig_index));
|
|
self.info
|
|
.imported_funcs
|
|
.push((String::from(module), String::from(field)));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_func_type(&mut self, sig_index: SignatureIndex) -> WasmResult<()> {
|
|
self.info.functions.push(Exportable::new(sig_index));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_global(&mut self, global: Global) -> WasmResult<()> {
|
|
self.info.globals.push(Exportable::new(global));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_global_import(
|
|
&mut self,
|
|
global: Global,
|
|
module: &'data str,
|
|
field: &'data str,
|
|
) -> WasmResult<()> {
|
|
self.info.globals.push(Exportable::new(global));
|
|
self.info
|
|
.imported_globals
|
|
.push((String::from(module), String::from(field)));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_table(&mut self, table: Table) -> WasmResult<()> {
|
|
self.info.tables.push(Exportable::new(table));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_table_import(
|
|
&mut self,
|
|
table: Table,
|
|
module: &'data str,
|
|
field: &'data str,
|
|
) -> WasmResult<()> {
|
|
self.info.tables.push(Exportable::new(table));
|
|
self.info
|
|
.imported_tables
|
|
.push((String::from(module), String::from(field)));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_table_elements(
|
|
&mut self,
|
|
_table_index: TableIndex,
|
|
_base: Option<GlobalIndex>,
|
|
_offset: usize,
|
|
_elements: Box<[FuncIndex]>,
|
|
) -> WasmResult<()> {
|
|
// We do nothing
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_passive_element(
|
|
&mut self,
|
|
_elem_index: ElemIndex,
|
|
_segments: Box<[FuncIndex]>,
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_passive_data(
|
|
&mut self,
|
|
_elem_index: DataIndex,
|
|
_segments: &'data [u8],
|
|
) -> WasmResult<()> {
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_memory(&mut self, memory: Memory) -> WasmResult<()> {
|
|
self.info.memories.push(Exportable::new(memory));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_memory_import(
|
|
&mut self,
|
|
memory: Memory,
|
|
module: &'data str,
|
|
field: &'data str,
|
|
) -> WasmResult<()> {
|
|
self.info.memories.push(Exportable::new(memory));
|
|
self.info
|
|
.imported_memories
|
|
.push((String::from(module), String::from(field)));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_data_initialization(
|
|
&mut self,
|
|
_memory_index: MemoryIndex,
|
|
_base: Option<GlobalIndex>,
|
|
_offset: usize,
|
|
_data: &'data [u8],
|
|
) -> WasmResult<()> {
|
|
// We do nothing
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_func_export(&mut self, func_index: FuncIndex, name: &'data str) -> WasmResult<()> {
|
|
self.info.functions[func_index]
|
|
.export_names
|
|
.push(String::from(name));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_table_export(
|
|
&mut self,
|
|
table_index: TableIndex,
|
|
name: &'data str,
|
|
) -> WasmResult<()> {
|
|
self.info.tables[table_index]
|
|
.export_names
|
|
.push(String::from(name));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_memory_export(
|
|
&mut self,
|
|
memory_index: MemoryIndex,
|
|
name: &'data str,
|
|
) -> WasmResult<()> {
|
|
self.info.memories[memory_index]
|
|
.export_names
|
|
.push(String::from(name));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_global_export(
|
|
&mut self,
|
|
global_index: GlobalIndex,
|
|
name: &'data str,
|
|
) -> WasmResult<()> {
|
|
self.info.globals[global_index]
|
|
.export_names
|
|
.push(String::from(name));
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_start_func(&mut self, func_index: FuncIndex) -> WasmResult<()> {
|
|
debug_assert!(self.info.start_func.is_none());
|
|
self.info.start_func = Some(func_index);
|
|
Ok(())
|
|
}
|
|
|
|
fn define_function_body(
|
|
&mut self,
|
|
module_translation_state: &ModuleTranslationState,
|
|
body_bytes: &'data [u8],
|
|
body_offset: usize,
|
|
) -> WasmResult<()> {
|
|
let func = {
|
|
let mut func_environ = DummyFuncEnvironment::new(&self.info, self.return_mode);
|
|
let func_index =
|
|
FuncIndex::new(self.get_num_func_imports() + self.info.function_bodies.len());
|
|
let name = get_func_name(func_index);
|
|
let sig = func_environ.vmctx_sig(self.get_func_type(func_index));
|
|
let mut func = ir::Function::with_name_signature(name, sig);
|
|
if self.debug_info {
|
|
func.collect_debug_info();
|
|
}
|
|
self.trans.translate(
|
|
module_translation_state,
|
|
body_bytes,
|
|
body_offset,
|
|
&mut func,
|
|
&mut func_environ,
|
|
)?;
|
|
func
|
|
};
|
|
self.func_bytecode_sizes.push(body_bytes.len());
|
|
self.info.function_bodies.push(func);
|
|
Ok(())
|
|
}
|
|
|
|
fn declare_module_name(&mut self, name: &'data str) {
|
|
self.module_name = Some(String::from(name));
|
|
}
|
|
|
|
fn declare_func_name(&mut self, func_index: FuncIndex, name: &'data str) {
|
|
self.function_names[func_index] = String::from(name);
|
|
}
|
|
}
|