mirror of
https://github.com/fluencelabs/wasm-utils
synced 2025-05-04 19:22:17 +00:00
various fixes
This commit is contained in:
parent
05d12be585
commit
beea9addfc
@ -1,38 +1,7 @@
|
|||||||
extern crate parity_wasm;
|
extern crate parity_wasm;
|
||||||
|
extern crate wasm_utils;
|
||||||
|
|
||||||
use std::env;
|
use std::env;
|
||||||
use parity_wasm::{builder, elements};
|
|
||||||
|
|
||||||
pub fn update_call_index(opcodes: &mut elements::Opcodes, inserted_index: u32) {
|
|
||||||
use parity_wasm::elements::Opcode::*;
|
|
||||||
for opcode in opcodes.elements_mut().iter_mut() {
|
|
||||||
match opcode {
|
|
||||||
&mut Block(_, ref mut block) | &mut If(_, ref mut block) | &mut Loop(_, ref mut block) => {
|
|
||||||
update_call_index(block, inserted_index)
|
|
||||||
},
|
|
||||||
&mut Call(ref mut call_index) => {
|
|
||||||
if *call_index >= inserted_index { *call_index += 1}
|
|
||||||
},
|
|
||||||
_ => { }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn inject_counter(opcodes: &mut elements::Opcodes, gas_func: u32) {
|
|
||||||
use parity_wasm::elements::Opcode::*;
|
|
||||||
for opcode in opcodes.elements_mut().iter_mut() {
|
|
||||||
match opcode {
|
|
||||||
&mut Block(_, ref mut block) | &mut If(_, ref mut block) | &mut Loop(_, ref mut block) => {
|
|
||||||
inject_counter(block, gas_func)
|
|
||||||
},
|
|
||||||
_ => { }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
let ops = opcodes.elements_mut().len() as u32;
|
|
||||||
opcodes.elements_mut().insert(0, I32Const(ops as i32));
|
|
||||||
opcodes.elements_mut().insert(1, Call(gas_func));
|
|
||||||
}
|
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
|
|
||||||
@ -43,69 +12,9 @@ fn main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Loading module
|
// Loading module
|
||||||
let module = parity_wasm::deserialize_file(&args[1]).unwrap();
|
let module = parity_wasm::deserialize_file(&args[1]).expect("Module deserialization to succeed");
|
||||||
|
|
||||||
// Injecting gas counting external
|
let result = wasm_utils::inject_gas_counter(module);
|
||||||
let mut mbuilder = builder::from_module(module);
|
|
||||||
let import_sig = mbuilder.push_signature(
|
|
||||||
builder::signature()
|
|
||||||
.param().i32()
|
|
||||||
.build_sig()
|
|
||||||
);
|
|
||||||
|
|
||||||
let mut gas_func = mbuilder.push_import(
|
parity_wasm::serialize_to_file(&args[2], result).expect("Module serialization to succeed")
|
||||||
builder::import()
|
|
||||||
.module("env")
|
|
||||||
.field("gas")
|
|
||||||
.external().func(import_sig)
|
|
||||||
.build()
|
|
||||||
);
|
|
||||||
|
|
||||||
// back to plain module
|
|
||||||
let mut module = mbuilder.build();
|
|
||||||
|
|
||||||
assert!(module.global_section().is_some());
|
|
||||||
|
|
||||||
// calculate actual function index of the imported definition
|
|
||||||
// (substract all imports that are NOT functions)
|
|
||||||
|
|
||||||
for import_entry in module.import_section().expect("Builder should have insert the import section").entries() {
|
|
||||||
match *import_entry.external() {
|
|
||||||
elements::External::Function(_) => {},
|
|
||||||
_ => { gas_func -= 1; }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Updating calling addresses (all calls to function index >= `gas_func` should be incremented)
|
|
||||||
for section in module.sections_mut() {
|
|
||||||
match section {
|
|
||||||
&mut elements::Section::Code(ref mut code_section) => {
|
|
||||||
for ref mut func_body in code_section.bodies_mut() {
|
|
||||||
update_call_index(func_body.code_mut(), gas_func);
|
|
||||||
inject_counter(func_body.code_mut(), gas_func);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
&mut elements::Section::Export(ref mut export_section) => {
|
|
||||||
for ref mut export in export_section.entries_mut() {
|
|
||||||
match export.internal_mut() {
|
|
||||||
&mut elements::Internal::Function(ref mut func_index) => {
|
|
||||||
if *func_index >= gas_func { *func_index += 1}
|
|
||||||
},
|
|
||||||
_ => {}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
&mut elements::Section::Element(ref mut elements_section) => {
|
|
||||||
for ref mut segment in elements_section.entries_mut() {
|
|
||||||
// update all indirect call addresses initial values
|
|
||||||
for func_index in segment.members_mut() {
|
|
||||||
if *func_index >= gas_func { *func_index += 1}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
_ => { }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
parity_wasm::serialize_to_file(&args[2], module).unwrap();
|
|
||||||
}
|
}
|
||||||
|
@ -21,7 +21,7 @@ then
|
|||||||
|
|
||||||
else
|
else
|
||||||
# c/c++ can be compiled directly by emcc
|
# c/c++ can be compiled directly by emcc
|
||||||
emcc $file -O3 -s WASM=1 -s SIDE_MODULE=1 -o out/contract.wasm -s
|
emcc $file -O3 -s WASM=1 -s SIDE_MODULE=1 -o out/contract.wasm
|
||||||
|
|
||||||
# Gas injector
|
# Gas injector
|
||||||
cargo run --manifest-path=./../gas/Cargo.toml --release -- ./out/contract.wasm ./out/contract.wasm
|
cargo run --manifest-path=./../gas/Cargo.toml --release -- ./out/contract.wasm ./out/contract.wasm
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
|
|
||||||
use std::slice;
|
use std::slice;
|
||||||
|
|
||||||
#[link_args = "-s WASM=1 -s NO_EXIT_RUNTIME=1 -s NO_FILESYSTEM=1"]
|
#[link_args = "-s NO_EXIT_RUNTIME=1 -s NO_FILESYSTEM=1"]
|
||||||
extern {}
|
extern {}
|
||||||
|
|
||||||
/// Wrapper over storage read/write/size externs
|
/// Wrapper over storage read/write/size externs
|
||||||
|
99
src/gas.rs
Normal file
99
src/gas.rs
Normal file
@ -0,0 +1,99 @@
|
|||||||
|
use parity_wasm::{elements, builder};
|
||||||
|
|
||||||
|
|
||||||
|
pub fn update_call_index(opcodes: &mut elements::Opcodes, inserted_index: u32) {
|
||||||
|
use parity_wasm::elements::Opcode::*;
|
||||||
|
for opcode in opcodes.elements_mut().iter_mut() {
|
||||||
|
match opcode {
|
||||||
|
&mut Block(_, ref mut block) | &mut If(_, ref mut block) | &mut Loop(_, ref mut block) => {
|
||||||
|
update_call_index(block, inserted_index)
|
||||||
|
},
|
||||||
|
&mut Call(ref mut call_index) => {
|
||||||
|
if *call_index >= inserted_index { *call_index += 1}
|
||||||
|
},
|
||||||
|
_ => { }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn inject_counter(opcodes: &mut elements::Opcodes, gas_func: u32) {
|
||||||
|
use parity_wasm::elements::Opcode::*;
|
||||||
|
for opcode in opcodes.elements_mut().iter_mut() {
|
||||||
|
match opcode {
|
||||||
|
&mut Block(_, ref mut block) | &mut If(_, ref mut block) | &mut Loop(_, ref mut block) => {
|
||||||
|
inject_counter(block, gas_func)
|
||||||
|
},
|
||||||
|
_ => { }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let ops = opcodes.elements_mut().len() as u32;
|
||||||
|
opcodes.elements_mut().insert(0, I32Const(ops as i32));
|
||||||
|
opcodes.elements_mut().insert(1, Call(gas_func));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn inject_gas_counter(module: elements::Module) -> elements::Module {
|
||||||
|
// Injecting gas counting external
|
||||||
|
let mut mbuilder = builder::from_module(module);
|
||||||
|
let import_sig = mbuilder.push_signature(
|
||||||
|
builder::signature()
|
||||||
|
.param().i32()
|
||||||
|
.build_sig()
|
||||||
|
);
|
||||||
|
|
||||||
|
let mut gas_func = mbuilder.push_import(
|
||||||
|
builder::import()
|
||||||
|
.module("env")
|
||||||
|
.field("gas")
|
||||||
|
.external().func(import_sig)
|
||||||
|
.build()
|
||||||
|
);
|
||||||
|
|
||||||
|
// back to plain module
|
||||||
|
let mut module = mbuilder.build();
|
||||||
|
|
||||||
|
assert!(module.global_section().is_some());
|
||||||
|
|
||||||
|
// calculate actual function index of the imported definition
|
||||||
|
// (substract all imports that are NOT functions)
|
||||||
|
|
||||||
|
for import_entry in module.import_section().expect("Builder should have insert the import section").entries() {
|
||||||
|
match *import_entry.external() {
|
||||||
|
elements::External::Function(_) => {},
|
||||||
|
_ => { gas_func -= 1; }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Updating calling addresses (all calls to function index >= `gas_func` should be incremented)
|
||||||
|
for section in module.sections_mut() {
|
||||||
|
match section {
|
||||||
|
&mut elements::Section::Code(ref mut code_section) => {
|
||||||
|
for ref mut func_body in code_section.bodies_mut() {
|
||||||
|
update_call_index(func_body.code_mut(), gas_func);
|
||||||
|
inject_counter(func_body.code_mut(), gas_func);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
&mut elements::Section::Export(ref mut export_section) => {
|
||||||
|
for ref mut export in export_section.entries_mut() {
|
||||||
|
match export.internal_mut() {
|
||||||
|
&mut elements::Internal::Function(ref mut func_index) => {
|
||||||
|
if *func_index >= gas_func { *func_index += 1}
|
||||||
|
},
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
&mut elements::Section::Element(ref mut elements_section) => {
|
||||||
|
for ref mut segment in elements_section.entries_mut() {
|
||||||
|
// update all indirect call addresses initial values
|
||||||
|
for func_index in segment.members_mut() {
|
||||||
|
if *func_index >= gas_func { *func_index += 1}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
_ => { }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
module
|
||||||
|
}
|
@ -1,6 +1,8 @@
|
|||||||
extern crate parity_wasm;
|
extern crate parity_wasm;
|
||||||
|
|
||||||
mod optimizer;
|
mod optimizer;
|
||||||
pub mod symbols;
|
mod gas;
|
||||||
|
mod symbols;
|
||||||
|
|
||||||
pub use optimizer::optimize;
|
pub use optimizer::optimize;
|
||||||
|
pub use gas::inject_gas_counter;
|
Loading…
x
Reference in New Issue
Block a user