wasmer/lib/middleware-common/src/metering.rs

252 lines
7.9 KiB
Rust
Raw Normal View History

2019-05-16 17:10:45 +00:00
use wasmer_runtime_core::{
codegen::{Event, EventSink, FunctionMiddleware, InternalEvent},
module::ModuleInfo,
2019-05-23 12:10:17 +00:00
vm::InternalField,
2019-05-23 12:10:34 +00:00
wasmparser::{Operator, Type as WpType},
2019-05-16 17:10:45 +00:00
};
2019-05-23 12:10:17 +00:00
static INTERNAL_FIELD: InternalField = InternalField::allocate();
2019-05-16 17:10:45 +00:00
pub struct Metering {
limit: u64,
current_block: u64,
}
impl Metering {
pub fn new(limit: u64) -> Metering {
Metering {
limit,
current_block: 0,
}
}
}
impl FunctionMiddleware for Metering {
type Error = String;
fn feed_event<'a, 'b: 'a>(
&mut self,
op: Event<'a, 'b>,
_module_info: &ModuleInfo,
sink: &mut EventSink<'a, 'b>,
) -> Result<(), Self::Error> {
match op {
Event::Internal(InternalEvent::FunctionBegin(_)) => {
self.current_block = 0;
2019-05-16 17:15:05 +00:00
}
2019-05-16 17:10:45 +00:00
Event::Wasm(&ref op) | Event::WasmOwned(ref op) => {
self.current_block += 1;
match *op {
Operator::Loop { .. }
2019-05-16 17:15:05 +00:00
| Operator::Block { .. }
| Operator::End
| Operator::If { .. }
| Operator::Else
| Operator::Unreachable
| Operator::Br { .. }
| Operator::BrTable { .. }
| Operator::BrIf { .. }
| Operator::Call { .. }
2019-05-16 17:20:09 +00:00
| Operator::CallIndirect { .. }
| Operator::Return => {
2019-05-23 12:10:34 +00:00
sink.push(Event::Internal(InternalEvent::GetInternal(
INTERNAL_FIELD.index() as _,
)));
2019-05-16 17:15:05 +00:00
sink.push(Event::WasmOwned(Operator::I64Const {
value: self.current_block as i64,
}));
sink.push(Event::WasmOwned(Operator::I64Add));
2019-05-23 12:10:34 +00:00
sink.push(Event::Internal(InternalEvent::SetInternal(
INTERNAL_FIELD.index() as _,
)));
2019-05-16 17:15:05 +00:00
self.current_block = 0;
}
2019-05-16 17:10:45 +00:00
_ => {}
}
match *op {
2019-05-16 17:15:05 +00:00
Operator::Br { .. }
| Operator::BrTable { .. }
| Operator::BrIf { .. }
| Operator::Call { .. }
| Operator::CallIndirect { .. } => {
2019-05-23 12:10:34 +00:00
sink.push(Event::Internal(InternalEvent::GetInternal(
INTERNAL_FIELD.index() as _,
)));
2019-05-16 17:15:05 +00:00
sink.push(Event::WasmOwned(Operator::I64Const {
value: self.limit as i64,
}));
sink.push(Event::WasmOwned(Operator::I64GeU));
sink.push(Event::WasmOwned(Operator::If {
ty: WpType::EmptyBlockType,
}));
sink.push(Event::Internal(InternalEvent::Breakpoint(Box::new(
move |ctx| {
2019-05-16 17:10:45 +00:00
eprintln!("execution limit reached");
unsafe {
(ctx.throw)();
}
2019-05-16 17:15:05 +00:00
},
))));
sink.push(Event::WasmOwned(Operator::End));
}
2019-05-16 17:10:45 +00:00
_ => {}
}
2019-05-16 17:15:05 +00:00
}
2019-05-16 17:10:45 +00:00
_ => {}
}
sink.push(op);
Ok(())
}
}
#[cfg(test)]
mod tests {
use wabt::wat2wasm;
use wasmer_runtime_core::{
backend::{Compiler, CompilerConfig},
compile_with_config, imports, Func,
};
#[cfg(feature = "llvm")]
fn get_compiler() -> impl Compiler {
use wasmer_llvm_backend::LLVMCompiler;
LLVMCompiler::new()
}
#[cfg(feature = "singlepass")]
fn get_compiler() -> impl Compiler {
use wasmer_singlepass_backend::SinglePassCompiler;
SinglePassCompiler::new()
}
#[cfg(not(any(feature = "llvm", feature = "clif", feature = "singlepass")))]
fn get_compiler() -> impl Compiler {
panic!("compiler not specified, activate a compiler via features");
use wasmer_clif_backend::CraneliftCompiler;
CraneliftCompiler::new()
}
// Assemblyscript
// export function add_to(x: i32, y: i32): i32 {
// for(var i = 0; i < x; i++){
// if(i % 1 == 0){
// y += i;
// } else {
// y *= i
// }
// }
// return y;
// }
static WAT: &'static str = r#"
(module
(type $t0 (func (param i32 i32) (result i32)))
(type $t1 (func))
(func $add_to (export "add_to") (type $t0) (param $p0 i32) (param $p1 i32) (result i32)
(local $l0 i32)
block $B0
i32.const 0
set_local $l0
loop $L1
get_local $l0
get_local $p0
i32.lt_s
i32.eqz
br_if $B0
get_local $l0
i32.const 1
i32.rem_s
i32.const 0
i32.eq
if $I2
get_local $p1
get_local $l0
i32.add
set_local $p1
else
get_local $p1
get_local $l0
i32.mul
set_local $p1
end
get_local $l0
i32.const 1
i32.add
set_local $l0
br $L1
unreachable
end
unreachable
end
get_local $p1)
(func $f1 (type $t1))
(table $table (export "table") 1 anyfunc)
(memory $memory (export "memory") 0)
(global $g0 i32 (i32.const 8))
(elem (i32.const 0) $f1))
"#;
#[test]
fn test_points_reduced_after_call() {
let wasm_binary = wat2wasm(WAT).unwrap();
let limit = 100u64;
let module = compile_with_config(
&wasm_binary,
&get_compiler(),
CompilerConfig {
points_limit: Some(limit),
..Default::default()
},
)
.unwrap();
let import_object = imports! {};
let mut instance = module.instantiate(&import_object).unwrap();
instance.context_mut().set_points_used(0u64);
let add_to: Func<(i32, i32), i32> = instance.func("add_to").unwrap();
let value = add_to.call(3, 4).unwrap();
// verify it returns the correct value
assert_eq!(value, 7);
// verify is uses the correct number of points
assert_eq!(instance.context().get_points_used(), 42); // TODO need to update assertion to actual points used.
}
#[test]
fn test_traps_after_costly_call() {
let wasm_binary = wat2wasm(WAT).unwrap();
let limit = 100u64;
let module = compile_with_config(
&wasm_binary,
&get_compiler(),
CompilerConfig {
points_limit: Some(limit),
..Default::default()
},
)
.unwrap();
let import_object = imports! {};
let mut instance = module.instantiate(&import_object).unwrap();
instance.context_mut().set_points_used(0u64);
let add_to: Func<(i32, i32), i32> = instance.func("add_to").unwrap();
let result = add_to.call(10_000_000, 4);
// verify it errors
assert_eq!(result.is_err(), true); // TODO assert that the trap is caused by PointsExausted
// verify is uses the correct number of points
assert_eq!(instance.context().get_points_used(), 99); // TODO need to update assertion to actual points used.
// TODO should points used be close to limit or at limit when trapping
}
}