Calculating gas using usize (if supplied gaslimit fits in usize) (#1518)
* Spliting gasometer out of interpreter * Choosing right gas calculations implementation based on supplied gas * Moving verification out of gasometer * MemGasCost benchmark. Conflicts: ethcore/src/evm/benches/mod.rs * Some simple benchmarks * Benchmark for simple loop * Calculating gas_for_memory only when it's actually needed * Removing superfluous newline [ci skip]
This commit is contained in:
		
							parent
							
								
									45d532368d
								
							
						
					
					
						commit
						4c1b74a42e
					
				| @ -43,3 +43,4 @@ json-tests = [] | |||||||
| test-heavy = [] | test-heavy = [] | ||||||
| dev = ["clippy"] | dev = ["clippy"] | ||||||
| default = [] | default = [] | ||||||
|  | benches = [] | ||||||
|  | |||||||
							
								
								
									
										126
									
								
								ethcore/src/evm/benches/mod.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										126
									
								
								ethcore/src/evm/benches/mod.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,126 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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/>.
 | ||||||
|  | 
 | ||||||
|  | //! benchmarking for EVM
 | ||||||
|  | //! should be started with:
 | ||||||
|  | //! ```bash
 | ||||||
|  | //! multirust run nightly cargo bench
 | ||||||
|  | //! ```
 | ||||||
|  | 
 | ||||||
|  | extern crate test; | ||||||
|  | 
 | ||||||
|  | use self::test::{Bencher, black_box}; | ||||||
|  | 
 | ||||||
|  | use common::*; | ||||||
|  | use evm::{self, Factory, VMType}; | ||||||
|  | use evm::tests::FakeExt; | ||||||
|  | 
 | ||||||
|  | #[bench] | ||||||
|  | fn simple_loop_log0_usize(b: &mut Bencher) { | ||||||
|  | 	simple_loop_log0(U256::from(::std::usize::MAX), b) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[bench] | ||||||
|  | fn simple_loop_log0_u256(b: &mut Bencher) { | ||||||
|  | 	simple_loop_log0(!U256::zero(), b) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn simple_loop_log0(gas: U256, b: &mut Bencher) { | ||||||
|  | 	let mut vm = Factory::new(VMType::Interpreter).create(gas); | ||||||
|  | 	let mut ext = FakeExt::new(); | ||||||
|  | 
 | ||||||
|  | 	let address = Address::from_str("0f572e5295c57f15886f9b263e2f6d2d6c7b5ec6").unwrap(); | ||||||
|  | 	let code = black_box( | ||||||
|  | 		"62ffffff5b600190036000600fa0600357".from_hex().unwrap() | ||||||
|  | 	); | ||||||
|  | 
 | ||||||
|  | 	b.iter(|| { | ||||||
|  | 		let mut params = ActionParams::default(); | ||||||
|  | 		params.address = address.clone(); | ||||||
|  | 		params.gas = gas; | ||||||
|  | 		params.code = Some(code.clone()); | ||||||
|  | 
 | ||||||
|  | 		result(vm.exec(params, &mut ext)) | ||||||
|  | 	}); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[bench] | ||||||
|  | fn mem_gas_calculation_same_usize(b: &mut Bencher) { | ||||||
|  | 	mem_gas_calculation_same(U256::from(::std::usize::MAX), b) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[bench] | ||||||
|  | fn mem_gas_calculation_same_u256(b: &mut Bencher) { | ||||||
|  | 	mem_gas_calculation_same(!U256::zero(), b) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn mem_gas_calculation_same(gas: U256, b: &mut Bencher) { | ||||||
|  | 	let mut vm = Factory::new(VMType::Interpreter).create(gas); | ||||||
|  | 	let mut ext = FakeExt::new(); | ||||||
|  | 
 | ||||||
|  | 	let address = Address::from_str("0f572e5295c57f15886f9b263e2f6d2d6c7b5ec6").unwrap(); | ||||||
|  | 
 | ||||||
|  | 	b.iter(|| { | ||||||
|  | 		let code = black_box( | ||||||
|  | 			"6110006001556001546000555b610fff805560016000540380600055600c57".from_hex().unwrap() | ||||||
|  | 		); | ||||||
|  | 
 | ||||||
|  | 		let mut params = ActionParams::default(); | ||||||
|  | 		params.address = address.clone(); | ||||||
|  | 		params.gas = gas; | ||||||
|  | 		params.code = Some(code.clone()); | ||||||
|  | 
 | ||||||
|  | 		result(vm.exec(params, &mut ext)) | ||||||
|  | 	}); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[bench] | ||||||
|  | fn mem_gas_calculation_increasing_usize(b: &mut Bencher) { | ||||||
|  | 	mem_gas_calculation_increasing(U256::from(::std::usize::MAX), b) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[bench] | ||||||
|  | fn mem_gas_calculation_increasing_u256(b: &mut Bencher) { | ||||||
|  | 	mem_gas_calculation_increasing(!U256::zero(), b) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn mem_gas_calculation_increasing(gas: U256, b: &mut Bencher) { | ||||||
|  | 	let mut vm = Factory::new(VMType::Interpreter).create(gas); | ||||||
|  | 	let mut ext = FakeExt::new(); | ||||||
|  | 
 | ||||||
|  | 	let address = Address::from_str("0f572e5295c57f15886f9b263e2f6d2d6c7b5ec6").unwrap(); | ||||||
|  | 
 | ||||||
|  | 	b.iter(|| { | ||||||
|  | 		let code = black_box( | ||||||
|  | 			"6110006001556001546000555b610fff60005401805560016000540380600055600c57".from_hex().unwrap() | ||||||
|  | 		); | ||||||
|  | 
 | ||||||
|  | 		let mut params = ActionParams::default(); | ||||||
|  | 		params.address = address.clone(); | ||||||
|  | 		params.gas = gas; | ||||||
|  | 		params.code = Some(code.clone()); | ||||||
|  | 
 | ||||||
|  | 		result(vm.exec(params, &mut ext)) | ||||||
|  | 	}); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | fn result(r: evm::Result<evm::GasLeft>) -> U256 { | ||||||
|  | 	match r { | ||||||
|  | 		Ok(evm::GasLeft::Known(v)) => v, | ||||||
|  | 		Ok(evm::GasLeft::NeedsReturn(v, _)) => v, | ||||||
|  | 		_ => U256::zero(), | ||||||
|  | 	} | ||||||
|  | } | ||||||
| @ -95,6 +95,61 @@ impl<'a> Finalize for Result<GasLeft<'a>> { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | pub trait CostType: ops::Mul<Output=Self> + ops::Div<Output=Self> + ops::Add<Output=Self> + ops::Sub<Output=Self> + ops::Shr<usize, Output=Self> + ops::Shl<usize, Output=Self> + cmp::Ord + Sized + From<usize> + Copy { | ||||||
|  | 	fn as_u256(&self) -> U256; | ||||||
|  | 	fn from_u256(val: U256) -> Result<Self>; | ||||||
|  | 	fn as_usize(&self) -> usize; | ||||||
|  | 	fn overflow_add(self, other: Self) -> (Self, bool); | ||||||
|  | 	fn overflow_mul(self, other: Self) -> (Self, bool); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl CostType for U256 { | ||||||
|  | 	fn as_u256(&self) -> U256 { | ||||||
|  | 		*self | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn from_u256(val: U256) -> Result<Self> { | ||||||
|  | 		Ok(val) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn as_usize(&self) -> usize { | ||||||
|  | 		self.as_u64() as usize | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn overflow_add(self, other: Self) -> (Self, bool) { | ||||||
|  | 		Uint::overflowing_add(self, other) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn overflow_mul(self, other: Self) -> (Self, bool) { | ||||||
|  | 		Uint::overflowing_mul(self, other) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl CostType for usize { | ||||||
|  | 	fn as_u256(&self) -> U256 { | ||||||
|  | 		U256::from(*self) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn from_u256(val: U256) -> Result<Self> { | ||||||
|  | 		if U256::from(val.low_u64()) != val { | ||||||
|  | 			return Err(Error::OutOfGas); | ||||||
|  | 		} | ||||||
|  | 		Ok(val.low_u64() as usize) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn as_usize(&self) -> usize { | ||||||
|  | 		*self | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn overflow_add(self, other: Self) -> (Self, bool) { | ||||||
|  | 		self.overflowing_add(other) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn overflow_mul(self, other: Self) -> (Self, bool) { | ||||||
|  | 		self.overflowing_mul(other) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| /// Evm interface
 | /// Evm interface
 | ||||||
| pub trait Evm { | pub trait Evm { | ||||||
| 	/// This function should be used to execute transaction.
 | 	/// This function should be used to execute transaction.
 | ||||||
|  | |||||||
| @ -19,6 +19,7 @@ | |||||||
| //! TODO: consider spliting it into two separate files.
 | //! TODO: consider spliting it into two separate files.
 | ||||||
| use std::fmt; | use std::fmt; | ||||||
| use evm::Evm; | use evm::Evm; | ||||||
|  | use util::{U256, Uint}; | ||||||
| 
 | 
 | ||||||
| #[derive(Debug, Clone)] | #[derive(Debug, Clone)] | ||||||
| /// Type of EVM to use.
 | /// Type of EVM to use.
 | ||||||
| @ -85,24 +86,30 @@ pub struct Factory { | |||||||
| 
 | 
 | ||||||
| impl Factory { | impl Factory { | ||||||
| 	/// Create fresh instance of VM
 | 	/// Create fresh instance of VM
 | ||||||
|  | 	/// Might choose implementation depending on supplied gas.
 | ||||||
| 	#[cfg(feature = "jit")] | 	#[cfg(feature = "jit")] | ||||||
| 	pub fn create(&self) -> Box<Evm> { | 	pub fn create(&self, gas: U256) -> Box<Evm> { | ||||||
| 		match self.evm { | 		match self.evm { | ||||||
| 			VMType::Jit => { | 			VMType::Jit => { | ||||||
| 				Box::new(super::jit::JitEvm::default()) | 				Box::new(super::jit::JitEvm::default()) | ||||||
| 			}, | 			}, | ||||||
| 			VMType::Interpreter => { | 			VMType::Interpreter => if Self::can_fit_in_usize(gas) { | ||||||
| 				Box::new(super::interpreter::Interpreter::default()) | 				Box::new(super::interpreter::Interpreter::<usize>::default()) | ||||||
|  | 			} else { | ||||||
|  | 				Box::new(super::interpreter::Interpreter::<U256>::default()) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	/// Create fresh instance of VM
 | 	/// Create fresh instance of VM
 | ||||||
|  | 	/// Might choose implementation depending on supplied gas.
 | ||||||
| 	#[cfg(not(feature = "jit"))] | 	#[cfg(not(feature = "jit"))] | ||||||
| 	pub fn create(&self) -> Box<Evm> { | 	pub fn create(&self, gas: U256) -> Box<Evm> { | ||||||
| 		match self.evm { | 		match self.evm { | ||||||
| 			VMType::Interpreter => { | 			VMType::Interpreter => if Self::can_fit_in_usize(gas) { | ||||||
| 				Box::new(super::interpreter::Interpreter::default()) | 				Box::new(super::interpreter::Interpreter::<usize>::default()) | ||||||
|  | 			} else { | ||||||
|  | 				Box::new(super::interpreter::Interpreter::<U256>::default()) | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| @ -113,6 +120,10 @@ impl Factory { | |||||||
| 			evm: evm | 			evm: evm | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
|  | 
 | ||||||
|  | 	fn can_fit_in_usize(gas: U256) -> bool { | ||||||
|  | 		gas == U256::from(gas.low_u64() as usize) | ||||||
|  | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Default for Factory { | impl Default for Factory { | ||||||
| @ -135,7 +146,7 @@ impl Default for Factory { | |||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
| fn test_create_vm() { | fn test_create_vm() { | ||||||
| 	let _vm = Factory::default().create(); | 	let _vm = Factory::default().create(U256::zero()); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| /// Create tests by injecting different VM factories
 | /// Create tests by injecting different VM factories
 | ||||||
|  | |||||||
| @ -79,7 +79,7 @@ fn test_get_log_topics() { | |||||||
| 	assert_eq!(get_log_topics(LOG4), 4); | 	assert_eq!(get_log_topics(LOG4), 4); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(PartialEq)] | #[derive(PartialEq, Clone, Copy)] | ||||||
| pub enum GasPriceTier { | pub enum GasPriceTier { | ||||||
| 	/// 0 Zero
 | 	/// 0 Zero
 | ||||||
| 	Zero, | 	Zero, | ||||||
|  | |||||||
							
								
								
									
										261
									
								
								ethcore/src/evm/interpreter/gasometer.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										261
									
								
								ethcore/src/evm/interpreter/gasometer.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,261 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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/>.
 | ||||||
|  | 
 | ||||||
|  | use common::*; | ||||||
|  | use super::u256_to_address; | ||||||
|  | use evm::{self, CostType}; | ||||||
|  | use evm::instructions::{self, Instruction, InstructionInfo}; | ||||||
|  | use evm::interpreter::stack::Stack; | ||||||
|  | 
 | ||||||
|  | macro_rules! overflowing { | ||||||
|  | 	($x: expr) => {{ | ||||||
|  | 		let (v, overflow) = $x; | ||||||
|  | 		if overflow { return Err(evm::Error::OutOfGas); } | ||||||
|  | 		v | ||||||
|  | 	}} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[cfg_attr(feature="dev", allow(enum_variant_names))] | ||||||
|  | enum InstructionCost<Cost: CostType> { | ||||||
|  | 	Gas(Cost), | ||||||
|  | 	GasMem(Cost, Cost), | ||||||
|  | 	GasMemCopy(Cost, Cost, Cost) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub struct Gasometer<Gas: CostType> { | ||||||
|  | 	pub current_gas: Gas, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<Gas: CostType> Gasometer<Gas> { | ||||||
|  | 
 | ||||||
|  | 	pub fn new(current_gas: Gas) -> Self { | ||||||
|  | 		Gasometer { | ||||||
|  | 			current_gas: current_gas, | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	pub fn verify_gas(&self, gas_cost: &Gas) -> evm::Result<()> { | ||||||
|  | 		match &self.current_gas < gas_cost { | ||||||
|  | 			true => Err(evm::Error::OutOfGas), | ||||||
|  | 			false => Ok(()) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	#[cfg_attr(feature="dev", allow(cyclomatic_complexity))] | ||||||
|  | 	pub fn get_gas_cost_mem( | ||||||
|  | 		&mut self, | ||||||
|  | 		ext: &evm::Ext, | ||||||
|  | 		instruction: Instruction, | ||||||
|  | 		info: &InstructionInfo, | ||||||
|  | 		stack: &Stack<U256>, | ||||||
|  | 		current_mem_size: usize, | ||||||
|  | 	) -> evm::Result<(Gas, usize)> { | ||||||
|  | 		let schedule = ext.schedule(); | ||||||
|  | 		let tier = instructions::get_tier_idx(info.tier); | ||||||
|  | 		let default_gas = Gas::from(schedule.tier_step_gas[tier]); | ||||||
|  | 
 | ||||||
|  | 		let cost = match instruction { | ||||||
|  | 			instructions::SSTORE => { | ||||||
|  | 				let address = H256::from(stack.peek(0)); | ||||||
|  | 				let newval = stack.peek(1); | ||||||
|  | 				let val = U256::from(ext.storage_at(&address).as_slice()); | ||||||
|  | 
 | ||||||
|  | 				let gas = if U256::zero() == val && &U256::zero() != newval { | ||||||
|  | 					schedule.sstore_set_gas | ||||||
|  | 				} else { | ||||||
|  | 					// Refund for below case is added when actually executing sstore
 | ||||||
|  | 					// !self.is_zero(&val) && self.is_zero(newval)
 | ||||||
|  | 					schedule.sstore_reset_gas | ||||||
|  | 				}; | ||||||
|  | 				InstructionCost::Gas(Gas::from(gas)) | ||||||
|  | 			}, | ||||||
|  | 			instructions::SLOAD => { | ||||||
|  | 				InstructionCost::Gas(Gas::from(schedule.sload_gas)) | ||||||
|  | 			}, | ||||||
|  | 			instructions::MSTORE | instructions::MLOAD => { | ||||||
|  | 				InstructionCost::GasMem(default_gas, try!(self.mem_needed_const(stack.peek(0), 32))) | ||||||
|  | 			}, | ||||||
|  | 			instructions::MSTORE8 => { | ||||||
|  | 				InstructionCost::GasMem(default_gas, try!(self.mem_needed_const(stack.peek(0), 1))) | ||||||
|  | 			}, | ||||||
|  | 			instructions::RETURN => { | ||||||
|  | 				InstructionCost::GasMem(default_gas, try!(self.mem_needed(stack.peek(0), stack.peek(1)))) | ||||||
|  | 			}, | ||||||
|  | 			instructions::SHA3 => { | ||||||
|  | 				let w = overflowing!(add_gas_usize(try!(Gas::from_u256(*stack.peek(1))), 31)); | ||||||
|  | 				let words = w >> 5; | ||||||
|  | 				let gas = Gas::from(schedule.sha3_gas) + (Gas::from(schedule.sha3_word_gas) * words); | ||||||
|  | 				InstructionCost::GasMem(gas, try!(self.mem_needed(stack.peek(0), stack.peek(1)))) | ||||||
|  | 			}, | ||||||
|  | 			instructions::CALLDATACOPY | instructions::CODECOPY => { | ||||||
|  | 				InstructionCost::GasMemCopy(default_gas, try!(self.mem_needed(stack.peek(0), stack.peek(2))), try!(Gas::from_u256(*stack.peek(2)))) | ||||||
|  | 			}, | ||||||
|  | 			instructions::EXTCODECOPY => { | ||||||
|  | 				InstructionCost::GasMemCopy(default_gas, try!(self.mem_needed(stack.peek(1), stack.peek(3))), try!(Gas::from_u256(*stack.peek(3)))) | ||||||
|  | 			}, | ||||||
|  | 			instructions::JUMPDEST => { | ||||||
|  | 				InstructionCost::Gas(Gas::from(1)) | ||||||
|  | 			}, | ||||||
|  | 			instructions::LOG0...instructions::LOG4 => { | ||||||
|  | 				let no_of_topics = instructions::get_log_topics(instruction); | ||||||
|  | 				let log_gas = schedule.log_gas + schedule.log_topic_gas * no_of_topics; | ||||||
|  | 
 | ||||||
|  | 				let data_gas = overflowing!(try!(Gas::from_u256(*stack.peek(1))).overflow_mul(Gas::from(schedule.log_data_gas))); | ||||||
|  | 				let gas = overflowing!(data_gas.overflow_add(Gas::from(log_gas))); | ||||||
|  | 				InstructionCost::GasMem(gas, try!(self.mem_needed(stack.peek(0), stack.peek(1)))) | ||||||
|  | 			}, | ||||||
|  | 			instructions::CALL | instructions::CALLCODE => { | ||||||
|  | 				let mut gas  = overflowing!(add_gas_usize(try!(Gas::from_u256(*stack.peek(0))), schedule.call_gas)); | ||||||
|  | 				let mem = cmp::max( | ||||||
|  | 					try!(self.mem_needed(stack.peek(5), stack.peek(6))), | ||||||
|  | 					try!(self.mem_needed(stack.peek(3), stack.peek(4))) | ||||||
|  | 				); | ||||||
|  | 
 | ||||||
|  | 				let address = u256_to_address(stack.peek(1)); | ||||||
|  | 
 | ||||||
|  | 				if instruction == instructions::CALL && !ext.exists(&address) { | ||||||
|  | 					gas = overflowing!(gas.overflow_add(Gas::from(schedule.call_new_account_gas))); | ||||||
|  | 				}; | ||||||
|  | 
 | ||||||
|  | 				if stack.peek(2) > &U256::zero() { | ||||||
|  | 					gas = overflowing!(gas.overflow_add(Gas::from(schedule.call_value_transfer_gas))); | ||||||
|  | 				}; | ||||||
|  | 
 | ||||||
|  | 				InstructionCost::GasMem(gas,mem) | ||||||
|  | 			}, | ||||||
|  | 			instructions::DELEGATECALL => { | ||||||
|  | 				let gas = overflowing!(add_gas_usize(try!(Gas::from_u256(*stack.peek(0))), schedule.call_gas)); | ||||||
|  | 				let mem = cmp::max( | ||||||
|  | 					try!(self.mem_needed(stack.peek(4), stack.peek(5))), | ||||||
|  | 					try!(self.mem_needed(stack.peek(2), stack.peek(3))) | ||||||
|  | 				); | ||||||
|  | 				InstructionCost::GasMem(gas, mem) | ||||||
|  | 			}, | ||||||
|  | 			instructions::CREATE => { | ||||||
|  | 				let gas = Gas::from(schedule.create_gas); | ||||||
|  | 				let mem = try!(self.mem_needed(stack.peek(1), stack.peek(2))); | ||||||
|  | 				InstructionCost::GasMem(gas, mem) | ||||||
|  | 			}, | ||||||
|  | 			instructions::EXP => { | ||||||
|  | 				let expon = stack.peek(1); | ||||||
|  | 				let bytes = ((expon.bits() + 7) / 8) as usize; | ||||||
|  | 				let gas = Gas::from(schedule.exp_gas + schedule.exp_byte_gas * bytes); | ||||||
|  | 				InstructionCost::Gas(gas) | ||||||
|  | 			}, | ||||||
|  | 			_ => InstructionCost::Gas(default_gas) | ||||||
|  | 		}; | ||||||
|  | 
 | ||||||
|  | 		match cost { | ||||||
|  | 			InstructionCost::Gas(gas) => { | ||||||
|  | 				Ok((gas, 0)) | ||||||
|  | 			}, | ||||||
|  | 			InstructionCost::GasMem(gas, mem_size) => { | ||||||
|  | 				let (mem_gas, new_mem_size) = try!(self.mem_gas_cost(schedule, current_mem_size, &mem_size)); | ||||||
|  | 				let gas = overflowing!(gas.overflow_add(mem_gas)); | ||||||
|  | 				Ok((gas, new_mem_size)) | ||||||
|  | 			}, | ||||||
|  | 			InstructionCost::GasMemCopy(gas, mem_size, copy) => { | ||||||
|  | 				let (mem_gas, new_mem_size) = try!(self.mem_gas_cost(schedule, current_mem_size, &mem_size)); | ||||||
|  | 				let copy = overflowing!(add_gas_usize(copy, 31)); | ||||||
|  | 				let copy_gas = Gas::from(schedule.copy_gas) * (copy / Gas::from(32 as usize)); | ||||||
|  | 				let gas = overflowing!(gas.overflow_add(copy_gas)); | ||||||
|  | 				let gas = overflowing!(gas.overflow_add(mem_gas)); | ||||||
|  | 				Ok((gas, new_mem_size)) | ||||||
|  | 			} | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn is_zero(&self, val: &Gas) -> bool { | ||||||
|  | 		&Gas::from(0) == val | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn mem_needed_const(&self, mem: &U256, add: usize) -> evm::Result<Gas> { | ||||||
|  | 		Gas::from_u256(overflowing!(mem.overflowing_add(U256::from(add)))) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn mem_needed(&self, offset: &U256, size: &U256) -> evm::Result<Gas> { | ||||||
|  | 		if self.is_zero(&try!(Gas::from_u256(*size))) { | ||||||
|  | 			return Ok(Gas::from(0)); | ||||||
|  | 		} | ||||||
|  | 
 | ||||||
|  | 		Gas::from_u256(overflowing!(offset.overflowing_add(*size))) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn mem_gas_cost(&self, schedule: &evm::Schedule, current_mem_size: usize, mem_size: &Gas) -> evm::Result<(Gas, usize)> { | ||||||
|  | 		let gas_for_mem = |mem_size: Gas| { | ||||||
|  | 			let s = mem_size >> 5; | ||||||
|  | 			// s * memory_gas + s * s / quad_coeff_div
 | ||||||
|  | 			let a = overflowing!(s.overflow_mul(Gas::from(schedule.memory_gas))); | ||||||
|  | 			// We need to go to U512 to calculate s*s/quad_coeff_div
 | ||||||
|  | 			let b = U512::from(s.as_u256()) * U512::from(s.as_u256()) / U512::from(schedule.quad_coeff_div); | ||||||
|  | 			if b > U512::from(!U256::zero()) { | ||||||
|  | 				Err(evm::Error::OutOfGas) | ||||||
|  | 			} else { | ||||||
|  | 				Ok(overflowing!(a.overflow_add(try!(Gas::from_u256(U256::from(b)))))) | ||||||
|  | 			} | ||||||
|  | 		}; | ||||||
|  | 		let current_mem_size = Gas::from(current_mem_size); | ||||||
|  | 		let req_mem_size_rounded = (overflowing!(mem_size.overflow_add(Gas::from(31 as usize))) >> 5) << 5; | ||||||
|  | 
 | ||||||
|  | 		let mem_gas_cost = if req_mem_size_rounded > current_mem_size { | ||||||
|  | 			let new_mem_gas = try!(gas_for_mem(req_mem_size_rounded)); | ||||||
|  | 			let current_mem_gas = try!(gas_for_mem(current_mem_size)); | ||||||
|  | 			new_mem_gas - current_mem_gas | ||||||
|  | 		} else { | ||||||
|  | 			Gas::from(0) | ||||||
|  | 		}; | ||||||
|  | 
 | ||||||
|  | 		Ok((mem_gas_cost, req_mem_size_rounded.as_usize())) | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[inline] | ||||||
|  | fn add_gas_usize<Gas: CostType>(value: Gas, num: usize) -> (Gas, bool) { | ||||||
|  | 	value.overflow_add(Gas::from(num)) | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn test_mem_gas_cost() { | ||||||
|  | 	// given
 | ||||||
|  | 	let gasometer = Gasometer::<U256>::new(U256::zero()); | ||||||
|  | 	let schedule = evm::Schedule::default(); | ||||||
|  | 	let current_mem_size = 5; | ||||||
|  | 	let mem_size = !U256::zero(); | ||||||
|  | 
 | ||||||
|  | 	// when
 | ||||||
|  | 	let result = gasometer.mem_gas_cost(&schedule, current_mem_size, &mem_size); | ||||||
|  | 
 | ||||||
|  | 	// then
 | ||||||
|  | 	if let Ok(_) = result { | ||||||
|  | 		assert!(false, "Should fail with OutOfGas"); | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn test_calculate_mem_cost() { | ||||||
|  | 	// given
 | ||||||
|  | 	let gasometer = Gasometer::<usize>::new(0); | ||||||
|  | 	let schedule = evm::Schedule::default(); | ||||||
|  | 	let current_mem_size = 0; | ||||||
|  | 	let mem_size = 5; | ||||||
|  | 
 | ||||||
|  | 	// when
 | ||||||
|  | 	let (mem_cost, mem_size) = gasometer.mem_gas_cost(&schedule, current_mem_size, &mem_size).unwrap(); | ||||||
|  | 
 | ||||||
|  | 	// then
 | ||||||
|  | 	assert_eq!(mem_cost, 3); | ||||||
|  | 	assert_eq!(mem_size, 32); | ||||||
|  | } | ||||||
							
								
								
									
										150
									
								
								ethcore/src/evm/interpreter/memory.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										150
									
								
								ethcore/src/evm/interpreter/memory.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,150 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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/>.
 | ||||||
|  | 
 | ||||||
|  | use util::{U256, Uint}; | ||||||
|  | 
 | ||||||
|  | pub trait Memory { | ||||||
|  | 	/// Retrieve current size of the memory
 | ||||||
|  | 	fn size(&self) -> usize; | ||||||
|  | 	/// Resize (shrink or expand) the memory to specified size (fills 0)
 | ||||||
|  | 	fn resize(&mut self, new_size: usize); | ||||||
|  | 	/// Resize the memory only if its smaller
 | ||||||
|  | 	fn expand(&mut self, new_size: usize); | ||||||
|  | 	/// Write single byte to memory
 | ||||||
|  | 	fn write_byte(&mut self, offset: U256, value: U256); | ||||||
|  | 	/// Write a word to memory. Does not resize memory!
 | ||||||
|  | 	fn write(&mut self, offset: U256, value: U256); | ||||||
|  | 	/// Read a word from memory
 | ||||||
|  | 	fn read(&self, offset: U256) -> U256; | ||||||
|  | 	/// Write slice of bytes to memory. Does not resize memory!
 | ||||||
|  | 	fn write_slice(&mut self, offset: U256, &[u8]); | ||||||
|  | 	/// Retrieve part of the memory between offset and offset + size
 | ||||||
|  | 	fn read_slice(&self, offset: U256, size: U256) -> &[u8]; | ||||||
|  | 	/// Retrieve writeable part of memory
 | ||||||
|  | 	fn writeable_slice(&mut self, offset: U256, size: U256) -> &mut[u8]; | ||||||
|  | 	fn dump(&self); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | /// Checks whether offset and size is valid memory range
 | ||||||
|  | fn is_valid_range(off: usize, size: usize)  -> bool { | ||||||
|  | 	// When size is zero we haven't actually expanded the memory
 | ||||||
|  | 	let overflow = off.overflowing_add(size).1; | ||||||
|  | 	size > 0 && !overflow | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl Memory for Vec<u8> { | ||||||
|  | 	fn dump(&self) { | ||||||
|  | 		println!("MemoryDump:"); | ||||||
|  | 		for i in self.iter() { | ||||||
|  | 			println!("{:02x} ", i); | ||||||
|  | 		} | ||||||
|  | 		println!(""); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn size(&self) -> usize { | ||||||
|  | 		self.len() | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn read_slice(&self, init_off_u: U256, init_size_u: U256) -> &[u8] { | ||||||
|  | 		let off = init_off_u.low_u64() as usize; | ||||||
|  | 		let size = init_size_u.low_u64() as usize; | ||||||
|  | 		if !is_valid_range(off, size) { | ||||||
|  | 			&self[0..0] | ||||||
|  | 		} else { | ||||||
|  | 			&self[off..off+size] | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn read(&self, offset: U256) -> U256 { | ||||||
|  | 		let off = offset.low_u64() as usize; | ||||||
|  | 		U256::from(&self[off..off+32]) | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn writeable_slice(&mut self, offset: U256, size: U256) -> &mut [u8] { | ||||||
|  | 		let off = offset.low_u64() as usize; | ||||||
|  | 		let s = size.low_u64() as usize; | ||||||
|  | 		if !is_valid_range(off, s) { | ||||||
|  | 			&mut self[0..0] | ||||||
|  | 		} else { | ||||||
|  | 			&mut self[off..off+s] | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn write_slice(&mut self, offset: U256, slice: &[u8]) { | ||||||
|  | 		let off = offset.low_u64() as usize; | ||||||
|  | 
 | ||||||
|  | 		// TODO [todr] Optimize?
 | ||||||
|  | 		for pos in off..off+slice.len() { | ||||||
|  | 			self[pos] = slice[pos - off]; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn write(&mut self, offset: U256, value: U256) { | ||||||
|  | 		let off = offset.low_u64() as usize; | ||||||
|  | 		let mut val = value; | ||||||
|  | 
 | ||||||
|  | 		let end = off + 32; | ||||||
|  | 		for pos in 0..32 { | ||||||
|  | 			self[end - pos - 1] = val.low_u64() as u8; | ||||||
|  | 			val = val >> 8; | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn write_byte(&mut self, offset: U256, value: U256) { | ||||||
|  | 		let off = offset.low_u64() as usize; | ||||||
|  | 		let val = value.low_u64() as u64; | ||||||
|  | 		self[off] = val as u8; | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn resize(&mut self, new_size: usize) { | ||||||
|  | 		self.resize(new_size, 0); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn expand(&mut self, size: usize) { | ||||||
|  | 		if size > self.len() { | ||||||
|  | 			Memory::resize(self, size) | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn test_memory_read_and_write() { | ||||||
|  | 	// given
 | ||||||
|  | 	let mem: &mut Memory = &mut vec![]; | ||||||
|  | 	mem.resize(0x80 + 32); | ||||||
|  | 
 | ||||||
|  | 	// when
 | ||||||
|  | 	mem.write(U256::from(0x80), U256::from(0xabcdef)); | ||||||
|  | 
 | ||||||
|  | 	// then
 | ||||||
|  | 	assert_eq!(mem.read(U256::from(0x80)), U256::from(0xabcdef)); | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[test] | ||||||
|  | fn test_memory_read_and_write_byte() { | ||||||
|  | 	// given
 | ||||||
|  | 	let mem: &mut Memory = &mut vec![]; | ||||||
|  | 	mem.resize(32); | ||||||
|  | 
 | ||||||
|  | 	// when
 | ||||||
|  | 	mem.write_byte(U256::from(0x1d), U256::from(0xab)); | ||||||
|  | 	mem.write_byte(U256::from(0x1e), U256::from(0xcd)); | ||||||
|  | 	mem.write_byte(U256::from(0x1f), U256::from(0xef)); | ||||||
|  | 
 | ||||||
|  | 	// then
 | ||||||
|  | 	assert_eq!(mem.read(U256::from(0x00)), U256::from(0xabcdef)); | ||||||
|  | } | ||||||
| @ -16,12 +16,6 @@ | |||||||
| 
 | 
 | ||||||
| ///! Rust VM implementation
 | ///! Rust VM implementation
 | ||||||
| 
 | 
 | ||||||
| use common::*; |  | ||||||
| use super::instructions as instructions; |  | ||||||
| use super::instructions::{Instruction, get_info}; |  | ||||||
| use std::marker::Copy; |  | ||||||
| use evm::{self, MessageCallResult, ContractCreateResult, GasLeft}; |  | ||||||
| 
 |  | ||||||
| #[cfg(not(feature = "evm-debug"))] | #[cfg(not(feature = "evm-debug"))] | ||||||
| macro_rules! evm_debug { | macro_rules! evm_debug { | ||||||
| 	($x: expr) => {} | 	($x: expr) => {} | ||||||
| @ -34,6 +28,19 @@ macro_rules! evm_debug { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | mod gasometer; | ||||||
|  | mod stack; | ||||||
|  | mod memory; | ||||||
|  | 
 | ||||||
|  | use self::gasometer::Gasometer; | ||||||
|  | use self::stack::{Stack, VecStack}; | ||||||
|  | use self::memory::Memory; | ||||||
|  | 
 | ||||||
|  | use std::marker::PhantomData; | ||||||
|  | use common::*; | ||||||
|  | use super::instructions::{self, Instruction, InstructionInfo}; | ||||||
|  | use evm::{self, MessageCallResult, ContractCreateResult, GasLeft, CostType}; | ||||||
|  | 
 | ||||||
| #[cfg(feature = "evm-debug")] | #[cfg(feature = "evm-debug")] | ||||||
| fn color(instruction: Instruction, name: &'static str) -> String { | fn color(instruction: Instruction, name: &'static str) -> String { | ||||||
| 	let c = instruction as usize % 6; | 	let c = instruction as usize % 6; | ||||||
| @ -41,209 +48,9 @@ fn color(instruction: Instruction, name: &'static str) -> String { | |||||||
| 	format!("\x1B[1;{}m{}\x1B[0m", colors[c], name) | 	format!("\x1B[1;{}m{}\x1B[0m", colors[c], name) | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| macro_rules! overflowing { |  | ||||||
| 	($x: expr) => {{ |  | ||||||
| 		let (v, overflow) = $x; |  | ||||||
| 		if overflow { return Err(evm::Error::OutOfGas); } |  | ||||||
| 		v |  | ||||||
| 	}} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| type CodePosition = usize; | type CodePosition = usize; | ||||||
| type Gas = U256; |  | ||||||
| type ProgramCounter = usize; | type ProgramCounter = usize; | ||||||
| 
 | 
 | ||||||
| /// Stack trait with VM-friendly API
 |  | ||||||
| trait Stack<T> { |  | ||||||
| 	/// Returns `Stack[len(Stack) - no_from_top]`
 |  | ||||||
| 	fn peek(&self, no_from_top: usize) -> &T; |  | ||||||
| 	/// Swaps Stack[len(Stack)] and Stack[len(Stack) - no_from_top]
 |  | ||||||
| 	fn swap_with_top(&mut self, no_from_top: usize); |  | ||||||
| 	/// Returns true if Stack has at least `no_of_elems` elements
 |  | ||||||
| 	fn has(&self, no_of_elems: usize) -> bool; |  | ||||||
| 	/// Get element from top and remove it from Stack. Panics if stack is empty.
 |  | ||||||
| 	fn pop_back(&mut self) -> T; |  | ||||||
| 	/// Get (up to `instructions::MAX_NO_OF_TOPICS`) elements from top and remove them from Stack. Panics if stack is empty.
 |  | ||||||
| 	fn pop_n(&mut self, no_of_elems: usize) -> &[T]; |  | ||||||
| 	/// Add element on top of the Stack
 |  | ||||||
| 	fn push(&mut self, elem: T); |  | ||||||
| 	/// Get number of elements on Stack
 |  | ||||||
| 	fn size(&self) -> usize; |  | ||||||
| 	/// Returns all data on stack.
 |  | ||||||
| 	fn peek_top(&mut self, no_of_elems: usize) -> &[T]; |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| struct VecStack<S> { |  | ||||||
| 	stack: Vec<S>, |  | ||||||
| 	logs: [S; instructions::MAX_NO_OF_TOPICS] |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<S : Copy> VecStack<S> { |  | ||||||
| 	fn with_capacity(capacity: usize, zero: S) -> Self { |  | ||||||
| 		VecStack { |  | ||||||
| 			stack: Vec::with_capacity(capacity), |  | ||||||
| 			logs: [zero; instructions::MAX_NO_OF_TOPICS] |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl<S : fmt::Display> Stack<S> for VecStack<S> { |  | ||||||
| 	fn peek(&self, no_from_top: usize) -> &S { |  | ||||||
| 		&self.stack[self.stack.len() - no_from_top - 1] |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn swap_with_top(&mut self, no_from_top: usize) { |  | ||||||
| 		let len = self.stack.len(); |  | ||||||
| 		self.stack.swap(len - no_from_top - 1, len - 1); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn has(&self, no_of_elems: usize) -> bool { |  | ||||||
| 		self.stack.len() >= no_of_elems |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn pop_back(&mut self) -> S { |  | ||||||
| 		let val = self.stack.pop(); |  | ||||||
| 		match val { |  | ||||||
| 			Some(x) => { |  | ||||||
| 				evm_debug!({ |  | ||||||
| 					println!("   POP: {}", x) |  | ||||||
| 				}); |  | ||||||
| 				x |  | ||||||
| 			}, |  | ||||||
| 			None => panic!("Tried to pop from empty stack.") |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn pop_n(&mut self, no_of_elems: usize) -> &[S] { |  | ||||||
| 		assert!(no_of_elems <= instructions::MAX_NO_OF_TOPICS); |  | ||||||
| 
 |  | ||||||
| 		for i in 0..no_of_elems { |  | ||||||
| 			self.logs[i] = self.pop_back(); |  | ||||||
| 		} |  | ||||||
| 		&self.logs[0..no_of_elems] |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn push(&mut self, elem: S) { |  | ||||||
| 		evm_debug!({ |  | ||||||
| 			println!("  PUSH: {}", elem) |  | ||||||
| 		}); |  | ||||||
| 		self.stack.push(elem); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn size(&self) -> usize { |  | ||||||
| 		self.stack.len() |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn peek_top(&mut self, no_from_top: usize) -> &[S] { |  | ||||||
| 		assert!(self.stack.len() >= no_from_top, "peek_top asked for more items than exist."); |  | ||||||
| 		&self.stack[self.stack.len() - no_from_top .. self.stack.len()] |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| trait Memory { |  | ||||||
| 	/// Retrieve current size of the memory
 |  | ||||||
| 	fn size(&self) -> usize; |  | ||||||
| 	/// Resize (shrink or expand) the memory to specified size (fills 0)
 |  | ||||||
| 	fn resize(&mut self, new_size: usize); |  | ||||||
| 	/// Resize the memory only if its smaller
 |  | ||||||
| 	fn expand(&mut self, new_size: usize); |  | ||||||
| 	/// Write single byte to memory
 |  | ||||||
| 	fn write_byte(&mut self, offset: U256, value: U256); |  | ||||||
| 	/// Write a word to memory. Does not resize memory!
 |  | ||||||
| 	fn write(&mut self, offset: U256, value: U256); |  | ||||||
| 	/// Read a word from memory
 |  | ||||||
| 	fn read(&self, offset: U256) -> U256; |  | ||||||
| 	/// Write slice of bytes to memory. Does not resize memory!
 |  | ||||||
| 	fn write_slice(&mut self, offset: U256, &[u8]); |  | ||||||
| 	/// Retrieve part of the memory between offset and offset + size
 |  | ||||||
| 	fn read_slice(&self, offset: U256, size: U256) -> &[u8]; |  | ||||||
| 	/// Retrieve writeable part of memory
 |  | ||||||
| 	fn writeable_slice(&mut self, offset: U256, size: U256) -> &mut[u8]; |  | ||||||
| 	fn dump(&self); |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// Checks whether offset and size is valid memory range
 |  | ||||||
| fn is_valid_range(off: usize, size: usize)  -> bool { |  | ||||||
| 	// When size is zero we haven't actually expanded the memory
 |  | ||||||
| 	let overflow = off.overflowing_add(size).1; |  | ||||||
| 	size > 0 && !overflow |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| impl Memory for Vec<u8> { |  | ||||||
| 	fn dump(&self) { |  | ||||||
| 		println!("MemoryDump:"); |  | ||||||
| 		for i in self.iter() { |  | ||||||
| 			println!("{:02x} ", i); |  | ||||||
| 		} |  | ||||||
| 		println!(""); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn size(&self) -> usize { |  | ||||||
| 		self.len() |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn read_slice(&self, init_off_u: U256, init_size_u: U256) -> &[u8] { |  | ||||||
| 		let off = init_off_u.low_u64() as usize; |  | ||||||
| 		let size = init_size_u.low_u64() as usize; |  | ||||||
| 		if !is_valid_range(off, size) { |  | ||||||
| 			&self[0..0] |  | ||||||
| 		} else { |  | ||||||
| 			&self[off..off+size] |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn read(&self, offset: U256) -> U256 { |  | ||||||
| 		let off = offset.low_u64() as usize; |  | ||||||
| 		U256::from(&self[off..off+32]) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn writeable_slice(&mut self, offset: U256, size: U256) -> &mut [u8] { |  | ||||||
| 		let off = offset.low_u64() as usize; |  | ||||||
| 		let s = size.low_u64() as usize; |  | ||||||
| 		if !is_valid_range(off, s) { |  | ||||||
| 			&mut self[0..0] |  | ||||||
| 		} else { |  | ||||||
| 			&mut self[off..off+s] |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn write_slice(&mut self, offset: U256, slice: &[u8]) { |  | ||||||
| 		let off = offset.low_u64() as usize; |  | ||||||
| 
 |  | ||||||
| 		// TODO [todr] Optimize?
 |  | ||||||
| 		for pos in off..off+slice.len() { |  | ||||||
| 			self[pos] = slice[pos - off]; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn write(&mut self, offset: U256, value: U256) { |  | ||||||
| 		let off = offset.low_u64() as usize; |  | ||||||
| 		let mut val = value; |  | ||||||
| 
 |  | ||||||
| 		let end = off + 32; |  | ||||||
| 		for pos in 0..32 { |  | ||||||
| 			self[end - pos - 1] = val.low_u64() as u8; |  | ||||||
| 			val = val >> 8; |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn write_byte(&mut self, offset: U256, value: U256) { |  | ||||||
| 		let off = offset.low_u64() as usize; |  | ||||||
| 		let val = value.low_u64() as u64; |  | ||||||
| 		self[off] = val as u8; |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn resize(&mut self, new_size: usize) { |  | ||||||
| 		self.resize(new_size, 0); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn expand(&mut self, size: usize) { |  | ||||||
| 		if size > self.len() { |  | ||||||
| 			Memory::resize(self, size) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| /// Abstraction over raw vector of Bytes. Easier state management of PC.
 | /// Abstraction over raw vector of Bytes. Easier state management of PC.
 | ||||||
| struct CodeReader<'a> { | struct CodeReader<'a> { | ||||||
| 	position: ProgramCounter, | 	position: ProgramCounter, | ||||||
| @ -265,38 +72,33 @@ impl<'a> CodeReader<'a> { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[cfg_attr(feature="dev", allow(enum_variant_names))] | enum InstructionResult<Gas> { | ||||||
| enum InstructionCost { |  | ||||||
| 	Gas(U256), |  | ||||||
| 	GasMem(U256, U256), |  | ||||||
| 	GasMemCopy(U256, U256, U256) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| enum InstructionResult { |  | ||||||
| 	Ok, | 	Ok, | ||||||
| 	UseAllGas, | 	UseAllGas, | ||||||
| 	GasLeft(U256), | 	GasLeft(Gas), | ||||||
| 	UnusedGas(U256), | 	UnusedGas(Gas), | ||||||
| 	JumpToPosition(U256), | 	JumpToPosition(U256), | ||||||
| 	// gas left, init_orf, init_size
 | 	// gas left, init_orf, init_size
 | ||||||
| 	StopExecutionNeedsReturn(U256, U256, U256), | 	StopExecutionNeedsReturn(Gas, U256, U256), | ||||||
| 	StopExecution, | 	StopExecution, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| /// Intepreter EVM implementation
 | /// Intepreter EVM implementation
 | ||||||
| #[derive(Default)] | #[derive(Default)] | ||||||
| pub struct Interpreter { | pub struct Interpreter<Cost: CostType> { | ||||||
| 	mem: Vec<u8>, | 	mem: Vec<u8>, | ||||||
|  | 	_type: PhantomData<Cost>, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl evm::Evm for Interpreter { | impl<Cost: CostType> evm::Evm for Interpreter<Cost> { | ||||||
| 	fn exec(&mut self, params: ActionParams, ext: &mut evm::Ext) -> evm::Result<GasLeft> { | 	fn exec(&mut self, params: ActionParams, ext: &mut evm::Ext) -> evm::Result<GasLeft> { | ||||||
| 		self.mem.clear(); | 		self.mem.clear(); | ||||||
| 
 | 
 | ||||||
| 		let code = ¶ms.code.as_ref().unwrap(); | 		let code = ¶ms.code.as_ref().unwrap(); | ||||||
| 		let valid_jump_destinations = self.find_jump_destinations(&code); | 		let valid_jump_destinations = self.find_jump_destinations(&code); | ||||||
| 
 | 
 | ||||||
| 		let mut current_gas = params.gas; | 		let mut gasometer = Gasometer::<Cost>::new(try!(Cost::from_u256(params.gas))); | ||||||
| 		let mut stack = VecStack::with_capacity(ext.schedule().stack_limit, U256::zero()); | 		let mut stack = VecStack::with_capacity(ext.schedule().stack_limit, U256::zero()); | ||||||
| 		let mut reader = CodeReader { | 		let mut reader = CodeReader { | ||||||
| 			position: 0, | 			position: 0, | ||||||
| @ -305,26 +107,27 @@ impl evm::Evm for Interpreter { | |||||||
| 
 | 
 | ||||||
| 		while reader.position < code.len() { | 		while reader.position < code.len() { | ||||||
| 			let instruction = code[reader.position]; | 			let instruction = code[reader.position]; | ||||||
| 
 |  | ||||||
| 			// Calculate gas cost
 |  | ||||||
| 			let (gas_cost, mem_size) = try!(self.get_gas_cost_mem(ext, instruction, &stack)); |  | ||||||
| 
 |  | ||||||
| 			// TODO: make compile-time removable if too much of a performance hit.
 |  | ||||||
| 			let trace_executed = ext.trace_prepare_execute(reader.position, instruction, &gas_cost); |  | ||||||
| 
 |  | ||||||
| 			reader.position += 1; | 			reader.position += 1; | ||||||
| 
 | 
 | ||||||
| 			try!(self.verify_gas(¤t_gas, &gas_cost)); | 			let info = instructions::get_info(instruction); | ||||||
|  | 			try!(self.verify_instruction(ext, instruction, &info, &stack)); | ||||||
|  | 
 | ||||||
|  | 			// Calculate gas cost
 | ||||||
|  | 			let (gas_cost, mem_size) = try!(gasometer.get_gas_cost_mem(ext, instruction, &info, &stack, self.mem.size())); | ||||||
|  | 			// TODO: make compile-time removable if too much of a performance hit.
 | ||||||
|  | 			let trace_executed = ext.trace_prepare_execute(reader.position - 1, instruction, &gas_cost.as_u256()); | ||||||
|  | 
 | ||||||
|  | 			try!(gasometer.verify_gas(&gas_cost)); | ||||||
| 			self.mem.expand(mem_size); | 			self.mem.expand(mem_size); | ||||||
| 			current_gas = current_gas - gas_cost; //TODO: use operator -=
 | 			gasometer.current_gas = gasometer.current_gas - gas_cost; | ||||||
| 
 | 
 | ||||||
| 			evm_debug!({ | 			evm_debug!({ | ||||||
| 				println!("[0x{:x}][{}(0x{:x}) Gas: {:x}\n  Gas Before: {:x}", | 				println!("[0x{:x}][{}(0x{:x}) Gas: {:x}\n  Gas Before: {:x}", | ||||||
| 					reader.position, | 					reader.position, | ||||||
| 					color(instruction, instructions::get_info(instruction).name), | 					color(instruction, info.name), | ||||||
| 					instruction, | 					instruction, | ||||||
| 					gas_cost, | 					gas_cost, | ||||||
| 					current_gas + gas_cost | 					gasometer.current_gas + gas_cost | ||||||
| 				); | 				); | ||||||
| 			}); | 			}); | ||||||
| 
 | 
 | ||||||
| @ -335,50 +138,44 @@ impl evm::Evm for Interpreter { | |||||||
| 
 | 
 | ||||||
| 			// Execute instruction
 | 			// Execute instruction
 | ||||||
| 			let result = try!(self.exec_instruction( | 			let result = try!(self.exec_instruction( | ||||||
| 				current_gas, ¶ms, ext, instruction, &mut reader, &mut stack | 				gasometer.current_gas, ¶ms, ext, instruction, &mut reader, &mut stack | ||||||
| 			)); | 			)); | ||||||
| 
 | 
 | ||||||
| 			if trace_executed { | 			if trace_executed { | ||||||
| 				ext.trace_executed(current_gas, stack.peek_top(get_info(instruction).ret), mem_written.map(|(o, s)| (o, &(self.mem[o..(o + s)]))), store_written); | 				ext.trace_executed(gasometer.current_gas.as_u256(), stack.peek_top(info.ret), mem_written.map(|(o, s)| (o, &(self.mem[o..(o + s)]))), store_written); | ||||||
| 			} | 			} | ||||||
| 
 | 
 | ||||||
| 			// Advance
 | 			// Advance
 | ||||||
| 			match result { | 			match result { | ||||||
| 				InstructionResult::Ok => {}, | 				InstructionResult::Ok => {}, | ||||||
| 				InstructionResult::UnusedGas(gas) => { | 				InstructionResult::UnusedGas(gas) => { | ||||||
| 					current_gas = current_gas + gas; //TODO: use operator +=
 | 					gasometer.current_gas = gasometer.current_gas + gas; | ||||||
| 				}, | 				}, | ||||||
| 				InstructionResult::UseAllGas => { | 				InstructionResult::UseAllGas => { | ||||||
| 					current_gas = U256::zero(); | 					gasometer.current_gas = Cost::from(0); | ||||||
| 				}, | 				}, | ||||||
| 				InstructionResult::GasLeft(gas_left) => { | 				InstructionResult::GasLeft(gas_left) => { | ||||||
| 					current_gas = gas_left; | 					gasometer.current_gas = gas_left; | ||||||
| 				}, | 				}, | ||||||
| 				InstructionResult::JumpToPosition(position) => { | 				InstructionResult::JumpToPosition(position) => { | ||||||
| 					let pos = try!(self.verify_jump(position, &valid_jump_destinations)); | 					let pos = try!(self.verify_jump(position, &valid_jump_destinations)); | ||||||
| 					reader.position = pos; | 					reader.position = pos; | ||||||
| 				}, | 				}, | ||||||
| 				InstructionResult::StopExecutionNeedsReturn(gas, off, size) => { | 				InstructionResult::StopExecutionNeedsReturn(gas, off, size) => { | ||||||
| 					return Ok(GasLeft::NeedsReturn(gas, self.mem.read_slice(off, size))); | 					return Ok(GasLeft::NeedsReturn(gas.as_u256(), self.mem.read_slice(off, size))); | ||||||
| 				}, | 				}, | ||||||
| 				InstructionResult::StopExecution => break, | 				InstructionResult::StopExecution => break, | ||||||
| 			} | 			} | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		Ok(GasLeft::Known(current_gas)) | 		Ok(GasLeft::Known(gasometer.current_gas.as_u256())) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl Interpreter { | impl<Cost: CostType> Interpreter<Cost> { | ||||||
| 	#[cfg_attr(feature="dev", allow(cyclomatic_complexity))] | 
 | ||||||
| 	fn get_gas_cost_mem( | 	fn verify_instruction(&self, ext: &evm::Ext, instruction: Instruction, info: &InstructionInfo, stack: &Stack<U256>) -> evm::Result<()> { | ||||||
| 		&mut self, |  | ||||||
| 		ext: &evm::Ext, |  | ||||||
| 		instruction: Instruction, |  | ||||||
| 		stack: &Stack<U256> |  | ||||||
| 	) -> evm::Result<(U256, usize)> { |  | ||||||
| 		let schedule = ext.schedule(); | 		let schedule = ext.schedule(); | ||||||
| 		let info = instructions::get_info(instruction); |  | ||||||
| 
 | 
 | ||||||
| 		if !schedule.have_delegate_call && instruction == instructions::DELEGATECALL { | 		if !schedule.have_delegate_call && instruction == instructions::DELEGATECALL { | ||||||
| 			return Err(evm::Error::BadInstruction { | 			return Err(evm::Error::BadInstruction { | ||||||
| @ -391,119 +188,20 @@ impl Interpreter { | |||||||
| 			}); | 			}); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		try!(self.verify_instructions_requirements(&info, schedule.stack_limit, stack)); | 		if !stack.has(info.args) { | ||||||
| 
 | 			Err(evm::Error::StackUnderflow { | ||||||
| 		let tier = instructions::get_tier_idx(info.tier); | 				instruction: info.name, | ||||||
| 		let default_gas = U256::from(schedule.tier_step_gas[tier]); | 				wanted: info.args, | ||||||
| 
 | 				on_stack: stack.size() | ||||||
| 		let cost = match instruction { | 			}) | ||||||
| 			instructions::SSTORE => { | 		} else if stack.size() - info.args + info.ret > schedule.stack_limit { | ||||||
| 				let address = H256::from(stack.peek(0)); | 			Err(evm::Error::OutOfStack { | ||||||
| 				let newval = stack.peek(1); | 				instruction: info.name, | ||||||
| 				let val = U256::from(ext.storage_at(&address).as_slice()); | 				wanted: info.ret - info.args, | ||||||
| 
 | 				limit: schedule.stack_limit | ||||||
| 				let gas = if self.is_zero(&val) && !self.is_zero(newval) { | 			}) | ||||||
| 					schedule.sstore_set_gas |  | ||||||
| 		} else { | 		} else { | ||||||
| 					// Refund for below case is added when actually executing sstore
 | 			Ok(()) | ||||||
| 					// !self.is_zero(&val) && self.is_zero(newval)
 |  | ||||||
| 					schedule.sstore_reset_gas |  | ||||||
| 				}; |  | ||||||
| 				InstructionCost::Gas(U256::from(gas)) |  | ||||||
| 			}, |  | ||||||
| 			instructions::SLOAD => { |  | ||||||
| 				InstructionCost::Gas(U256::from(schedule.sload_gas)) |  | ||||||
| 			}, |  | ||||||
| 			instructions::MSTORE | instructions::MLOAD => { |  | ||||||
| 				InstructionCost::GasMem(default_gas, try!(self.mem_needed_const(stack.peek(0), 32))) |  | ||||||
| 			}, |  | ||||||
| 			instructions::MSTORE8 => { |  | ||||||
| 				InstructionCost::GasMem(default_gas, try!(self.mem_needed_const(stack.peek(0), 1))) |  | ||||||
| 			}, |  | ||||||
| 			instructions::RETURN => { |  | ||||||
| 				InstructionCost::GasMem(default_gas, try!(self.mem_needed(stack.peek(0), stack.peek(1)))) |  | ||||||
| 			}, |  | ||||||
| 			instructions::SHA3 => { |  | ||||||
| 				let w = overflowing!(add_u256_usize(stack.peek(1), 31)); |  | ||||||
| 				let words = w >> 5; |  | ||||||
| 				let gas = U256::from(schedule.sha3_gas) + (U256::from(schedule.sha3_word_gas) * words); |  | ||||||
| 				InstructionCost::GasMem(gas, try!(self.mem_needed(stack.peek(0), stack.peek(1)))) |  | ||||||
| 			}, |  | ||||||
| 			instructions::CALLDATACOPY | instructions::CODECOPY => { |  | ||||||
| 				InstructionCost::GasMemCopy(default_gas, try!(self.mem_needed(stack.peek(0), stack.peek(2))), stack.peek(2).clone()) |  | ||||||
| 			}, |  | ||||||
| 			instructions::EXTCODECOPY => { |  | ||||||
| 				InstructionCost::GasMemCopy(default_gas, try!(self.mem_needed(stack.peek(1), stack.peek(3))), stack.peek(3).clone()) |  | ||||||
| 			}, |  | ||||||
| 			instructions::JUMPDEST => { |  | ||||||
| 				InstructionCost::Gas(U256::one()) |  | ||||||
| 			}, |  | ||||||
| 			instructions::LOG0...instructions::LOG4 => { |  | ||||||
| 				let no_of_topics = instructions::get_log_topics(instruction); |  | ||||||
| 				let log_gas = schedule.log_gas + schedule.log_topic_gas * no_of_topics; |  | ||||||
| 
 |  | ||||||
| 				let data_gas = overflowing!(stack.peek(1).overflowing_mul(U256::from(schedule.log_data_gas))); |  | ||||||
| 				let gas = overflowing!(data_gas.overflowing_add(U256::from(log_gas))); |  | ||||||
| 				InstructionCost::GasMem(gas, try!(self.mem_needed(stack.peek(0), stack.peek(1)))) |  | ||||||
| 			}, |  | ||||||
| 			instructions::CALL | instructions::CALLCODE => { |  | ||||||
| 				let mut gas  = overflowing!(add_u256_usize(stack.peek(0), schedule.call_gas)); |  | ||||||
| 				let mem = cmp::max( |  | ||||||
| 					try!(self.mem_needed(stack.peek(5), stack.peek(6))), |  | ||||||
| 					try!(self.mem_needed(stack.peek(3), stack.peek(4))) |  | ||||||
| 				); |  | ||||||
| 
 |  | ||||||
| 				let address = u256_to_address(stack.peek(1)); |  | ||||||
| 
 |  | ||||||
| 				if instruction == instructions::CALL && !ext.exists(&address) { |  | ||||||
| 					gas = overflowing!(gas.overflowing_add(U256::from(schedule.call_new_account_gas))); |  | ||||||
| 				}; |  | ||||||
| 
 |  | ||||||
| 				if stack.peek(2).clone() > U256::zero() { |  | ||||||
| 					gas = overflowing!(gas.overflowing_add(U256::from(schedule.call_value_transfer_gas))); |  | ||||||
| 				}; |  | ||||||
| 
 |  | ||||||
| 				InstructionCost::GasMem(gas,mem) |  | ||||||
| 			}, |  | ||||||
| 			instructions::DELEGATECALL => { |  | ||||||
| 				let gas = overflowing!(add_u256_usize(stack.peek(0), schedule.call_gas)); |  | ||||||
| 				let mem = cmp::max( |  | ||||||
| 					try!(self.mem_needed(stack.peek(4), stack.peek(5))), |  | ||||||
| 					try!(self.mem_needed(stack.peek(2), stack.peek(3))) |  | ||||||
| 				); |  | ||||||
| 				InstructionCost::GasMem(gas, mem) |  | ||||||
| 			}, |  | ||||||
| 			instructions::CREATE => { |  | ||||||
| 				let gas = U256::from(schedule.create_gas); |  | ||||||
| 				let mem = try!(self.mem_needed(stack.peek(1), stack.peek(2))); |  | ||||||
| 				InstructionCost::GasMem(gas, mem) |  | ||||||
| 			}, |  | ||||||
| 			instructions::EXP => { |  | ||||||
| 				let expon = stack.peek(1); |  | ||||||
| 				let bytes = ((expon.bits() + 7) / 8) as usize; |  | ||||||
| 				let gas = U256::from(schedule.exp_gas + schedule.exp_byte_gas * bytes); |  | ||||||
| 				InstructionCost::Gas(gas) |  | ||||||
| 			}, |  | ||||||
| 			_ => InstructionCost::Gas(default_gas) |  | ||||||
| 		}; |  | ||||||
| 
 |  | ||||||
| 		match cost { |  | ||||||
| 			InstructionCost::Gas(gas) => { |  | ||||||
| 				Ok((gas, 0)) |  | ||||||
| 			}, |  | ||||||
| 			InstructionCost::GasMem(gas, mem_size) => { |  | ||||||
| 				let (mem_gas, new_mem_size) = try!(self.mem_gas_cost(schedule, self.mem.size(), &mem_size)); |  | ||||||
| 				let gas = overflowing!(gas.overflowing_add(mem_gas)); |  | ||||||
| 				Ok((gas, new_mem_size)) |  | ||||||
| 			}, |  | ||||||
| 			InstructionCost::GasMemCopy(gas, mem_size, copy) => { |  | ||||||
| 				let (mem_gas, new_mem_size) = try!(self.mem_gas_cost(schedule, self.mem.size(), &mem_size)); |  | ||||||
| 				let copy = overflowing!(add_u256_usize(©, 31)); |  | ||||||
| 				let copy_gas = U256::from(schedule.copy_gas) * (copy / U256::from(32)); |  | ||||||
| 				let gas = overflowing!(gas.overflowing_add(copy_gas)); |  | ||||||
| 				let gas = overflowing!(gas.overflowing_add(mem_gas)); |  | ||||||
| 				Ok((gas, new_mem_size)) |  | ||||||
| 			} |  | ||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| @ -532,53 +230,16 @@ impl Interpreter { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	fn mem_gas_cost(&self, schedule: &evm::Schedule, current_mem_size: usize, mem_size: &U256) -> evm::Result<(U256, usize)> { |  | ||||||
| 		let gas_for_mem = |mem_size: U256| { |  | ||||||
| 			let s = mem_size >> 5; |  | ||||||
| 			// s * memory_gas + s * s / quad_coeff_div
 |  | ||||||
| 			let a = overflowing!(s.overflowing_mul(U256::from(schedule.memory_gas))); |  | ||||||
| 			// We need to go to U512 to calculate s*s/quad_coeff_div
 |  | ||||||
| 			let b = U512::from(s) * U512::from(s) / U512::from(schedule.quad_coeff_div); |  | ||||||
| 			if b > U512::from(!U256::zero()) { |  | ||||||
| 				Err(evm::Error::OutOfGas) |  | ||||||
| 			} else { |  | ||||||
| 				Ok(overflowing!(a.overflowing_add(U256::from(b)))) |  | ||||||
| 			} |  | ||||||
| 		}; |  | ||||||
| 		let current_mem_size = U256::from(current_mem_size); |  | ||||||
| 		let req_mem_size_rounded = (overflowing!(mem_size.overflowing_add(U256::from(31))) >> 5) << 5; |  | ||||||
| 		let new_mem_gas = try!(gas_for_mem(U256::from(req_mem_size_rounded))); |  | ||||||
| 		let current_mem_gas = try!(gas_for_mem(current_mem_size)); |  | ||||||
| 
 |  | ||||||
| 		Ok((if req_mem_size_rounded > current_mem_size { |  | ||||||
| 			new_mem_gas - current_mem_gas |  | ||||||
| 		} else { |  | ||||||
| 			U256::zero() |  | ||||||
| 		}, req_mem_size_rounded.low_u64() as usize)) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn mem_needed_const(&self, mem: &U256, add: usize) -> evm::Result<U256> { |  | ||||||
| 		Ok(overflowing!(mem.overflowing_add(U256::from(add)))) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn mem_needed(&self, offset: &U256, size: &U256) -> evm::Result<U256> { |  | ||||||
| 		if self.is_zero(size) { |  | ||||||
| 			return Ok(U256::zero()); |  | ||||||
| 		} |  | ||||||
| 
 |  | ||||||
| 		Ok(overflowing!(offset.overflowing_add(size.clone()))) |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	#[cfg_attr(feature="dev", allow(too_many_arguments))] | 	#[cfg_attr(feature="dev", allow(too_many_arguments))] | ||||||
| 	fn exec_instruction( | 	fn exec_instruction( | ||||||
| 		&mut self, | 		&mut self, | ||||||
| 		gas: Gas, | 		gas: Cost, | ||||||
| 		params: &ActionParams, | 		params: &ActionParams, | ||||||
| 		ext: &mut evm::Ext, | 		ext: &mut evm::Ext, | ||||||
| 		instruction: Instruction, | 		instruction: Instruction, | ||||||
| 		code: &mut CodeReader, | 		code: &mut CodeReader, | ||||||
| 		stack: &mut Stack<U256> | 		stack: &mut Stack<U256> | ||||||
| 	) -> evm::Result<InstructionResult> { | 	) -> evm::Result<InstructionResult<Cost>> { | ||||||
| 		match instruction { | 		match instruction { | ||||||
| 			instructions::JUMP => { | 			instructions::JUMP => { | ||||||
| 				let jump = stack.pop_back(); | 				let jump = stack.pop_back(); | ||||||
| @ -611,11 +272,11 @@ impl Interpreter { | |||||||
| 					return Ok(InstructionResult::Ok); | 					return Ok(InstructionResult::Ok); | ||||||
| 				} | 				} | ||||||
| 
 | 
 | ||||||
| 				let create_result = ext.create(&gas, &endowment, &contract_code); | 				let create_result = ext.create(&gas.as_u256(), &endowment, &contract_code); | ||||||
| 				return match create_result { | 				return match create_result { | ||||||
| 					ContractCreateResult::Created(address, gas_left) => { | 					ContractCreateResult::Created(address, gas_left) => { | ||||||
| 						stack.push(address_to_u256(address)); | 						stack.push(address_to_u256(address)); | ||||||
| 						Ok(InstructionResult::GasLeft(gas_left)) | 						Ok(InstructionResult::GasLeft(Cost::from_u256(gas_left).expect("Gas left cannot be greater."))) | ||||||
| 					}, | 					}, | ||||||
| 					ContractCreateResult::Failed => { | 					ContractCreateResult::Failed => { | ||||||
| 						stack.push(U256::zero()); | 						stack.push(U256::zero()); | ||||||
| @ -626,7 +287,7 @@ impl Interpreter { | |||||||
| 			}, | 			}, | ||||||
| 			instructions::CALL | instructions::CALLCODE | instructions::DELEGATECALL => { | 			instructions::CALL | instructions::CALLCODE | instructions::DELEGATECALL => { | ||||||
| 				assert!(ext.schedule().call_value_transfer_gas > ext.schedule().call_stipend, "overflow possible"); | 				assert!(ext.schedule().call_value_transfer_gas > ext.schedule().call_stipend, "overflow possible"); | ||||||
| 				let call_gas = stack.pop_back(); | 				let call_gas = Cost::from_u256(stack.pop_back()).expect("Gas is already validated."); | ||||||
| 				let code_address = stack.pop_back(); | 				let code_address = stack.pop_back(); | ||||||
| 				let code_address = u256_to_address(&code_address); | 				let code_address = u256_to_address(&code_address); | ||||||
| 
 | 
 | ||||||
| @ -642,9 +303,9 @@ impl Interpreter { | |||||||
| 				let out_size = stack.pop_back(); | 				let out_size = stack.pop_back(); | ||||||
| 
 | 
 | ||||||
| 				// Add stipend (only CALL|CALLCODE when value > 0)
 | 				// Add stipend (only CALL|CALLCODE when value > 0)
 | ||||||
| 				let call_gas = call_gas + value.map_or_else(U256::zero, |val| match val > U256::zero() { | 				let call_gas = call_gas + value.map_or_else(|| Cost::from(0), |val| match val > U256::zero() { | ||||||
| 					true => U256::from(ext.schedule().call_stipend), | 					true => Cost::from(ext.schedule().call_stipend), | ||||||
| 					false => U256::zero() | 					false => Cost::from(0) | ||||||
| 				}); | 				}); | ||||||
| 
 | 
 | ||||||
| 				// Get sender & receive addresses, check if we have balance
 | 				// Get sender & receive addresses, check if we have balance
 | ||||||
| @ -672,13 +333,13 @@ impl Interpreter { | |||||||
| 					// and we don't want to copy
 | 					// and we don't want to copy
 | ||||||
| 					let input = unsafe { ::std::mem::transmute(self.mem.read_slice(in_off, in_size)) }; | 					let input = unsafe { ::std::mem::transmute(self.mem.read_slice(in_off, in_size)) }; | ||||||
| 					let output = self.mem.writeable_slice(out_off, out_size); | 					let output = self.mem.writeable_slice(out_off, out_size); | ||||||
| 					ext.call(&call_gas, sender_address, receive_address, value, input, &code_address, output) | 					ext.call(&call_gas.as_u256(), sender_address, receive_address, value, input, &code_address, output) | ||||||
| 				}; | 				}; | ||||||
| 
 | 
 | ||||||
| 				return match call_result { | 				return match call_result { | ||||||
| 					MessageCallResult::Success(gas_left) => { | 					MessageCallResult::Success(gas_left) => { | ||||||
| 						stack.push(U256::one()); | 						stack.push(U256::one()); | ||||||
| 						Ok(InstructionResult::UnusedGas(gas_left)) | 						Ok(InstructionResult::UnusedGas(Cost::from_u256(gas_left).expect("Gas left cannot be greater then current one"))) | ||||||
| 					}, | 					}, | ||||||
| 					MessageCallResult::Failed  => { | 					MessageCallResult::Failed  => { | ||||||
| 						stack.push(U256::zero()); | 						stack.push(U256::zero()); | ||||||
| @ -759,7 +420,7 @@ impl Interpreter { | |||||||
| 				stack.push(U256::from(code.position - 1)); | 				stack.push(U256::from(code.position - 1)); | ||||||
| 			}, | 			}, | ||||||
| 			instructions::GAS => { | 			instructions::GAS => { | ||||||
| 				stack.push(gas.clone()); | 				stack.push(gas.as_u256()); | ||||||
| 			}, | 			}, | ||||||
| 			instructions::ADDRESS => { | 			instructions::ADDRESS => { | ||||||
| 				stack.push(address_to_u256(params.address.clone())); | 				stack.push(address_to_u256(params.address.clone())); | ||||||
| @ -876,36 +537,6 @@ impl Interpreter { | |||||||
| 		} | 		} | ||||||
| 	} | 	} | ||||||
| 
 | 
 | ||||||
| 	fn verify_instructions_requirements( |  | ||||||
| 		&self, |  | ||||||
| 		info: &instructions::InstructionInfo, |  | ||||||
| 		stack_limit: usize, |  | ||||||
| 		stack: &Stack<U256> |  | ||||||
| 	) -> evm::Result<()> { |  | ||||||
| 		if !stack.has(info.args) { |  | ||||||
| 			Err(evm::Error::StackUnderflow { |  | ||||||
| 				instruction: info.name, |  | ||||||
| 				wanted: info.args, |  | ||||||
| 				on_stack: stack.size() |  | ||||||
| 			}) |  | ||||||
| 		} else if stack.size() - info.args + info.ret > stack_limit { |  | ||||||
| 			Err(evm::Error::OutOfStack { |  | ||||||
| 				instruction: info.name, |  | ||||||
| 				wanted: info.ret - info.args, |  | ||||||
| 				limit: stack_limit |  | ||||||
| 			}) |  | ||||||
| 		} else { |  | ||||||
| 			Ok(()) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn verify_gas(&self, current_gas: &U256, gas_cost: &U256) -> evm::Result<()> { |  | ||||||
| 		match current_gas < gas_cost { |  | ||||||
| 			true => Err(evm::Error::OutOfGas), |  | ||||||
| 			false => Ok(()) |  | ||||||
| 		} |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	fn verify_jump(&self, jump_u: U256, valid_jump_destinations: &HashSet<usize>) -> evm::Result<usize> { | 	fn verify_jump(&self, jump_u: U256, valid_jump_destinations: &HashSet<usize>) -> evm::Result<usize> { | ||||||
| 		let jump = jump_u.low_u64() as usize; | 		let jump = jump_u.low_u64() as usize; | ||||||
| 
 | 
 | ||||||
| @ -1163,11 +794,6 @@ fn set_sign(value: U256, sign: bool) -> U256 { | |||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[inline] |  | ||||||
| fn add_u256_usize(value: &U256, num: usize) -> (U256, bool) { |  | ||||||
| 	value.clone().overflowing_add(U256::from(num)) |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[inline] | #[inline] | ||||||
| fn u256_to_address(value: &U256) -> Address { | fn u256_to_address(value: &U256) -> Address { | ||||||
| 	Address::from(H256::from(value)) | 	Address::from(H256::from(value)) | ||||||
| @ -1179,32 +805,9 @@ fn address_to_u256(value: Address) -> U256 { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[test] | #[test] | ||||||
| fn test_mem_gas_cost() { | fn test_find_jump_destinations() { | ||||||
| 	// given
 | 	// given
 | ||||||
| 	let interpreter = Interpreter::default(); | 	let interpreter = Interpreter::<U256>::default(); | ||||||
| 	let schedule = evm::Schedule::default(); |  | ||||||
| 	let current_mem_size = 5; |  | ||||||
| 	let mem_size = !U256::zero(); |  | ||||||
| 
 |  | ||||||
| 	// when
 |  | ||||||
| 	let result = interpreter.mem_gas_cost(&schedule, current_mem_size, &mem_size); |  | ||||||
| 
 |  | ||||||
| 	// then
 |  | ||||||
| 	if let Ok(_) = result { |  | ||||||
| 		assert!(false, "Should fail with OutOfGas"); |  | ||||||
| 	} |  | ||||||
| } |  | ||||||
| 
 |  | ||||||
| #[cfg(test)] |  | ||||||
| mod tests { |  | ||||||
| 	use common::*; |  | ||||||
| 	use super::*; |  | ||||||
| 	use evm; |  | ||||||
| 
 |  | ||||||
| 	#[test] |  | ||||||
| 	fn test_find_jump_destinations() { |  | ||||||
| 		// given
 |  | ||||||
| 		let interpreter = Interpreter::default(); |  | ||||||
| 	let code = "7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff5b01600055".from_hex().unwrap(); | 	let code = "7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff7fffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffffff5b01600055".from_hex().unwrap(); | ||||||
| 
 | 
 | ||||||
| 	// when
 | 	// when
 | ||||||
| @ -1212,49 +815,4 @@ mod tests { | |||||||
| 
 | 
 | ||||||
| 	// then
 | 	// then
 | ||||||
| 	assert!(valid_jump_destinations.contains(&66)); | 	assert!(valid_jump_destinations.contains(&66)); | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	#[test] |  | ||||||
| 	fn test_calculate_mem_cost() { |  | ||||||
| 		// given
 |  | ||||||
| 		let interpreter = Interpreter::default(); |  | ||||||
| 		let schedule = evm::Schedule::default(); |  | ||||||
| 		let current_mem_size = 0; |  | ||||||
| 		let mem_size = U256::from(5); |  | ||||||
| 
 |  | ||||||
| 		// when
 |  | ||||||
| 		let (mem_cost, mem_size) = interpreter.mem_gas_cost(&schedule, current_mem_size, &mem_size).unwrap(); |  | ||||||
| 
 |  | ||||||
| 		// then
 |  | ||||||
| 		assert_eq!(mem_cost, U256::from(3)); |  | ||||||
| 		assert_eq!(mem_size, 32); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	#[test] |  | ||||||
| 	fn test_memory_read_and_write() { |  | ||||||
| 		// given
 |  | ||||||
| 		let mem: &mut super::Memory = &mut vec![]; |  | ||||||
| 		mem.resize(0x80 + 32); |  | ||||||
| 
 |  | ||||||
| 		// when
 |  | ||||||
| 		mem.write(U256::from(0x80), U256::from(0xabcdef)); |  | ||||||
| 
 |  | ||||||
| 		// then
 |  | ||||||
| 		assert_eq!(mem.read(U256::from(0x80)), U256::from(0xabcdef)); |  | ||||||
| 	} |  | ||||||
| 
 |  | ||||||
| 	#[test] |  | ||||||
| 	fn test_memory_read_and_write_byte() { |  | ||||||
| 		// given
 |  | ||||||
| 		let mem: &mut super::Memory = &mut vec![]; |  | ||||||
| 		mem.resize(32); |  | ||||||
| 
 |  | ||||||
| 		// when
 |  | ||||||
| 		mem.write_byte(U256::from(0x1d), U256::from(0xab)); |  | ||||||
| 		mem.write_byte(U256::from(0x1e), U256::from(0xcd)); |  | ||||||
| 		mem.write_byte(U256::from(0x1f), U256::from(0xef)); |  | ||||||
| 
 |  | ||||||
| 		// then
 |  | ||||||
| 		assert_eq!(mem.read(U256::from(0x00)), U256::from(0xabcdef)); |  | ||||||
| 	} |  | ||||||
| } | } | ||||||
							
								
								
									
										106
									
								
								ethcore/src/evm/interpreter/stack.rs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										106
									
								
								ethcore/src/evm/interpreter/stack.rs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,106 @@ | |||||||
|  | // Copyright 2015, 2016 Ethcore (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/>.
 | ||||||
|  | 
 | ||||||
|  | use std::fmt; | ||||||
|  | use evm::instructions; | ||||||
|  | 
 | ||||||
|  | /// Stack trait with VM-friendly API
 | ||||||
|  | pub trait Stack<T> { | ||||||
|  | 	/// Returns `Stack[len(Stack) - no_from_top]`
 | ||||||
|  | 	fn peek(&self, no_from_top: usize) -> &T; | ||||||
|  | 	/// Swaps Stack[len(Stack)] and Stack[len(Stack) - no_from_top]
 | ||||||
|  | 	fn swap_with_top(&mut self, no_from_top: usize); | ||||||
|  | 	/// Returns true if Stack has at least `no_of_elems` elements
 | ||||||
|  | 	fn has(&self, no_of_elems: usize) -> bool; | ||||||
|  | 	/// Get element from top and remove it from Stack. Panics if stack is empty.
 | ||||||
|  | 	fn pop_back(&mut self) -> T; | ||||||
|  | 	/// Get (up to `instructions::MAX_NO_OF_TOPICS`) elements from top and remove them from Stack. Panics if stack is empty.
 | ||||||
|  | 	fn pop_n(&mut self, no_of_elems: usize) -> &[T]; | ||||||
|  | 	/// Add element on top of the Stack
 | ||||||
|  | 	fn push(&mut self, elem: T); | ||||||
|  | 	/// Get number of elements on Stack
 | ||||||
|  | 	fn size(&self) -> usize; | ||||||
|  | 	/// Returns all data on stack.
 | ||||||
|  | 	fn peek_top(&mut self, no_of_elems: usize) -> &[T]; | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | pub struct VecStack<S> { | ||||||
|  | 	stack: Vec<S>, | ||||||
|  | 	logs: [S; instructions::MAX_NO_OF_TOPICS] | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<S : Copy> VecStack<S> { | ||||||
|  | 	pub fn with_capacity(capacity: usize, zero: S) -> Self { | ||||||
|  | 		VecStack { | ||||||
|  | 			stack: Vec::with_capacity(capacity), | ||||||
|  | 			logs: [zero; instructions::MAX_NO_OF_TOPICS] | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl<S : fmt::Display> Stack<S> for VecStack<S> { | ||||||
|  | 	fn peek(&self, no_from_top: usize) -> &S { | ||||||
|  | 		&self.stack[self.stack.len() - no_from_top - 1] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn swap_with_top(&mut self, no_from_top: usize) { | ||||||
|  | 		let len = self.stack.len(); | ||||||
|  | 		self.stack.swap(len - no_from_top - 1, len - 1); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn has(&self, no_of_elems: usize) -> bool { | ||||||
|  | 		self.stack.len() >= no_of_elems | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn pop_back(&mut self) -> S { | ||||||
|  | 		let val = self.stack.pop(); | ||||||
|  | 		match val { | ||||||
|  | 			Some(x) => { | ||||||
|  | 				evm_debug!({ | ||||||
|  | 					println!("   POP: {}", x) | ||||||
|  | 				}); | ||||||
|  | 				x | ||||||
|  | 			}, | ||||||
|  | 			None => panic!("Tried to pop from empty stack.") | ||||||
|  | 		} | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn pop_n(&mut self, no_of_elems: usize) -> &[S] { | ||||||
|  | 		assert!(no_of_elems <= instructions::MAX_NO_OF_TOPICS); | ||||||
|  | 
 | ||||||
|  | 		for i in 0..no_of_elems { | ||||||
|  | 			self.logs[i] = self.pop_back(); | ||||||
|  | 		} | ||||||
|  | 		&self.logs[0..no_of_elems] | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn push(&mut self, elem: S) { | ||||||
|  | 		evm_debug!({ | ||||||
|  | 			println!("  PUSH: {}", elem) | ||||||
|  | 		}); | ||||||
|  | 		self.stack.push(elem); | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn size(&self) -> usize { | ||||||
|  | 		self.stack.len() | ||||||
|  | 	} | ||||||
|  | 
 | ||||||
|  | 	fn peek_top(&mut self, no_from_top: usize) -> &[S] { | ||||||
|  | 		assert!(self.stack.len() >= no_from_top, "peek_top asked for more items than exist."); | ||||||
|  | 		&self.stack[self.stack.len() - no_from_top .. self.stack.len()] | ||||||
|  | 	} | ||||||
|  | } | ||||||
|  | 
 | ||||||
| @ -28,8 +28,10 @@ mod jit; | |||||||
| 
 | 
 | ||||||
| #[cfg(test)] | #[cfg(test)] | ||||||
| mod tests; | mod tests; | ||||||
|  | #[cfg(all(feature="benches", test))] | ||||||
|  | mod benches; | ||||||
| 
 | 
 | ||||||
| pub use self::evm::{Evm, Error, Finalize, GasLeft, Result}; | pub use self::evm::{Evm, Error, Finalize, GasLeft, Result, CostType}; | ||||||
| pub use self::ext::{Ext, ContractCreateResult, MessageCallResult}; | pub use self::ext::{Ext, ContractCreateResult, MessageCallResult}; | ||||||
| pub use self::factory::{Factory, VMType}; | pub use self::factory::{Factory, VMType}; | ||||||
| pub use self::schedule::Schedule; | pub use self::schedule::Schedule; | ||||||
|  | |||||||
| @ -18,18 +18,18 @@ use common::*; | |||||||
| use evm::{self, Ext, Schedule, Factory, GasLeft, VMType, ContractCreateResult, MessageCallResult}; | use evm::{self, Ext, Schedule, Factory, GasLeft, VMType, ContractCreateResult, MessageCallResult}; | ||||||
| use std::fmt::Debug; | use std::fmt::Debug; | ||||||
| 
 | 
 | ||||||
| struct FakeLogEntry { | pub struct FakeLogEntry { | ||||||
| 	topics: Vec<H256>, | 	topics: Vec<H256>, | ||||||
| 	data: Bytes | 	data: Bytes | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(PartialEq, Eq, Hash, Debug)] | #[derive(PartialEq, Eq, Hash, Debug)] | ||||||
| enum FakeCallType { | pub enum FakeCallType { | ||||||
| 	Call, Create | 	Call, Create | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(PartialEq, Eq, Hash, Debug)] | #[derive(PartialEq, Eq, Hash, Debug)] | ||||||
| struct FakeCall { | pub struct FakeCall { | ||||||
| 	call_type: FakeCallType, | 	call_type: FakeCallType, | ||||||
| 	gas: U256, | 	gas: U256, | ||||||
| 	sender_address: Option<Address>, | 	sender_address: Option<Address>, | ||||||
| @ -43,7 +43,7 @@ struct FakeCall { | |||||||
| ///
 | ///
 | ||||||
| /// Can't do recursive calls.
 | /// Can't do recursive calls.
 | ||||||
| #[derive(Default)] | #[derive(Default)] | ||||||
| struct FakeExt { | pub struct FakeExt { | ||||||
| 	sstore_clears: usize, | 	sstore_clears: usize, | ||||||
| 	depth: usize, | 	depth: usize, | ||||||
| 	store: HashMap<H256, H256>, | 	store: HashMap<H256, H256>, | ||||||
| @ -67,7 +67,7 @@ fn test_finalize(res: Result<GasLeft, evm::Error>) -> Result<U256, evm::Error> { | |||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl FakeExt { | impl FakeExt { | ||||||
| 	fn new() -> Self { | 	pub fn new() -> Self { | ||||||
| 		FakeExt::default() | 		FakeExt::default() | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| @ -181,7 +181,7 @@ fn test_stack_underflow() { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let err = { | 	let err = { | ||||||
| 		let mut vm : Box<evm::Evm> = Box::new(super::interpreter::Interpreter::default()); | 		let mut vm : Box<evm::Evm> = Box::new(super::interpreter::Interpreter::<usize>::default()); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap_err() | 		test_finalize(vm.exec(params, &mut ext)).unwrap_err() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -208,7 +208,7 @@ fn test_add(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -228,7 +228,7 @@ fn test_sha3(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -248,7 +248,7 @@ fn test_address(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -270,7 +270,7 @@ fn test_origin(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -292,7 +292,7 @@ fn test_sender(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -327,7 +327,7 @@ fn test_extcodecopy(factory: super::Factory) { | |||||||
| 	ext.codes.insert(sender, sender_code); | 	ext.codes.insert(sender, sender_code); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -347,7 +347,7 @@ fn test_log_empty(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -379,7 +379,7 @@ fn test_log_sender(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -404,7 +404,7 @@ fn test_blockhash(factory: super::Factory) { | |||||||
| 	ext.blockhashes.insert(U256::zero(), blockhash.clone()); | 	ext.blockhashes.insert(U256::zero(), blockhash.clone()); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -426,7 +426,7 @@ fn test_calldataload(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -447,7 +447,7 @@ fn test_author(factory: super::Factory) { | |||||||
| 	ext.info.author = author; | 	ext.info.author = author; | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -467,7 +467,7 @@ fn test_timestamp(factory: super::Factory) { | |||||||
| 	ext.info.timestamp = timestamp; | 	ext.info.timestamp = timestamp; | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -487,7 +487,7 @@ fn test_number(factory: super::Factory) { | |||||||
| 	ext.info.number = number; | 	ext.info.number = number; | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -507,7 +507,7 @@ fn test_difficulty(factory: super::Factory) { | |||||||
| 	ext.info.difficulty = difficulty; | 	ext.info.difficulty = difficulty; | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -527,7 +527,7 @@ fn test_gas_limit(factory: super::Factory) { | |||||||
| 	ext.info.gas_limit = gas_limit; | 	ext.info.gas_limit = gas_limit; | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -545,7 +545,7 @@ fn test_mul(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -563,7 +563,7 @@ fn test_sub(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -581,7 +581,7 @@ fn test_div(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -599,7 +599,7 @@ fn test_div_zero(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -617,7 +617,7 @@ fn test_mod(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -636,7 +636,7 @@ fn test_smod(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -655,7 +655,7 @@ fn test_sdiv(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -674,7 +674,7 @@ fn test_exp(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -694,7 +694,7 @@ fn test_comparison(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -715,7 +715,7 @@ fn test_signed_comparison(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -736,7 +736,7 @@ fn test_bitops(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -759,7 +759,7 @@ fn test_addmod_mulmod(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -780,7 +780,7 @@ fn test_byte(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -799,7 +799,7 @@ fn test_signextend(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -819,7 +819,7 @@ fn test_badinstruction_int() { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let err = { | 	let err = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap_err() | 		test_finalize(vm.exec(params, &mut ext)).unwrap_err() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -839,7 +839,7 @@ fn test_pop(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -859,7 +859,7 @@ fn test_extops(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -882,7 +882,7 @@ fn test_jumps(factory: super::Factory) { | |||||||
| 	let mut ext = FakeExt::new(); | 	let mut ext = FakeExt::new(); | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| @ -911,7 +911,7 @@ fn test_calls(factory: super::Factory) { | |||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
| 	let gas_left = { | 	let gas_left = { | ||||||
| 		let mut vm = factory.create(); | 		let mut vm = factory.create(params.gas); | ||||||
| 		test_finalize(vm.exec(params, &mut ext)).unwrap() | 		test_finalize(vm.exec(params, &mut ext)).unwrap() | ||||||
| 	}; | 	}; | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -211,7 +211,7 @@ impl<'a> Executive<'a> { | |||||||
| 			let vm_factory = self.vm_factory; | 			let vm_factory = self.vm_factory; | ||||||
| 			let mut ext = self.as_externalities(OriginInfo::from(¶ms), unconfirmed_substate, output_policy, tracer, vm_tracer); | 			let mut ext = self.as_externalities(OriginInfo::from(¶ms), unconfirmed_substate, output_policy, tracer, vm_tracer); | ||||||
| 			trace!(target: "executive", "ext.schedule.have_delegate_call: {}", ext.schedule().have_delegate_call); | 			trace!(target: "executive", "ext.schedule.have_delegate_call: {}", ext.schedule().have_delegate_call); | ||||||
| 			return vm_factory.create().exec(params, &mut ext).finalize(ext); | 			return vm_factory.create(params.gas).exec(params, &mut ext).finalize(ext); | ||||||
| 		} | 		} | ||||||
| 
 | 
 | ||||||
| 		// Start in new thread to reset stack
 | 		// Start in new thread to reset stack
 | ||||||
| @ -222,7 +222,7 @@ impl<'a> Executive<'a> { | |||||||
| 			let mut ext = self.as_externalities(OriginInfo::from(¶ms), unconfirmed_substate, output_policy, tracer, vm_tracer); | 			let mut ext = self.as_externalities(OriginInfo::from(¶ms), unconfirmed_substate, output_policy, tracer, vm_tracer); | ||||||
| 
 | 
 | ||||||
| 			scope.spawn(move || { | 			scope.spawn(move || { | ||||||
| 				vm_factory.create().exec(params, &mut ext).finalize(ext) | 				vm_factory.create(params.gas).exec(params, &mut ext).finalize(ext) | ||||||
| 			}) | 			}) | ||||||
| 		}).join() | 		}).join() | ||||||
| 	} | 	} | ||||||
|  | |||||||
| @ -208,7 +208,7 @@ fn do_json_test_for(vm_type: &VMType, json_data: &[u8]) -> Vec<String> { | |||||||
| 				&mut tracer, | 				&mut tracer, | ||||||
| 				&mut vm_tracer, | 				&mut vm_tracer, | ||||||
| 			); | 			); | ||||||
| 			let mut evm = vm_factory.create(); | 			let mut evm = vm_factory.create(params.gas); | ||||||
| 			let res = evm.exec(params, &mut ex); | 			let res = evm.exec(params, &mut ex); | ||||||
| 			// a return in finalize will not alter callcreates
 | 			// a return in finalize will not alter callcreates
 | ||||||
| 			let callcreates = ex.callcreates.clone(); | 			let callcreates = ex.callcreates.clone(); | ||||||
|  | |||||||
| @ -31,6 +31,7 @@ | |||||||
| #![cfg_attr(feature="dev", allow(needless_borrow))] | #![cfg_attr(feature="dev", allow(needless_borrow))] | ||||||
| #![cfg_attr(feature="dev", allow(assign_op_pattern))] | #![cfg_attr(feature="dev", allow(assign_op_pattern))] | ||||||
| 
 | 
 | ||||||
|  | #![cfg_attr(feature="benches", feature(test))] | ||||||
| 
 | 
 | ||||||
| //! Ethcore library
 | //! Ethcore library
 | ||||||
| //!
 | //!
 | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user