Documentation ¶
Overview ¶
Package vm implements the Ethereum Virtual Machine.
The vm package implements two EVMs, a byte code VM and a JIT VM. The BC (Byte Code) VM loops over a set of bytes and executes them according to the set of rules defined in the Ethereum yellow paper. When the BC VM is invoked it invokes the JIT VM in a separate goroutine and compiles the byte code in JIT instructions.
The JIT VM, when invoked, loops around a set of pre-defined instructions until it either runs of gas, causes an internal error, returns or stops.
The JIT optimiser attempts to pre-compile instructions in to chunks or segments such as multiple PUSH operations and static JUMPs. It does this by analysing the opcodes and attempts to match certain regions to known sets. Whenever the optimiser finds said segments it creates a new instruction and replaces the first occurrence in the sequence.
Index ¶
- Constants
- Variables
- func Disassemble(script []byte) (asm []string)
- func PrecompiledContracts() map[string]*PrecompiledAccount
- func PrecompiledContractsAtlantis() map[string]*PrecompiledAccount
- type Account
- type Contract
- func (c *Contract) Address() common.Address
- func (c *Contract) AsDelegate() *Contract
- func (c *Contract) Caller() common.Address
- func (c *Contract) Finalise()
- func (self *Contract) ForEachStorage(cb func(key, value common.Hash) bool)
- func (c *Contract) GetByte(n uint64) byte
- func (c *Contract) GetOp(n uint64) OpCode
- func (c *Contract) ReturnGas(gas, price *big.Int)
- func (self *Contract) SetCallCode(addr *common.Address, hash common.Hash, code []byte)
- func (self *Contract) SetCode(hash common.Hash, code []byte)
- func (c *Contract) UseGas(gas *big.Int) (ok bool)
- func (c *Contract) Value() *big.Int
- type ContractRef
- type Database
- type EVM
- type Environment
- type GasTable
- type Log
- type LogForStorage
- type Logs
- type Memory
- type OpCode
- type PrecompiledAccount
- type RuleSet
- type Type
- type VirtualMachine
- type Vm
Constants ¶
const ( EcrecoverGas uint64 = 3000 // Elliptic curve sender recovery gas price Sha256BaseGas uint64 = 60 // Base price for a SHA256 operation Sha256PerWordGas uint64 = 12 // Per-word price for a SHA256 operation Ripemd160BaseGas uint64 = 600 // Base price for a RIPEMD160 operation Ripemd160PerWordGas uint64 = 120 // Per-word price for a RIPEMD160 operation IdentityBaseGas uint64 = 15 // Base price for a data copy operation IdentityPerWordGas uint64 = 3 // Per-work price for a data copy operation ModExpQuadCoeffDiv uint64 = 20 // Divisor for the quadratic particle of the big int modular exponentiation Bn256AddGas uint64 = 500 // Gas needed for an elliptic curve addition Bn256ScalarMulGas uint64 = 40000 // Gas needed for an elliptic curve scalar multiplication Bn256PairingBaseGas uint64 = 100000 // Base price for an elliptic curve pairing check Bn256PairingPerPointGas uint64 = 80000 // Per-point price for an elliptic curve pairing check )
const ( // 0xf0 range - closures CREATE OpCode = 0xf0 + iota CALL CALLCODE RETURN DELEGATECALL STATICCALL = 0xfa REVERT = 0xfd SUICIDE = 0xff )
Variables ¶
var ( Pow256 = common.BigPow(2, 256) // Pow256 is 2**256 U256 = common.U256 // Shortcut to common.U256 S256 = common.S256 // Shortcut to common.S256 One = common.Big1 // Shortcut to common.Big1 )
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) )
var ( OutOfGasError = errors.New("Out of gas") CodeStoreOutOfGasError = errors.New("Contract creation code storage out of gas") ErrRevert = errors.New("Execution reverted") )
var PrecompiledAtlantis = func() map[string]*PrecompiledAccount { a := PrecompiledContracts() b := PrecompiledContractsAtlantis() precompiles := make(map[string]*PrecompiledAccount) for k, c := range a { precompiles[k] = c } for k, c := range b { precompiles[k] = c } return precompiles }()
var PrecompiledPreAtlantis = PrecompiledContracts()
Precompiled contains the default set of ethereum contracts
Functions ¶
func Disassemble ¶
Disassemble disassembles the byte code and returns the string representation (human readable opcodes).
func PrecompiledContracts ¶
func PrecompiledContracts() map[string]*PrecompiledAccount
PrecompiledContractsPreAtlantis returns the default set of precompiled ethereum contracts defined by the ethereum yellow paper pre-Atlantis.
func PrecompiledContractsAtlantis ¶
func PrecompiledContractsAtlantis() map[string]*PrecompiledAccount
PrecompiledContractsAtlantis returns the set of precompiled contracts introducted in Atlantis
Types ¶
type Account ¶
type Account interface { SubBalance(amount *big.Int) AddBalance(amount *big.Int) SetBalance(*big.Int) SetNonce(uint64) Balance() *big.Int Address() common.Address ReturnGas(*big.Int, *big.Int) SetCode(common.Hash, []byte) ForEachStorage(cb func(key, value common.Hash) bool) Value() *big.Int }
Account represents a contract or basic ethereum account.
type Contract ¶
type Contract struct { // CallerAddress is the result of the caller which initialised this // contract. However when the "call method" is delegated this value // needs to be initialised to that of the caller's caller. CallerAddress common.Address Code []byte CodeHash common.Hash CodeAddr *common.Address Input []byte Gas, UsedGas, Price *big.Int Args []byte DelegateCall bool // contains filtered or unexported fields }
Contract represents an ethereum contract in the state database. It contains the the contract code, calling arguments. Contract implements ContractRef
func NewContract ¶
func NewContract(caller ContractRef, object ContractRef, value, gas, price *big.Int) *Contract
NewContract returns a new contract environment for the execution of EVM.
func (*Contract) AsDelegate ¶
AsDelegate sets the contract to be a delegate call and returns the current contract (for chaining calls)
func (*Contract) Caller ¶
Caller returns the caller of the contract.
Caller will recursively call caller when the contract is a delegate call, including that of caller's caller.
func (*Contract) Finalise ¶
func (c *Contract) Finalise()
Finalise finalises the contract and returning any remaining gas to the original caller.
func (*Contract) ForEachStorage ¶
EachStorage iterates the contract's storage and calls a method for every key value pair.
func (*Contract) SetCallCode ¶
SetCallCode sets the code of the contract and address of the backing data object
type ContractRef ¶
type ContractRef interface { ReturnGas(*big.Int, *big.Int) Address() common.Address Value() *big.Int SetCode(common.Hash, []byte) ForEachStorage(callback func(key, value common.Hash) bool) }
ContractRef is a reference to the contract's backing object
type Database ¶
type Database interface { GetAccount(common.Address) Account CreateAccount(common.Address) Account AddBalance(common.Address, *big.Int) GetBalance(common.Address) *big.Int GetNonce(common.Address) uint64 SetNonce(common.Address, uint64) GetCodeHash(common.Address) common.Hash GetCodeSize(common.Address) int GetCode(common.Address) []byte SetCode(common.Address, []byte) AddRefund(*big.Int) GetRefund() *big.Int GetState(common.Address, common.Hash) common.Hash SetState(common.Address, common.Hash, common.Hash) Suicide(common.Address) bool HasSuicided(common.Address) bool // Exist reports whether the given account exists in state. // Notably this should also return true for suicided accounts. Exist(common.Address) bool Empty(common.Address) bool }
Database is a EVM database for full state querying.
type EVM ¶
type EVM struct {
// contains filtered or unexported fields
}
EVM is used to run Ethereum based contracts and will utilise the passed environment to query external sources for state information. The EVM will run the byte code VM or JIT VM based on the passed configuration.
func (*EVM) RunPrecompiled ¶
func (evm *EVM) RunPrecompiled(p *PrecompiledAccount, input []byte, contract *Contract) (ret []byte, err error)
RunPrecompile runs and evaluate the output of a precompiled contract defined in contracts.go
type Environment ¶
type Environment interface { // The current ruleset RuleSet() RuleSet // The state database Db() Database // Creates a restorable snapshot SnapshotDatabase() int // Set database to previous snapshot RevertToSnapshot(int) // Address of the original invoker (first occurrence of the VM invoker) Origin() common.Address // The block number this VM is invoked on BlockNumber() *big.Int // The n'th hash ago from this block number GetHash(uint64) common.Hash // The handler's address Coinbase() common.Address // The current time (block time) Time() *big.Int // Difficulty set on the current block Difficulty() *big.Int // The gas limit of the block GasLimit() *big.Int // Determines whether it's possible to transact CanTransfer(from common.Address, balance *big.Int) bool // Transfers amount from one account to the other Transfer(from, to Account, amount *big.Int) // Adds a LOG to the state AddLog(*Log) // Type of the VM Vm() Vm // Get the curret calling depth Depth() int // Set the current calling depth SetDepth(i int) // Get previous return data ReturnData() []byte // Set previous return data SetReturnData([]byte) // Call another contract Call(me ContractRef, addr common.Address, data []byte, gas, price, value *big.Int) ([]byte, error) // Take another's contract code and execute within our own context CallCode(me ContractRef, addr common.Address, data []byte, gas, price, value *big.Int) ([]byte, error) // Same as CallCode except sender and value is propagated from parent to child scope DelegateCall(me ContractRef, addr common.Address, data []byte, gas, price *big.Int) ([]byte, error) // Call another contract and disallow any state changing operations StaticCall(me ContractRef, addr common.Address, data []byte, gas, price *big.Int) ([]byte, error) // Create a new contract Create(me ContractRef, data []byte, gas, price, value *big.Int) ([]byte, common.Address, error) }
Environment is an EVM requirement and helper which allows access to outside information such as states.
type GasTable ¶
type GasTable struct { ExtcodeSize *big.Int ExtcodeCopy *big.Int Balance *big.Int SLoad *big.Int Calls *big.Int Suicide *big.Int ExpByte *big.Int // CreateBySuicide occurs when the // refunded account is one that does // not exist. This logic is similar // to call. May be left nil. Nil means // not charged. CreateBySuicide *big.Int }
type Log ¶
type Log struct { // Consensus fields Address common.Address Topics []common.Hash Data []byte // Derived fields (don't reorder!) BlockNumber uint64 TxHash common.Hash TxIndex uint BlockHash common.Hash Index uint }
func (*Log) MarshalJSON ¶
type LogForStorage ¶
type LogForStorage Log
LogForStorage is a wrapper around a Log that flattens and parses the entire content of a log, as opposed to only the consensus fields originally (by hiding the rlp interface methods).
type Memory ¶
type Memory struct {
// contains filtered or unexported fields
}
Memory implements a simple memory model for the ethereum virtual machine.
type OpCode ¶
type OpCode byte
OpCode is an EVM opcode
const ( // 0x60 range PUSH1 OpCode = 0x60 + iota PUSH2 PUSH3 PUSH4 PUSH5 PUSH6 PUSH7 PUSH8 PUSH9 PUSH10 PUSH11 PUSH12 PUSH13 PUSH14 PUSH15 PUSH16 PUSH17 PUSH18 PUSH19 PUSH20 PUSH21 PUSH22 PUSH23 PUSH24 PUSH25 PUSH26 PUSH27 PUSH28 PUSH29 PUSH30 PUSH31 PUSH32 DUP1 DUP2 DUP3 DUP4 DUP5 DUP6 DUP7 DUP8 DUP9 DUP10 DUP11 DUP12 DUP13 DUP14 DUP15 DUP16 SWAP1 SWAP2 SWAP3 SWAP4 SWAP5 SWAP6 SWAP7 SWAP8 SWAP9 SWAP10 SWAP11 SWAP12 SWAP13 SWAP14 SWAP15 SWAP16 )
func StringToOp ¶
func (OpCode) IsStaticJump ¶
type PrecompiledAccount ¶
type PrecompiledAccount struct { Gas func(in []byte) *big.Int // contains filtered or unexported fields }
PrecompiledAccount represents a native ethereum contract
type RuleSet ¶
type RuleSet interface { IsHomestead(*big.Int) bool IsAtlantis(*big.Int) bool // GasTable returns the gas prices for this phase, which is based on // block number passed in. GasTable(*big.Int) *GasTable }
RuleSet is an interface that defines the current rule set during the execution of the EVM instructions (e.g. whether it's homestead)
type VirtualMachine ¶
VirtualMachine is an EVM interface