eips.go 5.38 KB
Newer Older
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
// Copyright 2019 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"
21
	"sort"
22

23
	"github.com/ethereum/go-ethereum/params"
24
	"github.com/holiman/uint256"
25 26
)

27
var activators = map[int]func(*JumpTable){
28
	3529: enable3529,
29
	3198: enable3198,
30
	2929: enable2929,
31 32 33 34 35
	2200: enable2200,
	1884: enable1884,
	1344: enable1344,
}

36 37 38 39
// EnableEIP enables the given EIP on the config.
// This operation writes in-place, and callers need to ensure that the globally
// defined jump tables are not polluted.
func EnableEIP(eipNum int, jt *JumpTable) error {
40 41
	enablerFn, ok := activators[eipNum]
	if !ok {
42 43
		return fmt.Errorf("undefined eip %d", eipNum)
	}
44
	enablerFn(jt)
45 46 47
	return nil
}

48 49 50 51 52 53 54 55 56 57 58 59 60
func ValidEip(eipNum int) bool {
	_, ok := activators[eipNum]
	return ok
}
func ActivateableEips() []string {
	var nums []string
	for k := range activators {
		nums = append(nums, fmt.Sprintf("%d", k))
	}
	sort.Strings(nums)
	return nums
}

61 62 63 64 65 66 67
// enable1884 applies EIP-1884 to the given jump table:
// - Increase cost of BALANCE to 700
// - Increase cost of EXTCODEHASH to 700
// - Increase cost of SLOAD to 800
// - Define SELFBALANCE, with cost GasFastStep (5)
func enable1884(jt *JumpTable) {
	// Gas cost changes
68
	jt[SLOAD].constantGas = params.SloadGasEIP1884
69 70 71 72
	jt[BALANCE].constantGas = params.BalanceGasEIP1884
	jt[EXTCODEHASH].constantGas = params.ExtcodeHashGasEIP1884

	// New opcode
73
	jt[SELFBALANCE] = &operation{
74 75 76 77 78 79 80
		execute:     opSelfBalance,
		constantGas: GasFastStep,
		minStack:    minStack(0, 1),
		maxStack:    maxStack(0, 1),
	}
}

81 82 83
func opSelfBalance(pc *uint64, interpreter *EVMInterpreter, scope *ScopeContext) ([]byte, error) {
	balance, _ := uint256.FromBig(interpreter.evm.StateDB.GetBalance(scope.Contract.Address()))
	scope.Stack.push(balance)
84 85
	return nil, nil
}
86 87 88 89 90

// enable1344 applies EIP-1344 (ChainID Opcode)
// - Adds an opcode that returns the current chain’s EIP-155 unique identifier
func enable1344(jt *JumpTable) {
	// New opcode
91
	jt[CHAINID] = &operation{
92 93 94 95 96 97 98 99
		execute:     opChainID,
		constantGas: GasQuickStep,
		minStack:    minStack(0, 1),
		maxStack:    maxStack(0, 1),
	}
}

// opChainID implements CHAINID opcode
100
func opChainID(pc *uint64, interpreter *EVMInterpreter, scope *ScopeContext) ([]byte, error) {
101
	chainId, _ := uint256.FromBig(interpreter.evm.chainConfig.ChainID)
102
	scope.Stack.push(chainId)
103 104
	return nil, nil
}
105 106 107

// enable2200 applies EIP-2200 (Rebalance net-metered SSTORE)
func enable2200(jt *JumpTable) {
108
	jt[SLOAD].constantGas = params.SloadGasEIP2200
109 110
	jt[SSTORE].dynamicGas = gasSStoreEIP2200
}
111

112 113 114 115 116 117 118 119
// enable2929 enables "EIP-2929: Gas cost increases for state access opcodes"
// https://eips.ethereum.org/EIPS/eip-2929
func enable2929(jt *JumpTable) {
	jt[SSTORE].dynamicGas = gasSStoreEIP2929

	jt[SLOAD].constantGas = 0
	jt[SLOAD].dynamicGas = gasSLoadEIP2929

120
	jt[EXTCODECOPY].constantGas = params.WarmStorageReadCostEIP2929
121 122
	jt[EXTCODECOPY].dynamicGas = gasExtCodeCopyEIP2929

123
	jt[EXTCODESIZE].constantGas = params.WarmStorageReadCostEIP2929
124 125
	jt[EXTCODESIZE].dynamicGas = gasEip2929AccountCheck

126
	jt[EXTCODEHASH].constantGas = params.WarmStorageReadCostEIP2929
127 128
	jt[EXTCODEHASH].dynamicGas = gasEip2929AccountCheck

129
	jt[BALANCE].constantGas = params.WarmStorageReadCostEIP2929
130 131
	jt[BALANCE].dynamicGas = gasEip2929AccountCheck

132
	jt[CALL].constantGas = params.WarmStorageReadCostEIP2929
133 134
	jt[CALL].dynamicGas = gasCallEIP2929

135
	jt[CALLCODE].constantGas = params.WarmStorageReadCostEIP2929
136 137
	jt[CALLCODE].dynamicGas = gasCallCodeEIP2929

138
	jt[STATICCALL].constantGas = params.WarmStorageReadCostEIP2929
139 140
	jt[STATICCALL].dynamicGas = gasStaticCallEIP2929

141
	jt[DELEGATECALL].constantGas = params.WarmStorageReadCostEIP2929
142 143 144 145 146 147 148
	jt[DELEGATECALL].dynamicGas = gasDelegateCallEIP2929

	// This was previously part of the dynamic cost, but we're using it as a constantGas
	// factor here
	jt[SELFDESTRUCT].constantGas = params.SelfdestructGasEIP150
	jt[SELFDESTRUCT].dynamicGas = gasSelfdestructEIP2929
}
149 150 151 152 153 154 155 156 157

// enable3529 enabled "EIP-3529: Reduction in refunds":
// - Removes refunds for selfdestructs
// - Reduces refunds for SSTORE
// - Reduces max refunds to 20% gas
func enable3529(jt *JumpTable) {
	jt[SSTORE].dynamicGas = gasSStoreEIP3529
	jt[SELFDESTRUCT].dynamicGas = gasSelfdestructEIP3529
}
158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176

// enable3198 applies EIP-3198 (BASEFEE Opcode)
// - Adds an opcode that returns the current block's base fee.
func enable3198(jt *JumpTable) {
	// New opcode
	jt[BASEFEE] = &operation{
		execute:     opBaseFee,
		constantGas: GasQuickStep,
		minStack:    minStack(0, 1),
		maxStack:    maxStack(0, 1),
	}
}

// opBaseFee implements BASEFEE opcode
func opBaseFee(pc *uint64, interpreter *EVMInterpreter, scope *ScopeContext) ([]byte, error) {
	baseFee, _ := uint256.FromBig(interpreter.evm.Context.BaseFee)
	scope.Stack.push(baseFee)
	return nil, nil
}