mirror of
https://github.com/fluencelabs/wasm-utils
synced 2025-04-25 23:02:21 +00:00
907 lines
24 KiB
Rust
907 lines
24 KiB
Rust
//! Wasm binary graph format
|
|
|
|
#![warn(missing_docs)]
|
|
|
|
use parity_wasm::elements;
|
|
use super::ref_list::{RefList, EntryRef};
|
|
use std::vec::Vec;
|
|
use std::borrow::ToOwned;
|
|
use std::string::String;
|
|
use std::collections::BTreeMap;
|
|
|
|
/// Imported or declared variant of the same thing.
|
|
///
|
|
/// In WebAssembly, function/global/memory/table instances can either be
|
|
/// imported or declared internally, forming united index space.
|
|
#[derive(Debug)]
|
|
pub enum ImportedOrDeclared<T=()> {
|
|
/// Variant for imported instances.
|
|
Imported(String, String),
|
|
/// Variant for instances declared internally in the module.
|
|
Declared(T),
|
|
}
|
|
|
|
impl<T> From<&elements::ImportEntry> for ImportedOrDeclared<T> {
|
|
fn from(v: &elements::ImportEntry) -> Self {
|
|
ImportedOrDeclared::Imported(v.module().to_owned(), v.field().to_owned())
|
|
}
|
|
}
|
|
|
|
/// Function origin (imported or internal).
|
|
pub type FuncOrigin = ImportedOrDeclared<FuncBody>;
|
|
/// Global origin (imported or internal).
|
|
pub type GlobalOrigin = ImportedOrDeclared<Vec<Instruction>>;
|
|
/// Memory origin (imported or internal).
|
|
pub type MemoryOrigin = ImportedOrDeclared;
|
|
/// Table origin (imported or internal).
|
|
pub type TableOrigin = ImportedOrDeclared;
|
|
|
|
/// Function body.
|
|
///
|
|
/// Function consist of declaration (signature, i.e. type reference)
|
|
/// and the actual code. This part is the actual code.
|
|
#[derive(Debug)]
|
|
pub struct FuncBody {
|
|
pub locals: Vec<elements::Local>,
|
|
pub code: Vec<Instruction>,
|
|
}
|
|
|
|
/// Function declaration.
|
|
///
|
|
/// As with other instances, functions can be either imported or declared
|
|
/// within the module - `origin` field is handling this.
|
|
#[derive(Debug)]
|
|
pub struct Func {
|
|
/// Function signature/type reference.
|
|
pub type_ref: EntryRef<elements::Type>,
|
|
/// Where this function comes from (imported or declared).
|
|
pub origin: FuncOrigin,
|
|
}
|
|
|
|
/// Global declaration.
|
|
///
|
|
/// As with other instances, globals can be either imported or declared
|
|
/// within the module - `origin` field is handling this.
|
|
#[derive(Debug)]
|
|
pub struct Global {
|
|
pub content: elements::ValueType,
|
|
pub is_mut: bool,
|
|
pub origin: GlobalOrigin,
|
|
}
|
|
|
|
/// Instruction.
|
|
///
|
|
/// Some instructions don't reference any entities within the WebAssembly module,
|
|
/// while others do. This enum is for tracking references when required.
|
|
#[derive(Debug)]
|
|
pub enum Instruction {
|
|
/// WebAssembly instruction that does not reference any module entities.
|
|
Plain(elements::Instruction),
|
|
/// Call instruction which references the function.
|
|
Call(EntryRef<Func>),
|
|
/// Indirect call instruction which references function type (function signature).
|
|
CallIndirect(EntryRef<elements::Type>, u8),
|
|
/// get_global instruction which references the global.
|
|
GetGlobal(EntryRef<Global>),
|
|
/// set_global instruction which references the global.
|
|
SetGlobal(EntryRef<Global>),
|
|
}
|
|
|
|
/// Memory instance decriptor.
|
|
///
|
|
/// As with other similar instances, memory instances can be either imported
|
|
/// or declared within the module - `origin` field is handling this.
|
|
#[derive(Debug)]
|
|
pub struct Memory {
|
|
/// Declared limits of the table instance.
|
|
pub limits: elements::ResizableLimits,
|
|
/// Origin of the memory instance (internal or imported).
|
|
pub origin: MemoryOrigin,
|
|
}
|
|
|
|
/// Memory instance decriptor.
|
|
///
|
|
/// As with other similar instances, memory instances can be either imported
|
|
/// or declared within the module - `origin` field is handling this.
|
|
#[derive(Debug)]
|
|
pub struct Table {
|
|
/// Declared limits of the table instance.
|
|
pub limits: elements::ResizableLimits,
|
|
/// Origin of the table instance (internal or imported).
|
|
pub origin: TableOrigin,
|
|
}
|
|
|
|
/// Segment location.
|
|
///
|
|
/// Reserved for future use. Currenty only `Default` variant is supported.
|
|
#[derive(Debug)]
|
|
pub enum SegmentLocation {
|
|
/// Not used currently.
|
|
Passive,
|
|
/// Default segment location with index `0`.
|
|
Default(Vec<Instruction>),
|
|
/// Not used currently.
|
|
WithIndex(u32, Vec<Instruction>),
|
|
}
|
|
|
|
/// Data segment of data section.
|
|
#[derive(Debug)]
|
|
pub struct DataSegment {
|
|
/// Location of the segment in the linear memory.
|
|
pub location: SegmentLocation,
|
|
/// Raw value of the data segment.
|
|
pub value: Vec<u8>,
|
|
}
|
|
|
|
/// Element segment of element section.
|
|
#[derive(Debug)]
|
|
pub struct ElementSegment {
|
|
/// Location of the segment in the table space.
|
|
pub location: SegmentLocation,
|
|
/// Raw value (function indices) of the element segment.
|
|
pub value: Vec<EntryRef<Func>>,
|
|
}
|
|
|
|
/// Export entry reference.
|
|
///
|
|
/// Module can export function, global, table or memory instance
|
|
/// under specific name (field).
|
|
#[derive(Debug)]
|
|
pub enum ExportLocal {
|
|
/// Function reference.
|
|
Func(EntryRef<Func>),
|
|
/// Global reference.
|
|
Global(EntryRef<Global>),
|
|
/// Table reference.
|
|
Table(EntryRef<Table>),
|
|
/// Memory reference.
|
|
Memory(EntryRef<Memory>),
|
|
}
|
|
|
|
/// Export entry description.
|
|
#[derive(Debug)]
|
|
pub struct Export {
|
|
/// Name (field) of the export entry.
|
|
pub name: String,
|
|
/// What entity is exported.
|
|
pub local: ExportLocal,
|
|
}
|
|
|
|
/// Module
|
|
#[derive(Debug, Default)]
|
|
pub struct Module {
|
|
/// Refence-tracking list of types.
|
|
pub types: RefList<elements::Type>,
|
|
/// Refence-tracking list of funcs.
|
|
pub funcs: RefList<Func>,
|
|
/// Refence-tracking list of memory instances.
|
|
pub memory: RefList<Memory>,
|
|
/// Refence-tracking list of table instances.
|
|
pub tables: RefList<Table>,
|
|
/// Refence-tracking list of globals.
|
|
pub globals: RefList<Global>,
|
|
/// Reference to start function.
|
|
pub start: Option<EntryRef<Func>>,
|
|
/// References to exported objects.
|
|
pub exports: Vec<Export>,
|
|
/// List of element segments.
|
|
pub elements: Vec<ElementSegment>,
|
|
/// List of data segments.
|
|
pub data: Vec<DataSegment>,
|
|
/// Other module functions that are not decoded or processed.
|
|
pub other: BTreeMap<usize, elements::Section>,
|
|
}
|
|
|
|
impl Module {
|
|
|
|
fn map_instructions(&self, instructions: &[elements::Instruction]) -> Vec<Instruction> {
|
|
use parity_wasm::elements::Instruction::*;
|
|
instructions.iter().map(|instruction| match instruction {
|
|
Call(func_idx) => Instruction::Call(self.funcs.clone_ref(*func_idx as usize)),
|
|
CallIndirect(type_idx, arg2) =>
|
|
Instruction::CallIndirect(
|
|
self.types.clone_ref(*type_idx as usize),
|
|
*arg2,
|
|
),
|
|
SetGlobal(global_idx) =>
|
|
Instruction::SetGlobal(self.globals.clone_ref(*global_idx as usize)),
|
|
GetGlobal(global_idx) =>
|
|
Instruction::GetGlobal(self.globals.clone_ref(*global_idx as usize)),
|
|
other_instruction => Instruction::Plain(other_instruction.clone()),
|
|
}).collect()
|
|
}
|
|
|
|
fn generate_instructions(&self, instructions: &[Instruction]) -> Vec<elements::Instruction> {
|
|
use parity_wasm::elements::Instruction::*;
|
|
instructions.iter().map(|instruction| match instruction {
|
|
Instruction::Call(func_ref) => Call(func_ref.order().expect("detached instruction!") as u32),
|
|
Instruction::CallIndirect(type_ref, arg2) => CallIndirect(type_ref.order().expect("detached instruction!") as u32, *arg2),
|
|
Instruction::SetGlobal(global_ref) => SetGlobal(global_ref.order().expect("detached instruction!") as u32),
|
|
Instruction::GetGlobal(global_ref) => GetGlobal(global_ref.order().expect("detached instruction!") as u32),
|
|
Instruction::Plain(plain) => plain.clone(),
|
|
}).collect()
|
|
}
|
|
|
|
/// Initialize module from parity-wasm `Module`.
|
|
pub fn from_elements(module: &elements::Module) -> Self {
|
|
|
|
let mut idx = 0;
|
|
let mut res = Module::default();
|
|
|
|
let mut imported_functions = 0;
|
|
|
|
for section in module.sections() {
|
|
match section {
|
|
elements::Section::Type(type_section) => {
|
|
res.types = RefList::from_slice(type_section.types());
|
|
},
|
|
elements::Section::Import(import_section) => {
|
|
for entry in import_section.entries() {
|
|
match *entry.external() {
|
|
elements::External::Function(f) => {
|
|
res.funcs.push(Func {
|
|
type_ref: res.types.get(f as usize).expect("validated; qed").clone(),
|
|
origin: entry.into(),
|
|
});
|
|
imported_functions += 1;
|
|
},
|
|
elements::External::Memory(m) => {
|
|
res.memory.push(Memory {
|
|
limits: m.limits().clone(),
|
|
origin: entry.into(),
|
|
});
|
|
},
|
|
elements::External::Global(g) => {
|
|
res.globals.push(Global {
|
|
content: g.content_type(),
|
|
is_mut: g.is_mutable(),
|
|
origin: entry.into(),
|
|
});
|
|
},
|
|
elements::External::Table(t) => {
|
|
res.tables.push(Table {
|
|
limits: t.limits().clone(),
|
|
origin: entry.into(),
|
|
});
|
|
},
|
|
};
|
|
}
|
|
},
|
|
elements::Section::Function(function_section) => {
|
|
for f in function_section.entries() {
|
|
res.funcs.push(Func {
|
|
type_ref: res.types.get(f.type_ref() as usize).expect("validated; qed").clone(),
|
|
origin: ImportedOrDeclared::Declared(FuncBody {
|
|
locals: Vec::new(),
|
|
// code will be populated later
|
|
code: Vec::new(),
|
|
}),
|
|
});
|
|
};
|
|
},
|
|
elements::Section::Table(table_section) => {
|
|
for t in table_section.entries() {
|
|
res.tables.push(Table {
|
|
limits: t.limits().clone(),
|
|
origin: ImportedOrDeclared::Declared(()),
|
|
});
|
|
}
|
|
},
|
|
elements::Section::Memory(table_section) => {
|
|
for t in table_section.entries() {
|
|
res.memory.push(Memory {
|
|
limits: t.limits().clone(),
|
|
origin: ImportedOrDeclared::Declared(()),
|
|
});
|
|
}
|
|
},
|
|
elements::Section::Global(global_section) => {
|
|
for g in global_section.entries() {
|
|
let init_code = res.map_instructions(g.init_expr().code());
|
|
res.globals.push(Global {
|
|
content: g.global_type().content_type(),
|
|
is_mut: g.global_type().is_mutable(),
|
|
origin: ImportedOrDeclared::Declared(init_code),
|
|
});
|
|
}
|
|
},
|
|
elements::Section::Export(export_section) => {
|
|
for e in export_section.entries() {
|
|
let local = match e.internal() {
|
|
&elements::Internal::Function(func_idx) => {
|
|
ExportLocal::Func(res.funcs.clone_ref(func_idx as usize))
|
|
},
|
|
&elements::Internal::Global(global_idx) => {
|
|
ExportLocal::Global(res.globals.clone_ref(global_idx as usize))
|
|
},
|
|
&elements::Internal::Memory(mem_idx) => {
|
|
ExportLocal::Memory(res.memory.clone_ref(mem_idx as usize))
|
|
},
|
|
&elements::Internal::Table(table_idx) => {
|
|
ExportLocal::Table(res.tables.clone_ref(table_idx as usize))
|
|
},
|
|
};
|
|
|
|
res.exports.push(Export { local: local, name: e.field().to_owned() })
|
|
}
|
|
},
|
|
elements::Section::Start(start_func) => {
|
|
res.start = Some(res.funcs.clone_ref(*start_func as usize));
|
|
},
|
|
elements::Section::Element(element_section) => {
|
|
for element_segment in element_section.entries() {
|
|
|
|
// let location = if element_segment.passive() {
|
|
// SegmentLocation::Passive
|
|
// } else if element_segment.index() == 0 {
|
|
// SegmentLocation::Default(Vec::new())
|
|
// } else {
|
|
// SegmentLocation::WithIndex(element_segment.index(), Vec::new())
|
|
// };
|
|
|
|
// TODO: update parity-wasm and uncomment the above instead
|
|
let location = SegmentLocation::Default(
|
|
res.map_instructions(element_segment.offset().code())
|
|
);
|
|
|
|
let funcs_map = element_segment
|
|
.members().iter()
|
|
.map(|idx| res.funcs.clone_ref(*idx as usize))
|
|
.collect::<Vec<EntryRef<Func>>>();
|
|
|
|
res.elements.push(ElementSegment {
|
|
value: funcs_map,
|
|
location: location,
|
|
});
|
|
}
|
|
},
|
|
elements::Section::Code(code_section) => {
|
|
let mut idx = 0;
|
|
for func_body in code_section.bodies() {
|
|
let code = res.map_instructions(func_body.code().elements());
|
|
|
|
let mut func = res.funcs.get_ref(imported_functions + idx).write();
|
|
match func.origin {
|
|
ImportedOrDeclared::Declared(ref mut body) => {
|
|
body.code = code;
|
|
body.locals = func_body.locals().iter().cloned().collect();
|
|
},
|
|
_ => unreachable!("All declared functions added after imported; qed"),
|
|
}
|
|
|
|
idx += 1;
|
|
}
|
|
},
|
|
elements::Section::Data(data_section) => {
|
|
for data_segment in data_section.entries() {
|
|
// TODO: update parity-wasm and use the same logic as in
|
|
// commented element segment branch
|
|
let location = SegmentLocation::Default(
|
|
res.map_instructions(data_segment.offset().code())
|
|
);
|
|
|
|
res.data.push(DataSegment {
|
|
value: data_segment.value().to_vec(),
|
|
location: location,
|
|
});
|
|
}
|
|
},
|
|
_ => {
|
|
res.other.insert(idx, section.clone());
|
|
}
|
|
}
|
|
idx += 1;
|
|
}
|
|
|
|
res
|
|
}
|
|
|
|
fn generate(&self) -> elements::Module {
|
|
use self::ImportedOrDeclared::*;
|
|
|
|
let mut idx = 0;
|
|
let mut sections = Vec::new();
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
|
|
if self.types.len() > 0 {
|
|
// TYPE SECTION (1)
|
|
let mut type_section = elements::TypeSection::default();
|
|
{
|
|
let types = type_section.types_mut();
|
|
|
|
for type_entry in self.types.iter() {
|
|
types.push(type_entry.read().clone())
|
|
}
|
|
}
|
|
sections.push(elements::Section::Type(type_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
// IMPORT SECTION (2)
|
|
let mut import_section = elements::ImportSection::default();
|
|
|
|
let add = {
|
|
let imports = import_section.entries_mut();
|
|
for func in self.funcs.iter() {
|
|
match func.read().origin {
|
|
Imported(ref module, ref field) => {
|
|
imports.push(
|
|
elements::ImportEntry::new(
|
|
module.to_owned(),
|
|
field.to_owned(),
|
|
elements::External::Function(
|
|
func.read().type_ref.order()
|
|
.expect("detached func encountered somehow!") as u32
|
|
),
|
|
)
|
|
)
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
|
|
for global in self.globals.iter() {
|
|
match global.read().origin {
|
|
Imported(ref module, ref field) => {
|
|
imports.push(
|
|
elements::ImportEntry::new(
|
|
module.to_owned(),
|
|
field.to_owned(),
|
|
elements::External::Global(
|
|
elements::GlobalType::new(
|
|
global.read().content,
|
|
global.read().is_mut,
|
|
)
|
|
),
|
|
)
|
|
)
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
|
|
for memory in self.memory.iter() {
|
|
match memory.read().origin {
|
|
Imported(ref module, ref field) => {
|
|
imports.push(
|
|
elements::ImportEntry::new(
|
|
module.to_owned(),
|
|
field.to_owned(),
|
|
elements::External::Memory(
|
|
elements::MemoryType::new(
|
|
memory.read().limits.initial(),
|
|
memory.read().limits.maximum(),
|
|
)
|
|
),
|
|
)
|
|
)
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
|
|
for table in self.tables.iter() {
|
|
match table.read().origin {
|
|
Imported(ref module, ref field) => {
|
|
imports.push(
|
|
elements::ImportEntry::new(
|
|
module.to_owned(),
|
|
field.to_owned(),
|
|
elements::External::Table(
|
|
elements::TableType::new(
|
|
table.read().limits.initial(),
|
|
table.read().limits.maximum(),
|
|
)
|
|
),
|
|
)
|
|
)
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
imports.len() > 0
|
|
};
|
|
|
|
if add {
|
|
sections.push(elements::Section::Import(import_section));
|
|
idx += 1;
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
if self.funcs.len() > 0 {
|
|
// FUNC SECTION (3)
|
|
let mut func_section = elements::FunctionSection::default();
|
|
{
|
|
let funcs = func_section.entries_mut();
|
|
|
|
for func in self.funcs.iter() {
|
|
match func.read().origin {
|
|
Declared(_) => {
|
|
funcs.push(elements::Func::new(
|
|
func.read().type_ref.order()
|
|
.expect("detached func encountered somehow!") as u32
|
|
));
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
}
|
|
sections.push(elements::Section::Function(func_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
if self.tables.len() > 0 {
|
|
// TABLE SECTION (4)
|
|
let mut table_section = elements::TableSection::default();
|
|
{
|
|
let tables = table_section.entries_mut();
|
|
|
|
for table in self.tables.iter() {
|
|
match table.read().origin {
|
|
Declared(_) => {
|
|
tables.push(elements::TableType::new(
|
|
table.read().limits.initial(),
|
|
table.read().limits.maximum(),
|
|
));
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
}
|
|
sections.push(elements::Section::Table(table_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
if self.memory.len() > 0 {
|
|
// MEMORY SECTION (5)
|
|
let mut memory_section = elements::MemorySection::default();
|
|
{
|
|
let memories = memory_section.entries_mut();
|
|
|
|
for memory in self.memory.iter() {
|
|
match memory.read().origin {
|
|
Declared(_) => {
|
|
memories.push(elements::MemoryType::new(
|
|
memory.read().limits.initial(),
|
|
memory.read().limits.maximum(),
|
|
));
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
}
|
|
sections.push(elements::Section::Memory(memory_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
if self.globals.len() > 0 {
|
|
// GLOBAL SECTION (6)
|
|
let mut global_section = elements::GlobalSection::default();
|
|
{
|
|
let globals = global_section.entries_mut();
|
|
|
|
for global in self.globals.iter() {
|
|
match global.read().origin {
|
|
Declared(ref init_code) => {
|
|
globals.push(elements::GlobalEntry::new(
|
|
elements::GlobalType::new(global.read().content, global.read().is_mut),
|
|
elements::InitExpr::new(self.generate_instructions(&init_code[..])),
|
|
));
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
}
|
|
sections.push(elements::Section::Global(global_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
if self.exports.len() > 0 {
|
|
// EXPORT SECTION (7)
|
|
let mut export_section = elements::ExportSection::default();
|
|
{
|
|
let exports = export_section.entries_mut();
|
|
|
|
for export in self.exports.iter() {
|
|
let internal = match export.local {
|
|
ExportLocal::Func(ref func_ref) => {
|
|
elements::Internal::Function(func_ref.order().expect("detached func ref") as u32)
|
|
},
|
|
ExportLocal::Global(ref global_ref) => {
|
|
elements::Internal::Global(global_ref.order().expect("detached global ref") as u32)
|
|
},
|
|
ExportLocal::Table(ref table_ref) => {
|
|
elements::Internal::Table(table_ref.order().expect("detached table ref") as u32)
|
|
},
|
|
ExportLocal::Memory(ref memory_ref) => {
|
|
elements::Internal::Memory(memory_ref.order().expect("detached memory ref") as u32)
|
|
},
|
|
};
|
|
|
|
exports.push(elements::ExportEntry::new(export.name.to_owned(), internal));
|
|
}
|
|
}
|
|
sections.push(elements::Section::Export(export_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
if let Some(ref func_ref) = self.start {
|
|
// START SECTION (8)
|
|
sections.push(elements::Section::Start(
|
|
func_ref.order().expect("detached start func") as u32
|
|
));
|
|
}
|
|
|
|
if self.elements.len() > 0 {
|
|
// START SECTION (9)
|
|
let mut element_section = elements::ElementSection::default();
|
|
{
|
|
let element_segments = element_section.entries_mut();
|
|
|
|
for element in self.elements.iter() {
|
|
match element.location {
|
|
SegmentLocation::Default(ref offset_expr) => {
|
|
let elements_map = element.value.iter()
|
|
.map(|f| f.order().expect("Detached func in element segment!") as u32)
|
|
.collect();
|
|
|
|
element_segments.push(
|
|
elements::ElementSegment::new(
|
|
0,
|
|
elements::InitExpr::new(self.generate_instructions(&offset_expr[..])),
|
|
elements_map,
|
|
)
|
|
);
|
|
},
|
|
_ => unreachable!("Other segment location types are never added"),
|
|
}
|
|
}
|
|
}
|
|
|
|
sections.push(elements::Section::Element(element_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
if self.funcs.len() > 0 {
|
|
// CODE SECTION (10)
|
|
let mut code_section = elements::CodeSection::default();
|
|
{
|
|
let funcs = code_section.bodies_mut();
|
|
|
|
for func in self.funcs.iter() {
|
|
match func.read().origin {
|
|
Declared(ref body) => {
|
|
funcs.push(elements::FuncBody::new(
|
|
body.locals.clone(),
|
|
elements::Instructions::new(self.generate_instructions(&body.code[..])),
|
|
));
|
|
},
|
|
_ => continue,
|
|
}
|
|
}
|
|
}
|
|
sections.push(elements::Section::Code(code_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
|
|
if self.data.len() > 0 {
|
|
// DATA SECTION (11)
|
|
let mut data_section = elements::DataSection::default();
|
|
{
|
|
let data_segments = data_section.entries_mut();
|
|
|
|
for data_entry in self.data.iter() {
|
|
match data_entry.location {
|
|
SegmentLocation::Default(ref offset_expr) => {
|
|
data_segments.push(
|
|
elements::DataSegment::new(
|
|
0,
|
|
elements::InitExpr::new(self.generate_instructions(&offset_expr[..])),
|
|
data_entry.value.clone(),
|
|
)
|
|
);
|
|
},
|
|
_ => unreachable!("Other segment location types are never added"),
|
|
}
|
|
}
|
|
}
|
|
|
|
sections.push(elements::Section::Data(data_section));
|
|
idx += 1;
|
|
|
|
custom_round(&self.other, &mut idx, &mut sections);
|
|
}
|
|
|
|
elements::Module::new(sections)
|
|
}
|
|
}
|
|
|
|
fn custom_round(
|
|
map: &BTreeMap<usize, elements::Section>,
|
|
idx: &mut usize,
|
|
sections: &mut Vec<elements::Section>,
|
|
) {
|
|
while let Some(other_section) = map.get(&idx) {
|
|
sections.push(other_section.clone());
|
|
*idx += 1;
|
|
}
|
|
}
|
|
|
|
/// New module from parity-wasm `Module`
|
|
pub fn parse(wasm: &[u8]) -> Module {
|
|
Module::from_elements(&::parity_wasm::deserialize_buffer(wasm).expect("failed to parse wasm"))
|
|
}
|
|
|
|
/// Generate parity-wasm `Module`
|
|
pub fn generate(f: &Module) -> Vec<u8> {
|
|
let pm = f.generate();
|
|
::parity_wasm::serialize(pm).expect("failed to generate wasm")
|
|
}
|
|
|
|
#[cfg(test)]
|
|
mod tests {
|
|
|
|
extern crate wabt;
|
|
|
|
use parity_wasm::elements;
|
|
|
|
fn load_sample(wat: &'static str) -> super::Module {
|
|
super::parse(&wabt::wat2wasm(wat).expect("faled to parse wat!")[..])
|
|
}
|
|
|
|
fn validate_sample(module: &super::Module) {
|
|
let binary = super::generate(module);
|
|
wabt::Module::read_binary(&binary, &Default::default())
|
|
.expect("Wabt failed to read final binary")
|
|
.validate()
|
|
.expect("Invalid module");
|
|
}
|
|
|
|
#[test]
|
|
fn smoky() {
|
|
let sample = load_sample(r#"
|
|
(module
|
|
(type (func))
|
|
(func (type 0))
|
|
(memory 0 1)
|
|
(export "simple" (func 0))
|
|
)
|
|
"#
|
|
);
|
|
|
|
assert_eq!(sample.types.len(), 1);
|
|
assert_eq!(sample.funcs.len(), 1);
|
|
assert_eq!(sample.tables.len(), 0);
|
|
assert_eq!(sample.memory.len(), 1);
|
|
assert_eq!(sample.exports.len(), 1);
|
|
|
|
assert_eq!(sample.types.get_ref(0).link_count(), 1);
|
|
assert_eq!(sample.funcs.get_ref(0).link_count(), 1);
|
|
}
|
|
|
|
#[test]
|
|
fn table() {
|
|
let mut sample = load_sample(r#"
|
|
(module
|
|
(import "env" "foo" (func $foo))
|
|
(func (param i32)
|
|
get_local 0
|
|
i32.const 0
|
|
call $i32.add
|
|
drop
|
|
)
|
|
(func $i32.add (export "i32.add") (param i32 i32) (result i32)
|
|
get_local 0
|
|
get_local 1
|
|
i32.add
|
|
)
|
|
(table 10 anyfunc)
|
|
|
|
;; Refer all types of functions: imported, defined not exported and defined exported.
|
|
(elem (i32.const 0) 0 1 2)
|
|
)"#
|
|
);
|
|
|
|
{
|
|
let element_func = &sample.elements[0].value[1];
|
|
let rfunc = element_func.read();
|
|
let rtype = &**rfunc.type_ref.read();
|
|
let elements::Type::Function(ref ftype) = rtype;
|
|
|
|
// it's func#1 in the function space
|
|
assert_eq!(rfunc.order(), Some(1));
|
|
// it's type#1
|
|
assert_eq!(ftype.params().len(), 1);
|
|
}
|
|
|
|
sample.funcs.begin_delete().push(0).done();
|
|
|
|
{
|
|
let element_func = &sample.elements[0].value[1];
|
|
let rfunc = element_func.read();
|
|
let rtype = &**rfunc.type_ref.read();
|
|
let elements::Type::Function(ref ftype) = rtype;
|
|
|
|
/// import deleted so now it's func #0
|
|
assert_eq!(rfunc.order(), Some(0));
|
|
/// type should be the same, #1
|
|
assert_eq!(ftype.params().len(), 1);
|
|
}
|
|
}
|
|
|
|
#[test]
|
|
fn new_import() {
|
|
let mut sample = load_sample(r#"
|
|
(module
|
|
(type (;0;) (func))
|
|
(type (;1;) (func (param i32 i32) (result i32)))
|
|
(import "env" "foo" (func (type 1)))
|
|
(func (param i32)
|
|
get_local 0
|
|
i32.const 0
|
|
call 0
|
|
drop
|
|
)
|
|
(func (type 0)
|
|
i32.const 0
|
|
call 1
|
|
)
|
|
)"#
|
|
);
|
|
|
|
{
|
|
let type_ref_0 = sample.types.clone_ref(0);
|
|
let declared_func_2 = sample.funcs.clone_ref(2);
|
|
|
|
let mut tx = sample.funcs.begin_insert_not_until(
|
|
|f| match f.origin {
|
|
super::ImportedOrDeclared::Imported(_, _) => true,
|
|
_ => false,
|
|
}
|
|
);
|
|
|
|
let new_import_func = tx.push(super::Func {
|
|
type_ref: type_ref_0,
|
|
origin: super::ImportedOrDeclared::Imported("env".to_owned(), "bar".to_owned()),
|
|
});
|
|
|
|
tx.done();
|
|
|
|
assert_eq!(new_import_func.order(), Some(1));
|
|
assert_eq!(declared_func_2.order(), Some(3));
|
|
assert_eq!(
|
|
match &declared_func_2.read().origin {
|
|
super::ImportedOrDeclared::Declared(ref body) => {
|
|
match body.code[1] {
|
|
super::Instruction::Call(ref called_func) => called_func.order(),
|
|
_ => panic!("instruction #2 should be a call!"),
|
|
}
|
|
},
|
|
_ => panic!("func #4 should be declared!"),
|
|
},
|
|
Some(2),
|
|
"Call should be recalculated to 2"
|
|
);
|
|
}
|
|
|
|
validate_sample(&sample);
|
|
}
|
|
} |