Simple Gas calculation based on instruction gas price tier
This commit is contained in:
parent
d7176faddc
commit
0a5666f2c0
@ -15,6 +15,13 @@ pub enum Error {
|
|||||||
/// `BadJumpDestination` is returned when execution tried to move
|
/// `BadJumpDestination` is returned when execution tried to move
|
||||||
/// to position that wasn't marked with JUMPDEST instruction
|
/// to position that wasn't marked with JUMPDEST instruction
|
||||||
BadJumpDestination,
|
BadJumpDestination,
|
||||||
|
/// `BadInstructions` is returned when given instruction is not supported
|
||||||
|
BadInstruction,
|
||||||
|
/// `StackUnderflow` when there is not enough stack elements to execute instruction
|
||||||
|
/// First parameter says how many elements were needed and the second how many were actually on Stack
|
||||||
|
StackUnderflow(/*wanted*/usize, /*on_stack*/usize),
|
||||||
|
/// When execution would exceed defined Stack Limit
|
||||||
|
OutOfStack(/*wanted*/usize, /*stack_limit*/usize),
|
||||||
/// Returned on evm internal error. Should never be ignored during development.
|
/// Returned on evm internal error. Should never be ignored during development.
|
||||||
/// Likely to cause consensus issues.
|
/// Likely to cause consensus issues.
|
||||||
Internal,
|
Internal,
|
||||||
|
@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
use util::hash::*;
|
use util::hash::*;
|
||||||
use util::uint::*;
|
use util::uint::*;
|
||||||
use util::bytes::*;
|
|
||||||
use evm::{Schedule, Error};
|
use evm::{Schedule, Error};
|
||||||
use env_info::*;
|
use env_info::*;
|
||||||
|
|
||||||
|
@ -2,10 +2,6 @@
|
|||||||
|
|
||||||
pub type Instruction = u8;
|
pub type Instruction = u8;
|
||||||
|
|
||||||
pub fn is_jump (i : Instruction) -> bool {
|
|
||||||
i == JUMP || i == JUMPI || i == JUMPDEST
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn is_push (i : Instruction) -> bool {
|
pub fn is_push (i : Instruction) -> bool {
|
||||||
i >= PUSH1 && i <= PUSH32
|
i >= PUSH1 && i <= PUSH32
|
||||||
}
|
}
|
||||||
@ -15,25 +11,239 @@ pub fn get_push_bytes (i : Instruction) -> usize {
|
|||||||
(i - PUSH1 + 1) as usize
|
(i - PUSH1 + 1) as usize
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_get_push_bytes() {
|
||||||
|
assert_eq!(get_push_bytes(PUSH1), 1);
|
||||||
|
assert_eq!(get_push_bytes(PUSH3), 3);
|
||||||
|
assert_eq!(get_push_bytes(PUSH32), 32);
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_dup_position (i: Instruction) -> usize {
|
pub fn get_dup_position (i: Instruction) -> usize {
|
||||||
// TODO [todr] range checking?
|
// TODO [todr] range checking?
|
||||||
(i - DUP1) as usize
|
(i - DUP1) as usize
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_get_dup_position() {
|
||||||
|
assert_eq!(get_dup_position(DUP1), 1);
|
||||||
|
assert_eq!(get_dup_position(DUP5), 5);
|
||||||
|
assert_eq!(get_dup_position(DUP10), 10);
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_swap_position (i : Instruction) -> usize {
|
pub fn get_swap_position (i : Instruction) -> usize {
|
||||||
// TODO [todr] range checking?
|
// TODO [todr] range checking?
|
||||||
(i - SWAP1 + 1) as usize
|
(i - SWAP1 + 1) as usize
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_get_swap_position() {
|
||||||
|
assert_eq!(get_swap_position(SWAP1), 1);
|
||||||
|
assert_eq!(get_swap_position(SWAP5), 5);
|
||||||
|
assert_eq!(get_swap_position(SWAP10), 10);
|
||||||
|
}
|
||||||
|
|
||||||
pub fn get_log_topics (i: Instruction) -> usize {
|
pub fn get_log_topics (i: Instruction) -> usize {
|
||||||
(i - LOG0) as usize
|
(i - LOG0) as usize
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
fn test_get_push_bytes() {
|
fn test_get_log_topics() {
|
||||||
assert_eq!(get_push_bytes(PUSH1), 1);
|
assert_eq!(get_log_topics(LOG0), 0);
|
||||||
assert_eq!(get_push_bytes(PUSH3), 3);
|
assert_eq!(get_log_topics(LOG2), 2);
|
||||||
assert_eq!(get_push_bytes(PUSH32), 32);
|
assert_eq!(get_log_topics(LOG4), 4);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(PartialEq)]
|
||||||
|
pub enum GasPriceTier {
|
||||||
|
/// 0 Zero
|
||||||
|
ZeroTier,
|
||||||
|
/// 2 Quick
|
||||||
|
BaseTier,
|
||||||
|
/// 3 Fastest
|
||||||
|
VeryLowTier,
|
||||||
|
/// 5 Fast
|
||||||
|
LowTier,
|
||||||
|
/// 8 Mid
|
||||||
|
MidTier,
|
||||||
|
/// 10 Slow
|
||||||
|
HighTier,
|
||||||
|
/// 20 Ext
|
||||||
|
ExtTier,
|
||||||
|
/// Multiparam or otherwise special
|
||||||
|
SpecialTier,
|
||||||
|
/// Invalid
|
||||||
|
InvalidTier
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_tier_idx (tier: GasPriceTier) -> usize {
|
||||||
|
match tier {
|
||||||
|
GasPriceTier::ZeroTier => 0,
|
||||||
|
GasPriceTier::BaseTier => 1,
|
||||||
|
GasPriceTier::VeryLowTier => 2,
|
||||||
|
GasPriceTier::LowTier => 3,
|
||||||
|
GasPriceTier::MidTier => 4,
|
||||||
|
GasPriceTier::HighTier => 5,
|
||||||
|
GasPriceTier::ExtTier => 6,
|
||||||
|
GasPriceTier::SpecialTier => 7,
|
||||||
|
GasPriceTier::InvalidTier => 8
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct InstructionInfo {
|
||||||
|
pub name: &'static str,
|
||||||
|
pub additional: usize,
|
||||||
|
pub args: usize,
|
||||||
|
pub ret: usize,
|
||||||
|
pub side_effects: bool,
|
||||||
|
pub tier: GasPriceTier
|
||||||
|
}
|
||||||
|
impl InstructionInfo {
|
||||||
|
pub fn new(name: &'static str, additional: usize, args: usize, ret: usize, side_effects: bool, tier: GasPriceTier) -> InstructionInfo {
|
||||||
|
InstructionInfo {
|
||||||
|
name: name,
|
||||||
|
additional: additional,
|
||||||
|
args: args,
|
||||||
|
ret: ret,
|
||||||
|
side_effects: side_effects,
|
||||||
|
tier: tier
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_info (instruction: Instruction) -> InstructionInfo {
|
||||||
|
match instruction {
|
||||||
|
STOP => InstructionInfo::new("STOP", 0, 0, 0, true, GasPriceTier::ZeroTier),
|
||||||
|
ADD => InstructionInfo::new("ADD", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
SUB => InstructionInfo::new("SUB", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
MUL => InstructionInfo::new("MUL", 0, 2, 1, false, GasPriceTier::LowTier),
|
||||||
|
DIV => InstructionInfo::new("DIV", 0, 2, 1, false, GasPriceTier::LowTier),
|
||||||
|
SDIV => InstructionInfo::new("SDIV", 0, 2, 1, false, GasPriceTier::LowTier),
|
||||||
|
MOD => InstructionInfo::new("MOD", 0, 2, 1, false, GasPriceTier::LowTier),
|
||||||
|
SMOD => InstructionInfo::new("SMOD", 0, 2, 1, false, GasPriceTier::LowTier),
|
||||||
|
EXP => InstructionInfo::new("EXP", 0, 2, 1, false, GasPriceTier::SpecialTier),
|
||||||
|
NOT => InstructionInfo::new("NOT", 0, 1, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
LT => InstructionInfo::new("LT", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
GT => InstructionInfo::new("GT", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
SLT => InstructionInfo::new("SLT", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
SGT => InstructionInfo::new("SGT", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
EQ => InstructionInfo::new("EQ", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
ISZERO => InstructionInfo::new("ISZERO", 0, 1, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
AND => InstructionInfo::new("AND", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
OR => InstructionInfo::new("OR", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
XOR => InstructionInfo::new("XOR", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
BYTE => InstructionInfo::new("BYTE", 0, 2, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
ADDMOD => InstructionInfo::new("ADDMOD", 0, 3, 1, false, GasPriceTier::MidTier),
|
||||||
|
MULMOD => InstructionInfo::new("MULMOD", 0, 3, 1, false, GasPriceTier::MidTier),
|
||||||
|
SIGNEXTEND => InstructionInfo::new("SIGNEXTEND", 0, 2, 1, false, GasPriceTier::LowTier),
|
||||||
|
SHA3 => InstructionInfo::new("SHA3", 0, 2, 1, false, GasPriceTier::SpecialTier),
|
||||||
|
ADDRESS => InstructionInfo::new("ADDRESS", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
BALANCE => InstructionInfo::new("BALANCE", 0, 1, 1, false, GasPriceTier::ExtTier),
|
||||||
|
ORIGIN => InstructionInfo::new("ORIGIN", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
CALLER => InstructionInfo::new("CALLER", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
CALLVALUE => InstructionInfo::new("CALLVALUE", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
CALLDATALOAD => InstructionInfo::new("CALLDATALOAD", 0, 1, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
CALLDATASIZE => InstructionInfo::new("CALLDATASIZE", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
CALLDATACOPY => InstructionInfo::new("CALLDATACOPY", 0, 3, 0, true, GasPriceTier::VeryLowTier),
|
||||||
|
CODESIZE => InstructionInfo::new("CODESIZE", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
CODECOPY => InstructionInfo::new("CODECOPY", 0, 3, 0, true, GasPriceTier::VeryLowTier),
|
||||||
|
GASPRICE => InstructionInfo::new("GASPRICE", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
EXTCODESIZE => InstructionInfo::new("EXTCODESIZE", 0, 1, 1, false, GasPriceTier::ExtTier),
|
||||||
|
EXTCODECOPY => InstructionInfo::new("EXTCODECOPY", 0, 4, 0, true, GasPriceTier::ExtTier),
|
||||||
|
BLOCKHASH => InstructionInfo::new("BLOCKHASH", 0, 1, 1, false, GasPriceTier::ExtTier),
|
||||||
|
COINBASE => InstructionInfo::new("COINBASE", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
TIMESTAMP => InstructionInfo::new("TIMESTAMP", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
NUMBER => InstructionInfo::new("NUMBER", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
DIFFICULTY => InstructionInfo::new("DIFFICULTY", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
GASLIMIT => InstructionInfo::new("GASLIMIT", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
POP => InstructionInfo::new("POP", 0, 1, 0, false, GasPriceTier::BaseTier),
|
||||||
|
MLOAD => InstructionInfo::new("MLOAD", 0, 1, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
MSTORE => InstructionInfo::new("MSTORE", 0, 2, 0, true, GasPriceTier::VeryLowTier),
|
||||||
|
MSTORE8 => InstructionInfo::new("MSTORE8", 0, 2, 0, true, GasPriceTier::VeryLowTier),
|
||||||
|
SLOAD => InstructionInfo::new("SLOAD", 0, 1, 1, false, GasPriceTier::SpecialTier),
|
||||||
|
SSTORE => InstructionInfo::new("SSTORE", 0, 2, 0, true, GasPriceTier::SpecialTier),
|
||||||
|
JUMP => InstructionInfo::new("JUMP", 0, 1, 0, true, GasPriceTier::MidTier),
|
||||||
|
JUMPI => InstructionInfo::new("JUMPI", 0, 2, 0, true, GasPriceTier::HighTier),
|
||||||
|
PC => InstructionInfo::new("PC", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
MSIZE => InstructionInfo::new("MSIZE", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
GAS => InstructionInfo::new("GAS", 0, 0, 1, false, GasPriceTier::BaseTier),
|
||||||
|
JUMPDEST => InstructionInfo::new("JUMPDEST", 0, 0, 0, true, GasPriceTier::SpecialTier),
|
||||||
|
PUSH1 => InstructionInfo::new("PUSH1", 1, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH2 => InstructionInfo::new("PUSH2", 2, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH3 => InstructionInfo::new("PUSH3", 3, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH4 => InstructionInfo::new("PUSH4", 4, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH5 => InstructionInfo::new("PUSH5", 5, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH6 => InstructionInfo::new("PUSH6", 6, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH7 => InstructionInfo::new("PUSH7", 7, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH8 => InstructionInfo::new("PUSH8", 8, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH9 => InstructionInfo::new("PUSH9", 9, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH10 => InstructionInfo::new("PUSH10", 10, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH11 => InstructionInfo::new("PUSH11", 11, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH12 => InstructionInfo::new("PUSH12", 12, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH13 => InstructionInfo::new("PUSH13", 13, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH14 => InstructionInfo::new("PUSH14", 14, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH15 => InstructionInfo::new("PUSH15", 15, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH16 => InstructionInfo::new("PUSH16", 16, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH17 => InstructionInfo::new("PUSH17", 17, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH18 => InstructionInfo::new("PUSH18", 18, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH19 => InstructionInfo::new("PUSH19", 19, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH20 => InstructionInfo::new("PUSH20", 20, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH21 => InstructionInfo::new("PUSH21", 21, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH22 => InstructionInfo::new("PUSH22", 22, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH23 => InstructionInfo::new("PUSH23", 23, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH24 => InstructionInfo::new("PUSH24", 24, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH25 => InstructionInfo::new("PUSH25", 25, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH26 => InstructionInfo::new("PUSH26", 26, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH27 => InstructionInfo::new("PUSH27", 27, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH28 => InstructionInfo::new("PUSH28", 28, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH29 => InstructionInfo::new("PUSH29", 29, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH30 => InstructionInfo::new("PUSH30", 30, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH31 => InstructionInfo::new("PUSH31", 31, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
PUSH32 => InstructionInfo::new("PUSH32", 32, 0, 1, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP1 => InstructionInfo::new("DUP1", 0, 1, 2, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP2 => InstructionInfo::new("DUP2", 0, 2, 3, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP3 => InstructionInfo::new("DUP3", 0, 3, 4, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP4 => InstructionInfo::new("DUP4", 0, 4, 5, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP5 => InstructionInfo::new("DUP5", 0, 5, 6, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP6 => InstructionInfo::new("DUP6", 0, 6, 7, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP7 => InstructionInfo::new("DUP7", 0, 7, 8, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP8 => InstructionInfo::new("DUP8", 0, 8, 9, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP9 => InstructionInfo::new("DUP9", 0, 9, 10, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP10 => InstructionInfo::new("DUP10", 0, 10, 11, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP11 => InstructionInfo::new("DUP11", 0, 11, 12, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP12 => InstructionInfo::new("DUP12", 0, 12, 13, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP13 => InstructionInfo::new("DUP13", 0, 13, 14, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP14 => InstructionInfo::new("DUP14", 0, 14, 15, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP15 => InstructionInfo::new("DUP15", 0, 15, 16, false, GasPriceTier::VeryLowTier),
|
||||||
|
DUP16 => InstructionInfo::new("DUP16", 0, 16, 17, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP1 => InstructionInfo::new("SWAP1", 0, 2, 2, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP2 => InstructionInfo::new("SWAP2", 0, 3, 3, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP3 => InstructionInfo::new("SWAP3", 0, 4, 4, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP4 => InstructionInfo::new("SWAP4", 0, 5, 5, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP5 => InstructionInfo::new("SWAP5", 0, 6, 6, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP6 => InstructionInfo::new("SWAP6", 0, 7, 7, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP7 => InstructionInfo::new("SWAP7", 0, 8, 8, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP8 => InstructionInfo::new("SWAP8", 0, 9, 9, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP9 => InstructionInfo::new("SWAP9", 0, 10, 10, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP10 => InstructionInfo::new("SWAP10", 0, 11, 11, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP11 => InstructionInfo::new("SWAP11", 0, 12, 12, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP12 => InstructionInfo::new("SWAP12", 0, 13, 13, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP13 => InstructionInfo::new("SWAP13", 0, 14, 14, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP14 => InstructionInfo::new("SWAP14", 0, 15, 15, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP15 => InstructionInfo::new("SWAP15", 0, 16, 16, false, GasPriceTier::VeryLowTier),
|
||||||
|
SWAP16 => InstructionInfo::new("SWAP16", 0, 17, 17, false, GasPriceTier::VeryLowTier),
|
||||||
|
LOG0 => InstructionInfo::new("LOG0", 0, 2, 0, true, GasPriceTier::SpecialTier),
|
||||||
|
LOG1 => InstructionInfo::new("LOG1", 0, 3, 0, true, GasPriceTier::SpecialTier),
|
||||||
|
LOG2 => InstructionInfo::new("LOG2", 0, 4, 0, true, GasPriceTier::SpecialTier),
|
||||||
|
LOG3 => InstructionInfo::new("LOG3", 0, 5, 0, true, GasPriceTier::SpecialTier),
|
||||||
|
LOG4 => InstructionInfo::new("LOG4", 0, 6, 0, true, GasPriceTier::SpecialTier),
|
||||||
|
CREATE => InstructionInfo::new("CREATE", 0, 3, 1, true, GasPriceTier::SpecialTier),
|
||||||
|
CALL => InstructionInfo::new("CALL", 0, 7, 1, true, GasPriceTier::SpecialTier),
|
||||||
|
CALLCODE => InstructionInfo::new("CALLCODE", 0, 7, 1, true, GasPriceTier::SpecialTier),
|
||||||
|
RETURN => InstructionInfo::new("RETURN", 0, 2, 0, true, GasPriceTier::ZeroTier),
|
||||||
|
DELEGATECALL => InstructionInfo::new("DELEGATECALL", 0, 6, 1, true, GasPriceTier::SpecialTier),
|
||||||
|
SUICIDE => InstructionInfo::new("SUICIDE", 0, 1, 0, true, GasPriceTier::ZeroTier),
|
||||||
|
_ => panic!(format!("Undefined instruction: {}", instruction))
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Virtual machine bytecode instruction.
|
// Virtual machine bytecode instruction.
|
||||||
@ -99,136 +309,75 @@ pub const GAS: Instruction = 0x5a; //< get the amount of available gas
|
|||||||
pub const JUMPDEST: Instruction = 0x5b; //< set a potential jump destination
|
pub const JUMPDEST: Instruction = 0x5b; //< set a potential jump destination
|
||||||
|
|
||||||
pub const PUSH1: Instruction = 0x60; //< place 1 byte item on stack
|
pub const PUSH1: Instruction = 0x60; //< place 1 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH2: Instruction = 0x61; //< place 2 byte item on stack
|
pub const PUSH2: Instruction = 0x61; //< place 2 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH3: Instruction = 0x62; //< place 3 byte item on stack
|
pub const PUSH3: Instruction = 0x62; //< place 3 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH4: Instruction = 0x63; //< place 4 byte item on stack
|
pub const PUSH4: Instruction = 0x63; //< place 4 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH5: Instruction = 0x64; //< place 5 byte item on stack
|
pub const PUSH5: Instruction = 0x64; //< place 5 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH6: Instruction = 0x65; //< place 6 byte item on stack
|
pub const PUSH6: Instruction = 0x65; //< place 6 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH7: Instruction = 0x66; //< place 7 byte item on stack
|
pub const PUSH7: Instruction = 0x66; //< place 7 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH8: Instruction = 0x67; //< place 8 byte item on stack
|
pub const PUSH8: Instruction = 0x67; //< place 8 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH9: Instruction = 0x68; //< place 9 byte item on stack
|
pub const PUSH9: Instruction = 0x68; //< place 9 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH10: Instruction = 0x69; //< place 10 byte item on stack
|
pub const PUSH10: Instruction = 0x69; //< place 10 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH11: Instruction = 0x6a; //< place 11 byte item on stack
|
pub const PUSH11: Instruction = 0x6a; //< place 11 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH12: Instruction = 0x6b; //< place 12 byte item on stack
|
pub const PUSH12: Instruction = 0x6b; //< place 12 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH13: Instruction = 0x6c; //< place 13 byte item on stack
|
pub const PUSH13: Instruction = 0x6c; //< place 13 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH14: Instruction = 0x6d; //< place 14 byte item on stack
|
pub const PUSH14: Instruction = 0x6d; //< place 14 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH15: Instruction = 0x6e; //< place 15 byte item on stack
|
pub const PUSH15: Instruction = 0x6e; //< place 15 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH16: Instruction = 0x6f; //< place 16 byte item on stack
|
pub const PUSH16: Instruction = 0x6f; //< place 16 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH17: Instruction = 0x70; //< place 17 byte item on stack
|
pub const PUSH17: Instruction = 0x70; //< place 17 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH18: Instruction = 0x71; //< place 18 byte item on stack
|
pub const PUSH18: Instruction = 0x71; //< place 18 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH19: Instruction = 0x72; //< place 19 byte item on stack
|
pub const PUSH19: Instruction = 0x72; //< place 19 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH20: Instruction = 0x73; //< place 20 byte item on stack
|
pub const PUSH20: Instruction = 0x73; //< place 20 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH21: Instruction = 0x74; //< place 21 byte item on stack
|
pub const PUSH21: Instruction = 0x74; //< place 21 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH22: Instruction = 0x75; //< place 22 byte item on stack
|
pub const PUSH22: Instruction = 0x75; //< place 22 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH23: Instruction = 0x76; //< place 23 byte item on stack
|
pub const PUSH23: Instruction = 0x76; //< place 23 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH24: Instruction = 0x77; //< place 24 byte item on stack
|
pub const PUSH24: Instruction = 0x77; //< place 24 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH25: Instruction = 0x78; //< place 25 byte item on stack
|
pub const PUSH25: Instruction = 0x78; //< place 25 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH26: Instruction = 0x79; //< place 26 byte item on stack
|
pub const PUSH26: Instruction = 0x79; //< place 26 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH27: Instruction = 0x7a; //< place 27 byte item on stack
|
pub const PUSH27: Instruction = 0x7a; //< place 27 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH28: Instruction = 0x7b; //< place 28 byte item on stack
|
pub const PUSH28: Instruction = 0x7b; //< place 28 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH29: Instruction = 0x7c; //< place 29 byte item on stack
|
pub const PUSH29: Instruction = 0x7c; //< place 29 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH30: Instruction = 0x7d; //< place 30 byte item on stack
|
pub const PUSH30: Instruction = 0x7d; //< place 30 byte item on stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const PUSH31: Instruction = 0x7e; //< place 31 byte item on stack
|
pub const PUSH31: Instruction = 0x7e; //< place 31 byte item on stack
|
||||||
pub const PUSH32: Instruction = 0x7f; //< place 32 byte item on stack
|
pub const PUSH32: Instruction = 0x7f; //< place 32 byte item on stack
|
||||||
|
|
||||||
pub const DUP1: Instruction = 0x80; //< copies the highest item in the stack to the top of the stack
|
pub const DUP1: Instruction = 0x80; //< copies the highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP2: Instruction = 0x81; //< copies the second highest item in the stack to the top of the stack
|
pub const DUP2: Instruction = 0x81; //< copies the second highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP3: Instruction = 0x82; //< copies the third highest item in the stack to the top of the stack
|
pub const DUP3: Instruction = 0x82; //< copies the third highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP4: Instruction = 0x83; //< copies the 4th highest item in the stack to the top of the stack
|
pub const DUP4: Instruction = 0x83; //< copies the 4th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP5: Instruction = 0x84; //< copies the 5th highest item in the stack to the top of the stack
|
pub const DUP5: Instruction = 0x84; //< copies the 5th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP6: Instruction = 0x85; //< copies the 6th highest item in the stack to the top of the stack
|
pub const DUP6: Instruction = 0x85; //< copies the 6th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP7: Instruction = 0x86; //< copies the 7th highest item in the stack to the top of the stack
|
pub const DUP7: Instruction = 0x86; //< copies the 7th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP8: Instruction = 0x87; //< copies the 8th highest item in the stack to the top of the stack
|
pub const DUP8: Instruction = 0x87; //< copies the 8th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP9: Instruction = 0x88; //< copies the 9th highest item in the stack to the top of the stack
|
pub const DUP9: Instruction = 0x88; //< copies the 9th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP10: Instruction = 0x89; //< copies the 10th highest item in the stack to the top of the stack
|
pub const DUP10: Instruction = 0x89; //< copies the 10th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP11: Instruction = 0x8a; //< copies the 11th highest item in the stack to the top of the stack
|
pub const DUP11: Instruction = 0x8a; //< copies the 11th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP12: Instruction = 0x8b; //< copies the 12th highest item in the stack to the top of the stack
|
pub const DUP12: Instruction = 0x8b; //< copies the 12th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP13: Instruction = 0x8c; //< copies the 13th highest item in the stack to the top of the stack
|
pub const DUP13: Instruction = 0x8c; //< copies the 13th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP14: Instruction = 0x8d; //< copies the 14th highest item in the stack to the top of the stack
|
pub const DUP14: Instruction = 0x8d; //< copies the 14th highest item in the stack to the top of the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const DUP15: Instruction = 0x8e; //< copies the 15th highest item in the stack to the top of the stack
|
pub const DUP15: Instruction = 0x8e; //< copies the 15th highest item in the stack to the top of the stack
|
||||||
pub const DUP16: Instruction = 0x8f; //< copies the 16th highest item in the stack to the top of the stack
|
pub const DUP16: Instruction = 0x8f; //< copies the 16th highest item in the stack to the top of the stack
|
||||||
|
|
||||||
pub const SWAP1: Instruction = 0x90; //< swaps the highest and second highest value on the stack
|
pub const SWAP1: Instruction = 0x90; //< swaps the highest and second highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP2: Instruction = 0x91; //< swaps the highest and third highest value on the stack
|
pub const SWAP2: Instruction = 0x91; //< swaps the highest and third highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP3: Instruction = 0x92; //< swaps the highest and 4th highest value on the stack
|
pub const SWAP3: Instruction = 0x92; //< swaps the highest and 4th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP4: Instruction = 0x93; //< swaps the highest and 5th highest value on the stack
|
pub const SWAP4: Instruction = 0x93; //< swaps the highest and 5th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP5: Instruction = 0x94; //< swaps the highest and 6th highest value on the stack
|
pub const SWAP5: Instruction = 0x94; //< swaps the highest and 6th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP6: Instruction = 0x95; //< swaps the highest and 7th highest value on the stack
|
pub const SWAP6: Instruction = 0x95; //< swaps the highest and 7th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP7: Instruction = 0x96; //< swaps the highest and 8th highest value on the stack
|
pub const SWAP7: Instruction = 0x96; //< swaps the highest and 8th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP8: Instruction = 0x97; //< swaps the highest and 9th highest value on the stack
|
pub const SWAP8: Instruction = 0x97; //< swaps the highest and 9th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP9: Instruction = 0x98; //< swaps the highest and 10th highest value on the stack
|
pub const SWAP9: Instruction = 0x98; //< swaps the highest and 10th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP10: Instruction = 0x99; //< swaps the highest and 11th highest value on the stack
|
pub const SWAP10: Instruction = 0x99; //< swaps the highest and 11th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP11: Instruction = 0x9a; //< swaps the highest and 12th highest value on the stack
|
pub const SWAP11: Instruction = 0x9a; //< swaps the highest and 12th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP12: Instruction = 0x9b; //< swaps the highest and 13th highest value on the stack
|
pub const SWAP12: Instruction = 0x9b; //< swaps the highest and 13th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP13: Instruction = 0x9c; //< swaps the highest and 14th highest value on the stack
|
pub const SWAP13: Instruction = 0x9c; //< swaps the highest and 14th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP14: Instruction = 0x9d; //< swaps the highest and 15th highest value on the stack
|
pub const SWAP14: Instruction = 0x9d; //< swaps the highest and 15th highest value on the stack
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const SWAP15: Instruction = 0x9e; //< swaps the highest and 16th highest value on the stack
|
pub const SWAP15: Instruction = 0x9e; //< swaps the highest and 16th highest value on the stack
|
||||||
pub const SWAP16: Instruction = 0x9f; //< swaps the highest and 17th highest value on the stack
|
pub const SWAP16: Instruction = 0x9f; //< swaps the highest and 17th highest value on the stack
|
||||||
|
|
||||||
pub const LOG0: Instruction = 0xa0; //< Makes a log entry; no topics.
|
pub const LOG0: Instruction = 0xa0; //< Makes a log entry; no topics.
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const LOG1: Instruction = 0xa1; //< Makes a log entry; 1 topic.
|
pub const LOG1: Instruction = 0xa1; //< Makes a log entry; 1 topic.
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const LOG2: Instruction = 0xa2; //< Makes a log entry; 2 topics.
|
pub const LOG2: Instruction = 0xa2; //< Makes a log entry; 2 topics.
|
||||||
#[allow(dead_code)]
|
|
||||||
pub const LOG3: Instruction = 0xa3; //< Makes a log entry; 3 topics.
|
pub const LOG3: Instruction = 0xa3; //< Makes a log entry; 3 topics.
|
||||||
pub const LOG4: Instruction = 0xa4; //< Makes a log entry; 4 topics.
|
pub const LOG4: Instruction = 0xa4; //< Makes a log entry; 4 topics.
|
||||||
|
|
||||||
|
@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
use common::*;
|
use common::*;
|
||||||
use evm;
|
use evm;
|
||||||
use super::schedule::Schedule;
|
|
||||||
use super::instructions as instructions;
|
use super::instructions as instructions;
|
||||||
use super::instructions::Instruction;
|
use super::instructions::Instruction;
|
||||||
|
|
||||||
@ -24,6 +23,8 @@ trait Stack<T> {
|
|||||||
fn pop_n(&mut self, no_of_elems: usize) -> Vec<T>;
|
fn pop_n(&mut self, no_of_elems: usize) -> Vec<T>;
|
||||||
/// Add element on top of the Stack
|
/// Add element on top of the Stack
|
||||||
fn push(&mut self, elem: T);
|
fn push(&mut self, elem: T);
|
||||||
|
/// Get number of elements on Stack
|
||||||
|
fn size(&self) -> usize;
|
||||||
}
|
}
|
||||||
impl<S : fmt::Display> Stack<S> for Vec<S> {
|
impl<S : fmt::Display> Stack<S> for Vec<S> {
|
||||||
fn peek(&self, no_from_top: usize) -> &S {
|
fn peek(&self, no_from_top: usize) -> &S {
|
||||||
@ -55,11 +56,14 @@ impl<S : fmt::Display> Stack<S> for Vec<S> {
|
|||||||
vec
|
vec
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
fn push(&mut self, elem: S) {
|
fn push(&mut self, elem: S) {
|
||||||
println!("Pushing to stack: {}", elem);
|
println!("Pushing to stack: {}", elem);
|
||||||
self.push(elem);
|
self.push(elem);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
fn size(&self) -> usize {
|
||||||
|
self.len()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
/// 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> {
|
||||||
@ -94,25 +98,24 @@ impl<'a> CodeReader<'a> {
|
|||||||
let init_size = init_size_u.low_u64() as usize;
|
let init_size = init_size_u.low_u64() as usize;
|
||||||
&self.code[self.position + init_off..self.position + init_off + init_size]
|
&self.code[self.position + init_off..self.position + init_off + init_size]
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/// Stop any further execution (move PC to the end)
|
enum InstructionResult {
|
||||||
fn stop_execution(&mut self) {
|
AdditionalGasCost(U256),
|
||||||
self.position = self.code.len();
|
JumpToPosition(U256),
|
||||||
}
|
StopExecutionWithGasCost(U256),
|
||||||
|
StopExecution
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Interpreter;
|
pub struct Interpreter;
|
||||||
|
|
||||||
impl evm::Evm for Interpreter {
|
impl evm::Evm for Interpreter {
|
||||||
fn exec(&self, params: &ActionParams, ext: &mut evm::Ext) -> evm::Result {
|
fn exec(&self, params: &ActionParams, ext: &mut evm::Ext) -> evm::Result {
|
||||||
// TODO schedule?
|
|
||||||
// TODO reserve stack
|
|
||||||
|
|
||||||
// let schedule = ext.schedule();
|
|
||||||
let code = ¶ms.code;
|
let code = ¶ms.code;
|
||||||
let valid_jump_destinations = self.find_jump_destinations(&code);
|
let valid_jump_destinations = self.find_jump_destinations(&code);
|
||||||
|
|
||||||
let mut gas = params.gas.clone();
|
// TODO reserve stack
|
||||||
|
let mut current_gas = params.gas.clone();
|
||||||
let mut stack = vec![];
|
let mut stack = vec![];
|
||||||
let mut reader = CodeReader {
|
let mut reader = CodeReader {
|
||||||
position: 0,
|
position: 0,
|
||||||
@ -121,51 +124,110 @@ 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];
|
||||||
let gas_usage = self.check_and_get_gas_usage(instruction/*, schedule*/);
|
|
||||||
// TODO check if we have enough
|
|
||||||
|
|
||||||
reader.position += 1;
|
reader.position += 1;
|
||||||
// Handle jumps
|
|
||||||
match instruction {
|
// Calculate gas cost
|
||||||
instructions::JUMP => {
|
let gas_cost = try!(self.get_gas_cost(current_gas, params, ext, instruction, &stack));
|
||||||
let jump = stack.pop_back();
|
try!(self.verify_gas(¤t_gas, &gas_cost));
|
||||||
reader.position = try!(self.verify_jump(jump, &valid_jump_destinations));
|
current_gas = current_gas - gas_cost;
|
||||||
|
|
||||||
|
// Execute instruction
|
||||||
|
let result = try!(self.exec_instruction(
|
||||||
|
current_gas, params, ext, instruction, &mut reader, &mut stack
|
||||||
|
));
|
||||||
|
|
||||||
|
// Advance
|
||||||
|
match result {
|
||||||
|
InstructionResult::JumpToPosition(position) => {
|
||||||
|
let pos = try!(self.verify_jump(position, &valid_jump_destinations));
|
||||||
|
reader.position = pos;
|
||||||
},
|
},
|
||||||
instructions::JUMPI => {
|
InstructionResult::AdditionalGasCost(gas_cost) => {
|
||||||
let condition = stack.pop_back();
|
current_gas = current_gas - gas_cost;
|
||||||
let jump = stack.pop_back();
|
|
||||||
if !self.is_zero(condition) {
|
|
||||||
reader.position = try!(self.verify_jump(jump, &valid_jump_destinations));
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
instructions::JUMPDEST => {
|
InstructionResult::StopExecutionWithGasCost(gas_cost) => {
|
||||||
// ignore
|
current_gas = current_gas - gas_cost;
|
||||||
|
reader.position = code.len();
|
||||||
},
|
},
|
||||||
_ => {
|
InstructionResult::StopExecution => {
|
||||||
// Execute all other instructions
|
reader.position = code.len();
|
||||||
self.exec_instruction(params, ext, gas, instruction, &mut reader, &mut stack);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
Ok(U256::from(79_988))
|
|
||||||
|
Ok(current_gas)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Interpreter {
|
impl Interpreter {
|
||||||
|
|
||||||
fn check_and_get_gas_usage(&self, instruction: Instruction/*, schedule: &Schedule*/) -> Gas {
|
fn get_gas_cost(&self,
|
||||||
U256::zero()
|
gas: Gas,
|
||||||
|
params: &ActionParams,
|
||||||
|
ext: &evm::Ext,
|
||||||
|
instruction: Instruction,
|
||||||
|
stack: &Stack<U256>
|
||||||
|
) -> evm::Result {
|
||||||
|
|
||||||
|
let schedule = ext.schedule();
|
||||||
|
let info = instructions::get_info(instruction);
|
||||||
|
|
||||||
|
if !schedule.have_delegate_call && instruction == instructions::DELEGATECALL {
|
||||||
|
return Err(evm::Error::BadInstruction);
|
||||||
|
}
|
||||||
|
if info.tier == instructions::GasPriceTier::InvalidTier {
|
||||||
|
return Err(evm::Error::BadInstruction);
|
||||||
|
}
|
||||||
|
|
||||||
|
try!(self.verify_instructions_requirements(&info, schedule.stack_limit, stack));
|
||||||
|
|
||||||
|
let tier = instructions::get_tier_idx(info.tier);
|
||||||
|
let run_gas = schedule.tier_step_gas[tier];
|
||||||
|
|
||||||
|
|
||||||
|
Ok(Gas::from(run_gas))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn verify_instructions_requirements(&self,
|
||||||
|
info: &instructions::InstructionInfo,
|
||||||
|
stack_limit: usize,
|
||||||
|
stack: &Stack<U256>) -> Result<(), evm::Error> {
|
||||||
|
if !stack.has(info.args) {
|
||||||
|
Err(evm::Error::StackUnderflow(info.args, stack.size()))
|
||||||
|
} else if stack.size() - info.args + info.ret > stack_limit {
|
||||||
|
Err(evm::Error::OutOfStack(info.ret - info.args, stack_limit))
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn exec_instruction(&self,
|
fn exec_instruction(&self,
|
||||||
|
gas: Gas,
|
||||||
params: &ActionParams,
|
params: &ActionParams,
|
||||||
ext: &mut evm::Ext,
|
ext: &mut evm::Ext,
|
||||||
gas: Gas,
|
|
||||||
instruction: Instruction,
|
instruction: Instruction,
|
||||||
code: &mut CodeReader,
|
code: &mut CodeReader,
|
||||||
stack: &mut Stack<U256>
|
stack: &mut Stack<U256>
|
||||||
) -> evm::Result {
|
) -> Result<InstructionResult, evm::Error> {
|
||||||
match instruction {
|
match instruction {
|
||||||
|
instructions::JUMP => {
|
||||||
|
let jump = stack.pop_back();
|
||||||
|
return Ok(InstructionResult::JumpToPosition(
|
||||||
|
jump
|
||||||
|
));
|
||||||
|
},
|
||||||
|
instructions::JUMPI => {
|
||||||
|
let condition = stack.pop_back();
|
||||||
|
let jump = stack.pop_back();
|
||||||
|
if !self.is_zero(condition) {
|
||||||
|
return Ok(InstructionResult::JumpToPosition(
|
||||||
|
jump
|
||||||
|
));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
instructions::JUMPDEST => {
|
||||||
|
// ignore
|
||||||
|
},
|
||||||
instructions::CREATE => {
|
instructions::CREATE => {
|
||||||
let endowment = stack.pop_back();
|
let endowment = stack.pop_back();
|
||||||
let init_off = stack.pop_back();
|
let init_off = stack.pop_back();
|
||||||
@ -174,35 +236,38 @@ impl Interpreter {
|
|||||||
// TODO [todr] Fix u64 for gas
|
// TODO [todr] Fix u64 for gas
|
||||||
let contract_code = code.get_slice(init_off, init_size);
|
let contract_code = code.get_slice(init_off, init_size);
|
||||||
// TODO [todr] Fix u64 for gasLeft
|
// TODO [todr] Fix u64 for gasLeft
|
||||||
let (gas_left, maybe_address) = try!(ext.create(gas.low_u64(), &endowment, &contract_code));
|
let (gas_left, maybe_address) = try!(
|
||||||
|
ext.create(gas.low_u64(), &endowment, &contract_code)
|
||||||
|
);
|
||||||
match maybe_address {
|
match maybe_address {
|
||||||
Some(address) => stack.push(address_to_u256(address)),
|
Some(address) => stack.push(address_to_u256(address)),
|
||||||
None => stack.push(U256::zero())
|
None => stack.push(U256::zero())
|
||||||
}
|
}
|
||||||
Ok(U256::from(gas_left))
|
return Ok(InstructionResult::AdditionalGasCost(
|
||||||
|
gas - Gas::from(gas_left)
|
||||||
|
));
|
||||||
},
|
},
|
||||||
// CALL, CALLCODE, DELEGATECALL
|
// CALL, CALLCODE, DELEGATECALL
|
||||||
instructions::RETURN => {
|
instructions::RETURN => {
|
||||||
let init_off = stack.pop_back();
|
let init_off = stack.pop_back();
|
||||||
let init_size = stack.pop_back();
|
let init_size = stack.pop_back();
|
||||||
code.stop_execution();
|
|
||||||
let return_code = code.get_slice(init_off, init_size);
|
let return_code = code.get_slice(init_off, init_size);
|
||||||
// TODO [todr] Fix u64 for gas
|
// TODO [todr] Fix u64 for gas
|
||||||
let gas_left = try!(ext.ret(gas.low_u64(), &return_code));
|
let gas_left = try!(ext.ret(gas.low_u64(), &return_code));
|
||||||
// TODO [todr] Fix u64 for gasLeft
|
// TODO [todr] Fix u64 for gasLeft
|
||||||
Ok(U256::from(gas_left))
|
return Ok(InstructionResult::StopExecutionWithGasCost(
|
||||||
|
gas - Gas::from(gas_left)
|
||||||
|
));
|
||||||
},
|
},
|
||||||
instructions::STOP => {
|
instructions::STOP => {
|
||||||
code.stop_execution();
|
return Ok(InstructionResult::StopExecution);
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::SUICIDE => {
|
instructions::SUICIDE => {
|
||||||
// TODO [todr] Suicide should have argument with address of contract that funds should be transfered to
|
// TODO [todr] Suicide should have argument with address of contract that funds should be transfered to
|
||||||
let address = stack.pop_back();
|
let address = stack.pop_back();
|
||||||
// ext.suicide(Address::from(address));
|
// ext.suicide(Address::from(address));
|
||||||
ext.suicide();
|
ext.suicide();
|
||||||
code.stop_execution();
|
return Ok(InstructionResult::StopExecution);
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::LOG0...instructions::LOG4 => {
|
instructions::LOG0...instructions::LOG4 => {
|
||||||
let no_of_topics = instructions::get_log_topics(instruction);
|
let no_of_topics = instructions::get_log_topics(instruction);
|
||||||
@ -216,7 +281,6 @@ impl Interpreter {
|
|||||||
.map(H256::from)
|
.map(H256::from)
|
||||||
.collect();
|
.collect();
|
||||||
ext.log(topics, code.get_slice(offset, size));
|
ext.log(topics, code.get_slice(offset, size));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::PUSH1...instructions::PUSH32 => {
|
instructions::PUSH1...instructions::PUSH32 => {
|
||||||
// Load to stack
|
// Load to stack
|
||||||
@ -224,87 +288,69 @@ impl Interpreter {
|
|||||||
// TODO [todr] move positions management outside of CodeReader
|
// TODO [todr] move positions management outside of CodeReader
|
||||||
let val = code.read(bytes);
|
let val = code.read(bytes);
|
||||||
stack.push(val);
|
stack.push(val);
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::MLOAD => {
|
instructions::MLOAD => {
|
||||||
// TODO [ToDr] load word from mem?
|
// TODO [ToDr] load word from mem?
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::MSTORE => {
|
instructions::MSTORE => {
|
||||||
// TODO [ToDr] save word to mem?
|
// TODO [ToDr] save word to mem?
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::MSTORE8 => {
|
instructions::MSTORE8 => {
|
||||||
// TODO [ToDr] save byte to mem?
|
// TODO [ToDr] save byte to mem?
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::MSIZE => {
|
instructions::MSIZE => {
|
||||||
// Size of memry to stack
|
// Size of memry to stack
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::SHA3 => {
|
instructions::SHA3 => {
|
||||||
let offset = stack.pop_back();
|
let offset = stack.pop_back();
|
||||||
let size = stack.pop_back();
|
let size = stack.pop_back();
|
||||||
let sha3 = code.get_slice(offset, size).sha3();
|
let sha3 = code.get_slice(offset, size).sha3();
|
||||||
stack.push(U256::from(sha3.as_slice()));
|
stack.push(U256::from(sha3.as_slice()));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::SLOAD => {
|
instructions::SLOAD => {
|
||||||
let key = H256::from(&stack.pop_back());
|
let key = H256::from(&stack.pop_back());
|
||||||
let word = U256::from(ext.sload(&key).as_slice());
|
let word = U256::from(ext.sload(&key).as_slice());
|
||||||
stack.push(word);
|
stack.push(word);
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::SSTORE => {
|
instructions::SSTORE => {
|
||||||
let key = H256::from(&stack.pop_back());
|
let key = H256::from(&stack.pop_back());
|
||||||
let word = H256::from(&stack.pop_back());
|
let word = H256::from(&stack.pop_back());
|
||||||
ext.sstore(key, word);
|
ext.sstore(key, word);
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::PC => {
|
instructions::PC => {
|
||||||
stack.push(U256::from(code.position));
|
stack.push(U256::from(code.position));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::GAS => {
|
instructions::GAS => {
|
||||||
stack.push(U256::from(gas));
|
stack.push(gas.clone());
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::ADDRESS => {
|
instructions::ADDRESS => {
|
||||||
stack.push(address_to_u256(params.address.clone()));
|
stack.push(address_to_u256(params.address.clone()));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::ORIGIN => {
|
instructions::ORIGIN => {
|
||||||
stack.push(address_to_u256(params.origin.clone()));
|
stack.push(address_to_u256(params.origin.clone()));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::BALANCE => {
|
instructions::BALANCE => {
|
||||||
let address = u256_to_address(&stack.pop_back());
|
let address = u256_to_address(&stack.pop_back());
|
||||||
let balance = ext.balance(&address);
|
let balance = ext.balance(&address);
|
||||||
stack.push(balance);
|
stack.push(balance);
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::CALLER => {
|
instructions::CALLER => {
|
||||||
stack.push(address_to_u256(params.sender.clone()));
|
stack.push(address_to_u256(params.sender.clone()));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::CALLVALUE => {
|
instructions::CALLVALUE => {
|
||||||
stack.push(params.value.clone());
|
stack.push(params.value.clone());
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
// instructions::CALLDATALOAD
|
// instructions::CALLDATALOAD
|
||||||
instructions::CALLDATASIZE => {
|
instructions::CALLDATASIZE => {
|
||||||
stack.push(U256::from(params.data.len()));
|
stack.push(U256::from(params.data.len()));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::CODESIZE => {
|
instructions::CODESIZE => {
|
||||||
stack.push(U256::from(code.len()));
|
stack.push(U256::from(code.len()));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::EXTCODESIZE => {
|
instructions::EXTCODESIZE => {
|
||||||
let address = u256_to_address(&stack.pop_back());
|
let address = u256_to_address(&stack.pop_back());
|
||||||
let len = ext.extcode(&address).len();
|
let len = ext.extcode(&address).len();
|
||||||
stack.push(U256::from(len));
|
stack.push(U256::from(len));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
// instructions::CALLDATACOPY => {},
|
// instructions::CALLDATACOPY => {},
|
||||||
// instructions::CODECOPY => {},
|
// instructions::CODECOPY => {},
|
||||||
@ -314,38 +360,39 @@ impl Interpreter {
|
|||||||
// },
|
// },
|
||||||
instructions::GASPRICE => {
|
instructions::GASPRICE => {
|
||||||
stack.push(params.gas_price.clone());
|
stack.push(params.gas_price.clone());
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::BLOCKHASH => {
|
instructions::BLOCKHASH => {
|
||||||
let block_number = stack.pop_back();
|
let block_number = stack.pop_back();
|
||||||
let block_hash = ext.blockhash(&block_number);
|
let block_hash = ext.blockhash(&block_number);
|
||||||
stack.push(U256::from(block_hash.as_slice()));
|
stack.push(U256::from(block_hash.as_slice()));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::COINBASE => {
|
instructions::COINBASE => {
|
||||||
stack.push(address_to_u256(ext.env_info().author.clone()));
|
stack.push(address_to_u256(ext.env_info().author.clone()));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::TIMESTAMP => {
|
instructions::TIMESTAMP => {
|
||||||
stack.push(U256::from(ext.env_info().timestamp));
|
stack.push(U256::from(ext.env_info().timestamp));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::NUMBER => {
|
instructions::NUMBER => {
|
||||||
stack.push(U256::from(ext.env_info().number));
|
stack.push(U256::from(ext.env_info().number));
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::DIFFICULTY => {
|
instructions::DIFFICULTY => {
|
||||||
stack.push(ext.env_info().difficulty.clone());
|
stack.push(ext.env_info().difficulty.clone());
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
instructions::GASLIMIT => {
|
instructions::GASLIMIT => {
|
||||||
stack.push(ext.env_info().gas_limit.clone());
|
stack.push(ext.env_info().gas_limit.clone());
|
||||||
Ok(gas)
|
|
||||||
},
|
},
|
||||||
_ => {
|
_ => {
|
||||||
self.exec_stack_instruction(instruction, stack);
|
self.exec_stack_instruction(instruction, stack);
|
||||||
Ok(gas)
|
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
Ok(InstructionResult::AdditionalGasCost(U256::zero()))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn verify_gas(&self, current_gas: &U256, gas_cost: &U256) -> Result<(), evm::Error> {
|
||||||
|
if current_gas < gas_cost {
|
||||||
|
Err(evm::Error::OutOfGas)
|
||||||
|
} else {
|
||||||
|
Ok(())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -18,11 +18,20 @@ struct FakeExt {
|
|||||||
codes: HashMap<Address, Bytes>,
|
codes: HashMap<Address, Bytes>,
|
||||||
logs: Vec<FakeLogEntry>,
|
logs: Vec<FakeLogEntry>,
|
||||||
_suicides: HashSet<Address>,
|
_suicides: HashSet<Address>,
|
||||||
info: EnvInfo
|
info: EnvInfo,
|
||||||
|
_schedule: Schedule
|
||||||
}
|
}
|
||||||
|
|
||||||
impl FakeExt {
|
impl FakeExt {
|
||||||
fn new() -> Self { FakeExt::default() }
|
fn new() -> Self {
|
||||||
|
FakeExt::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for Schedule {
|
||||||
|
fn default() -> Self {
|
||||||
|
Schedule::new_homestead()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Ext for FakeExt {
|
impl Ext for FakeExt {
|
||||||
@ -77,7 +86,7 @@ impl Ext for FakeExt {
|
|||||||
}
|
}
|
||||||
|
|
||||||
fn schedule(&self) -> &Schedule {
|
fn schedule(&self) -> &Schedule {
|
||||||
unimplemented!();
|
&self._schedule
|
||||||
}
|
}
|
||||||
|
|
||||||
fn env_info(&self) -> &EnvInfo {
|
fn env_info(&self) -> &EnvInfo {
|
||||||
@ -85,6 +94,34 @@ impl Ext for FakeExt {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
|
||||||
|
fn test_stack_underflow() {
|
||||||
|
let address = Address::from_str("0f572e5295c57f15886f9b263e2f6d2d6c7b5ec6").unwrap();
|
||||||
|
let code = "01600055".from_hex().unwrap();
|
||||||
|
|
||||||
|
let mut params = ActionParams::new();
|
||||||
|
params.address = address.clone();
|
||||||
|
params.gas = U256::from(100_000);
|
||||||
|
params.code = code;
|
||||||
|
let mut ext = FakeExt::new();
|
||||||
|
|
||||||
|
let err = {
|
||||||
|
let vm : Box<evm::Evm> = Box::new(super::interpreter::Interpreter);
|
||||||
|
vm.exec(¶ms, &mut ext).unwrap_err()
|
||||||
|
};
|
||||||
|
|
||||||
|
match err {
|
||||||
|
evm::Error::StackUnderflow(wanted, stack) => {
|
||||||
|
assert_eq!(wanted, 2);
|
||||||
|
assert_eq!(stack, 0);
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
assert!(false, "Expected StackUndeflow")
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
macro_rules! evm_test(
|
macro_rules! evm_test(
|
||||||
($name_test: ident: $name_jit: ident, $name_int: ident) => {
|
($name_test: ident: $name_jit: ident, $name_int: ident) => {
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -216,7 +216,11 @@ impl<'a> Executive<'a> {
|
|||||||
match result {
|
match result {
|
||||||
Err(evm::Error::Internal) => Err(ExecutionError::Internal),
|
Err(evm::Error::Internal) => Err(ExecutionError::Internal),
|
||||||
// TODO [ToDr] BadJumpDestination @debris - how to handle that?
|
// TODO [ToDr] BadJumpDestination @debris - how to handle that?
|
||||||
Err(evm::Error::OutOfGas) | Err(evm::Error::BadJumpDestination) => {
|
Err(evm::Error::OutOfGas)
|
||||||
|
| Err(evm::Error::BadJumpDestination)
|
||||||
|
| Err(evm::Error::BadInstruction)
|
||||||
|
| Err(evm::Error::StackUnderflow(_, _))
|
||||||
|
| Err(evm::Error::OutOfStack(_, _)) => {
|
||||||
*self.state = backup;
|
*self.state = backup;
|
||||||
Ok(Executed {
|
Ok(Executed {
|
||||||
gas: t.gas,
|
gas: t.gas,
|
||||||
|
Loading…
Reference in New Issue
Block a user