state_transition.go 11.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
	feeCap     *big.Int
	tip        *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
	FeeCap() *big.Int
	Tip() *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
		gp:       gp,
		evm:      evm,
		msg:      msg,
		gasPrice: msg.GasPrice(),
162 163
		feeCap:   msg.FeeCap(),
		tip:      msg.Tip(),
164 165 166
		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
	mgval := new(big.Int).Mul(new(big.Int).SetUint64(st.msg.Gas()), st.gasPrice)
191 192
	if have, want := st.state.GetBalance(st.msg.From()), mgval; have.Cmp(want) < 0 {
		return fmt.Errorf("%w: address %v have %v want %v", ErrInsufficientFunds, st.msg.From().Hex(), have, want)
193
	}
194
	if err := st.gp.SubGas(st.msg.Gas()); err != nil {
195 196
		return err
	}
197
	st.gas += st.msg.Gas()
198

199
	st.initialGas = st.msg.Gas()
200
	st.state.SubBalance(st.msg.From(), mgval)
201 202 203
	return nil
}

C
changhong 已提交
204
func (st *StateTransition) preCheck() error {
205 206
	// Make sure this transaction's nonce is correct.
	if st.msg.CheckNonce() {
207 208 209 210 211 212 213
		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)
214
		}
O
obscuren 已提交
215
	}
216 217 218 219 220 221 222 223 224
	// Make sure that transaction feeCap is greater than the baseFee (post london)
	if st.evm.ChainConfig().IsLondon(st.evm.Context.BlockNumber) {
		// This will panic if baseFee is nil, but basefee presence is verified
		// as part of header validation.
		if st.feeCap.Cmp(st.evm.Context.BaseFee) < 0 {
			return fmt.Errorf("%w: address %v, feeCap: %s baseFee: %s", ErrFeeCapTooLow,
				st.msg.From().Hex(), st.feeCap, st.evm.Context.BaseFee)
		}
	}
C
changhong 已提交
225
	return st.buyGas()
O
obscuren 已提交
226 227
}

228
// TransitionDb will transition the state by applying the current message and
229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254
// 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 已提交
255
	}
C
changhong 已提交
256
	msg := st.msg
257
	sender := vm.AccountRef(msg.From())
258 259
	homestead := st.evm.ChainConfig().IsHomestead(st.evm.Context.BlockNumber)
	istanbul := st.evm.ChainConfig().IsIstanbul(st.evm.Context.BlockNumber)
260
	eip3529 := st.evm.ChainConfig().IsLondon(st.evm.Context.BlockNumber)
261
	contractCreation := msg.To() == nil
C
changhong 已提交
262

263
	// Check clauses 4-5, subtract intrinsic gas if everything is correct
264
	gas, err := IntrinsicGas(st.data, st.msg.AccessList(), contractCreation, homestead, istanbul)
265
	if err != nil {
266
		return nil, err
267
	}
268
	if st.gas < gas {
269
		return nil, fmt.Errorf("%w: have %d, want %d", ErrIntrinsicGas, st.gas, gas)
270
	}
271
	st.gas -= gas
272

273
	// Check clause 6
274
	if msg.Value().Sign() > 0 && !st.evm.Context.CanTransfer(st.state, msg.From(), msg.Value()) {
275
		return nil, fmt.Errorf("%w: address %v", ErrInsufficientFundsForTransfer, msg.From().Hex())
276
	}
277 278

	// Set up the initial access list.
279 280
	if rules := st.evm.ChainConfig().Rules(st.evm.Context.BlockNumber); rules.IsBerlin {
		st.state.PrepareAccessList(msg.From(), msg.To(), vm.ActivePrecompiles(rules), msg.AccessList())
281
	}
282
	var (
283 284
		ret   []byte
		vmerr error // vm errors do not effect consensus and are therefore not assigned to err
285
	)
286
	if contractCreation {
287
		ret, _, st.gas, vmerr = st.evm.Create(sender, st.data, st.gas, st.value)
O
obscuren 已提交
288
	} else {
O
obscuren 已提交
289
		// Increment the nonce for the next transaction
290
		st.state.SetNonce(msg.From(), st.state.GetNonce(sender.Address())+1)
291
		ret, st.gas, vmerr = st.evm.Call(sender, st.to(), st.data, st.gas, st.value)
292
	}
293 294 295 296 297 298 299
	if !eip3529 {
		// 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)
	}
300 301 302 303 304
	effectiveTip := st.gasPrice
	if st.evm.ChainConfig().IsLondon(st.evm.Context.BlockNumber) {
		effectiveTip = cmath.BigMin(st.tip, new(big.Int).Sub(st.feeCap, st.evm.Context.BaseFee))
	}
	st.state.AddBalance(st.evm.Context.Coinbase, new(big.Int).Mul(new(big.Int).SetUint64(st.gasUsed()), effectiveTip))
O
obscuren 已提交
305

306 307 308 309 310
	return &ExecutionResult{
		UsedGas:    st.gasUsed(),
		Err:        vmerr,
		ReturnData: ret,
	}, nil
O
obscuren 已提交
311
}
O
obscuren 已提交
312

313 314 315
func (st *StateTransition) refundGas(refundQuotient uint64) {
	// Apply refund counter, capped to a refund quotient
	refund := st.gasUsed() / refundQuotient
316 317 318 319
	if refund > st.state.GetRefund() {
		refund = st.state.GetRefund()
	}
	st.gas += refund
320

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

325 326
	// Also return remaining gas to the block gas counter so it is
	// available for the next transaction.
327
	st.gp.AddGas(st.gas)
O
obscuren 已提交
328 329
}

330 331 332
// gasUsed returns the amount of gas used up by the state transition.
func (st *StateTransition) gasUsed() uint64 {
	return st.initialGas - st.gas
O
obscuren 已提交
333
}