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(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(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) }