mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2024-12-31 23:41:22 +00:00
143 lines
4.4 KiB
Go
143 lines
4.4 KiB
Go
|
// Copyright 2015 The go-ethereum Authors
|
||
|
// This file is part of the go-ethereum library.
|
||
|
//
|
||
|
// The go-ethereum library is free software: you can redistribute it and/or modify
|
||
|
// it under the terms of the GNU Lesser General Public License as published by
|
||
|
// the Free Software Foundation, either version 3 of the License, or
|
||
|
// (at your option) any later version.
|
||
|
//
|
||
|
// The go-ethereum library 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 Lesser General Public License for more details.
|
||
|
//
|
||
|
// You should have received a copy of the GNU Lesser General Public License
|
||
|
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
package vm
|
||
|
|
||
|
import (
|
||
|
"fmt"
|
||
|
"math/big"
|
||
|
|
||
|
"github.com/ethereum/go-ethereum/params"
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
GasQuickStep = big.NewInt(2)
|
||
|
GasFastestStep = big.NewInt(3)
|
||
|
GasFastStep = big.NewInt(5)
|
||
|
GasMidStep = big.NewInt(8)
|
||
|
GasSlowStep = big.NewInt(10)
|
||
|
GasExtStep = big.NewInt(20)
|
||
|
|
||
|
GasReturn = big.NewInt(0)
|
||
|
GasStop = big.NewInt(0)
|
||
|
|
||
|
GasContractByte = big.NewInt(200)
|
||
|
)
|
||
|
|
||
|
func baseCheck(op OpCode, stack *stack, gas *big.Int) error {
|
||
|
// PUSH and DUP are a bit special. They all cost the same but we do want to have checking on stack push limit
|
||
|
// PUSH is also allowed to calculate the same price for all PUSHes
|
||
|
// DUP requirements are handled elsewhere (except for the stack limit check)
|
||
|
if op >= PUSH1 && op <= PUSH32 {
|
||
|
op = PUSH1
|
||
|
}
|
||
|
if op >= DUP1 && op <= DUP16 {
|
||
|
op = DUP1
|
||
|
}
|
||
|
|
||
|
if r, ok := _baseCheck[op]; ok {
|
||
|
err := stack.require(r.stackPop)
|
||
|
if err != nil {
|
||
|
return err
|
||
|
}
|
||
|
|
||
|
if r.stackPush > 0 && len(stack.data)-r.stackPop+r.stackPush > int(params.StackLimit.Int64())+1 {
|
||
|
return fmt.Errorf("stack limit reached %d (%d)", len(stack.data), params.StackLimit.Int64())
|
||
|
}
|
||
|
|
||
|
gas.Add(gas, r.gas)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func toWordSize(size *big.Int) *big.Int {
|
||
|
tmp := new(big.Int)
|
||
|
tmp.Add(size, u256(31))
|
||
|
tmp.Div(tmp, u256(32))
|
||
|
return tmp
|
||
|
}
|
||
|
|
||
|
type req struct {
|
||
|
stackPop int
|
||
|
gas *big.Int
|
||
|
stackPush int
|
||
|
}
|
||
|
|
||
|
var _baseCheck = map[OpCode]req{
|
||
|
// opcode | stack pop | gas price | stack push
|
||
|
ADD: {2, GasFastestStep, 1},
|
||
|
LT: {2, GasFastestStep, 1},
|
||
|
GT: {2, GasFastestStep, 1},
|
||
|
SLT: {2, GasFastestStep, 1},
|
||
|
SGT: {2, GasFastestStep, 1},
|
||
|
EQ: {2, GasFastestStep, 1},
|
||
|
ISZERO: {1, GasFastestStep, 1},
|
||
|
SUB: {2, GasFastestStep, 1},
|
||
|
AND: {2, GasFastestStep, 1},
|
||
|
OR: {2, GasFastestStep, 1},
|
||
|
XOR: {2, GasFastestStep, 1},
|
||
|
NOT: {1, GasFastestStep, 1},
|
||
|
BYTE: {2, GasFastestStep, 1},
|
||
|
CALLDATALOAD: {1, GasFastestStep, 1},
|
||
|
CALLDATACOPY: {3, GasFastestStep, 1},
|
||
|
MLOAD: {1, GasFastestStep, 1},
|
||
|
MSTORE: {2, GasFastestStep, 0},
|
||
|
MSTORE8: {2, GasFastestStep, 0},
|
||
|
CODECOPY: {3, GasFastestStep, 0},
|
||
|
MUL: {2, GasFastStep, 1},
|
||
|
DIV: {2, GasFastStep, 1},
|
||
|
SDIV: {2, GasFastStep, 1},
|
||
|
MOD: {2, GasFastStep, 1},
|
||
|
SMOD: {2, GasFastStep, 1},
|
||
|
SIGNEXTEND: {2, GasFastStep, 1},
|
||
|
ADDMOD: {3, GasMidStep, 1},
|
||
|
MULMOD: {3, GasMidStep, 1},
|
||
|
JUMP: {1, GasMidStep, 0},
|
||
|
JUMPI: {2, GasSlowStep, 0},
|
||
|
EXP: {2, GasSlowStep, 1},
|
||
|
ADDRESS: {0, GasQuickStep, 1},
|
||
|
ORIGIN: {0, GasQuickStep, 1},
|
||
|
CALLER: {0, GasQuickStep, 1},
|
||
|
CALLVALUE: {0, GasQuickStep, 1},
|
||
|
CODESIZE: {0, GasQuickStep, 1},
|
||
|
GASPRICE: {0, GasQuickStep, 1},
|
||
|
COINBASE: {0, GasQuickStep, 1},
|
||
|
TIMESTAMP: {0, GasQuickStep, 1},
|
||
|
NUMBER: {0, GasQuickStep, 1},
|
||
|
CALLDATASIZE: {0, GasQuickStep, 1},
|
||
|
DIFFICULTY: {0, GasQuickStep, 1},
|
||
|
GASLIMIT: {0, GasQuickStep, 1},
|
||
|
POP: {1, GasQuickStep, 0},
|
||
|
PC: {0, GasQuickStep, 1},
|
||
|
MSIZE: {0, GasQuickStep, 1},
|
||
|
GAS: {0, GasQuickStep, 1},
|
||
|
BLOCKHASH: {1, GasExtStep, 1},
|
||
|
BALANCE: {1, GasExtStep, 1},
|
||
|
EXTCODESIZE: {1, GasExtStep, 1},
|
||
|
EXTCODECOPY: {4, GasExtStep, 0},
|
||
|
SLOAD: {1, params.SloadGas, 1},
|
||
|
SSTORE: {2, Zero, 0},
|
||
|
SHA3: {2, params.Sha3Gas, 1},
|
||
|
CREATE: {3, params.CreateGas, 1},
|
||
|
CALL: {7, params.CallGas, 1},
|
||
|
CALLCODE: {7, params.CallGas, 1},
|
||
|
JUMPDEST: {0, params.JumpdestGas, 0},
|
||
|
SUICIDE: {1, Zero, 0},
|
||
|
RETURN: {2, Zero, 0},
|
||
|
PUSH1: {0, GasFastestStep, 1},
|
||
|
DUP1: {0, Zero, 1},
|
||
|
}
|