mirror of
https://github.com/fluencelabs/wasmer
synced 2024-12-13 06:15:33 +00:00
Metering middleware.
This commit is contained in:
parent
6aa87a0bbf
commit
7e79dd2cfe
@ -8,4 +8,4 @@ authors = ["The Wasmer Engineering Team <engineering@wasmer.io>"]
|
||||
edition = "2018"
|
||||
|
||||
[dependencies]
|
||||
wasmer-runtime-core = { path = "../runtime-core", version = "0.4.1" }
|
||||
wasmer-runtime-core = { path = "../runtime-core" }
|
||||
|
@ -1,3 +1,4 @@
|
||||
#![deny(unused_imports, unused_variables, unused_unsafe, unreachable_patterns)]
|
||||
|
||||
pub mod call_trace;
|
||||
pub mod metering;
|
||||
|
83
lib/middleware-common/src/metering.rs
Normal file
83
lib/middleware-common/src/metering.rs
Normal file
@ -0,0 +1,83 @@
|
||||
use wasmer_runtime_core::{
|
||||
codegen::{Event, EventSink, FunctionMiddleware, InternalEvent},
|
||||
module::ModuleInfo,
|
||||
wasmparser::{Operator, Type as WpType},
|
||||
};
|
||||
|
||||
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;
|
||||
},
|
||||
Event::Wasm(&ref op) | Event::WasmOwned(ref op) => {
|
||||
self.current_block += 1;
|
||||
match *op {
|
||||
Operator::Loop { .. }
|
||||
| Operator::Block { .. }
|
||||
| Operator::End
|
||||
| Operator::If { .. }
|
||||
| Operator::Else
|
||||
| Operator::Unreachable
|
||||
| Operator::Br { .. }
|
||||
| Operator::BrTable { .. }
|
||||
| Operator::BrIf { .. }
|
||||
| Operator::Call { .. }
|
||||
| Operator::CallIndirect { .. }
|
||||
=> {
|
||||
sink.push(Event::Internal(InternalEvent::GetInternal(0)));
|
||||
sink.push(Event::WasmOwned(Operator::I64Const { value: self.current_block as i64 }));
|
||||
sink.push(Event::WasmOwned(Operator::I64Add));
|
||||
sink.push(Event::Internal(InternalEvent::SetInternal(0)));
|
||||
self.current_block = 0;
|
||||
},
|
||||
_ => {}
|
||||
}
|
||||
match *op {
|
||||
Operator::Br { .. }
|
||||
| Operator::BrTable { .. }
|
||||
| Operator::BrIf { .. }
|
||||
| Operator::Call { .. }
|
||||
| Operator::CallIndirect { .. }
|
||||
=> {
|
||||
sink.push(Event::Internal(InternalEvent::GetInternal(0)));
|
||||
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| {
|
||||
eprintln!("execution limit reached");
|
||||
unsafe {
|
||||
(ctx.throw)();
|
||||
}
|
||||
}))));
|
||||
sink.push(Event::WasmOwned(Operator::End));
|
||||
},
|
||||
_ => {}
|
||||
}
|
||||
},
|
||||
_ => {}
|
||||
}
|
||||
sink.push(op);
|
||||
Ok(())
|
||||
}
|
||||
}
|
Loading…
Reference in New Issue
Block a user