state_transition.go 12.1 KB
Newer Older
F
Felix Lange 已提交
1
// Copyright 2014 The go-ethereum Authors
2
// This file is part of the go-ethereum library.
F
Felix Lange 已提交
3
//
4
// The go-ethereum library is free software: you can redistribute it and/or modify
F
Felix Lange 已提交
5 6 7 8
// 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.
//
9
// The go-ethereum library is distributed in the hope that it will be useful,
F
Felix Lange 已提交
10
// but WITHOUT ANY WARRANTY; without even the implied warranty of
11
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
F
Felix Lange 已提交
12 13 14
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
15
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
F
Felix Lange 已提交
16

O
obscuren 已提交
17
package core
18 19

import (
20
	"fmt"
21
	"math"
22
	"math/big"
23

O
obscuren 已提交
24
	"github.com/ethereum/go-ethereum/common"
25
	cmath "github.com/ethereum/go-ethereum/common/math"
26
	"github.com/ethereum/go-ethereum/core/types"
O
obscuren 已提交
27
	"github.com/ethereum/go-ethereum/core/vm"
28
	"github.com/ethereum/go-ethereum/params"
29 30
)

O
obscuren 已提交
31
/*
32 33 34
The State Transitioning Model

A state transition is a change made when a transaction is applied to the current world state
W
Wenbiao Zheng 已提交
35
The state transitioning model does all the necessary work to work out a valid new state root.
36 37 38 39 40 41 42 43 44 45 46 47

1) Nonce handling
2) Pre pay gas
3) Create a new state object if the recipient is \0*32
4) Value transfer
== If contract creation ==
  4a) Attempt to run transaction data
  4b) If valid, use result as code for the new state object
== end ==
5) Run Script section
6) Derive new state root
*/
48
type StateTransition struct {
49 50 51 52
	gp         *GasPool
	msg        Message
	gas        uint64
	gasPrice   *big.Int
53 54
	gasFeeCap  *big.Int
	gasTipCap  *big.Int
55
	initialGas uint64
56 57 58
	value      *big.Int
	data       []byte
	state      vm.StateDB
59
	evm        *vm.EVM
60 61
}

62
// Message represents a message sent to a contract.
63
type Message interface {
J
Jeffrey Wilcke 已提交
64
	From() common.Address
65
	To() *common.Address
66 67

	GasPrice() *big.Int
68 69
	GasFeeCap() *big.Int
	GasTipCap() *big.Int
70
	Gas() uint64
71 72 73
	Value() *big.Int

	Nonce() uint64
74
	CheckNonce() bool
75
	Data() []byte
76
	AccessList() types.AccessList
77
}
O
obscuren 已提交
78

79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113
// ExecutionResult includes all output after executing given evm
// message no matter the execution itself is successful or not.
type ExecutionResult struct {
	UsedGas    uint64 // Total used gas but include the refunded gas
	Err        error  // Any error encountered during the execution(listed in core/vm/errors.go)
	ReturnData []byte // Returned data from evm(function result or data supplied with revert opcode)
}

// Unwrap returns the internal evm error which allows us for further
// analysis outside.
func (result *ExecutionResult) Unwrap() error {
	return result.Err
}

// Failed returns the indicator whether the execution is successful or not
func (result *ExecutionResult) Failed() bool { return result.Err != nil }

// Return is a helper function to help caller distinguish between revert reason
// and function return. Return returns the data after execution if no error occurs.
func (result *ExecutionResult) Return() []byte {
	if result.Err != nil {
		return nil
	}
	return common.CopyBytes(result.ReturnData)
}

// Revert returns the concrete revert reason if the execution is aborted by `REVERT`
// opcode. Note the reason can be nil if no data supplied with revert opcode.
func (result *ExecutionResult) Revert() []byte {
	if result.Err != vm.ErrExecutionReverted {
		return nil
	}
	return common.CopyBytes(result.ReturnData)
}

114
// IntrinsicGas computes the 'intrinsic gas' for a message with the given data.
115
func IntrinsicGas(data []byte, accessList types.AccessList, isContractCreation bool, isHomestead, isEIP2028 bool) (uint64, error) {
116 117
	// Set the starting gas for the raw transaction
	var gas uint64
118
	if isContractCreation && isHomestead {
119
		gas = params.TxGasContractCreation
120
	} else {
121
		gas = params.TxGas
122
	}
123
	// Bump the required gas by the amount of transactional data
F
Felix Lange 已提交
124
	if len(data) > 0 {
125 126
		// Zero and non-zero bytes are priced differently
		var nz uint64
F
Felix Lange 已提交
127 128 129 130
		for _, byt := range data {
			if byt != 0 {
				nz++
			}
O
obscuren 已提交
131
		}
132
		// Make sure we don't exceed uint64 for all data combinations
133 134 135 136 137
		nonZeroGas := params.TxDataNonZeroGasFrontier
		if isEIP2028 {
			nonZeroGas = params.TxDataNonZeroGasEIP2028
		}
		if (math.MaxUint64-gas)/nonZeroGas < nz {
138
			return 0, ErrGasUintOverflow
139
		}
140
		gas += nz * nonZeroGas
141 142 143

		z := uint64(len(data)) - nz
		if (math.MaxUint64-gas)/params.TxDataZeroGas < z {
144
			return 0, ErrGasUintOverflow
145 146
		}
		gas += z * params.TxDataZeroGas
O
obscuren 已提交
147
	}
148 149 150 151
	if accessList != nil {
		gas += uint64(len(accessList)) * params.TxAccessListAddressGas
		gas += uint64(accessList.StorageKeys()) * params.TxAccessListStorageKeyGas
	}
152
	return gas, nil
O
obscuren 已提交
153 154
}

155
// NewStateTransition initialises and returns a new state transition object.
156
func NewStateTransition(evm *vm.EVM, msg Message, gp *GasPool) *StateTransition {
157
	return &StateTransition{
158 159 160 161 162 163 164 165 166
		gp:        gp,
		evm:       evm,
		msg:       msg,
		gasPrice:  msg.GasPrice(),
		gasFeeCap: msg.GasFeeCap(),
		gasTipCap: msg.GasTipCap(),
		value:     msg.Value(),
		data:      msg.Data(),
		state:     evm.StateDB,
167
	}
168 169 170 171 172 173 174 175 176
}

// ApplyMessage computes the new state by applying the given message
// against the old state within the environment.
//
// ApplyMessage returns the bytes returned by any EVM execution (if it took place),
// the gas used (which includes gas refunds) and an error if it failed. An error always
// indicates a core error meaning that the message would always fail for that particular
// state and would never be accepted within a block.
177
func ApplyMessage(evm *vm.EVM, msg Message, gp *GasPool) (*ExecutionResult, error) {
178
	return NewStateTransition(evm, msg, gp).TransitionDb()
179 180
}

181 182 183 184
// to returns the recipient of the message.
func (st *StateTransition) to() common.Address {
	if st.msg == nil || st.msg.To() == nil /* contract creation */ {
		return common.Address{}
185
	}
186
	return *st.msg.To()
187 188
}

C
changhong 已提交
189
func (st *StateTransition) buyGas() error {
190 191 192
	mgval := new(big.Int).SetUint64(st.msg.Gas())
	mgval = mgval.Mul(mgval, st.gasPrice)
	balanceCheck := mgval
193
	if st.gasFeeCap != nil {
194
		balanceCheck = new(big.Int).SetUint64(st.msg.Gas())
195
		balanceCheck = balanceCheck.Mul(balanceCheck, st.gasFeeCap)
196
		balanceCheck.Add(balanceCheck, st.value)
197 198
	}
	if have, want := st.state.GetBalance(st.msg.From()), balanceCheck; have.Cmp(want) < 0 {
199
		return fmt.Errorf("%w: address %v have %v want %v", ErrInsufficientFunds, st.msg.From().Hex(), have, want)
200
	}
201
	if err := st.gp.SubGas(st.msg.Gas()); err != nil {
202 203
		return err
	}
204
	st.gas += st.msg.Gas()
205

206
	st.initialGas = st.msg.Gas()
207
	st.state.SubBalance(st.msg.From(), mgval)
208 209 210
	return nil
}

C
changhong 已提交
211
func (st *StateTransition) preCheck() error {
212 213
	// Make sure this transaction's nonce is correct.
	if st.msg.CheckNonce() {
214 215 216 217 218 219 220
		stNonce := st.state.GetNonce(st.msg.From())
		if msgNonce := st.msg.Nonce(); stNonce < msgNonce {
			return fmt.Errorf("%w: address %v, tx: %d state: %d", ErrNonceTooHigh,
				st.msg.From().Hex(), msgNonce, stNonce)
		} else if stNonce > msgNonce {
			return fmt.Errorf("%w: address %v, tx: %d state: %d", ErrNonceTooLow,
				st.msg.From().Hex(), msgNonce, stNonce)
221
		}
O
obscuren 已提交
222
	}
223
	// Make sure that transaction gasFeeCap is greater than the baseFee (post london)
224
	if st.evm.ChainConfig().IsLondon(st.evm.Context.BlockNumber) {
225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244
		// Skip the checks if gas fields are zero and baseFee was explicitly disabled (eth_call)
		if !st.evm.Config.NoBaseFee || st.gasFeeCap.BitLen() > 0 || st.gasTipCap.BitLen() > 0 {
			if l := st.gasFeeCap.BitLen(); l > 256 {
				return fmt.Errorf("%w: address %v, maxFeePerGas bit length: %d", ErrFeeCapVeryHigh,
					st.msg.From().Hex(), l)
			}
			if l := st.gasTipCap.BitLen(); l > 256 {
				return fmt.Errorf("%w: address %v, maxPriorityFeePerGas bit length: %d", ErrTipVeryHigh,
					st.msg.From().Hex(), l)
			}
			if st.gasFeeCap.Cmp(st.gasTipCap) < 0 {
				return fmt.Errorf("%w: address %v, maxPriorityFeePerGas: %s, maxFeePerGas: %s", ErrTipAboveFeeCap,
					st.msg.From().Hex(), st.gasTipCap, st.gasFeeCap)
			}
			// This will panic if baseFee is nil, but basefee presence is verified
			// as part of header validation.
			if st.gasFeeCap.Cmp(st.evm.Context.BaseFee) < 0 {
				return fmt.Errorf("%w: address %v, maxFeePerGas: %s baseFee: %s", ErrFeeCapTooLow,
					st.msg.From().Hex(), st.gasFeeCap, st.evm.Context.BaseFee)
			}
245 246
		}
	}
C
changhong 已提交
247
	return st.buyGas()
O
obscuren 已提交
248 249
}

250
// TransitionDb will transition the state by applying the current message and
251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276
// returning the evm execution result with following fields.
//
// - used gas:
//      total gas used (including gas being refunded)
// - returndata:
//      the returned data from evm
// - concrete execution error:
//      various **EVM** error which aborts the execution,
//      e.g. ErrOutOfGas, ErrExecutionReverted
//
// However if any consensus issue encountered, return the error directly with
// nil evm execution result.
func (st *StateTransition) TransitionDb() (*ExecutionResult, error) {
	// First check this message satisfies all consensus rules before
	// applying the message. The rules include these clauses
	//
	// 1. the nonce of the message caller is correct
	// 2. caller has enough balance to cover transaction fee(gaslimit * gasprice)
	// 3. the amount of gas required is available in the block
	// 4. the purchased gas is enough to cover intrinsic usage
	// 5. there is no overflow when calculating intrinsic gas
	// 6. caller has enough balance to cover asset transfer for **topmost** call

	// Check clauses 1-3, buy gas if everything is correct
	if err := st.preCheck(); err != nil {
		return nil, err
O
obscuren 已提交
277
	}
C
changhong 已提交
278
	msg := st.msg
279
	sender := vm.AccountRef(msg.From())
280 281
	homestead := st.evm.ChainConfig().IsHomestead(st.evm.Context.BlockNumber)
	istanbul := st.evm.ChainConfig().IsIstanbul(st.evm.Context.BlockNumber)
282
	contractCreation := msg.To() == nil
C
changhong 已提交
283

284
	// Check clauses 4-5, subtract intrinsic gas if everything is correct
285
	gas, err := IntrinsicGas(st.data, st.msg.AccessList(), contractCreation, homestead, istanbul)
286
	if err != nil {
287
		return nil, err
288
	}
289
	if st.gas < gas {
290
		return nil, fmt.Errorf("%w: have %d, want %d", ErrIntrinsicGas, st.gas, gas)
291
	}
292
	st.gas -= gas
293

294
	// Check clause 6
295
	if msg.Value().Sign() > 0 && !st.evm.Context.CanTransfer(st.state, msg.From(), msg.Value()) {
296
		return nil, fmt.Errorf("%w: address %v", ErrInsufficientFundsForTransfer, msg.From().Hex())
297
	}
298 299

	// Set up the initial access list.
300 301
	if rules := st.evm.ChainConfig().Rules(st.evm.Context.BlockNumber); rules.IsBerlin {
		st.state.PrepareAccessList(msg.From(), msg.To(), vm.ActivePrecompiles(rules), msg.AccessList())
302
	}
303
	var (
304 305
		ret   []byte
		vmerr error // vm errors do not effect consensus and are therefore not assigned to err
306
	)
307
	if contractCreation {
308
		ret, _, st.gas, vmerr = st.evm.Create(sender, st.data, st.gas, st.value)
O
obscuren 已提交
309
	} else {
O
obscuren 已提交
310
		// Increment the nonce for the next transaction
311
		st.state.SetNonce(msg.From(), st.state.GetNonce(sender.Address())+1)
312
		ret, st.gas, vmerr = st.evm.Call(sender, st.to(), st.data, st.gas, st.value)
313
	}
314
	if !st.evm.ChainConfig().IsLondon(st.evm.Context.BlockNumber) {
315 316 317 318 319 320
		// Before EIP-3529: refunds were capped to gasUsed / 2
		st.refundGas(params.RefundQuotient)
	} else {
		// After EIP-3529: refunds are capped to gasUsed / 5
		st.refundGas(params.RefundQuotientEIP3529)
	}
321 322
	effectiveTip := st.gasPrice
	if st.evm.ChainConfig().IsLondon(st.evm.Context.BlockNumber) {
323
		effectiveTip = cmath.BigMin(st.gasTipCap, new(big.Int).Sub(st.gasFeeCap, st.evm.Context.BaseFee))
324 325
	}
	st.state.AddBalance(st.evm.Context.Coinbase, new(big.Int).Mul(new(big.Int).SetUint64(st.gasUsed()), effectiveTip))
O
obscuren 已提交
326

327 328 329 330 331
	return &ExecutionResult{
		UsedGas:    st.gasUsed(),
		Err:        vmerr,
		ReturnData: ret,
	}, nil
O
obscuren 已提交
332
}
O
obscuren 已提交
333

334 335 336
func (st *StateTransition) refundGas(refundQuotient uint64) {
	// Apply refund counter, capped to a refund quotient
	refund := st.gasUsed() / refundQuotient
337 338 339 340
	if refund > st.state.GetRefund() {
		refund = st.state.GetRefund()
	}
	st.gas += refund
341

342 343
	// Return ETH for remaining gas, exchanged at the original rate.
	remaining := new(big.Int).Mul(new(big.Int).SetUint64(st.gas), st.gasPrice)
344
	st.state.AddBalance(st.msg.From(), remaining)
O
obscuren 已提交
345

346 347
	// Also return remaining gas to the block gas counter so it is
	// available for the next transaction.
348
	st.gp.AddGas(st.gas)
O
obscuren 已提交
349 350
}

351 352 353
// gasUsed returns the amount of gas used up by the state transition.
func (st *StateTransition) gasUsed() uint64 {
	return st.initialGas - st.gas
O
obscuren 已提交
354
}