use crate::abi::{align_to, ty_size, ABIArg, ABISig, LocalSlot, ABI};
use anyhow::Result;
use smallvec::SmallVec;
use std::ops::Range;
use wasmparser::{BinaryReader, FuncValidator, ValidatorResources};
use wasmtime_environ::{ModuleTranslation, TypeConvert};
pub(crate) type Locals = SmallVec<[LocalSlot; 16]>;
pub(crate) struct DefinedLocalsRange(Range<u32>);
impl DefinedLocalsRange {
pub fn as_range(&self) -> &Range<u32> {
&self.0
}
}
#[derive(Default)]
pub(crate) struct DefinedLocals {
pub defined_locals: Locals,
pub stack_size: u32,
}
impl DefinedLocals {
pub fn new(
translation: &ModuleTranslation<'_>,
reader: &mut BinaryReader<'_>,
validator: &mut FuncValidator<ValidatorResources>,
) -> Result<Self> {
let mut next_stack = 0;
let local_count = reader.read_var_u32()?;
let mut slots: Locals = Default::default();
for _ in 0..local_count {
let position = reader.original_position();
let count = reader.read_var_u32()?;
let ty = reader.read()?;
validator.define_locals(position, count, ty)?;
let ty = translation.module.convert_valtype(ty);
for _ in 0..count {
let ty_size = ty_size(&ty);
next_stack = align_to(next_stack, ty_size) + ty_size;
slots.push(LocalSlot::new(ty, next_stack));
}
}
Ok(Self {
defined_locals: slots,
stack_size: next_stack,
})
}
}
pub(crate) struct Frame {
pub locals_size: u32,
pub defined_locals_range: DefinedLocalsRange,
pub locals: Locals,
pub vmctx_slot: LocalSlot,
}
impl Frame {
pub fn new<A: ABI>(sig: &ABISig, defined_locals: &DefinedLocals) -> Result<Self> {
let (mut locals, defined_locals_start) = Self::compute_arg_slots::<A>(sig)?;
locals.extend(
defined_locals
.defined_locals
.iter()
.map(|l| LocalSlot::new(l.ty, l.offset + defined_locals_start)),
);
let vmctx_slots_size = <A as ABI>::word_bytes();
let vmctx_offset = defined_locals_start + defined_locals.stack_size + vmctx_slots_size;
let locals_size = align_to(vmctx_offset, <A as ABI>::stack_align().into());
Ok(Self {
locals,
locals_size,
vmctx_slot: LocalSlot::i64(vmctx_offset),
defined_locals_range: DefinedLocalsRange(
defined_locals_start..defined_locals.stack_size,
),
})
}
pub fn get_local(&self, index: u32) -> Option<&LocalSlot> {
self.locals.get(index as usize)
}
fn compute_arg_slots<A: ABI>(sig: &ABISig) -> Result<(Locals, u32)> {
let arg_base_offset = <A as ABI>::arg_base_offset().into();
let mut next_stack = 0u32;
let slots: Locals = sig
.params
.iter()
.map(|arg| Self::abi_arg_slot(&arg, &mut next_stack, arg_base_offset))
.collect();
Ok((slots, next_stack))
}
fn abi_arg_slot(arg: &ABIArg, next_stack: &mut u32, arg_base_offset: u32) -> LocalSlot {
match arg {
ABIArg::Reg { ty, reg: _ } => {
let ty_size = ty_size(&ty);
*next_stack = align_to(*next_stack, ty_size) + ty_size;
LocalSlot::new(*ty, *next_stack)
}
ABIArg::Stack { ty, offset } => LocalSlot::stack_arg(*ty, offset + arg_base_offset),
}
}
}