Add naive local and temporary spills
This commit is contained in:
parent
5622aa661e
commit
d33e4a6b3e
@ -4,13 +4,12 @@ use std::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use wasm_ast::node::{
|
use wasm_ast::node::{
|
||||||
BinOp, CmpOp, Expression, GetGlobal, GetLocal, GetTemporary, LoadAt, MemorySize, Select, UnOp,
|
BinOp, CmpOp, Expression, GetGlobal, LoadAt, Local, MemorySize, Select, Temporary, UnOp, Value,
|
||||||
Value,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::analyzer::as_symbol::AsSymbol;
|
use crate::analyzer::as_symbol::AsSymbol;
|
||||||
|
|
||||||
use super::manager::{write_separated, DriverNoContext};
|
use super::manager::{write_separated, Driver, Manager};
|
||||||
|
|
||||||
macro_rules! impl_write_number {
|
macro_rules! impl_write_number {
|
||||||
($name:tt, $numeric:ty) => {
|
($name:tt, $numeric:ty) => {
|
||||||
@ -26,43 +25,55 @@ macro_rules! impl_write_number {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for Select {
|
impl Driver for Select {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "(")?;
|
write!(w, "(")?;
|
||||||
Condition(self.condition()).write(w)?;
|
Condition(self.condition()).write(mng, w)?;
|
||||||
write!(w, " and ")?;
|
write!(w, " and ")?;
|
||||||
self.on_true().write(w)?;
|
self.on_true().write(mng, w)?;
|
||||||
write!(w, " or ")?;
|
write!(w, " or ")?;
|
||||||
self.on_false().write(w)?;
|
self.on_false().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for GetTemporary {
|
impl Driver for Temporary {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "reg_{}", self.var())
|
let var = self.var();
|
||||||
|
|
||||||
|
if let Some(var) = var.checked_sub(mng.num_temp()) {
|
||||||
|
write!(w, "reg_spill[{}]", var + 1)
|
||||||
|
} else {
|
||||||
|
write!(w, "reg_{var}")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for GetLocal {
|
impl Driver for Local {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "loc_{}", self.var())
|
let var = self.var();
|
||||||
|
|
||||||
|
if let Some(var) = var.checked_sub(mng.num_local()) {
|
||||||
|
write!(w, "loc_spill[{}]", var + 1)
|
||||||
|
} else {
|
||||||
|
write!(w, "loc_{var}")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for GetGlobal {
|
impl Driver for GetGlobal {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, _mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "GLOBAL_LIST[{}].value", self.var())
|
write!(w, "GLOBAL_LIST[{}].value", self.var())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for LoadAt {
|
impl Driver for LoadAt {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let name = self.load_type().as_name();
|
let name = self.load_type().as_name();
|
||||||
let memory = self.memory();
|
let memory = self.memory();
|
||||||
|
|
||||||
write!(w, "load_{name}(memory_at_{memory}, ")?;
|
write!(w, "load_{name}(memory_at_{memory}, ")?;
|
||||||
self.pointer().write(w)?;
|
self.pointer().write(mng, w)?;
|
||||||
|
|
||||||
if self.offset() != 0 {
|
if self.offset() != 0 {
|
||||||
write!(w, " + {}", self.offset())?;
|
write!(w, " + {}", self.offset())?;
|
||||||
@ -72,8 +83,8 @@ impl DriverNoContext for LoadAt {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for MemorySize {
|
impl Driver for MemorySize {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, _mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "memory_at_{}.min", self.memory())
|
write!(w, "memory_at_{}.min", self.memory())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -81,8 +92,8 @@ impl DriverNoContext for MemorySize {
|
|||||||
impl_write_number!(write_f32, f32);
|
impl_write_number!(write_f32, f32);
|
||||||
impl_write_number!(write_f64, f64);
|
impl_write_number!(write_f64, f64);
|
||||||
|
|
||||||
impl DriverNoContext for Value {
|
impl Driver for Value {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, _mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
match self {
|
match self {
|
||||||
Self::I32(i) => write!(w, "{i}"),
|
Self::I32(i) => write!(w, "{i}"),
|
||||||
Self::I64(i) => write!(w, "{i}LL"),
|
Self::I64(i) => write!(w, "{i}LL"),
|
||||||
@ -92,97 +103,97 @@ impl DriverNoContext for Value {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for UnOp {
|
impl Driver for UnOp {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let (a, b) = self.op_type().as_name();
|
let (a, b) = self.op_type().as_name();
|
||||||
|
|
||||||
write!(w, "{a}_{b}(")?;
|
write!(w, "{a}_{b}(")?;
|
||||||
self.rhs().write(w)?;
|
self.rhs().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for BinOp {
|
impl Driver for BinOp {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
if let Some(symbol) = self.op_type().as_symbol() {
|
if let Some(symbol) = self.op_type().as_symbol() {
|
||||||
write!(w, "(")?;
|
write!(w, "(")?;
|
||||||
self.lhs().write(w)?;
|
self.lhs().write(mng, w)?;
|
||||||
write!(w, " {symbol} ")?;
|
write!(w, " {symbol} ")?;
|
||||||
} else {
|
} else {
|
||||||
let (head, tail) = self.op_type().as_name();
|
let (head, tail) = self.op_type().as_name();
|
||||||
|
|
||||||
write!(w, "{head}_{tail}(")?;
|
write!(w, "{head}_{tail}(")?;
|
||||||
self.lhs().write(w)?;
|
self.lhs().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
self.rhs().write(w)?;
|
self.rhs().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct CmpOpBoolean<'a>(&'a CmpOp);
|
struct CmpOpBoolean<'a>(&'a CmpOp);
|
||||||
|
|
||||||
impl DriverNoContext for CmpOpBoolean<'_> {
|
impl Driver for CmpOpBoolean<'_> {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let cmp = self.0;
|
let cmp = self.0;
|
||||||
|
|
||||||
if let Some(symbol) = cmp.op_type().as_symbol() {
|
if let Some(symbol) = cmp.op_type().as_symbol() {
|
||||||
cmp.lhs().write(w)?;
|
cmp.lhs().write(mng, w)?;
|
||||||
write!(w, " {symbol} ")?;
|
write!(w, " {symbol} ")?;
|
||||||
cmp.rhs().write(w)
|
cmp.rhs().write(mng, w)
|
||||||
} else {
|
} else {
|
||||||
let (head, tail) = cmp.op_type().as_name();
|
let (head, tail) = cmp.op_type().as_name();
|
||||||
|
|
||||||
write!(w, "{head}_{tail}(")?;
|
write!(w, "{head}_{tail}(")?;
|
||||||
cmp.lhs().write(w)?;
|
cmp.lhs().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
cmp.rhs().write(w)?;
|
cmp.rhs().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for CmpOp {
|
impl Driver for CmpOp {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "(")?;
|
write!(w, "(")?;
|
||||||
CmpOpBoolean(self).write(w)?;
|
CmpOpBoolean(self).write(mng, w)?;
|
||||||
write!(w, " and 1 or 0)")
|
write!(w, " and 1 or 0)")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Condition<'a>(pub &'a Expression);
|
pub struct Condition<'a>(pub &'a Expression);
|
||||||
|
|
||||||
impl DriverNoContext for Condition<'_> {
|
impl Driver for Condition<'_> {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
if let Expression::CmpOp(node) = self.0 {
|
if let Expression::CmpOp(node) = self.0 {
|
||||||
CmpOpBoolean(node).write(w)
|
CmpOpBoolean(node).write(mng, w)
|
||||||
} else {
|
} else {
|
||||||
self.0.write(w)?;
|
self.0.write(mng, w)?;
|
||||||
write!(w, " ~= 0")
|
write!(w, " ~= 0")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for Expression {
|
impl Driver for Expression {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
match self {
|
match self {
|
||||||
Self::Select(e) => e.write(w),
|
Self::Select(e) => e.write(mng, w),
|
||||||
Self::GetTemporary(e) => e.write(w),
|
Self::GetTemporary(e) => e.write(mng, w),
|
||||||
Self::GetLocal(e) => e.write(w),
|
Self::GetLocal(e) => e.write(mng, w),
|
||||||
Self::GetGlobal(e) => e.write(w),
|
Self::GetGlobal(e) => e.write(mng, w),
|
||||||
Self::LoadAt(e) => e.write(w),
|
Self::LoadAt(e) => e.write(mng, w),
|
||||||
Self::MemorySize(e) => e.write(w),
|
Self::MemorySize(e) => e.write(mng, w),
|
||||||
Self::Value(e) => e.write(w),
|
Self::Value(e) => e.write(mng, w),
|
||||||
Self::UnOp(e) => e.write(w),
|
Self::UnOp(e) => e.write(mng, w),
|
||||||
Self::BinOp(e) => e.write(w),
|
Self::BinOp(e) => e.write(mng, w),
|
||||||
Self::CmpOp(e) => e.write(w),
|
Self::CmpOp(e) => e.write(mng, w),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for &[Expression] {
|
impl Driver for &[Expression] {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write_separated(self.iter(), |e, w| e.write(w), w)
|
write_separated(self.iter(), |e, w| e.write(mng, w), w)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
use std::{
|
use std::{
|
||||||
collections::HashMap,
|
collections::HashMap,
|
||||||
io::{Result, Write},
|
io::{Result, Write},
|
||||||
ops::Range,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
use wasm_ast::node::BrTable;
|
use wasm_ast::node::{BrTable, FuncData};
|
||||||
|
|
||||||
|
use crate::analyzer::{br_table, localize};
|
||||||
|
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! indentation {
|
macro_rules! indentation {
|
||||||
@ -31,23 +32,81 @@ macro_rules! line {
|
|||||||
}};
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
fn get_pinned_registers(
|
||||||
|
upvalues: usize,
|
||||||
|
params: usize,
|
||||||
|
locals: usize,
|
||||||
|
temporaries: usize,
|
||||||
|
) -> (usize, usize) {
|
||||||
|
const MAX_LOCAL_COUNT: usize = 170;
|
||||||
|
|
||||||
|
let available = MAX_LOCAL_COUNT
|
||||||
|
.saturating_sub(upvalues)
|
||||||
|
.saturating_sub(params);
|
||||||
|
|
||||||
|
let locals = available.min(locals);
|
||||||
|
let temporaries = available.saturating_sub(locals).min(temporaries);
|
||||||
|
|
||||||
|
(params + locals, temporaries)
|
||||||
|
}
|
||||||
|
|
||||||
pub struct Manager {
|
pub struct Manager {
|
||||||
table_map: HashMap<usize, usize>,
|
table_map: HashMap<usize, usize>,
|
||||||
label_list: Vec<usize>,
|
num_local: usize,
|
||||||
|
num_temp: usize,
|
||||||
num_label: usize,
|
num_label: usize,
|
||||||
|
label_list: Vec<usize>,
|
||||||
indentation: usize,
|
indentation: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Manager {
|
impl Manager {
|
||||||
|
pub fn empty() -> Self {
|
||||||
|
Self {
|
||||||
|
table_map: HashMap::new(),
|
||||||
|
num_local: 0,
|
||||||
|
num_temp: usize::MAX,
|
||||||
|
num_label: 0,
|
||||||
|
label_list: Vec::new(),
|
||||||
|
indentation: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn function(ast: &FuncData) -> Self {
|
||||||
|
let (upvalues, memories) = localize::visit(ast);
|
||||||
|
let table_map = br_table::visit(ast);
|
||||||
|
let (num_local, num_temp) = get_pinned_registers(
|
||||||
|
upvalues.len() + memories.len(),
|
||||||
|
ast.num_param(),
|
||||||
|
ast.local_data().len(),
|
||||||
|
ast.num_stack(),
|
||||||
|
);
|
||||||
|
|
||||||
|
Self {
|
||||||
|
table_map,
|
||||||
|
num_local,
|
||||||
|
num_temp,
|
||||||
|
num_label: 0,
|
||||||
|
label_list: Vec::new(),
|
||||||
|
indentation: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_table_index(&self, table: &BrTable) -> usize {
|
pub fn get_table_index(&self, table: &BrTable) -> usize {
|
||||||
let id = table as *const _ as usize;
|
let id = table as *const _ as usize;
|
||||||
|
|
||||||
self.table_map[&id]
|
self.table_map[&id]
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_table_map(&mut self, map: HashMap<usize, usize>) {
|
pub fn has_table(&self) -> bool {
|
||||||
self.table_map = map;
|
!self.table_map.is_empty()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn num_local(&self) -> usize {
|
||||||
|
self.num_local
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn num_temp(&self) -> usize {
|
||||||
|
self.num_temp
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn label_list(&self) -> &[usize] {
|
pub fn label_list(&self) -> &[usize] {
|
||||||
@ -82,10 +141,6 @@ pub trait Driver {
|
|||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()>;
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()>;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait DriverNoContext {
|
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn write_separated<I, T, M>(mut iter: I, mut func: M, w: &mut dyn Write) -> Result<()>
|
pub fn write_separated<I, T, M>(mut iter: I, mut func: M, w: &mut dyn Write) -> Result<()>
|
||||||
where
|
where
|
||||||
M: FnMut(T, &mut dyn Write) -> Result<()>,
|
M: FnMut(T, &mut dyn Write) -> Result<()>,
|
||||||
@ -101,7 +156,3 @@ where
|
|||||||
func(v, w)
|
func(v, w)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn write_ascending(prefix: &str, range: Range<usize>, w: &mut dyn Write) -> Result<()> {
|
|
||||||
write_separated(range, |i, w| write!(w, "{prefix}_{i}"), w)
|
|
||||||
}
|
|
||||||
|
@ -5,26 +5,32 @@ use std::{
|
|||||||
|
|
||||||
use wasm_ast::node::{
|
use wasm_ast::node::{
|
||||||
Block, Br, BrIf, BrTable, Call, CallIndirect, FuncData, If, LabelType, MemoryCopy, MemoryFill,
|
Block, Br, BrIf, BrTable, Call, CallIndirect, FuncData, If, LabelType, MemoryCopy, MemoryFill,
|
||||||
MemoryGrow, SetGlobal, SetLocal, SetTemporary, Statement, StoreAt, Terminator,
|
MemoryGrow, ResultList, SetGlobal, SetLocal, SetTemporary, Statement, StoreAt, Terminator,
|
||||||
};
|
};
|
||||||
use wasmparser::ValType;
|
use wasmparser::ValType;
|
||||||
|
|
||||||
use crate::{analyzer::br_table, indentation, indented, line};
|
use crate::{backend::manager::write_separated, indentation, indented, line};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
expression::Condition,
|
expression::Condition,
|
||||||
manager::{write_ascending, write_separated, Driver, DriverNoContext, Manager},
|
manager::{Driver, Manager},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
impl Driver for ResultList {
|
||||||
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
|
write_separated(self.iter(), |t, w| t.write(mng, w), w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Driver for Br {
|
impl Driver for Br {
|
||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let level = *mng.label_list().iter().nth_back(self.target()).unwrap();
|
let level = *mng.label_list().iter().nth_back(self.target()).unwrap();
|
||||||
|
|
||||||
if !self.align().is_aligned() {
|
if !self.align().is_aligned() {
|
||||||
indentation!(mng, w)?;
|
indentation!(mng, w)?;
|
||||||
write_ascending("reg", self.align().new_range(), w)?;
|
self.align().new_range().write(mng, w)?;
|
||||||
write!(w, " = ")?;
|
write!(w, " = ")?;
|
||||||
write_ascending("reg", self.align().old_range(), w)?;
|
self.align().old_range().write(mng, w)?;
|
||||||
writeln!(w)?;
|
writeln!(w)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -97,7 +103,7 @@ fn write_table_setup(table: &BrTable, mng: &mut Manager, w: &mut dyn Write) -> R
|
|||||||
line!(mng, w, "end")?;
|
line!(mng, w, "end")?;
|
||||||
|
|
||||||
indented!(mng, w, "temp = br_map[{id}][")?;
|
indented!(mng, w, "temp = br_map[{id}][")?;
|
||||||
table.condition().write(w)?;
|
table.condition().write(mng, w)?;
|
||||||
writeln!(w, "] or {}", table.default().target())
|
writeln!(w, "] or {}", table.default().target())
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -174,7 +180,7 @@ impl Driver for Block {
|
|||||||
impl Driver for BrIf {
|
impl Driver for BrIf {
|
||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
indented!(mng, w, "if ")?;
|
indented!(mng, w, "if ")?;
|
||||||
Condition(self.condition()).write(w)?;
|
Condition(self.condition()).write(mng, w)?;
|
||||||
writeln!(w, " then")?;
|
writeln!(w, " then")?;
|
||||||
mng.indent();
|
mng.indent();
|
||||||
self.target().write(mng, w)?;
|
self.target().write(mng, w)?;
|
||||||
@ -186,7 +192,7 @@ impl Driver for BrIf {
|
|||||||
impl Driver for If {
|
impl Driver for If {
|
||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
indented!(mng, w, "if ")?;
|
indented!(mng, w, "if ")?;
|
||||||
Condition(self.condition()).write(w)?;
|
Condition(self.condition()).write(mng, w)?;
|
||||||
writeln!(w, " then")?;
|
writeln!(w, " then")?;
|
||||||
|
|
||||||
mng.indent();
|
mng.indent();
|
||||||
@ -204,120 +210,119 @@ impl Driver for If {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_call_store(result: Range<usize>, w: &mut dyn Write) -> Result<()> {
|
impl Driver for Call {
|
||||||
if result.is_empty() {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
return Ok(());
|
if !self.result_list().is_empty() {
|
||||||
|
self.result_list().write(mng, w)?;
|
||||||
|
write!(w, " = ")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
write_ascending("reg", result, w)?;
|
|
||||||
write!(w, " = ")
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DriverNoContext for Call {
|
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
|
||||||
write_call_store(self.result(), w)?;
|
|
||||||
|
|
||||||
write!(w, "FUNC_LIST[{}](", self.function())?;
|
write!(w, "FUNC_LIST[{}](", self.function())?;
|
||||||
self.param_list().write(w)?;
|
self.param_list().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for CallIndirect {
|
impl Driver for CallIndirect {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write_call_store(self.result(), w)?;
|
if !self.result_list().is_empty() {
|
||||||
|
self.result_list().write(mng, w)?;
|
||||||
|
write!(w, " = ")?;
|
||||||
|
}
|
||||||
|
|
||||||
write!(w, "TABLE_LIST[{}].data[", self.table())?;
|
write!(w, "TABLE_LIST[{}].data[", self.table())?;
|
||||||
self.index().write(w)?;
|
self.index().write(mng, w)?;
|
||||||
write!(w, "](")?;
|
write!(w, "](")?;
|
||||||
self.param_list().write(w)?;
|
self.param_list().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for SetTemporary {
|
impl Driver for SetTemporary {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "reg_{} = ", self.var())?;
|
self.var().write(mng, w)?;
|
||||||
self.value().write(w)
|
write!(w, " = ")?;
|
||||||
|
self.value().write(mng, w)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for SetLocal {
|
impl Driver for SetLocal {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "loc_{} = ", self.var())?;
|
self.var().write(mng, w)?;
|
||||||
self.value().write(w)
|
write!(w, " = ")?;
|
||||||
|
self.value().write(mng, w)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for SetGlobal {
|
impl Driver for SetGlobal {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "GLOBAL_LIST[{}].value = ", self.var())?;
|
write!(w, "GLOBAL_LIST[{}].value = ", self.var())?;
|
||||||
self.value().write(w)
|
self.value().write(mng, w)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for StoreAt {
|
impl Driver for StoreAt {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let name = self.store_type().as_name();
|
let name = self.store_type().as_name();
|
||||||
let memory = self.memory();
|
let memory = self.memory();
|
||||||
|
|
||||||
write!(w, "store_{name}(memory_at_{memory}, ")?;
|
write!(w, "store_{name}(memory_at_{memory}, ")?;
|
||||||
|
|
||||||
self.pointer().write(w)?;
|
self.pointer().write(mng, w)?;
|
||||||
|
|
||||||
if self.offset() != 0 {
|
if self.offset() != 0 {
|
||||||
write!(w, " + {}", self.offset())?;
|
write!(w, " + {}", self.offset())?;
|
||||||
}
|
}
|
||||||
|
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
self.value().write(w)?;
|
self.value().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for MemoryGrow {
|
impl Driver for MemoryGrow {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let result = self.result();
|
|
||||||
let memory = self.memory();
|
let memory = self.memory();
|
||||||
|
|
||||||
write!(w, "reg_{result} = rt.allocator.grow(memory_at_{memory}, ")?;
|
self.result().write(mng, w)?;
|
||||||
self.size().write(w)?;
|
write!(w, " = rt.allocator.grow(memory_at_{memory}, ")?;
|
||||||
|
self.size().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for MemoryCopy {
|
impl Driver for MemoryCopy {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let memory_1 = self.destination().memory();
|
let memory_1 = self.destination().memory();
|
||||||
let memory_2 = self.source().memory();
|
let memory_2 = self.source().memory();
|
||||||
|
|
||||||
write!(w, "rt.store.copy(memory_at_{memory_1}, ")?;
|
write!(w, "rt.store.copy(memory_at_{memory_1}, ")?;
|
||||||
self.destination().pointer().write(w)?;
|
self.destination().pointer().write(mng, w)?;
|
||||||
write!(w, ", memory_at_{memory_2}, ")?;
|
write!(w, ", memory_at_{memory_2}, ")?;
|
||||||
self.source().pointer().write(w)?;
|
self.source().pointer().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
self.size().write(w)?;
|
self.size().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for MemoryFill {
|
impl Driver for MemoryFill {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let memory = self.destination().memory();
|
let memory = self.destination().memory();
|
||||||
|
|
||||||
write!(w, "rt.store.fill(memory_at_{memory}, ")?;
|
write!(w, "rt.store.fill(memory_at_{memory}, ")?;
|
||||||
self.destination().pointer().write(w)?;
|
self.destination().pointer().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
self.size().write(w)?;
|
self.size().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
self.value().write(w)?;
|
self.value().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_stat(stat: &dyn DriverNoContext, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write_stat(stat: &dyn Driver, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
indentation!(mng, w)?;
|
indentation!(mng, w)?;
|
||||||
stat.write(w)?;
|
stat.write(mng, w)?;
|
||||||
writeln!(w)
|
writeln!(w)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -342,30 +347,51 @@ impl Driver for Statement {
|
|||||||
|
|
||||||
fn write_parameter_list(ast: &FuncData, w: &mut dyn Write) -> Result<()> {
|
fn write_parameter_list(ast: &FuncData, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "function(")?;
|
write!(w, "function(")?;
|
||||||
write_ascending("loc", 0..ast.num_param(), w)?;
|
write_separated(0..ast.num_param(), |i, w| write!(w, "loc_{i}"), w)?;
|
||||||
writeln!(w, ")")
|
writeln!(w, ")")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const fn type_to_zero(typ: ValType) -> &'static str {
|
||||||
|
match typ {
|
||||||
|
ValType::F32 | ValType::F64 => "0.0",
|
||||||
|
ValType::I64 => "0LL",
|
||||||
|
_ => "0",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn write_variable_list(ast: &FuncData, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write_variable_list(ast: &FuncData, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let mut total = ast.num_param();
|
let mut locals = ast.local_data().iter().copied();
|
||||||
|
let num_local = mng.num_local() - ast.num_param();
|
||||||
|
|
||||||
for data in ast.local_data().iter().filter(|v| v.0 != 0) {
|
for (i, typ) in locals.by_ref().enumerate().take(num_local) {
|
||||||
let range = total..total + usize::try_from(data.0).unwrap();
|
let index = ast.num_param() + i;
|
||||||
let typed = if data.1 == ValType::I64 { "0LL" } else { "0" }.as_bytes();
|
let zero = type_to_zero(typ);
|
||||||
|
|
||||||
total = range.end;
|
line!(mng, w, "local loc_{index} = {zero}")?;
|
||||||
|
|
||||||
indented!(mng, w, "local ")?;
|
|
||||||
write_ascending("loc", range.clone(), w)?;
|
|
||||||
write!(w, " = ")?;
|
|
||||||
write_separated(range, |_, w| w.write_all(typed), w)?;
|
|
||||||
writeln!(w)?;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ast.num_stack() != 0 {
|
if locals.len() != 0 {
|
||||||
indented!(mng, w, "local ")?;
|
indented!(mng, w, "local loc_spill = {{ ")?;
|
||||||
write_ascending("reg", 0..ast.num_stack(), w)?;
|
|
||||||
writeln!(w)?;
|
for typ in locals {
|
||||||
|
let zero = type_to_zero(typ);
|
||||||
|
|
||||||
|
write!(w, "{zero}, ")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
writeln!(w, "}}")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut temporaries = 0..ast.num_stack();
|
||||||
|
|
||||||
|
for i in temporaries.by_ref().take(mng.num_temp()) {
|
||||||
|
line!(mng, w, "local reg_{i}")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if !temporaries.is_empty() {
|
||||||
|
let len = temporaries.len();
|
||||||
|
|
||||||
|
line!(mng, w, "local reg_spill = table.create({len})")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -373,23 +399,22 @@ fn write_variable_list(ast: &FuncData, mng: &mut Manager, w: &mut dyn Write) ->
|
|||||||
|
|
||||||
impl Driver for FuncData {
|
impl Driver for FuncData {
|
||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let br_map = br_table::visit(self);
|
|
||||||
|
|
||||||
mng.indent();
|
mng.indent();
|
||||||
|
|
||||||
write_parameter_list(self, w)?;
|
write_parameter_list(self, w)?;
|
||||||
write_variable_list(self, mng, w)?;
|
write_variable_list(self, mng, w)?;
|
||||||
|
|
||||||
if !br_map.is_empty() {
|
if mng.has_table() {
|
||||||
line!(mng, w, "local br_map, temp = {{}}, nil")?;
|
line!(mng, w, "local br_map, temp = {{}}, nil")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
mng.set_table_map(br_map);
|
|
||||||
self.code().write(mng, w)?;
|
self.code().write(mng, w)?;
|
||||||
|
|
||||||
if self.num_result() != 0 {
|
if self.num_result() != 0 {
|
||||||
indented!(mng, w, "return ")?;
|
indented!(mng, w, "return ")?;
|
||||||
write_ascending("reg", 0..self.num_result(), w)?;
|
|
||||||
|
ResultList::new(0, self.num_result()).write(mng, w)?;
|
||||||
|
|
||||||
writeln!(w)?;
|
writeln!(w)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ use wasmparser::{
|
|||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
analyzer::localize,
|
analyzer::localize,
|
||||||
backend::manager::{Driver, DriverNoContext, Manager},
|
backend::manager::{Driver, Manager},
|
||||||
};
|
};
|
||||||
|
|
||||||
trait AsIEName {
|
trait AsIEName {
|
||||||
@ -51,7 +51,7 @@ fn write_constant(init: &ConstExpr, type_info: &TypeInfo, w: &mut dyn Write) ->
|
|||||||
let func = Factory::from_type_info(type_info).create_anonymous(&code);
|
let func = Factory::from_type_info(type_info).create_anonymous(&code);
|
||||||
|
|
||||||
if let Some(Statement::SetTemporary(stat)) = func.code().code().last() {
|
if let Some(Statement::SetTemporary(stat)) = func.code().code().last() {
|
||||||
stat.value().write(w)
|
stat.value().write(&mut Manager::empty(), w)
|
||||||
} else {
|
} else {
|
||||||
writeln!(w, r#"error("Valueless constant")"#)
|
writeln!(w, r#"error("Valueless constant")"#)
|
||||||
}
|
}
|
||||||
@ -178,6 +178,7 @@ fn write_element_list(list: &[Element], type_info: &TypeInfo, w: &mut dyn Write)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
writeln!(w, " }}")?;
|
writeln!(w, " }}")?;
|
||||||
writeln!(w, "\t\ttable.move(data, 1, #data, offset, target)")?;
|
writeln!(w, "\t\ttable.move(data, 1, #data, offset, target)")?;
|
||||||
writeln!(w, "\tend")?;
|
writeln!(w, "\tend")?;
|
||||||
@ -270,7 +271,7 @@ fn write_func_list(wasm: &Module, func_list: &[FuncData], w: &mut dyn Write) ->
|
|||||||
|
|
||||||
write_func_start(wasm, index, w)?;
|
write_func_start(wasm, index, w)?;
|
||||||
|
|
||||||
v.write(&mut Manager::default(), w)
|
v.write(&mut Manager::function(v), w)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -309,9 +310,9 @@ fn write_module_start(
|
|||||||
/// # Errors
|
/// # Errors
|
||||||
/// Returns `Err` if writing to `Write` failed.
|
/// Returns `Err` if writing to `Write` failed.
|
||||||
pub fn from_inst_list(code: &[Operator], type_info: &TypeInfo, w: &mut dyn Write) -> Result<()> {
|
pub fn from_inst_list(code: &[Operator], type_info: &TypeInfo, w: &mut dyn Write) -> Result<()> {
|
||||||
Factory::from_type_info(type_info)
|
let ast = Factory::from_type_info(type_info).create_anonymous(code);
|
||||||
.create_anonymous(code)
|
|
||||||
.write(&mut Manager::default(), w)
|
ast.write(&mut Manager::function(&ast), w)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// # Errors
|
/// # Errors
|
||||||
|
@ -92,7 +92,7 @@ pub fn visit(ast: &FuncData) -> (BTreeSet<(&'static str, &'static str)>, BTreeSe
|
|||||||
memory_set: BTreeSet::new(),
|
memory_set: BTreeSet::new(),
|
||||||
};
|
};
|
||||||
|
|
||||||
if ast.local_data().iter().any(|v| v.1 == ValType::I64) {
|
if ast.local_data().iter().any(|&v| v == ValType::I64) {
|
||||||
visit.local_set.insert(("i64", "ZERO"));
|
visit.local_set.insert(("i64", "ZERO"));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -4,13 +4,12 @@ use std::{
|
|||||||
};
|
};
|
||||||
|
|
||||||
use wasm_ast::node::{
|
use wasm_ast::node::{
|
||||||
BinOp, CmpOp, Expression, GetGlobal, GetLocal, GetTemporary, LoadAt, MemorySize, Select, UnOp,
|
BinOp, CmpOp, Expression, GetGlobal, LoadAt, Local, MemorySize, Select, Temporary, UnOp, Value,
|
||||||
Value,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
use crate::analyzer::as_symbol::AsSymbol;
|
use crate::analyzer::as_symbol::AsSymbol;
|
||||||
|
|
||||||
use super::manager::{write_separated, DriverNoContext};
|
use super::manager::{write_separated, Driver, Manager};
|
||||||
|
|
||||||
macro_rules! impl_write_number {
|
macro_rules! impl_write_number {
|
||||||
($name:tt, $numeric:ty) => {
|
($name:tt, $numeric:ty) => {
|
||||||
@ -26,43 +25,55 @@ macro_rules! impl_write_number {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for Select {
|
impl Driver for Select {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "(if ")?;
|
write!(w, "(if ")?;
|
||||||
Condition(self.condition()).write(w)?;
|
Condition(self.condition()).write(mng, w)?;
|
||||||
write!(w, " then ")?;
|
write!(w, " then ")?;
|
||||||
self.on_true().write(w)?;
|
self.on_true().write(mng, w)?;
|
||||||
write!(w, " else ")?;
|
write!(w, " else ")?;
|
||||||
self.on_false().write(w)?;
|
self.on_false().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for GetTemporary {
|
impl Driver for Temporary {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "reg_{}", self.var())
|
let var = self.var();
|
||||||
|
|
||||||
|
if let Some(var) = var.checked_sub(mng.num_temp()) {
|
||||||
|
write!(w, "reg_spill[{}]", var + 1)
|
||||||
|
} else {
|
||||||
|
write!(w, "reg_{var}")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for GetLocal {
|
impl Driver for Local {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "loc_{}", self.var())
|
let var = self.var();
|
||||||
|
|
||||||
|
if let Some(var) = var.checked_sub(mng.num_local()) {
|
||||||
|
write!(w, "loc_spill[{}]", var + 1)
|
||||||
|
} else {
|
||||||
|
write!(w, "loc_{var}")
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for GetGlobal {
|
impl Driver for GetGlobal {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, _mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "GLOBAL_LIST[{}].value", self.var())
|
write!(w, "GLOBAL_LIST[{}].value", self.var())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for LoadAt {
|
impl Driver for LoadAt {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let name = self.load_type().as_name();
|
let name = self.load_type().as_name();
|
||||||
let memory = self.memory();
|
let memory = self.memory();
|
||||||
|
|
||||||
write!(w, "load_{name}(memory_at_{memory}, ")?;
|
write!(w, "load_{name}(memory_at_{memory}, ")?;
|
||||||
self.pointer().write(w)?;
|
self.pointer().write(mng, w)?;
|
||||||
|
|
||||||
if self.offset() != 0 {
|
if self.offset() != 0 {
|
||||||
write!(w, " + {}", self.offset())?;
|
write!(w, " + {}", self.offset())?;
|
||||||
@ -72,8 +83,8 @@ impl DriverNoContext for LoadAt {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for MemorySize {
|
impl Driver for MemorySize {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, _mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "memory_at_{}.min", self.memory())
|
write!(w, "memory_at_{}.min", self.memory())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -101,8 +112,8 @@ fn write_i64(number: i64, w: &mut dyn Write) -> Result<()> {
|
|||||||
impl_write_number!(write_f32, f32);
|
impl_write_number!(write_f32, f32);
|
||||||
impl_write_number!(write_f64, f64);
|
impl_write_number!(write_f64, f64);
|
||||||
|
|
||||||
impl DriverNoContext for Value {
|
impl Driver for Value {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, _mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
match self {
|
match self {
|
||||||
Self::I32(i) => write_i32(*i, w),
|
Self::I32(i) => write_i32(*i, w),
|
||||||
Self::I64(i) => write_i64(*i, w),
|
Self::I64(i) => write_i64(*i, w),
|
||||||
@ -112,97 +123,97 @@ impl DriverNoContext for Value {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for UnOp {
|
impl Driver for UnOp {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let (a, b) = self.op_type().as_name();
|
let (a, b) = self.op_type().as_name();
|
||||||
|
|
||||||
write!(w, "{a}_{b}(")?;
|
write!(w, "{a}_{b}(")?;
|
||||||
self.rhs().write(w)?;
|
self.rhs().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for BinOp {
|
impl Driver for BinOp {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
if let Some(symbol) = self.op_type().as_symbol() {
|
if let Some(symbol) = self.op_type().as_symbol() {
|
||||||
write!(w, "(")?;
|
write!(w, "(")?;
|
||||||
self.lhs().write(w)?;
|
self.lhs().write(mng, w)?;
|
||||||
write!(w, " {symbol} ")?;
|
write!(w, " {symbol} ")?;
|
||||||
} else {
|
} else {
|
||||||
let (head, tail) = self.op_type().as_name();
|
let (head, tail) = self.op_type().as_name();
|
||||||
|
|
||||||
write!(w, "{head}_{tail}(")?;
|
write!(w, "{head}_{tail}(")?;
|
||||||
self.lhs().write(w)?;
|
self.lhs().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
self.rhs().write(w)?;
|
self.rhs().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
struct CmpOpBoolean<'a>(&'a CmpOp);
|
struct CmpOpBoolean<'a>(&'a CmpOp);
|
||||||
|
|
||||||
impl DriverNoContext for CmpOpBoolean<'_> {
|
impl Driver for CmpOpBoolean<'_> {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let cmp = self.0;
|
let cmp = self.0;
|
||||||
|
|
||||||
if let Some(symbol) = cmp.op_type().as_symbol() {
|
if let Some(symbol) = cmp.op_type().as_symbol() {
|
||||||
cmp.lhs().write(w)?;
|
cmp.lhs().write(mng, w)?;
|
||||||
write!(w, " {symbol} ")?;
|
write!(w, " {symbol} ")?;
|
||||||
cmp.rhs().write(w)
|
cmp.rhs().write(mng, w)
|
||||||
} else {
|
} else {
|
||||||
let (head, tail) = cmp.op_type().as_name();
|
let (head, tail) = cmp.op_type().as_name();
|
||||||
|
|
||||||
write!(w, "{head}_{tail}(")?;
|
write!(w, "{head}_{tail}(")?;
|
||||||
cmp.lhs().write(w)?;
|
cmp.lhs().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
cmp.rhs().write(w)?;
|
cmp.rhs().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for CmpOp {
|
impl Driver for CmpOp {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "(if ")?;
|
write!(w, "(if ")?;
|
||||||
CmpOpBoolean(self).write(w)?;
|
CmpOpBoolean(self).write(mng, w)?;
|
||||||
write!(w, " then 1 else 0)")
|
write!(w, " then 1 else 0)")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Condition<'a>(pub &'a Expression);
|
pub struct Condition<'a>(pub &'a Expression);
|
||||||
|
|
||||||
impl DriverNoContext for Condition<'_> {
|
impl Driver for Condition<'_> {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
if let Expression::CmpOp(node) = self.0 {
|
if let Expression::CmpOp(node) = self.0 {
|
||||||
CmpOpBoolean(node).write(w)
|
CmpOpBoolean(node).write(mng, w)
|
||||||
} else {
|
} else {
|
||||||
self.0.write(w)?;
|
self.0.write(mng, w)?;
|
||||||
write!(w, " ~= 0")
|
write!(w, " ~= 0")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for Expression {
|
impl Driver for Expression {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
match self {
|
match self {
|
||||||
Self::Select(e) => e.write(w),
|
Self::Select(e) => e.write(mng, w),
|
||||||
Self::GetTemporary(e) => e.write(w),
|
Self::GetTemporary(e) => e.write(mng, w),
|
||||||
Self::GetLocal(e) => e.write(w),
|
Self::GetLocal(e) => e.write(mng, w),
|
||||||
Self::GetGlobal(e) => e.write(w),
|
Self::GetGlobal(e) => e.write(mng, w),
|
||||||
Self::LoadAt(e) => e.write(w),
|
Self::LoadAt(e) => e.write(mng, w),
|
||||||
Self::MemorySize(e) => e.write(w),
|
Self::MemorySize(e) => e.write(mng, w),
|
||||||
Self::Value(e) => e.write(w),
|
Self::Value(e) => e.write(mng, w),
|
||||||
Self::UnOp(e) => e.write(w),
|
Self::UnOp(e) => e.write(mng, w),
|
||||||
Self::BinOp(e) => e.write(w),
|
Self::BinOp(e) => e.write(mng, w),
|
||||||
Self::CmpOp(e) => e.write(w),
|
Self::CmpOp(e) => e.write(mng, w),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for &[Expression] {
|
impl Driver for &[Expression] {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write_separated(self.iter(), |e, w| e.write(w), w)
|
write_separated(self.iter(), |e, w| e.write(mng, w), w)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,10 +1,11 @@
|
|||||||
use std::{
|
use std::{
|
||||||
collections::HashMap,
|
collections::HashMap,
|
||||||
io::{Result, Write},
|
io::{Result, Write},
|
||||||
ops::Range,
|
|
||||||
};
|
};
|
||||||
|
|
||||||
use wasm_ast::node::{BrTable, LabelType};
|
use wasm_ast::node::{BrTable, FuncData, LabelType};
|
||||||
|
|
||||||
|
use crate::analyzer::{br_target, localize};
|
||||||
|
|
||||||
#[macro_export]
|
#[macro_export]
|
||||||
macro_rules! indentation {
|
macro_rules! indentation {
|
||||||
@ -31,30 +32,87 @@ macro_rules! line {
|
|||||||
}};
|
}};
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Default)]
|
fn get_pinned_registers(
|
||||||
|
upvalues: usize,
|
||||||
|
params: usize,
|
||||||
|
locals: usize,
|
||||||
|
temporaries: usize,
|
||||||
|
) -> (usize, usize) {
|
||||||
|
const MAX_LOCAL_COUNT: usize = 170;
|
||||||
|
|
||||||
|
let available = MAX_LOCAL_COUNT
|
||||||
|
.saturating_sub(upvalues)
|
||||||
|
.saturating_sub(params);
|
||||||
|
|
||||||
|
let locals = available.min(locals);
|
||||||
|
let temporaries = available.saturating_sub(locals).min(temporaries);
|
||||||
|
|
||||||
|
(params + locals, temporaries)
|
||||||
|
}
|
||||||
|
|
||||||
pub struct Manager {
|
pub struct Manager {
|
||||||
table_map: HashMap<usize, usize>,
|
table_map: HashMap<usize, usize>,
|
||||||
has_branch: bool,
|
has_branch: bool,
|
||||||
|
num_local: usize,
|
||||||
|
num_temp: usize,
|
||||||
label_list: Vec<Option<LabelType>>,
|
label_list: Vec<Option<LabelType>>,
|
||||||
indentation: usize,
|
indentation: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Manager {
|
impl Manager {
|
||||||
|
pub fn empty() -> Self {
|
||||||
|
Self {
|
||||||
|
table_map: HashMap::new(),
|
||||||
|
has_branch: false,
|
||||||
|
num_local: 0,
|
||||||
|
num_temp: usize::MAX,
|
||||||
|
label_list: Vec::new(),
|
||||||
|
indentation: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn function(ast: &FuncData) -> Self {
|
||||||
|
let (upvalues, memories) = localize::visit(ast);
|
||||||
|
let (table_map, has_branch) = br_target::visit(ast);
|
||||||
|
let (num_local, num_temp) = get_pinned_registers(
|
||||||
|
upvalues.len() + memories.len(),
|
||||||
|
ast.num_param(),
|
||||||
|
ast.local_data().len(),
|
||||||
|
ast.num_stack(),
|
||||||
|
);
|
||||||
|
|
||||||
|
Self {
|
||||||
|
table_map,
|
||||||
|
has_branch,
|
||||||
|
num_local,
|
||||||
|
num_temp,
|
||||||
|
label_list: Vec::new(),
|
||||||
|
indentation: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_table_index(&self, table: &BrTable) -> usize {
|
pub fn get_table_index(&self, table: &BrTable) -> usize {
|
||||||
let id = table as *const _ as usize;
|
let id = table as *const _ as usize;
|
||||||
|
|
||||||
self.table_map[&id]
|
self.table_map[&id]
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_branch_information(&mut self, table_map: HashMap<usize, usize>, has_branch: bool) {
|
pub fn has_table(&self) -> bool {
|
||||||
self.table_map = table_map;
|
!self.table_map.is_empty()
|
||||||
self.has_branch = has_branch;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub const fn has_branch(&self) -> bool {
|
pub const fn has_branch(&self) -> bool {
|
||||||
self.has_branch
|
self.has_branch
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub const fn num_local(&self) -> usize {
|
||||||
|
self.num_local
|
||||||
|
}
|
||||||
|
|
||||||
|
pub const fn num_temp(&self) -> usize {
|
||||||
|
self.num_temp
|
||||||
|
}
|
||||||
|
|
||||||
pub fn label_list(&self) -> &[Option<LabelType>] {
|
pub fn label_list(&self) -> &[Option<LabelType>] {
|
||||||
&self.label_list
|
&self.label_list
|
||||||
}
|
}
|
||||||
@ -84,10 +142,6 @@ pub trait Driver {
|
|||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()>;
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()>;
|
||||||
}
|
}
|
||||||
|
|
||||||
pub trait DriverNoContext {
|
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()>;
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn write_separated<I, T, M>(mut iter: I, mut func: M, w: &mut dyn Write) -> Result<()>
|
pub fn write_separated<I, T, M>(mut iter: I, mut func: M, w: &mut dyn Write) -> Result<()>
|
||||||
where
|
where
|
||||||
M: FnMut(T, &mut dyn Write) -> Result<()>,
|
M: FnMut(T, &mut dyn Write) -> Result<()>,
|
||||||
@ -103,7 +157,3 @@ where
|
|||||||
func(v, w)
|
func(v, w)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn write_ascending(prefix: &str, range: Range<usize>, w: &mut dyn Write) -> Result<()> {
|
|
||||||
write_separated(range, |i, w| write!(w, "{prefix}_{i}"), w)
|
|
||||||
}
|
|
||||||
|
@ -5,24 +5,30 @@ use std::{
|
|||||||
|
|
||||||
use wasm_ast::node::{
|
use wasm_ast::node::{
|
||||||
Block, Br, BrIf, BrTable, Call, CallIndirect, FuncData, If, LabelType, MemoryCopy, MemoryFill,
|
Block, Br, BrIf, BrTable, Call, CallIndirect, FuncData, If, LabelType, MemoryCopy, MemoryFill,
|
||||||
MemoryGrow, SetGlobal, SetLocal, SetTemporary, Statement, StoreAt, Terminator,
|
MemoryGrow, ResultList, SetGlobal, SetLocal, SetTemporary, Statement, StoreAt, Terminator,
|
||||||
};
|
};
|
||||||
use wasmparser::ValType;
|
use wasmparser::ValType;
|
||||||
|
|
||||||
use crate::{analyzer::br_target, indentation, indented, line};
|
use crate::{backend::manager::write_separated, indentation, indented, line};
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
expression::Condition,
|
expression::Condition,
|
||||||
manager::{write_ascending, write_separated, Driver, DriverNoContext, Manager},
|
manager::{Driver, Manager},
|
||||||
};
|
};
|
||||||
|
|
||||||
|
impl Driver for ResultList {
|
||||||
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
|
write_separated(self.iter(), |t, w| t.write(mng, w), w)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl Driver for Br {
|
impl Driver for Br {
|
||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
if !self.align().is_aligned() {
|
if !self.align().is_aligned() {
|
||||||
indentation!(mng, w)?;
|
indentation!(mng, w)?;
|
||||||
write_ascending("reg", self.align().new_range(), w)?;
|
self.align().new_range().write(mng, w)?;
|
||||||
write!(w, " = ")?;
|
write!(w, " = ")?;
|
||||||
write_ascending("reg", self.align().old_range(), w)?;
|
self.align().old_range().write(mng, w)?;
|
||||||
writeln!(w)?;
|
writeln!(w)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,7 +112,7 @@ fn write_table_setup(table: &BrTable, mng: &mut Manager, w: &mut dyn Write) -> R
|
|||||||
line!(mng, w, "end")?;
|
line!(mng, w, "end")?;
|
||||||
|
|
||||||
indented!(mng, w, "temp = br_map[{id}][")?;
|
indented!(mng, w, "temp = br_map[{id}][")?;
|
||||||
table.condition().write(w)?;
|
table.condition().write(mng, w)?;
|
||||||
writeln!(w, "] or {}", table.default().target())
|
writeln!(w, "] or {}", table.default().target())
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -193,7 +199,7 @@ impl Driver for Block {
|
|||||||
impl Driver for BrIf {
|
impl Driver for BrIf {
|
||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
indented!(mng, w, "if ")?;
|
indented!(mng, w, "if ")?;
|
||||||
Condition(self.condition()).write(w)?;
|
Condition(self.condition()).write(mng, w)?;
|
||||||
writeln!(w, " then")?;
|
writeln!(w, " then")?;
|
||||||
mng.indent();
|
mng.indent();
|
||||||
self.target().write(mng, w)?;
|
self.target().write(mng, w)?;
|
||||||
@ -205,7 +211,7 @@ impl Driver for BrIf {
|
|||||||
impl Driver for If {
|
impl Driver for If {
|
||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
indented!(mng, w, "if ")?;
|
indented!(mng, w, "if ")?;
|
||||||
Condition(self.condition()).write(w)?;
|
Condition(self.condition()).write(mng, w)?;
|
||||||
writeln!(w, " then")?;
|
writeln!(w, " then")?;
|
||||||
|
|
||||||
mng.indent();
|
mng.indent();
|
||||||
@ -223,120 +229,119 @@ impl Driver for If {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_call_store(result: Range<usize>, w: &mut dyn Write) -> Result<()> {
|
impl Driver for Call {
|
||||||
if result.is_empty() {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
return Ok(());
|
if !self.result_list().is_empty() {
|
||||||
|
self.result_list().write(mng, w)?;
|
||||||
|
write!(w, " = ")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
write_ascending("reg", result, w)?;
|
|
||||||
write!(w, " = ")
|
|
||||||
}
|
|
||||||
|
|
||||||
impl DriverNoContext for Call {
|
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
|
||||||
write_call_store(self.result(), w)?;
|
|
||||||
|
|
||||||
write!(w, "FUNC_LIST[{}](", self.function())?;
|
write!(w, "FUNC_LIST[{}](", self.function())?;
|
||||||
self.param_list().write(w)?;
|
self.param_list().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for CallIndirect {
|
impl Driver for CallIndirect {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write_call_store(self.result(), w)?;
|
if !self.result_list().is_empty() {
|
||||||
|
self.result_list().write(mng, w)?;
|
||||||
|
write!(w, " = ")?;
|
||||||
|
}
|
||||||
|
|
||||||
write!(w, "TABLE_LIST[{}].data[", self.table())?;
|
write!(w, "TABLE_LIST[{}].data[", self.table())?;
|
||||||
self.index().write(w)?;
|
self.index().write(mng, w)?;
|
||||||
write!(w, "](")?;
|
write!(w, "](")?;
|
||||||
self.param_list().write(w)?;
|
self.param_list().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for SetTemporary {
|
impl Driver for SetTemporary {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "reg_{} = ", self.var())?;
|
self.var().write(mng, w)?;
|
||||||
self.value().write(w)
|
write!(w, " = ")?;
|
||||||
|
self.value().write(mng, w)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for SetLocal {
|
impl Driver for SetLocal {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "loc_{} = ", self.var())?;
|
self.var().write(mng, w)?;
|
||||||
self.value().write(w)
|
write!(w, " = ")?;
|
||||||
|
self.value().write(mng, w)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for SetGlobal {
|
impl Driver for SetGlobal {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "GLOBAL_LIST[{}].value = ", self.var())?;
|
write!(w, "GLOBAL_LIST[{}].value = ", self.var())?;
|
||||||
self.value().write(w)
|
self.value().write(mng, w)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for StoreAt {
|
impl Driver for StoreAt {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let name = self.store_type().as_name();
|
let name = self.store_type().as_name();
|
||||||
let memory = self.memory();
|
let memory = self.memory();
|
||||||
|
|
||||||
write!(w, "store_{name}(memory_at_{memory}, ")?;
|
write!(w, "store_{name}(memory_at_{memory}, ")?;
|
||||||
|
|
||||||
self.pointer().write(w)?;
|
self.pointer().write(mng, w)?;
|
||||||
|
|
||||||
if self.offset() != 0 {
|
if self.offset() != 0 {
|
||||||
write!(w, " + {}", self.offset())?;
|
write!(w, " + {}", self.offset())?;
|
||||||
}
|
}
|
||||||
|
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
self.value().write(w)?;
|
self.value().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for MemoryGrow {
|
impl Driver for MemoryGrow {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let result = self.result();
|
|
||||||
let memory = self.memory();
|
let memory = self.memory();
|
||||||
|
|
||||||
write!(w, "reg_{result} = rt.allocator.grow(memory_at_{memory}, ")?;
|
self.result().write(mng, w)?;
|
||||||
self.size().write(w)?;
|
write!(w, " = rt.allocator.grow(memory_at_{memory}, ")?;
|
||||||
|
self.size().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for MemoryCopy {
|
impl Driver for MemoryCopy {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let memory_1 = self.destination().memory();
|
let memory_1 = self.destination().memory();
|
||||||
let memory_2 = self.source().memory();
|
let memory_2 = self.source().memory();
|
||||||
|
|
||||||
write!(w, "rt.store.copy(memory_at_{memory_1}, ")?;
|
write!(w, "rt.store.copy(memory_at_{memory_1}, ")?;
|
||||||
self.destination().pointer().write(w)?;
|
self.destination().pointer().write(mng, w)?;
|
||||||
write!(w, ", memory_at_{memory_2}, ")?;
|
write!(w, ", memory_at_{memory_2}, ")?;
|
||||||
self.source().pointer().write(w)?;
|
self.source().pointer().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
self.size().write(w)?;
|
self.size().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DriverNoContext for MemoryFill {
|
impl Driver for MemoryFill {
|
||||||
fn write(&self, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let memory = self.destination().memory();
|
let memory = self.destination().memory();
|
||||||
|
|
||||||
write!(w, "rt.store.fill(memory_at_{memory}, ")?;
|
write!(w, "rt.store.fill(memory_at_{memory}, ")?;
|
||||||
self.destination().pointer().write(w)?;
|
self.destination().pointer().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
self.size().write(w)?;
|
self.size().write(mng, w)?;
|
||||||
write!(w, ", ")?;
|
write!(w, ", ")?;
|
||||||
self.value().write(w)?;
|
self.value().write(mng, w)?;
|
||||||
write!(w, ")")
|
write!(w, ")")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn write_stat(stat: &dyn DriverNoContext, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write_stat(stat: &dyn Driver, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
indentation!(mng, w)?;
|
indentation!(mng, w)?;
|
||||||
stat.write(w)?;
|
stat.write(mng, w)?;
|
||||||
writeln!(w)
|
writeln!(w)
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -359,45 +364,53 @@ impl Driver for Statement {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn has_sane_variables(ast: &FuncData) -> bool {
|
|
||||||
let local_count = ast.local_data().iter().map(|v| v.0).sum::<u32>();
|
|
||||||
let local_count = usize::try_from(local_count).unwrap();
|
|
||||||
let param_count = ast.num_param();
|
|
||||||
let temp_count = ast.num_stack();
|
|
||||||
|
|
||||||
temp_count + param_count + local_count < 170
|
|
||||||
}
|
|
||||||
|
|
||||||
fn write_parameter_list(ast: &FuncData, w: &mut dyn Write) -> Result<()> {
|
fn write_parameter_list(ast: &FuncData, w: &mut dyn Write) -> Result<()> {
|
||||||
write!(w, "function(")?;
|
write!(w, "function(")?;
|
||||||
write_ascending("loc", 0..ast.num_param(), w)?;
|
write_separated(0..ast.num_param(), |i, w| write!(w, "loc_{i}"), w)?;
|
||||||
writeln!(w, ")")
|
writeln!(w, ")")
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const fn type_to_zero(typ: ValType) -> &'static str {
|
||||||
|
match typ {
|
||||||
|
ValType::F32 | ValType::F64 => "0.0",
|
||||||
|
ValType::I64 => "i64_ZERO",
|
||||||
|
_ => "0",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
fn write_variable_list(ast: &FuncData, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write_variable_list(ast: &FuncData, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
let mut total = ast.num_param();
|
let mut locals = ast.local_data().iter().copied();
|
||||||
|
let num_local = mng.num_local() - ast.num_param();
|
||||||
|
|
||||||
for data in ast.local_data().iter().filter(|v| v.0 != 0) {
|
for (i, typ) in locals.by_ref().enumerate().take(num_local) {
|
||||||
let range = total..total + usize::try_from(data.0).unwrap();
|
let index = ast.num_param() + i;
|
||||||
let zero = if data.1 == ValType::I64 {
|
let zero = type_to_zero(typ);
|
||||||
"i64_ZERO"
|
|
||||||
} else {
|
|
||||||
"0"
|
|
||||||
};
|
|
||||||
|
|
||||||
total = range.end;
|
line!(mng, w, "local loc_{index} = {zero}")?;
|
||||||
|
|
||||||
indented!(mng, w, "local ")?;
|
|
||||||
write_ascending("loc", range.clone(), w)?;
|
|
||||||
write!(w, " = ")?;
|
|
||||||
write_separated(range, |_, w| w.write_all(zero.as_bytes()), w)?;
|
|
||||||
writeln!(w)?;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ast.num_stack() != 0 {
|
if locals.len() != 0 {
|
||||||
indented!(mng, w, "local ")?;
|
indented!(mng, w, "local loc_spill = {{ ")?;
|
||||||
write_ascending("reg", 0..ast.num_stack(), w)?;
|
|
||||||
writeln!(w)?;
|
for typ in locals {
|
||||||
|
let zero = type_to_zero(typ);
|
||||||
|
|
||||||
|
write!(w, "{zero}, ")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
writeln!(w, "}}")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut temporaries = 0..ast.num_stack();
|
||||||
|
|
||||||
|
for i in temporaries.by_ref().take(mng.num_temp()) {
|
||||||
|
line!(mng, w, "local reg_{i}")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if !temporaries.is_empty() {
|
||||||
|
let len = temporaries.len();
|
||||||
|
|
||||||
|
line!(mng, w, "local reg_spill = table.create({len})")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
@ -405,34 +418,26 @@ fn write_variable_list(ast: &FuncData, mng: &mut Manager, w: &mut dyn Write) ->
|
|||||||
|
|
||||||
impl Driver for FuncData {
|
impl Driver for FuncData {
|
||||||
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
fn write(&self, mng: &mut Manager, w: &mut dyn Write) -> Result<()> {
|
||||||
if !has_sane_variables(self) {
|
|
||||||
return Err(std::io::Error::new(
|
|
||||||
std::io::ErrorKind::Other,
|
|
||||||
"too many variables in function",
|
|
||||||
));
|
|
||||||
}
|
|
||||||
|
|
||||||
let (table_map, has_branch) = br_target::visit(self);
|
|
||||||
|
|
||||||
mng.indent();
|
mng.indent();
|
||||||
|
|
||||||
write_parameter_list(self, w)?;
|
write_parameter_list(self, w)?;
|
||||||
write_variable_list(self, mng, w)?;
|
write_variable_list(self, mng, w)?;
|
||||||
|
|
||||||
if has_branch {
|
if mng.has_branch() {
|
||||||
line!(mng, w, "local desired")?;
|
line!(mng, w, "local desired")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
if !table_map.is_empty() {
|
if mng.has_table() {
|
||||||
line!(mng, w, "local br_map = {{}}")?;
|
line!(mng, w, "local br_map = {{}}")?;
|
||||||
}
|
}
|
||||||
|
|
||||||
mng.set_branch_information(table_map, has_branch);
|
|
||||||
self.code().write(mng, w)?;
|
self.code().write(mng, w)?;
|
||||||
|
|
||||||
if self.num_result() != 0 {
|
if self.num_result() != 0 {
|
||||||
indented!(mng, w, "return ")?;
|
indented!(mng, w, "return ")?;
|
||||||
write_ascending("reg", 0..self.num_result(), w)?;
|
|
||||||
|
ResultList::new(0, self.num_result()).write(mng, w)?;
|
||||||
|
|
||||||
writeln!(w)?;
|
writeln!(w)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -15,7 +15,7 @@ use wasmparser::{
|
|||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
analyzer::localize,
|
analyzer::localize,
|
||||||
backend::manager::{Driver, DriverNoContext, Manager},
|
backend::manager::{Driver, Manager},
|
||||||
};
|
};
|
||||||
|
|
||||||
trait AsIEName {
|
trait AsIEName {
|
||||||
@ -51,7 +51,7 @@ fn write_constant(init: &ConstExpr, type_info: &TypeInfo, w: &mut dyn Write) ->
|
|||||||
let func = Factory::from_type_info(type_info).create_anonymous(&code);
|
let func = Factory::from_type_info(type_info).create_anonymous(&code);
|
||||||
|
|
||||||
if let Some(Statement::SetTemporary(stat)) = func.code().code().last() {
|
if let Some(Statement::SetTemporary(stat)) = func.code().code().last() {
|
||||||
stat.value().write(w)
|
stat.value().write(&mut Manager::empty(), w)
|
||||||
} else {
|
} else {
|
||||||
writeln!(w, r#"error("Valueless constant")"#)
|
writeln!(w, r#"error("Valueless constant")"#)
|
||||||
}
|
}
|
||||||
@ -283,7 +283,7 @@ fn write_func_list(wasm: &Module, func_list: &[FuncData], w: &mut dyn Write) ->
|
|||||||
|
|
||||||
write_func_start(wasm, index, w)?;
|
write_func_start(wasm, index, w)?;
|
||||||
|
|
||||||
v.write(&mut Manager::default(), w)
|
v.write(&mut Manager::function(v), w)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -322,9 +322,9 @@ fn write_module_start(
|
|||||||
/// # Errors
|
/// # Errors
|
||||||
/// Returns `Err` if writing to `Write` failed.
|
/// Returns `Err` if writing to `Write` failed.
|
||||||
pub fn from_inst_list(code: &[Operator], type_info: &TypeInfo, w: &mut dyn Write) -> Result<()> {
|
pub fn from_inst_list(code: &[Operator], type_info: &TypeInfo, w: &mut dyn Write) -> Result<()> {
|
||||||
Factory::from_type_info(type_info)
|
let ast = Factory::from_type_info(type_info).create_anonymous(code);
|
||||||
.create_anonymous(code)
|
|
||||||
.write(&mut Manager::default(), w)
|
ast.write(&mut Manager::function(&ast), w)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// # Errors
|
/// # Errors
|
||||||
|
@ -1,10 +1,10 @@
|
|||||||
use wasmparser::{BlockType, FunctionBody, MemArg, Operator, Result};
|
use wasmparser::{BlockType, FunctionBody, MemArg, Operator, Result};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
module::{read_checked, TypeInfo},
|
module::{read_checked, read_checked_locals, TypeInfo},
|
||||||
node::{
|
node::{
|
||||||
BinOp, BinOpType, Block, Br, BrIf, BrTable, Call, CallIndirect, CmpOp, CmpOpType,
|
BinOp, BinOpType, Block, Br, BrIf, BrTable, Call, CallIndirect, CmpOp, CmpOpType,
|
||||||
Expression, FuncData, GetGlobal, GetLocal, If, LabelType, LoadAt, LoadType, MemoryArgument,
|
Expression, FuncData, GetGlobal, If, LabelType, LoadAt, LoadType, Local, MemoryArgument,
|
||||||
MemoryCopy, MemoryFill, MemoryGrow, MemorySize, Select, SetGlobal, SetLocal, Statement,
|
MemoryCopy, MemoryFill, MemoryGrow, MemorySize, Select, SetGlobal, SetLocal, Statement,
|
||||||
StoreAt, StoreType, Terminator, UnOp, UnOpType, Value,
|
StoreAt, StoreType, Terminator, UnOp, UnOpType, Value,
|
||||||
},
|
},
|
||||||
@ -253,13 +253,13 @@ impl<'a> Factory<'a> {
|
|||||||
/// Returns an error if the function is malformed.
|
/// Returns an error if the function is malformed.
|
||||||
pub fn create_indexed(&mut self, index: usize, func: &FunctionBody) -> Result<FuncData> {
|
pub fn create_indexed(&mut self, index: usize, func: &FunctionBody) -> Result<FuncData> {
|
||||||
let code = read_checked(func.get_operators_reader()?)?;
|
let code = read_checked(func.get_operators_reader()?)?;
|
||||||
let local = read_checked(func.get_locals_reader()?)?;
|
let local_data = read_checked_locals(func.get_locals_reader()?)?;
|
||||||
|
|
||||||
let (num_param, num_result) = self.type_info.by_func_index(index);
|
let (num_param, num_result) = self.type_info.by_func_index(index);
|
||||||
let data = self.build_stat_list(&code, num_result);
|
let data = self.build_stat_list(&code, num_result);
|
||||||
|
|
||||||
Ok(FuncData {
|
Ok(FuncData {
|
||||||
local_data: local,
|
local_data,
|
||||||
num_result,
|
num_result,
|
||||||
num_param,
|
num_param,
|
||||||
num_stack: data.stack.capacity,
|
num_stack: data.stack.capacity,
|
||||||
@ -279,7 +279,7 @@ impl<'a> Factory<'a> {
|
|||||||
BlockVariant::If => BlockData::If { num_result, ty },
|
BlockVariant::If => BlockData::If { num_result, ty },
|
||||||
BlockVariant::Else => {
|
BlockVariant::Else => {
|
||||||
old.stack.pop_len(num_result).for_each(drop);
|
old.stack.pop_len(num_result).for_each(drop);
|
||||||
old.stack.push_temporary(num_param);
|
old.stack.push_temporaries(num_param);
|
||||||
|
|
||||||
BlockData::Else { num_result }
|
BlockData::Else { num_result }
|
||||||
}
|
}
|
||||||
@ -287,7 +287,7 @@ impl<'a> Factory<'a> {
|
|||||||
|
|
||||||
self.target.stack = old.stack.split_last(num_param, num_result);
|
self.target.stack = old.stack.split_last(num_param, num_result);
|
||||||
|
|
||||||
old.stack.push_temporary(num_result);
|
old.stack.push_temporaries(num_result);
|
||||||
|
|
||||||
self.pending.push(old);
|
self.pending.push(old);
|
||||||
}
|
}
|
||||||
@ -358,12 +358,12 @@ impl<'a> Factory<'a> {
|
|||||||
|
|
||||||
self.target.leak_pre_call();
|
self.target.leak_pre_call();
|
||||||
|
|
||||||
let result = self.target.stack.push_temporary(num_result);
|
let result_list = self.target.stack.push_temporaries(num_result);
|
||||||
|
|
||||||
let data = Statement::Call(Call {
|
let data = Statement::Call(Call {
|
||||||
function,
|
function,
|
||||||
result,
|
|
||||||
param_list,
|
param_list,
|
||||||
|
result_list,
|
||||||
});
|
});
|
||||||
|
|
||||||
self.target.code.push(data);
|
self.target.code.push(data);
|
||||||
@ -376,13 +376,13 @@ impl<'a> Factory<'a> {
|
|||||||
|
|
||||||
self.target.leak_pre_call();
|
self.target.leak_pre_call();
|
||||||
|
|
||||||
let result = self.target.stack.push_temporary(num_result);
|
let result_list = self.target.stack.push_temporaries(num_result);
|
||||||
|
|
||||||
let data = Statement::CallIndirect(CallIndirect {
|
let data = Statement::CallIndirect(CallIndirect {
|
||||||
table,
|
table,
|
||||||
index,
|
index,
|
||||||
result,
|
|
||||||
param_list,
|
param_list,
|
||||||
|
result_list,
|
||||||
});
|
});
|
||||||
|
|
||||||
self.target.code.push(data);
|
self.target.code.push(data);
|
||||||
@ -521,14 +521,14 @@ impl<'a> Factory<'a> {
|
|||||||
}
|
}
|
||||||
Operator::LocalGet { local_index } => {
|
Operator::LocalGet { local_index } => {
|
||||||
let var = local_index.try_into().unwrap();
|
let var = local_index.try_into().unwrap();
|
||||||
let data = Expression::GetLocal(GetLocal { var });
|
let data = Expression::GetLocal(Local { var });
|
||||||
|
|
||||||
self.target.stack.push_with_single(data);
|
self.target.stack.push_with_single(data);
|
||||||
}
|
}
|
||||||
Operator::LocalSet { local_index } => {
|
Operator::LocalSet { local_index } => {
|
||||||
let var = local_index.try_into().unwrap();
|
let var = local_index.try_into().unwrap();
|
||||||
let data = Statement::SetLocal(SetLocal {
|
let data = Statement::SetLocal(SetLocal {
|
||||||
var,
|
var: Local { var },
|
||||||
value: self.target.stack.pop().into(),
|
value: self.target.stack.pop().into(),
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -537,9 +537,9 @@ impl<'a> Factory<'a> {
|
|||||||
}
|
}
|
||||||
Operator::LocalTee { local_index } => {
|
Operator::LocalTee { local_index } => {
|
||||||
let var = local_index.try_into().unwrap();
|
let var = local_index.try_into().unwrap();
|
||||||
let get = Expression::GetLocal(GetLocal { var });
|
let get = Expression::GetLocal(Local { var });
|
||||||
let set = Statement::SetLocal(SetLocal {
|
let set = Statement::SetLocal(SetLocal {
|
||||||
var,
|
var: Local { var },
|
||||||
value: self.target.stack.pop().into(),
|
value: self.target.stack.pop().into(),
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -594,7 +594,7 @@ impl<'a> Factory<'a> {
|
|||||||
}
|
}
|
||||||
Operator::MemoryGrow { mem, .. } => {
|
Operator::MemoryGrow { mem, .. } => {
|
||||||
let size = self.target.stack.pop().into();
|
let size = self.target.stack.pop().into();
|
||||||
let result = self.target.stack.push_temporary(1).start;
|
let result = self.target.stack.push_temporary();
|
||||||
let memory = mem.try_into().unwrap();
|
let memory = mem.try_into().unwrap();
|
||||||
|
|
||||||
let data = Statement::MemoryGrow(MemoryGrow {
|
let data = Statement::MemoryGrow(MemoryGrow {
|
||||||
|
@ -1,8 +1,9 @@
|
|||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
||||||
use wasmparser::{
|
use wasmparser::{
|
||||||
BlockType, Data, Element, Export, ExternalKind, FunctionBody, Global, Import, MemoryType, Name,
|
BlockType, Data, Element, Export, ExternalKind, FunctionBody, Global, Import, LocalsReader,
|
||||||
NameSectionReader, Parser, Payload, Result, TableType, Type, TypeRef,
|
MemoryType, Name, NameSectionReader, Parser, Payload, Result, TableType, Type, TypeRef,
|
||||||
|
ValType,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(PartialEq, Eq, Clone, Copy)]
|
#[derive(PartialEq, Eq, Clone, Copy)]
|
||||||
@ -45,6 +46,14 @@ where
|
|||||||
reader.into_iter().collect()
|
reader.into_iter().collect()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn read_checked_locals(reader: LocalsReader) -> Result<Vec<ValType>> {
|
||||||
|
read_checked(reader).map(|locals| {
|
||||||
|
let convert = |(a, b)| std::iter::repeat(b).take(usize::try_from(a).unwrap());
|
||||||
|
|
||||||
|
locals.into_iter().flat_map(convert).collect()
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
pub struct Module<'a> {
|
pub struct Module<'a> {
|
||||||
type_section: Vec<Type>,
|
type_section: Vec<Type>,
|
||||||
import_section: Vec<Import<'a>>,
|
import_section: Vec<Import<'a>>,
|
||||||
|
@ -1,5 +1,3 @@
|
|||||||
use std::ops::Range;
|
|
||||||
|
|
||||||
use wasmparser::{Operator, ValType};
|
use wasmparser::{Operator, ValType};
|
||||||
|
|
||||||
#[allow(non_camel_case_types)]
|
#[allow(non_camel_case_types)]
|
||||||
@ -627,22 +625,22 @@ impl Select {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct GetTemporary {
|
pub struct Temporary {
|
||||||
pub(crate) var: usize,
|
pub(crate) var: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GetTemporary {
|
impl Temporary {
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub const fn var(&self) -> usize {
|
pub const fn var(&self) -> usize {
|
||||||
self.var
|
self.var
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct GetLocal {
|
pub struct Local {
|
||||||
pub(crate) var: usize,
|
pub(crate) var: usize,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GetLocal {
|
impl Local {
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub const fn var(&self) -> usize {
|
pub const fn var(&self) -> usize {
|
||||||
self.var
|
self.var
|
||||||
@ -797,8 +795,8 @@ impl CmpOp {
|
|||||||
|
|
||||||
pub enum Expression {
|
pub enum Expression {
|
||||||
Select(Select),
|
Select(Select),
|
||||||
GetTemporary(GetTemporary),
|
GetTemporary(Temporary),
|
||||||
GetLocal(GetLocal),
|
GetLocal(Local),
|
||||||
GetGlobal(GetGlobal),
|
GetGlobal(GetGlobal),
|
||||||
LoadAt(LoadAt),
|
LoadAt(LoadAt),
|
||||||
MemorySize(MemorySize),
|
MemorySize(MemorySize),
|
||||||
@ -808,6 +806,28 @@ pub enum Expression {
|
|||||||
CmpOp(CmpOp),
|
CmpOp(CmpOp),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[derive(Clone, Copy)]
|
||||||
|
pub struct ResultList {
|
||||||
|
start: usize,
|
||||||
|
end: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl ResultList {
|
||||||
|
#[must_use]
|
||||||
|
pub const fn new(start: usize, end: usize) -> Self {
|
||||||
|
Self { start, end }
|
||||||
|
}
|
||||||
|
|
||||||
|
#[must_use]
|
||||||
|
pub const fn is_empty(self) -> bool {
|
||||||
|
self.start == self.end
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn iter(self) -> impl Iterator<Item = Temporary> {
|
||||||
|
(self.start..self.end).map(|var| Temporary { var })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub struct Align {
|
pub struct Align {
|
||||||
pub(crate) new: usize,
|
pub(crate) new: usize,
|
||||||
pub(crate) old: usize,
|
pub(crate) old: usize,
|
||||||
@ -821,13 +841,13 @@ impl Align {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub const fn new_range(&self) -> Range<usize> {
|
pub const fn new_range(&self) -> ResultList {
|
||||||
self.new..self.new + self.length
|
ResultList::new(self.new, self.new + self.length)
|
||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub const fn old_range(&self) -> Range<usize> {
|
pub const fn old_range(&self) -> ResultList {
|
||||||
self.old..self.old + self.length
|
ResultList::new(self.old, self.old + self.length)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -949,8 +969,8 @@ impl If {
|
|||||||
|
|
||||||
pub struct Call {
|
pub struct Call {
|
||||||
pub(crate) function: usize,
|
pub(crate) function: usize,
|
||||||
pub(crate) result: Range<usize>,
|
|
||||||
pub(crate) param_list: Vec<Expression>,
|
pub(crate) param_list: Vec<Expression>,
|
||||||
|
pub(crate) result_list: ResultList,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Call {
|
impl Call {
|
||||||
@ -960,21 +980,21 @@ impl Call {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn result(&self) -> Range<usize> {
|
pub fn param_list(&self) -> &[Expression] {
|
||||||
self.result.clone()
|
&self.param_list
|
||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn param_list(&self) -> &[Expression] {
|
pub const fn result_list(&self) -> ResultList {
|
||||||
&self.param_list
|
self.result_list
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct CallIndirect {
|
pub struct CallIndirect {
|
||||||
pub(crate) table: usize,
|
pub(crate) table: usize,
|
||||||
pub(crate) index: Box<Expression>,
|
pub(crate) index: Box<Expression>,
|
||||||
pub(crate) result: Range<usize>,
|
|
||||||
pub(crate) param_list: Vec<Expression>,
|
pub(crate) param_list: Vec<Expression>,
|
||||||
|
pub(crate) result_list: ResultList,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl CallIndirect {
|
impl CallIndirect {
|
||||||
@ -989,25 +1009,25 @@ impl CallIndirect {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn result(&self) -> Range<usize> {
|
pub fn param_list(&self) -> &[Expression] {
|
||||||
self.result.clone()
|
&self.param_list
|
||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn param_list(&self) -> &[Expression] {
|
pub const fn result_list(&self) -> ResultList {
|
||||||
&self.param_list
|
self.result_list
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct SetTemporary {
|
pub struct SetTemporary {
|
||||||
pub(crate) var: usize,
|
pub(crate) var: Temporary,
|
||||||
pub(crate) value: Box<Expression>,
|
pub(crate) value: Box<Expression>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SetTemporary {
|
impl SetTemporary {
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub const fn var(&self) -> usize {
|
pub const fn var(&self) -> &Temporary {
|
||||||
self.var
|
&self.var
|
||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
@ -1017,14 +1037,14 @@ impl SetTemporary {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub struct SetLocal {
|
pub struct SetLocal {
|
||||||
pub(crate) var: usize,
|
pub(crate) var: Local,
|
||||||
pub(crate) value: Box<Expression>,
|
pub(crate) value: Box<Expression>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl SetLocal {
|
impl SetLocal {
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub const fn var(&self) -> usize {
|
pub const fn var(&self) -> &Local {
|
||||||
self.var
|
&self.var
|
||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
@ -1087,7 +1107,7 @@ impl StoreAt {
|
|||||||
|
|
||||||
pub struct MemoryGrow {
|
pub struct MemoryGrow {
|
||||||
pub(crate) memory: usize,
|
pub(crate) memory: usize,
|
||||||
pub(crate) result: usize,
|
pub(crate) result: Temporary,
|
||||||
pub(crate) size: Box<Expression>,
|
pub(crate) size: Box<Expression>,
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1098,8 +1118,8 @@ impl MemoryGrow {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub const fn result(&self) -> usize {
|
pub const fn result(&self) -> &Temporary {
|
||||||
self.result
|
&self.result
|
||||||
}
|
}
|
||||||
|
|
||||||
#[must_use]
|
#[must_use]
|
||||||
@ -1187,7 +1207,7 @@ pub enum Statement {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub struct FuncData {
|
pub struct FuncData {
|
||||||
pub(crate) local_data: Vec<(u32, ValType)>,
|
pub(crate) local_data: Vec<ValType>,
|
||||||
pub(crate) num_result: usize,
|
pub(crate) num_result: usize,
|
||||||
pub(crate) num_param: usize,
|
pub(crate) num_param: usize,
|
||||||
pub(crate) num_stack: usize,
|
pub(crate) num_stack: usize,
|
||||||
@ -1196,7 +1216,7 @@ pub struct FuncData {
|
|||||||
|
|
||||||
impl FuncData {
|
impl FuncData {
|
||||||
#[must_use]
|
#[must_use]
|
||||||
pub fn local_data(&self) -> &[(u32, ValType)] {
|
pub fn local_data(&self) -> &[ValType] {
|
||||||
&self.local_data
|
&self.local_data
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
use std::{collections::HashSet, ops::Range};
|
use std::collections::HashSet;
|
||||||
|
|
||||||
use crate::node::{
|
use crate::node::{
|
||||||
Align, Expression, GetGlobal, GetLocal, GetTemporary, LoadAt, SetTemporary, Statement,
|
Align, Expression, GetGlobal, LoadAt, Local, ResultList, SetTemporary, Statement, Temporary,
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(Clone, Copy, PartialEq, Eq, Hash)]
|
#[derive(Clone, Copy, PartialEq, Eq, Hash)]
|
||||||
@ -68,7 +68,7 @@ impl Stack {
|
|||||||
pub fn push_with_single(&mut self, data: Expression) {
|
pub fn push_with_single(&mut self, data: Expression) {
|
||||||
let mut read = HashSet::new();
|
let mut read = HashSet::new();
|
||||||
let elem = match data {
|
let elem = match data {
|
||||||
Expression::GetLocal(GetLocal { var }) => ReadType::Local(var),
|
Expression::GetLocal(Local { var }) => ReadType::Local(var),
|
||||||
Expression::GetGlobal(GetGlobal { var }) => ReadType::Global(var),
|
Expression::GetGlobal(GetGlobal { var }) => ReadType::Global(var),
|
||||||
Expression::LoadAt(LoadAt { memory, .. }) => ReadType::Memory(memory),
|
Expression::LoadAt(LoadAt { memory, .. }) => ReadType::Memory(memory),
|
||||||
_ => unreachable!(),
|
_ => unreachable!(),
|
||||||
@ -94,19 +94,23 @@ impl Stack {
|
|||||||
self.var_list.drain(desired..).map(|v| v.data)
|
self.var_list.drain(desired..).map(|v| v.data)
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn push_temporary(&mut self, num: usize) -> Range<usize> {
|
pub fn push_temporaries(&mut self, num: usize) -> ResultList {
|
||||||
let start = self.previous + self.len();
|
let start = self.previous + self.len();
|
||||||
let range = start..start + num;
|
let range = start..start + num;
|
||||||
|
|
||||||
self.capacity = self.capacity.max(range.end);
|
self.capacity = self.capacity.max(range.end);
|
||||||
|
|
||||||
for var in range.clone() {
|
for var in range.clone() {
|
||||||
let data = Expression::GetTemporary(GetTemporary { var });
|
let data = Expression::GetTemporary(Temporary { var });
|
||||||
|
|
||||||
self.push(data);
|
self.push(data);
|
||||||
}
|
}
|
||||||
|
|
||||||
range
|
ResultList::new(range.start, range.end)
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn push_temporary(&mut self) -> Temporary {
|
||||||
|
self.push_temporaries(1).iter().next().unwrap()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Return the alignment necessary for this block to branch out to a
|
// Return the alignment necessary for this block to branch out to a
|
||||||
@ -136,9 +140,9 @@ impl Stack {
|
|||||||
|
|
||||||
old.read.clear();
|
old.read.clear();
|
||||||
|
|
||||||
let get = Expression::GetTemporary(GetTemporary { var });
|
let get = Expression::GetTemporary(Temporary { var });
|
||||||
let set = Statement::SetTemporary(SetTemporary {
|
let set = Statement::SetTemporary(SetTemporary {
|
||||||
var,
|
var: Temporary { var },
|
||||||
value: std::mem::replace(&mut old.data, get).into(),
|
value: std::mem::replace(&mut old.data, get).into(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -1,15 +1,15 @@
|
|||||||
use crate::node::{
|
use crate::node::{
|
||||||
BinOp, Block, Br, BrIf, BrTable, Call, CallIndirect, CmpOp, Expression, FuncData, GetGlobal,
|
BinOp, Block, Br, BrIf, BrTable, Call, CallIndirect, CmpOp, Expression, FuncData, GetGlobal,
|
||||||
GetLocal, GetTemporary, If, LoadAt, MemoryCopy, MemoryFill, MemoryGrow, MemorySize, Select,
|
If, LoadAt, Local, MemoryCopy, MemoryFill, MemoryGrow, MemorySize, Select, SetGlobal, SetLocal,
|
||||||
SetGlobal, SetLocal, SetTemporary, Statement, StoreAt, Terminator, UnOp, Value,
|
SetTemporary, Statement, StoreAt, Temporary, Terminator, UnOp, Value,
|
||||||
};
|
};
|
||||||
|
|
||||||
pub trait Visitor {
|
pub trait Visitor {
|
||||||
fn visit_select(&mut self, _: &Select) {}
|
fn visit_select(&mut self, _: &Select) {}
|
||||||
|
|
||||||
fn visit_get_temporary(&mut self, _: &GetTemporary) {}
|
fn visit_get_temporary(&mut self, _: &Temporary) {}
|
||||||
|
|
||||||
fn visit_get_local(&mut self, _: &GetLocal) {}
|
fn visit_get_local(&mut self, _: &Local) {}
|
||||||
|
|
||||||
fn visit_get_global(&mut self, _: &GetGlobal) {}
|
fn visit_get_global(&mut self, _: &GetGlobal) {}
|
||||||
|
|
||||||
@ -76,13 +76,13 @@ impl<T: Visitor> Driver<T> for Select {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Visitor> Driver<T> for GetTemporary {
|
impl<T: Visitor> Driver<T> for Temporary {
|
||||||
fn accept(&self, visitor: &mut T) {
|
fn accept(&self, visitor: &mut T) {
|
||||||
visitor.visit_get_temporary(self);
|
visitor.visit_get_temporary(self);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<T: Visitor> Driver<T> for GetLocal {
|
impl<T: Visitor> Driver<T> for Local {
|
||||||
fn accept(&self, visitor: &mut T) {
|
fn accept(&self, visitor: &mut T) {
|
||||||
visitor.visit_get_local(self);
|
visitor.visit_get_local(self);
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user