277 lines
6.4 KiB
Rust
Executable File
277 lines
6.4 KiB
Rust
Executable File
use std::io::Result;
|
|
|
|
use parity_wasm::elements::{External, ImportCountType, Internal, ResizableLimits};
|
|
|
|
use crate::{
|
|
backend::{edition::data::Edition, helper::writer::Writer},
|
|
data::Module,
|
|
};
|
|
|
|
use super::level_2::{gen_function, gen_init_expression};
|
|
|
|
const RUNTIME_DATA: &str = "
|
|
local add = rt.add
|
|
local sub = rt.sub
|
|
local mul = rt.mul
|
|
local div = rt.div
|
|
|
|
local le = rt.le
|
|
local lt = rt.lt
|
|
local ge = rt.ge
|
|
local gt = rt.gt
|
|
|
|
local band = rt.band
|
|
local bor = rt.bor
|
|
local bxor = rt.bxor
|
|
local bnot = rt.bnot
|
|
|
|
local shl = rt.shl
|
|
local shr = rt.shr
|
|
|
|
local extend = rt.extend
|
|
local wrap = rt.wrap
|
|
|
|
local load = rt.load
|
|
local store = rt.store
|
|
";
|
|
|
|
fn gen_import_of<T>(m: &Module, w: Writer, lower: &str, cond: T) -> Result<()>
|
|
where
|
|
T: Fn(&External) -> bool,
|
|
{
|
|
let import = match m.parent.import_section() {
|
|
Some(v) => v.entries(),
|
|
None => return Ok(()),
|
|
};
|
|
let upper = lower.to_uppercase();
|
|
|
|
for (i, v) in import.iter().filter(|v| cond(v.external())).enumerate() {
|
|
let field = v.field();
|
|
let module = v.module();
|
|
|
|
write!(w, "{}[{}] = wasm.{}.{}.{} ", upper, i, module, lower, field)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn aux_internal_index(internal: Internal) -> u32 {
|
|
match internal {
|
|
Internal::Function(v) | Internal::Table(v) | Internal::Memory(v) | Internal::Global(v) => v,
|
|
}
|
|
}
|
|
|
|
fn gen_export_of<T>(m: &Module, w: Writer, lower: &str, cond: T) -> Result<()>
|
|
where
|
|
T: Fn(&Internal) -> bool,
|
|
{
|
|
let export = match m.parent.export_section() {
|
|
Some(v) => v.entries(),
|
|
None => return Ok(()),
|
|
};
|
|
let upper = lower.to_uppercase();
|
|
|
|
write!(w, "{} = {{", lower)?;
|
|
|
|
for v in export.iter().filter(|v| cond(v.internal())) {
|
|
let field = v.field();
|
|
let index = aux_internal_index(*v.internal());
|
|
|
|
write!(w, "{} = {}[{}],", field, upper, index)?;
|
|
}
|
|
|
|
write!(w, "}},")
|
|
}
|
|
|
|
fn gen_import_list(m: &Module, w: Writer) -> Result<()> {
|
|
gen_import_of(m, w, "func_list", |v| matches!(v, External::Function(_)))?;
|
|
gen_import_of(m, w, "table_list", |v| matches!(v, External::Table(_)))?;
|
|
gen_import_of(m, w, "memory_list", |v| matches!(v, External::Memory(_)))?;
|
|
gen_import_of(m, w, "global_list", |v| matches!(v, External::Global(_)))
|
|
}
|
|
|
|
fn gen_export_list(m: &Module, w: Writer) -> Result<()> {
|
|
gen_export_of(m, w, "func_list", |v| matches!(v, Internal::Function(_)))?;
|
|
gen_export_of(m, w, "table_list", |v| matches!(v, Internal::Table(_)))?;
|
|
gen_export_of(m, w, "memory_list", |v| matches!(v, Internal::Memory(_)))?;
|
|
gen_export_of(m, w, "global_list", |v| matches!(v, Internal::Global(_)))
|
|
}
|
|
|
|
fn gen_table_init(limit: &ResizableLimits, w: Writer) -> Result<()> {
|
|
write!(w, "{{ min = {}", limit.initial())?;
|
|
|
|
if let Some(max) = limit.maximum() {
|
|
write!(w, ", max = {}", max)?;
|
|
}
|
|
|
|
write!(w, ", data = {{}} }}")
|
|
}
|
|
|
|
fn gen_memory_init(limit: &ResizableLimits, w: Writer) -> Result<()> {
|
|
write!(w, "rt.memory.new({}, ", limit.initial())?;
|
|
|
|
if let Some(max) = limit.maximum() {
|
|
write!(w, "{}", max)?;
|
|
} else {
|
|
write!(w, "nil")?;
|
|
}
|
|
|
|
write!(w, ")")
|
|
}
|
|
|
|
fn gen_table_list(m: &Module, w: Writer) -> Result<()> {
|
|
let table = match m.parent.table_section() {
|
|
Some(v) => v.entries(),
|
|
None => return Ok(()),
|
|
};
|
|
let offset = m.parent.import_count(ImportCountType::Table);
|
|
|
|
for (i, v) in table.iter().enumerate() {
|
|
let index = i + offset;
|
|
|
|
write!(w, "TABLE_LIST[{}] =", index)?;
|
|
gen_table_init(v.limits(), w)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn gen_memory_list(m: &Module, w: Writer) -> Result<()> {
|
|
let memory = match m.parent.memory_section() {
|
|
Some(v) => v.entries(),
|
|
None => return Ok(()),
|
|
};
|
|
let offset = m.parent.import_count(ImportCountType::Memory);
|
|
|
|
for (i, v) in memory.iter().enumerate() {
|
|
let index = i + offset;
|
|
|
|
write!(w, "MEMORY_LIST[{}] =", index)?;
|
|
gen_memory_init(v.limits(), w)?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn gen_global_list(m: &Module, w: Writer) -> Result<()> {
|
|
let global = match m.parent.global_section() {
|
|
Some(v) => v,
|
|
None => return Ok(()),
|
|
};
|
|
let offset = m.parent.import_count(ImportCountType::Global);
|
|
|
|
for (i, v) in global.entries().iter().enumerate() {
|
|
let index = i + offset;
|
|
|
|
write!(w, "GLOBAL_LIST[{}] = {{ value =", index)?;
|
|
gen_init_expression(v.init_expr().code(), w)?;
|
|
write!(w, "}}")?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn gen_element_list(m: &Module, w: Writer) -> Result<()> {
|
|
let element = match m.parent.elements_section() {
|
|
Some(v) => v.entries(),
|
|
None => return Ok(()),
|
|
};
|
|
|
|
for v in element {
|
|
write!(w, "do ")?;
|
|
write!(w, "local target = TABLE_LIST[{}].data ", v.index())?;
|
|
write!(w, "local offset =")?;
|
|
|
|
gen_init_expression(v.offset().as_ref().unwrap().code(), w)?;
|
|
|
|
for (i, f) in v.members().iter().enumerate() {
|
|
write!(w, "target[offset + {}] = FUNC_LIST[{}]", i, f)?;
|
|
}
|
|
|
|
write!(w, "end ")?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn gen_data_list(m: &Module, w: Writer) -> Result<()> {
|
|
let data = match m.parent.data_section() {
|
|
Some(v) => v.entries(),
|
|
None => return Ok(()),
|
|
};
|
|
|
|
for v in data {
|
|
write!(w, "do ")?;
|
|
write!(w, "local target = MEMORY_LIST[{}]", v.index())?;
|
|
write!(w, "local offset =")?;
|
|
|
|
gen_init_expression(v.offset().as_ref().unwrap().code(), w)?;
|
|
|
|
write!(w, "local data = \"")?;
|
|
|
|
v.value()
|
|
.iter()
|
|
.try_for_each(|v| write!(w, "\\x{:02X}", v))?;
|
|
|
|
write!(w, "\"")?;
|
|
|
|
write!(w, "rt.memory.init(target, offset, data)")?;
|
|
|
|
write!(w, "end ")?;
|
|
}
|
|
|
|
Ok(())
|
|
}
|
|
|
|
fn gen_start_point(m: &Module, w: Writer) -> Result<()> {
|
|
write!(w, "local function run_init_code()")?;
|
|
gen_table_list(m, w)?;
|
|
gen_memory_list(m, w)?;
|
|
gen_global_list(m, w)?;
|
|
gen_element_list(m, w)?;
|
|
gen_data_list(m, w)?;
|
|
write!(w, "end ")?;
|
|
|
|
write!(w, "return function(wasm)")?;
|
|
gen_import_list(m, w)?;
|
|
write!(w, "run_init_code()")?;
|
|
|
|
if let Some(start) = m.parent.start_section() {
|
|
write!(w, "FUNC_LIST[{}]()", start)?;
|
|
}
|
|
|
|
write!(w, "return {{")?;
|
|
gen_export_list(m, w)?;
|
|
write!(w, "}} end ")
|
|
}
|
|
|
|
fn gen_nil_array(name: &str, len: usize, w: Writer) -> Result<()> {
|
|
if len == 0 {
|
|
return Ok(());
|
|
}
|
|
|
|
let list = vec!["nil"; len].join(", ");
|
|
|
|
write!(w, "local {} = {{[0] = {}}}", name, list)
|
|
}
|
|
|
|
pub fn translate(spec: &dyn Edition, m: &Module, w: Writer) -> Result<()> {
|
|
write!(w, "local rt = require({})", spec.runtime())?;
|
|
write!(w, "{}", RUNTIME_DATA)?;
|
|
|
|
gen_nil_array("FUNC_LIST", m.in_arity.len(), w)?;
|
|
gen_nil_array("TABLE_LIST", m.parent.table_space(), w)?;
|
|
gen_nil_array("MEMORY_LIST", m.parent.memory_space(), w)?;
|
|
gen_nil_array("GLOBAL_LIST", m.parent.globals_space(), w)?;
|
|
|
|
let offset = m.ex_arity.len();
|
|
|
|
for i in 0..m.in_arity.len() {
|
|
write!(w, "FUNC_LIST[{}] =", i + offset)?;
|
|
|
|
gen_function(spec, i, m, w)?;
|
|
}
|
|
|
|
gen_start_point(m, w)
|
|
}
|