Merge pull request #13 from paritytech/cost-table

Cost table + some fixes
This commit is contained in:
Nikolay Volf
2017-07-27 13:39:26 +03:00
committed by GitHub
4 changed files with 294 additions and 18 deletions

View File

@ -8,7 +8,7 @@ pub fn update_call_index(opcodes: &mut elements::Opcodes, original_imports: usiz
match opcode { match opcode {
&mut Call(ref mut call_index) => { &mut Call(ref mut call_index) => {
if let Some(pos) = inserts.iter().position(|x| x.1 == *call_index) { if let Some(pos) = inserts.iter().position(|x| x.1 == *call_index) {
*call_index = (original_imports + pos) as u32; *call_index = (original_imports + pos) as u32;
} else if *call_index as usize > original_imports { } else if *call_index as usize > original_imports {
*call_index += inserts.len() as u32; *call_index += inserts.len() as u32;
} }
@ -86,9 +86,17 @@ pub fn externalize(
if *func_index >= import_funcs_total as u32 { *func_index += replaces.len() as u32; } if *func_index >= import_funcs_total as u32 { *func_index += replaces.len() as u32; }
}, },
_ => {} _ => {}
} }
} }
}, },
&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 >= import_funcs_total as u32 { *func_index += replaces.len() as u32; }
}
}
},
_ => { } _ => { }
} }
} }

View File

@ -22,7 +22,7 @@ enum InjectAction {
pub fn inject_counter(opcodes: &mut elements::Opcodes, gas_func: u32) { pub fn inject_counter(opcodes: &mut elements::Opcodes, gas_func: u32) {
use parity_wasm::elements::Opcode::*; use parity_wasm::elements::Opcode::*;
let mut stack: Vec<(usize, usize)> = Vec::new(); let mut stack: Vec<(usize, usize)> = Vec::new();
let mut cursor = 0; let mut cursor = 0;
stack.push((0, 1)); stack.push((0, 1));
@ -127,7 +127,7 @@ pub fn inject_gas_counter(module: elements::Module) -> elements::Module {
if *func_index >= gas_func { *func_index += 1} if *func_index >= gas_func { *func_index += 1}
}, },
_ => {} _ => {}
} }
} }
}, },
&mut elements::Section::Element(ref mut elements_section) => { &mut elements::Section::Element(ref mut elements_section) => {
@ -180,11 +180,11 @@ mod tests {
Call(0), Call(0),
GetGlobal(0), GetGlobal(0),
End End
][..], ][..],
injected_module injected_module
.code_section().expect("function section should exist").bodies()[0] .code_section().expect("function section should exist").bodies()[0]
.code().elements() .code().elements()
); );
} }
#[test] #[test]
@ -206,7 +206,7 @@ mod tests {
GetGlobal(0), GetGlobal(0),
GetGlobal(0), GetGlobal(0),
End, End,
GetGlobal(0), GetGlobal(0),
End End
] ]
)) ))
@ -215,7 +215,7 @@ mod tests {
.build(); .build();
let injected_module = inject_gas_counter(module); let injected_module = inject_gas_counter(module);
assert_eq!( assert_eq!(
&vec![ &vec![
I32Const(4), I32Const(4),
@ -230,7 +230,7 @@ mod tests {
End, End,
GetGlobal(0), GetGlobal(0),
End End
][..], ][..],
injected_module injected_module
.code_section().expect("function section should exist").bodies()[0] .code_section().expect("function section should exist").bodies()[0]
.code().elements() .code().elements()
@ -257,9 +257,9 @@ mod tests {
GetGlobal(0), GetGlobal(0),
Else, Else,
GetGlobal(0), GetGlobal(0),
GetGlobal(0), GetGlobal(0),
End, End,
GetGlobal(0), GetGlobal(0),
End End
] ]
)) ))
@ -288,12 +288,12 @@ mod tests {
End, End,
GetGlobal(0), GetGlobal(0),
End End
][..], ][..],
injected_module injected_module
.code_section().expect("function section should exist").bodies()[0] .code_section().expect("function section should exist").bodies()[0]
.code().elements() .code().elements()
); );
} }
#[test] #[test]
fn call_index() { fn call_index() {
@ -319,9 +319,9 @@ mod tests {
Call(0), Call(0),
Else, Else,
Call(0), Call(0),
Call(0), Call(0),
End, End,
Call(0), Call(0),
End End
] ]
)) ))
@ -350,11 +350,11 @@ mod tests {
End, End,
Call(1), Call(1),
End End
][..], ][..],
injected_module injected_module
.code_section().expect("function section should exist").bodies()[1] .code_section().expect("function section should exist").bodies()[1]
.code().elements() .code().elements()
); );
} }
} }

View File

@ -3,6 +3,8 @@ extern crate env_logger;
#[macro_use] extern crate log; #[macro_use] extern crate log;
#[macro_use] extern crate lazy_static; #[macro_use] extern crate lazy_static;
pub mod rules;
mod optimizer; mod optimizer;
mod gas; mod gas;
mod symbols; mod symbols;

266
src/rules.rs Normal file
View File

@ -0,0 +1,266 @@
use std::collections::HashMap;
use parity_wasm::elements;
pub struct UnknownInstruction;
#[derive(Debug, Hash, PartialEq, Eq, Copy, Clone)]
pub enum InstructionType {
Bit,
Add,
Mul,
Div,
Load,
Store,
Const,
Local,
Global,
ControlFlow,
IntegerComparsion,
FloatComparsion,
Numeric,
Conversion,
Reinterpretation,
Unreachable,
Nop,
CurrentMemory,
GrowMemory,
}
impl ::std::str::FromStr for InstructionType {
type Err = UnknownInstruction;
fn from_str(s: &str) -> Result<Self, Self::Err> {
match s {
"bit" => Ok(InstructionType::Bit),
"add" => Ok(InstructionType::Add),
"mul" => Ok(InstructionType::Mul),
"div" => Ok(InstructionType::Div),
"load" => Ok(InstructionType::Load),
"store" => Ok(InstructionType::Store),
"const" => Ok(InstructionType::Const),
"local" => Ok(InstructionType::Local),
"global" => Ok(InstructionType::Global),
"flow" => Ok(InstructionType::ControlFlow),
"integer_comp" => Ok(InstructionType::IntegerComparsion),
"float_comp" => Ok(InstructionType::FloatComparsion),
"numeric" => Ok(InstructionType::Numeric),
"conversion" => Ok(InstructionType::Conversion),
"reinterpret" => Ok(InstructionType::Reinterpretation),
"unreachable" => Ok(InstructionType::Unreachable),
"nop" => Ok(InstructionType::Nop),
"currrent_mem" => Ok(InstructionType::CurrentMemory),
"grow_mem" => Ok(InstructionType::GrowMemory),
_ => Err(UnknownInstruction),
}
}
}
impl InstructionType {
pub fn op(opcode: &elements::Opcode) -> Self {
use parity_wasm::elements::Opcode::*;
match *opcode {
Unreachable => InstructionType::Unreachable,
Nop => InstructionType::Nop,
Block(_) => InstructionType::ControlFlow,
Loop(_) => InstructionType::ControlFlow,
If(_) => InstructionType::ControlFlow,
Else => InstructionType::ControlFlow,
End => InstructionType::ControlFlow,
Br(_) => InstructionType::ControlFlow,
BrIf(_) => InstructionType::ControlFlow,
BrTable(_, _) => InstructionType::ControlFlow,
Return => InstructionType::ControlFlow,
Call(_) => InstructionType::ControlFlow,
CallIndirect(_, _) => InstructionType::ControlFlow,
Drop => InstructionType::ControlFlow,
Select => InstructionType::ControlFlow,
GetLocal(_) => InstructionType::Local,
SetLocal(_) => InstructionType::Local,
TeeLocal(_) => InstructionType::Local,
GetGlobal(_) => InstructionType::Local,
SetGlobal(_) => InstructionType::Local,
I32Load(_, _) => InstructionType::Load,
I64Load(_, _) => InstructionType::Load,
F32Load(_, _) => InstructionType::Load,
F64Load(_, _) => InstructionType::Load,
I32Load8S(_, _) => InstructionType::Load,
I32Load8U(_, _) => InstructionType::Load,
I32Load16S(_, _) => InstructionType::Load,
I32Load16U(_, _) => InstructionType::Load,
I64Load8S(_, _) => InstructionType::Load,
I64Load8U(_, _) => InstructionType::Load,
I64Load16S(_, _) => InstructionType::Load,
I64Load16U(_, _) => InstructionType::Load,
I64Load32S(_, _) => InstructionType::Load,
I64Load32U(_, _) => InstructionType::Load,
I32Store(_, _) => InstructionType::Store,
I64Store(_, _) => InstructionType::Store,
F32Store(_, _) => InstructionType::Store,
F64Store(_, _) => InstructionType::Store,
I32Store8(_, _) => InstructionType::Store,
I32Store16(_, _) => InstructionType::Store,
I64Store8(_, _) => InstructionType::Store,
I64Store16(_, _) => InstructionType::Store,
I64Store32(_, _) => InstructionType::Store,
CurrentMemory(_) => InstructionType::CurrentMemory,
GrowMemory(_) => InstructionType::GrowMemory,
I32Const(_) => InstructionType::Const,
I64Const(_) => InstructionType::Const,
F32Const(_) => InstructionType::Const,
F64Const(_) => InstructionType::Const,
I32Eqz => InstructionType::IntegerComparsion,
I32Eq => InstructionType::IntegerComparsion,
I32Ne => InstructionType::IntegerComparsion,
I32LtS => InstructionType::IntegerComparsion,
I32LtU => InstructionType::IntegerComparsion,
I32GtS => InstructionType::IntegerComparsion,
I32GtU => InstructionType::IntegerComparsion,
I32LeS => InstructionType::IntegerComparsion,
I32LeU => InstructionType::IntegerComparsion,
I32GeS => InstructionType::IntegerComparsion,
I32GeU => InstructionType::IntegerComparsion,
I64Eqz => InstructionType::IntegerComparsion,
I64Eq => InstructionType::IntegerComparsion,
I64Ne => InstructionType::IntegerComparsion,
I64LtS => InstructionType::IntegerComparsion,
I64LtU => InstructionType::IntegerComparsion,
I64GtS => InstructionType::IntegerComparsion,
I64GtU => InstructionType::IntegerComparsion,
I64LeS => InstructionType::IntegerComparsion,
I64LeU => InstructionType::IntegerComparsion,
I64GeS => InstructionType::IntegerComparsion,
I64GeU => InstructionType::IntegerComparsion,
F32Eq => InstructionType::FloatComparsion,
F32Ne => InstructionType::FloatComparsion,
F32Lt => InstructionType::FloatComparsion,
F32Gt => InstructionType::FloatComparsion,
F32Le => InstructionType::FloatComparsion,
F32Ge => InstructionType::FloatComparsion,
F64Eq => InstructionType::FloatComparsion,
F64Ne => InstructionType::FloatComparsion,
F64Lt => InstructionType::FloatComparsion,
F64Gt => InstructionType::FloatComparsion,
F64Le => InstructionType::FloatComparsion,
F64Ge => InstructionType::FloatComparsion,
I32Clz => InstructionType::Bit,
I32Ctz => InstructionType::Bit,
I32Popcnt => InstructionType::Bit,
I32Add => InstructionType::Add,
I32Sub => InstructionType::Add,
I32Mul => InstructionType::Mul,
I32DivS => InstructionType::Div,
I32DivU => InstructionType::Div,
I32RemS => InstructionType::Div,
I32RemU => InstructionType::Div,
I32And => InstructionType::Bit,
I32Or => InstructionType::Bit,
I32Xor => InstructionType::Bit,
I32Shl => InstructionType::Bit,
I32ShrS => InstructionType::Bit,
I32ShrU => InstructionType::Bit,
I32Rotl => InstructionType::Bit,
I32Rotr => InstructionType::Bit,
I64Clz => InstructionType::Bit,
I64Ctz => InstructionType::Bit,
I64Popcnt => InstructionType::Bit,
I64Add => InstructionType::Add,
I64Sub => InstructionType::Add,
I64Mul => InstructionType::Mul,
I64DivS => InstructionType::Div,
I64DivU => InstructionType::Div,
I64RemS => InstructionType::Div,
I64RemU => InstructionType::Div,
I64And => InstructionType::Bit,
I64Or => InstructionType::Bit,
I64Xor => InstructionType::Bit,
I64Shl => InstructionType::Bit,
I64ShrS => InstructionType::Bit,
I64ShrU => InstructionType::Bit,
I64Rotl => InstructionType::Bit,
I64Rotr => InstructionType::Bit,
F32Abs => InstructionType::Numeric,
F32Neg => InstructionType::Numeric,
F32Ceil => InstructionType::Numeric,
F32Floor => InstructionType::Numeric,
F32Trunc => InstructionType::Numeric,
F32Nearest => InstructionType::Numeric,
F32Sqrt => InstructionType::Numeric,
F32Add => InstructionType::Numeric,
F32Sub => InstructionType::Numeric,
F32Mul => InstructionType::Numeric,
F32Div => InstructionType::Numeric,
F32Min => InstructionType::Numeric,
F32Max => InstructionType::Numeric,
F32Copysign => InstructionType::Numeric,
F64Abs => InstructionType::Numeric,
F64Neg => InstructionType::Numeric,
F64Ceil => InstructionType::Numeric,
F64Floor => InstructionType::Numeric,
F64Trunc => InstructionType::Numeric,
F64Nearest => InstructionType::Numeric,
F64Sqrt => InstructionType::Numeric,
F64Add => InstructionType::Numeric,
F64Sub => InstructionType::Numeric,
F64Mul => InstructionType::Numeric,
F64Div => InstructionType::Numeric,
F64Min => InstructionType::Numeric,
F64Max => InstructionType::Numeric,
F64Copysign => InstructionType::Numeric,
I32WarpI64 => InstructionType::Conversion,
I32TruncSF32 => InstructionType::Conversion,
I32TruncUF32 => InstructionType::Conversion,
I32TruncSF64 => InstructionType::Conversion,
I32TruncUF64 => InstructionType::Conversion,
I64ExtendSI32 => InstructionType::Conversion,
I64ExtendUI32 => InstructionType::Conversion,
I64TruncSF32 => InstructionType::Conversion,
I64TruncUF32 => InstructionType::Conversion,
I64TruncSF64 => InstructionType::Conversion,
I64TruncUF64 => InstructionType::Conversion,
F32ConvertSI32 => InstructionType::Conversion,
F32ConvertUI32 => InstructionType::Conversion,
F32ConvertSI64 => InstructionType::Conversion,
F32ConvertUI64 => InstructionType::Conversion,
F32DemoteF64 => InstructionType::Conversion,
F64ConvertSI32 => InstructionType::Conversion,
F64ConvertUI32 => InstructionType::Conversion,
F64ConvertSI64 => InstructionType::Conversion,
F64ConvertUI64 => InstructionType::Conversion,
F64PromoteF32 => InstructionType::Conversion,
I32ReinterpretF32 => InstructionType::Reinterpretation,
I64ReinterpretF64 => InstructionType::Reinterpretation,
F32ReinterpretI32 => InstructionType::Reinterpretation,
F64ReinterpretI64 => InstructionType::Reinterpretation,
}
}
}
#[derive(Debug, Default)]
pub struct Set {
entries: HashMap<InstructionType, u32>,
}
impl Set {
pub fn new(entries: HashMap<InstructionType, u32>) -> Self {
Set { entries: entries }
}
pub fn process(&self, opcode: &elements::Opcode) -> u32 {
self.entries.get(&InstructionType::op(opcode)).map(|x| *x).unwrap_or(1)
}
}