2017-07-10 17:42:10 +02:00
|
|
|
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
//! Wasm Interpreter
|
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
extern crate byteorder;
|
|
|
|
extern crate ethcore_logger;
|
2017-08-01 12:37:57 +02:00
|
|
|
extern crate ethcore_util as util;
|
2017-09-04 16:36:49 +02:00
|
|
|
extern crate ethcore_bigint as bigint;
|
2017-08-01 12:37:57 +02:00
|
|
|
#[macro_use] extern crate log;
|
2018-02-14 16:13:38 +01:00
|
|
|
extern crate libc;
|
2017-08-01 12:37:57 +02:00
|
|
|
extern crate parity_wasm;
|
2018-02-14 16:13:38 +01:00
|
|
|
extern crate vm;
|
2017-08-01 12:37:57 +02:00
|
|
|
extern crate wasm_utils;
|
2018-02-14 16:13:38 +01:00
|
|
|
extern crate wasmi;
|
2017-08-01 12:37:57 +02:00
|
|
|
|
2017-07-10 17:42:10 +02:00
|
|
|
mod runtime;
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests;
|
|
|
|
mod env;
|
2017-11-15 10:49:07 +01:00
|
|
|
mod panic_payload;
|
2018-02-14 16:13:38 +01:00
|
|
|
mod parser;
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2017-08-01 12:37:57 +02:00
|
|
|
use vm::{GasLeft, ReturnData, ActionParams};
|
2018-02-14 16:13:38 +01:00
|
|
|
use wasmi::Error as InterpreterError;
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
use runtime::{Runtime, RuntimeContext};
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
use bigint::uint::U256;
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2017-09-10 18:02:31 +02:00
|
|
|
/// Wrapped interpreter error
|
|
|
|
#[derive(Debug)]
|
|
|
|
pub struct Error(InterpreterError);
|
|
|
|
|
|
|
|
impl From<InterpreterError> for Error {
|
|
|
|
fn from(e: InterpreterError) -> Self {
|
|
|
|
Error(e)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Error> for vm::Error {
|
|
|
|
fn from(e: Error) -> Self {
|
|
|
|
vm::Error::Wasm(format!("Wasm runtime error: {:?}", e.0))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-07-10 17:42:10 +02:00
|
|
|
/// Wasm interpreter instance
|
2018-02-14 16:13:38 +01:00
|
|
|
pub struct WasmInterpreter;
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
impl From<runtime::Error> for vm::Error {
|
|
|
|
fn from(e: runtime::Error) -> Self {
|
|
|
|
vm::Error::Wasm(format!("Wasm runtime error: {:?}", e))
|
2017-07-10 17:42:10 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-01 12:37:57 +02:00
|
|
|
impl vm::Vm for WasmInterpreter {
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2017-08-01 12:37:57 +02:00
|
|
|
fn exec(&mut self, params: ActionParams, ext: &mut vm::Ext) -> vm::Result<GasLeft> {
|
2018-02-14 16:13:38 +01:00
|
|
|
let (module, data) = parser::payload(¶ms, ext.schedule())?;
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
let loaded_module = wasmi::Module::from_parity_wasm_module(module).map_err(Error)?;
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
let instantiation_resolover = env::ImportResolver::with_limit(16);
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
let module_instance = wasmi::ModuleInstance::new(
|
|
|
|
&loaded_module,
|
|
|
|
&wasmi::ImportsBuilder::new().with_resolver("env", &instantiation_resolover)
|
|
|
|
).map_err(Error)?;
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
let adjusted_gas = params.gas * U256::from(ext.schedule().wasm.opcodes_div) /
|
|
|
|
U256::from(ext.schedule().wasm.opcodes_mul);
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
if adjusted_gas > ::std::u64::MAX.into()
|
|
|
|
{
|
|
|
|
return Err(vm::Error::Wasm("Wasm interpreter cannot run contracts with gas (wasm adjusted) >= 2^64".to_owned()));
|
2017-07-10 17:42:10 +02:00
|
|
|
}
|
2017-07-12 13:09:17 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
let initial_memory = instantiation_resolover.memory_size().map_err(Error)?;
|
|
|
|
trace!(target: "wasm", "Contract requested {:?} pages of initial memory", initial_memory);
|
|
|
|
|
|
|
|
let (gas_left, result) = {
|
|
|
|
let mut runtime = Runtime::with_params(
|
|
|
|
ext,
|
|
|
|
instantiation_resolover.memory_ref(),
|
|
|
|
// cannot overflow, checked above
|
|
|
|
adjusted_gas.low_u64(),
|
|
|
|
data.to_vec(),
|
|
|
|
RuntimeContext {
|
|
|
|
address: params.address,
|
|
|
|
sender: params.sender,
|
|
|
|
origin: params.origin,
|
|
|
|
code_address: params.code_address,
|
|
|
|
value: params.value.value(),
|
2017-11-02 12:49:57 +01:00
|
|
|
},
|
2018-02-14 16:13:38 +01:00
|
|
|
);
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
// cannot overflow if static_region < 2^16,
|
|
|
|
// initial_memory ∈ [0..2^32)
|
|
|
|
// total_charge <- static_region * 2^32 * 2^16
|
|
|
|
// total_charge ∈ [0..2^64) if static_region ∈ [0..2^16)
|
|
|
|
// qed
|
|
|
|
assert!(runtime.schedule().wasm.initial_mem < 1 << 16);
|
|
|
|
runtime.charge(|s| initial_memory as u64 * s.wasm.initial_mem as u64)?;
|
2017-07-12 13:09:17 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
let module_instance = module_instance.run_start(&mut runtime).map_err(Error)?;
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
match module_instance.invoke_export("call", &[], &mut runtime) {
|
2017-09-10 18:02:31 +02:00
|
|
|
Ok(_) => { },
|
2018-02-14 16:13:38 +01:00
|
|
|
Err(InterpreterError::Host(boxed)) => {
|
|
|
|
match boxed.downcast_ref::<runtime::Error>() {
|
|
|
|
None => {
|
|
|
|
return Err(vm::Error::Wasm("Invalid user error used in interpreter".to_owned()));
|
|
|
|
}
|
|
|
|
Some(runtime_err) => {
|
|
|
|
match *runtime_err {
|
|
|
|
runtime::Error::Suicide => {
|
|
|
|
// Suicide uses trap to break execution
|
|
|
|
}
|
|
|
|
ref any_err => {
|
|
|
|
trace!(target: "wasm", "Error executing contract: {:?}", boxed);
|
|
|
|
return Err(vm::Error::from(Error::from(InterpreterError::Host(Box::new(any_err.clone())))));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
2017-09-10 18:02:31 +02:00
|
|
|
Err(err) => {
|
2017-07-10 17:42:10 +02:00
|
|
|
trace!(target: "wasm", "Error executing contract: {:?}", err);
|
2018-02-14 16:13:38 +01:00
|
|
|
return Err(vm::Error::from(Error::from(err)))
|
2017-09-10 18:02:31 +02:00
|
|
|
}
|
|
|
|
}
|
2018-02-14 16:13:38 +01:00
|
|
|
(
|
|
|
|
runtime.gas_left().expect("Cannot fail since it was not updated since last charge"),
|
|
|
|
runtime.into_result(),
|
|
|
|
)
|
|
|
|
};
|
|
|
|
|
|
|
|
let gas_left =
|
|
|
|
U256::from(gas_left) * U256::from(ext.schedule().wasm.opcodes_mul)
|
|
|
|
/ U256::from(ext.schedule().wasm.opcodes_div);
|
2017-07-10 17:42:10 +02:00
|
|
|
|
2018-02-14 16:13:38 +01:00
|
|
|
if result.is_empty() {
|
2017-07-10 17:42:10 +02:00
|
|
|
trace!(target: "wasm", "Contract execution result is empty.");
|
2018-02-14 16:13:38 +01:00
|
|
|
Ok(GasLeft::Known(gas_left))
|
2017-07-10 17:42:10 +02:00
|
|
|
} else {
|
2018-02-14 16:13:38 +01:00
|
|
|
let len = result.len();
|
2017-07-12 13:09:17 +02:00
|
|
|
Ok(GasLeft::NeedsReturn {
|
2018-02-14 16:13:38 +01:00
|
|
|
gas_left: gas_left,
|
2017-07-10 17:42:10 +02:00
|
|
|
data: ReturnData::new(
|
2018-02-14 16:13:38 +01:00
|
|
|
result,
|
2017-07-12 13:09:17 +02:00
|
|
|
0,
|
2017-07-10 17:42:10 +02:00
|
|
|
len,
|
|
|
|
),
|
|
|
|
apply_state: true,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|