# Nimbus # Copyright (c) 2018 Status Research & Development GmbH # Licensed under either of # * Apache License, version 2.0, ([LICENSE-APACHE](LICENSE-APACHE) or # http://www.apache.org/licenses/LICENSE-2.0) # * MIT license ([LICENSE-MIT](LICENSE-MIT) or # http://opensource.org/licenses/MIT) # at your option. This file may not be copied, modified, or distributed except # according to those terms. import std/[sets], ../../db/accounts_cache, ../../forks, ../../transaction/call_evm, ../../vm_state, ../../vm_types, ../validate, ./executor_helpers, chronicles, eth/common, stew/results {.push raises: [Defect].} # ------------------------------------------------------------------------------ # Private functions # ------------------------------------------------------------------------------ proc eip1559TxNormalization(tx: Transaction): Transaction = result = tx if tx.txType < TxEip1559: result.maxPriorityFee = tx.gasPrice result.maxFee = tx.gasPrice proc eip1559BaseFee(header: BlockHeader; fork: Fork): UInt256 = ## Actually, `baseFee` should be 0 for pre-London headers already. But this ## function just plays safe. In particular, the `test_general_state_json.nim` ## module modifies this block header `baseFee` field unconditionally :(. if FkLondon <= fork: result = header.baseFee proc processTransactionImpl( vmState: BaseVMState; ## Parent accounts environment for transaction tx: Transaction; ## Transaction to validate sender: EthAddress; ## tx.getSender or tx.ecRecover header: BlockHeader; ## Header for the block containing the current tx fork: Fork): Result[GasInt,void] # wildcard exception, wrapped below {.gcsafe, raises: [Exception].} = ## Modelled after `https://eips.ethereum.org/EIPS/eip-1559#specification`_ ## which provides a backward compatible framwork for EIP1559. #trace "Sender", sender #trace "txHash", rlpHash = ty.rlpHash var tx = eip1559TxNormalization(tx) let roDB = vmState.readOnlyStateDB baseFee256 = header.eip1559BaseFee(fork) baseFee = baseFee256.truncate(int64) priorityFee = min(tx.maxPriorityFee, tx.maxFee - baseFee) miner = vmState.coinbase() if FkLondon <= fork: # The signer pays both the priority fee and the base fee. So tx.gasPrice # now is the effective gasPrice which also effects the `stateRoot` value. tx.gasPrice = priorityFee + baseFee # Return failure unless explicitely set `ok()` result = err() # Actually, the eip-1559 reference does not mention an early exit. # # Even though database was not changed yet but, a `persist()` directive # before leaving is crucial for some unit tests that us a direct/deep call # of the `processTransaction()` function. So there is no `return err()` # statement, here. if roDB.validateTransaction(tx, sender, header.gasLimit, baseFee256, fork): # Execute the transaction. let accTx = vmState.stateDB.beginSavepoint gasBurned = tx.txCallEvm(sender, vmState, fork) # Make sure that the tx does not exceed the maximum cumulative limit as # set in the block header. Again, the eip-1559 reference does not mention # an early stop. It would rather detect differing values for the block # header `gasUsed` and the `vmState.cumulativeGasUsed` at a later stage. if header.gasLimit < vmState.cumulativeGasUsed + gasBurned: vmState.stateDB.rollback(accTx) debug "invalid tx: block header gasLimit reached", maxLimit = header.gasLimit, gasUsed = vmState.cumulativeGasUsed, addition = gasBurned else: # Accept transaction and collect mining fee. vmState.stateDB.commit(accTx) vmState.stateDB.addBalance(miner, gasBurned.u256 * priorityFee.u256) vmState.cumulativeGasUsed += gasBurned result = ok(gasBurned) vmState.mutateStateDB: for deletedAccount in vmState.selfDestructs: db.deleteAccount deletedAccount if fork >= FkSpurious: vmState.touchedAccounts.incl(miner) # EIP158/161 state clearing for account in vmState.touchedAccounts: if db.accountExists(account) and db.isEmptyAccount(account): debug "state clearing", account db.deleteAccount(account) if vmState.generateWitness: vmState.stateDB.collectWitnessData() vmState.stateDB.persist(clearCache = false) # ------------------------------------------------------------------------------ # Public functions # ------------------------------------------------------------------------------ proc processTransaction*( vmState: BaseVMState; ## Parent accounts environment for transaction tx: Transaction; ## Transaction to validate sender: EthAddress; ## tx.getSender or tx.ecRecover header: BlockHeader; ## Header for the block containing the current tx fork: Fork): Result[GasInt,void] {.gcsafe, raises: [Defect,CatchableError].} = ## Process the transaction, write the results to accounts db. The function ## returns the amount of gas burned if executed. safeExecutor("processTransaction"): result = vmState.processTransactionImpl(tx, sender, header, fork) proc processTransaction*( vmState: BaseVMState; ## Parent accounts environment for transaction tx: Transaction; ## Transaction to validate sender: EthAddress; ## tx.getSender or tx.ecRecover header: BlockHeader): Result[GasInt,void] {.gcsafe, raises: [Defect,CatchableError].} = ## Variant of `processTransaction()` with `*fork* derived ## from the `vmState` argument. var fork: Fork safeExecutor("processTransaction"): fork = vmState.getForkUnsafe vmState.processTransaction(tx, sender, header, fork) # ------------------------------------------------------------------------------ # End # ------------------------------------------------------------------------------