2015-07-07 02:54:22 +02:00
// Copyright 2014 The go-ethereum Authors
2015-07-22 18:48:40 +02:00
// This file is part of the go-ethereum library.
2015-07-07 02:54:22 +02:00
//
2015-07-23 18:35:11 +02:00
// The go-ethereum library is free software: you can redistribute it and/or modify
2015-07-07 02:54:22 +02:00
// 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.
//
2015-07-22 18:48:40 +02:00
// The go-ethereum library is distributed in the hope that it will be useful,
2015-07-07 02:54:22 +02:00
// but WITHOUT ANY WARRANTY; without even the implied warranty of
2015-07-22 18:48:40 +02:00
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
2015-07-07 02:54:22 +02:00
// GNU Lesser General Public License for more details.
//
// You should have received a copy of the GNU Lesser General Public License
2015-07-22 18:48:40 +02:00
// along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
2015-07-07 02:54:22 +02:00
2014-12-04 10:28:02 +01:00
package core
2014-06-13 12:58:01 +02:00
import (
2020-05-19 13:28:20 +03:00
"fmt"
2021-12-07 17:24:59 +00:00
"math/bits"
2015-03-09 18:05:56 +01:00
2022-02-08 00:30:46 +03:00
"github.com/ledgerwatch/erigon/consensus"
2020-05-26 18:53:50 +02:00
"github.com/holiman/uint256"
2021-05-21 01:25:53 +07:00
"github.com/ledgerwatch/erigon/common"
cmath "github.com/ledgerwatch/erigon/common/math"
"github.com/ledgerwatch/erigon/core/types"
"github.com/ledgerwatch/erigon/core/vm"
2021-10-13 03:30:09 +02:00
"github.com/ledgerwatch/erigon/crypto"
2021-05-21 01:25:53 +07:00
"github.com/ledgerwatch/erigon/params"
2014-06-13 12:58:01 +02:00
)
2021-10-13 03:30:09 +02:00
var emptyCodeHash = crypto . Keccak256Hash ( nil )
2014-06-13 13:06:27 +02:00
/ *
2015-09-02 12:55:11 +02:00
The State Transitioning Model
A state transition is a change made when a transaction is applied to the current world state
2018-07-24 18:17:12 +08:00
The state transitioning model does all the necessary work to work out a valid new state root .
2015-09-02 12:55:11 +02:00
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 ==
2022-08-10 19:04:13 +07:00
4 a ) Attempt to run transaction data
4 b ) If valid , use result as code for the new state object
2015-09-02 12:55:11 +02:00
== end ==
5 ) Run Script section
6 ) Derive new state root
* /
2014-06-13 12:58:01 +02:00
type StateTransition struct {
2017-01-04 20:17:24 +01:00
gp * GasPool
msg Message
gas uint64
2020-06-04 09:43:08 +02:00
gasPrice * uint256 . Int
2021-07-11 04:05:56 +00:00
gasFeeCap * uint256 . Int
2021-04-22 18:11:37 +01:00
tip * uint256 . Int
2017-11-13 13:47:27 +02:00
initialGas uint64
2020-05-26 18:53:50 +02:00
value * uint256 . Int
2017-01-04 20:17:24 +01:00
data [ ] byte
2019-05-27 14:51:49 +01:00
state vm . IntraBlockState
2021-12-06 17:58:53 +03:00
evm vm . VMInterface
2021-07-14 21:34:14 +07:00
//some pre-allocated intermediate variables
sharedBuyGas * uint256 . Int
sharedBuyGasBalance * uint256 . Int
2022-02-08 00:30:46 +03:00
isParlia bool
isBor bool
2014-12-18 15:18:13 +01:00
}
2015-03-17 12:00:29 +01:00
// Message represents a message sent to a contract.
2014-12-18 15:18:13 +01:00
type Message interface {
2016-11-02 13:44:13 +01:00
From ( ) common . Address
2015-03-17 12:00:29 +01:00
To ( ) * common . Address
2014-12-18 15:18:13 +01:00
2020-06-04 09:43:08 +02:00
GasPrice ( ) * uint256 . Int
2021-04-22 18:11:37 +01:00
FeeCap ( ) * uint256 . Int
Tip ( ) * uint256 . Int
2017-11-13 13:47:27 +02:00
Gas ( ) uint64
2020-06-04 09:43:08 +02:00
Value ( ) * uint256 . Int
2014-12-18 15:18:13 +01:00
Nonce ( ) uint64
2016-07-11 11:58:10 +02:00
CheckNonce ( ) bool
2014-12-18 15:18:13 +01:00
Data ( ) [ ] byte
2021-02-25 07:26:57 -07:00
AccessList ( ) types . AccessList
2014-06-13 12:58:01 +02:00
}
2015-03-18 13:00:01 +01:00
2020-04-22 16:25:36 +08:00
// 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 )
}
2017-11-13 13:47:27 +02:00
// IntrinsicGas computes the 'intrinsic gas' for a message with the given data.
2021-02-25 07:26:57 -07:00
func IntrinsicGas ( data [ ] byte , accessList types . AccessList , isContractCreation bool , isHomestead , isEIP2028 bool ) ( uint64 , error ) {
2017-11-13 13:47:27 +02:00
// Set the starting gas for the raw transaction
var gas uint64
2021-02-25 07:26:57 -07:00
if isContractCreation && isHomestead {
2017-11-13 13:47:27 +02:00
gas = params . TxGasContractCreation
2015-11-27 15:40:29 +01:00
} else {
2017-11-13 13:47:27 +02:00
gas = params . TxGas
2015-11-27 15:40:29 +01:00
}
2021-12-07 17:24:59 +00:00
// Auxiliary variables for overflow protection
var product , overflow uint64
2017-11-13 13:47:27 +02:00
// Bump the required gas by the amount of transactional data
2015-06-25 14:46:44 +02:00
if len ( data ) > 0 {
2017-11-13 13:47:27 +02:00
// Zero and non-zero bytes are priced differently
var nz uint64
2015-06-25 14:46:44 +02:00
for _ , byt := range data {
if byt != 0 {
nz ++
}
2015-04-08 20:47:32 +02:00
}
2017-11-13 13:47:27 +02:00
// Make sure we don't exceed uint64 for all data combinations
2019-08-14 20:53:21 +08:00
nonZeroGas := params . TxDataNonZeroGasFrontier
if isEIP2028 {
nonZeroGas = params . TxDataNonZeroGasEIP2028
}
2021-12-07 17:24:59 +00:00
overflow , product = bits . Mul64 ( nz , nonZeroGas )
if overflow != 0 {
return 0 , ErrGasUintOverflow
}
gas , overflow = bits . Add64 ( gas , product , 0 )
if overflow != 0 {
2020-04-22 16:25:36 +08:00
return 0 , ErrGasUintOverflow
2017-11-13 13:47:27 +02:00
}
z := uint64 ( len ( data ) ) - nz
2021-12-07 17:24:59 +00:00
overflow , product = bits . Mul64 ( z , params . TxDataZeroGas )
if overflow != 0 {
return 0 , ErrGasUintOverflow
}
gas , overflow = bits . Add64 ( gas , product , 0 )
if overflow != 0 {
2020-04-22 16:25:36 +08:00
return 0 , ErrGasUintOverflow
2017-11-13 13:47:27 +02:00
}
2015-04-08 20:47:32 +02:00
}
2021-02-25 07:26:57 -07:00
if accessList != nil {
2021-12-07 17:24:59 +00:00
overflow , product = bits . Mul64 ( uint64 ( len ( accessList ) ) , params . TxAccessListAddressGas )
if overflow != 0 {
return 0 , ErrGasUintOverflow
}
gas , overflow = bits . Add64 ( gas , product , 0 )
if overflow != 0 {
return 0 , ErrGasUintOverflow
}
overflow , product = bits . Mul64 ( uint64 ( accessList . StorageKeys ( ) ) , params . TxAccessListStorageKeyGas )
if overflow != 0 {
return 0 , ErrGasUintOverflow
}
gas , overflow = bits . Add64 ( gas , product , 0 )
if overflow != 0 {
return 0 , ErrGasUintOverflow
}
2021-02-25 07:26:57 -07:00
}
2017-11-13 13:47:27 +02:00
return gas , nil
2015-04-08 20:47:32 +02:00
}
2016-04-01 00:30:21 +02:00
// NewStateTransition initialises and returns a new state transition object.
2021-12-06 17:58:53 +03:00
func NewStateTransition ( evm vm . VMInterface , msg Message , gp * GasPool ) * StateTransition {
2022-02-08 00:30:46 +03:00
isParlia := evm . ChainConfig ( ) . Parlia != nil
isBor := evm . ChainConfig ( ) . Bor != nil
2016-04-01 00:30:21 +02:00
return & StateTransition {
2021-07-11 04:05:56 +00:00
gp : gp ,
evm : evm ,
msg : msg ,
gasPrice : msg . GasPrice ( ) ,
gasFeeCap : msg . FeeCap ( ) ,
tip : msg . Tip ( ) ,
value : msg . Value ( ) ,
data : msg . Data ( ) ,
2021-12-06 17:58:53 +03:00
state : evm . IntraBlockState ( ) ,
2021-07-14 21:34:14 +07:00
sharedBuyGas : uint256 . NewInt ( 0 ) ,
sharedBuyGasBalance : uint256 . NewInt ( 0 ) ,
2022-02-08 00:30:46 +03:00
isParlia : isParlia ,
isBor : isBor ,
2014-12-18 15:18:13 +01:00
}
2016-04-01 00:30:21 +02:00
}
// 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.
2021-02-12 16:47:32 +00:00
// `refunds` is false when it is not required to apply gas refunds
// `gasBailout` is true when it is not required to fail transaction if the balance is not enough to pay gas.
// for trace_call to replicate OE/Pariry behaviour
2021-12-06 17:58:53 +03:00
func ApplyMessage ( evm vm . VMInterface , msg Message , gp * GasPool , refunds bool , gasBailout bool ) ( * ExecutionResult , error ) {
2021-02-12 16:47:32 +00:00
return NewStateTransition ( evm , msg , gp ) . TransitionDb ( refunds , gasBailout )
2014-06-13 12:58:01 +02:00
}
2018-04-10 15:33:25 +02:00
// 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 { }
2015-08-30 10:19:10 +02:00
}
2018-04-10 15:33:25 +02:00
return * st . msg . To ( )
2014-06-13 12:58:01 +02:00
}
2021-02-12 16:47:32 +00:00
func ( st * StateTransition ) buyGas ( gasBailout bool ) error {
2021-07-14 21:34:14 +07:00
mgval := st . sharedBuyGas
mgval . SetUint64 ( st . msg . Gas ( ) )
2021-12-06 08:00:54 +00:00
mgval , overflow := mgval . MulOverflow ( mgval , st . gasPrice )
if overflow {
return fmt . Errorf ( "%w: address %v" , ErrInsufficientFunds , st . msg . From ( ) . Hex ( ) )
}
2021-06-03 14:09:56 +07:00
balanceCheck := mgval
2021-07-11 04:05:56 +00:00
if st . gasFeeCap != nil {
2021-07-14 21:34:14 +07:00
balanceCheck = st . sharedBuyGasBalance . SetUint64 ( st . msg . Gas ( ) )
2021-12-06 08:00:54 +00:00
balanceCheck , overflow = balanceCheck . MulOverflow ( balanceCheck , st . gasFeeCap )
if overflow {
return fmt . Errorf ( "%w: address %v" , ErrInsufficientFunds , st . msg . From ( ) . Hex ( ) )
}
balanceCheck , overflow = balanceCheck . AddOverflow ( balanceCheck , st . value )
if overflow {
return fmt . Errorf ( "%w: address %v" , ErrInsufficientFunds , st . msg . From ( ) . Hex ( ) )
}
2021-06-03 14:09:56 +07:00
}
2022-08-10 19:04:13 +07:00
var subBalance = false
2021-06-03 14:09:56 +07:00
if have , want := st . state . GetBalance ( st . msg . From ( ) ) , balanceCheck ; have . Cmp ( want ) < 0 {
2021-02-12 16:47:32 +00:00
if ! gasBailout {
2021-03-12 17:26:06 +00:00
return fmt . Errorf ( "%w: address %v have %v want %v" , ErrInsufficientFunds , st . msg . From ( ) . Hex ( ) , have , want )
2021-02-12 16:47:32 +00:00
}
} else {
2022-06-15 01:03:30 +05:30
subBalance = true
2014-06-13 12:58:01 +02:00
}
2017-11-13 13:47:27 +02:00
if err := st . gp . SubGas ( st . msg . Gas ( ) ) ; err != nil {
2021-06-03 14:09:56 +07:00
if ! gasBailout {
return err
}
2014-06-13 12:58:01 +02:00
}
2017-11-13 13:47:27 +02:00
st . gas += st . msg . Gas ( )
2017-01-04 20:17:24 +01:00
2017-11-13 13:47:27 +02:00
st . initialGas = st . msg . Gas ( )
2022-06-15 01:03:30 +05:30
if subBalance {
st . state . SubBalance ( st . msg . From ( ) , mgval )
}
2014-06-13 12:58:01 +02:00
return nil
}
2022-05-30 14:56:07 +02:00
func CheckEip1559TxGasFeeCap ( from common . Address , gasFeeCap , tip , baseFee * uint256 . Int ) error {
if gasFeeCap . Cmp ( tip ) < 0 {
return fmt . Errorf ( "%w: address %v, tip: %s, gasFeeCap: %s" , ErrTipAboveFeeCap ,
2022-08-29 15:26:36 +02:00
from . Hex ( ) , tip , gasFeeCap )
2022-05-30 14:56:07 +02:00
}
if baseFee != nil && gasFeeCap . Cmp ( baseFee ) < 0 {
return fmt . Errorf ( "%w: address %v, gasFeeCap: %s baseFee: %s" , ErrFeeCapTooLow ,
from . Hex ( ) , gasFeeCap , baseFee )
}
return nil
}
2019-05-27 14:51:49 +01:00
// DESCRIBED: docs/programmers_guide/guide.md#nonce
2021-02-12 16:47:32 +00:00
func ( st * StateTransition ) preCheck ( gasBailout bool ) error {
2018-04-10 15:33:25 +02:00
// Make sure this transaction's nonce is correct.
if st . msg . CheckNonce ( ) {
2020-12-04 12:22:19 +01:00
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 )
2021-12-07 17:24:59 +00:00
} else if stNonce + 1 < stNonce {
return fmt . Errorf ( "%w: address %v, nonce: %d" , ErrNonceMax ,
st . msg . From ( ) . Hex ( ) , stNonce )
2016-07-11 11:58:10 +02:00
}
2021-07-27 09:39:41 +07:00
2022-02-25 12:51:29 +00:00
// Make sure the sender is an EOA (EIP-3607)
if codeHash := st . state . GetCodeHash ( st . msg . From ( ) ) ; codeHash != emptyCodeHash && codeHash != ( common . Hash { } ) {
// common.Hash{} means that the sender is not in the state.
// Historically there were transactions with 0 gas price and non-existing sender,
// so we have to allow that.
return fmt . Errorf ( "%w: address %v, codehash: %s" , ErrSenderNoEOA ,
st . msg . From ( ) . Hex ( ) , codeHash )
}
2021-10-13 03:30:09 +02:00
}
2021-07-11 04:05:56 +00:00
// Make sure the transaction gasFeeCap is greater than the block's baseFee.
2021-12-06 17:58:53 +03:00
if st . evm . ChainRules ( ) . IsLondon {
2021-07-11 04:05:56 +00:00
// Skip the checks if gas fields are zero and baseFee was explicitly disabled (eth_call)
2021-12-06 17:58:53 +03:00
if ! st . evm . Config ( ) . NoBaseFee || ! st . gasFeeCap . IsZero ( ) || ! st . tip . IsZero ( ) {
2022-05-30 14:56:07 +02:00
if err := CheckEip1559TxGasFeeCap ( st . msg . From ( ) , st . gasFeeCap , st . tip , st . evm . Context ( ) . BaseFee ) ; err != nil {
return err
2021-07-11 04:05:56 +00:00
}
2021-04-22 18:11:37 +01:00
}
}
2021-02-12 16:47:32 +00:00
return st . buyGas ( gasBailout )
2014-06-19 13:41:17 +02:00
}
2017-11-13 13:47:27 +02:00
// TransitionDb will transition the state by applying the current message and
2020-04-22 16:25:36 +08:00
// returning the evm execution result with following fields.
//
2022-08-10 19:04:13 +07:00
// - 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
2020-04-22 16:25:36 +08:00
//
// However if any consensus issue encountered, return the error directly with
// nil evm execution result.
2021-02-12 16:47:32 +00:00
func ( st * StateTransition ) TransitionDb ( refunds bool , gasBailout bool ) ( * ExecutionResult , error ) {
2022-02-08 00:30:46 +03:00
var input1 * uint256 . Int
var input2 * uint256 . Int
if st . isBor {
input1 = st . state . GetBalance ( st . msg . From ( ) ) . Clone ( )
input2 = st . state . GetBalance ( st . evm . Context ( ) . Coinbase ) . Clone ( )
}
2020-04-22 16:25:36 +08:00
// 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
2022-01-23 13:32:50 +03:00
// BSC always gave gas bailout due to system transactions that set 2^256/2 gas limit and
// for Parlia consensus this flag should be always be set
2022-02-08 00:30:46 +03:00
if st . isParlia {
2022-01-23 13:32:50 +03:00
gasBailout = true
}
2021-07-26 16:09:39 +02:00
// Check clauses 1-3 and 6, buy gas if everything is correct
2021-02-12 16:47:32 +00:00
if err := st . preCheck ( gasBailout ) ; err != nil {
2020-04-22 16:25:36 +08:00
return nil , err
2014-06-19 13:41:17 +02:00
}
2017-05-11 09:55:48 +08:00
msg := st . msg
2018-04-10 15:33:25 +02:00
sender := vm . AccountRef ( msg . From ( ) )
2021-12-06 17:58:53 +03:00
homestead := st . evm . ChainRules ( ) . IsHomestead
istanbul := st . evm . ChainRules ( ) . IsIstanbul
london := st . evm . ChainRules ( ) . IsLondon
2022-10-06 23:20:48 -07:00
nano := st . evm . ChainRules ( ) . IsNano
2017-05-08 12:09:35 +03:00
contractCreation := msg . To ( ) == nil
2017-05-11 09:55:48 +08:00
2022-10-06 23:20:48 -07:00
if nano {
for _ , blackListAddr := range types . NanoBlackList {
if blackListAddr == sender . Address ( ) {
return nil , fmt . Errorf ( "block blacklist account" )
}
if msg . To ( ) != nil && * msg . To ( ) == blackListAddr {
return nil , fmt . Errorf ( "block blacklist account" )
}
}
}
2020-04-22 16:25:36 +08:00
// Check clauses 4-5, subtract intrinsic gas if everything is correct
2021-02-25 07:26:57 -07:00
gas , err := IntrinsicGas ( st . data , st . msg . AccessList ( ) , contractCreation , homestead , istanbul )
2018-02-14 21:02:51 +01:00
if err != nil {
2020-04-22 16:25:36 +08:00
return nil , err
2018-02-14 21:02:51 +01:00
}
2020-04-22 16:25:36 +08:00
if st . gas < gas {
2020-12-04 12:22:19 +01:00
return nil , fmt . Errorf ( "%w: have %d, want %d" , ErrIntrinsicGas , st . gas , gas )
2014-06-13 12:58:01 +02:00
}
2020-04-22 16:25:36 +08:00
st . gas -= gas
2014-06-13 12:58:01 +02:00
2021-08-03 20:58:18 +01:00
var bailout bool
// Gas bailout (for trace_call) should only be applied if there is not sufficient balance to perform value transfer
if gasBailout {
2021-12-06 17:58:53 +03:00
if ! msg . Value ( ) . IsZero ( ) && ! st . evm . Context ( ) . CanTransfer ( st . state , msg . From ( ) , msg . Value ( ) ) {
2021-08-03 20:58:18 +01:00
bailout = true
}
}
2021-02-25 09:10:30 +02:00
// Set up the initial access list.
2021-12-06 17:58:53 +03:00
if st . evm . ChainRules ( ) . IsBerlin {
st . state . PrepareAccessList ( msg . From ( ) , msg . To ( ) , vm . ActivePrecompiles ( st . evm . ChainRules ( ) ) , msg . AccessList ( ) )
2020-04-22 16:25:36 +08:00
}
2021-02-25 07:26:57 -07:00
2016-12-06 02:16:03 +01:00
var (
2020-04-22 16:25:36 +08:00
ret [ ] byte
vmerr error // vm errors do not effect consensus and are therefore not assigned to err
2016-12-06 02:16:03 +01:00
)
2015-11-27 15:40:29 +01:00
if contractCreation {
2020-04-14 13:49:38 +01:00
// The reason why we don't increment nonce here is that we need the original
// nonce to calculate the address of the contract that is being created
// It does get incremented inside the `Create` call, after the computation
// of the contract's address, but before the execution of the code.
2020-04-22 16:25:36 +08:00
ret , _ , st . gas , vmerr = st . evm . Create ( sender , st . data , st . gas , st . value )
2014-07-07 13:59:09 +02:00
} else {
2015-03-24 15:23:16 +01:00
// Increment the nonce for the next transaction
2018-04-10 15:33:25 +02:00
st . state . SetNonce ( msg . From ( ) , st . state . GetNonce ( sender . Address ( ) ) + 1 )
2021-08-03 20:58:18 +01:00
ret , st . gas , vmerr = st . evm . Call ( sender , st . to ( ) , st . data , st . gas , st . value , bailout )
2015-07-06 11:54:11 +02:00
}
2020-12-09 18:24:08 +00:00
if refunds {
2021-05-11 10:54:20 +01:00
if london {
2021-05-14 08:57:18 +01:00
// After EIP-3529: refunds are capped to gasUsed / 5
2021-05-11 10:54:20 +01:00
st . refundGas ( params . RefundQuotientEIP3529 )
2021-05-02 10:34:19 +01:00
} else {
2021-05-14 08:57:18 +01:00
// Before EIP-3529: refunds were capped to gasUsed / 2
2021-05-11 10:54:20 +01:00
st . refundGas ( params . RefundQuotient )
2021-05-02 10:34:19 +01:00
}
2020-12-09 18:24:08 +00:00
}
2021-05-11 10:54:20 +01:00
effectiveTip := st . gasPrice
2022-02-21 12:42:23 +00:00
if london {
2021-12-06 17:58:53 +03:00
effectiveTip = cmath . Min256 ( st . tip , new ( uint256 . Int ) . Sub ( st . gasFeeCap , st . evm . Context ( ) . BaseFee ) )
2021-04-22 18:11:37 +01:00
}
2022-02-21 12:42:23 +00:00
amount := new ( uint256 . Int ) . SetUint64 ( st . gasUsed ( ) )
amount . Mul ( amount , effectiveTip ) // gasUsed * effectiveTip = how much goes to the block producer (miner, validator)
2022-02-08 00:30:46 +03:00
if st . isParlia {
2022-02-21 12:42:23 +00:00
st . state . AddBalance ( consensus . SystemAddress , amount )
2022-02-24 00:03:10 +00:00
} else {
st . state . AddBalance ( st . evm . Context ( ) . Coinbase , amount )
}
if st . isBor {
if london {
burntContractAddress := common . HexToAddress ( st . evm . ChainConfig ( ) . Bor . CalculateBurntContract ( st . evm . Context ( ) . BlockNumber ) )
burnAmount := new ( uint256 . Int ) . Mul ( new ( uint256 . Int ) . SetUint64 ( st . gasUsed ( ) ) , st . evm . Context ( ) . BaseFee )
st . state . AddBalance ( burntContractAddress , burnAmount )
}
2022-02-21 12:42:23 +00:00
// Deprecating transfer log and will be removed in future fork. PLEASE DO NOT USE this transfer log going forward. Parameters won't get updated as expected going forward with EIP1559
// add transfer log
2022-02-08 00:30:46 +03:00
output1 := input1 . Clone ( )
output2 := input2 . Clone ( )
AddFeeTransferLog (
st . state ,
msg . From ( ) ,
st . evm . Context ( ) . Coinbase ,
amount ,
input1 ,
input2 ,
output1 . Sub ( output1 , amount ) ,
output2 . Add ( output2 , amount ) ,
)
}
2020-04-22 16:25:36 +08:00
return & ExecutionResult {
UsedGas : st . gasUsed ( ) ,
Err : vmerr ,
ReturnData : ret ,
} , nil
2014-07-24 12:04:15 +02:00
}
2014-12-18 21:58:26 +01:00
2021-05-02 10:34:19 +01:00
func ( st * StateTransition ) refundGas ( refundQuotient uint64 ) {
2015-09-02 12:55:11 +02:00
// Apply refund counter, capped to half of the used gas.
2021-05-02 10:34:19 +01:00
refund := st . gasUsed ( ) / refundQuotient
2017-11-13 13:47:27 +02:00
if refund > st . state . GetRefund ( ) {
refund = st . state . GetRefund ( )
}
st . gas += refund
2017-01-04 20:17:24 +01:00
2017-11-13 13:47:27 +02:00
// Return ETH for remaining gas, exchanged at the original rate.
2021-05-25 23:26:25 +01:00
remaining := new ( uint256 . Int ) . Mul ( new ( uint256 . Int ) . SetUint64 ( st . gas ) , st . gasPrice )
2020-06-04 09:43:08 +02:00
st . state . AddBalance ( st . msg . From ( ) , remaining )
2014-12-18 21:58:26 +01:00
2015-09-02 12:55:11 +02:00
// Also return remaining gas to the block gas counter so it is
// available for the next transaction.
2017-11-13 13:47:27 +02:00
st . gp . AddGas ( st . gas )
2014-12-18 21:58:26 +01:00
}
2017-11-13 13:47:27 +02:00
// gasUsed returns the amount of gas used up by the state transition.
func ( st * StateTransition ) gasUsed ( ) uint64 {
return st . initialGas - st . gas
2014-12-18 21:58:26 +01:00
}