# Nimbus # Copyright (c) 2018-2024 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. {.push raises: [].} import std/strformat, results, stew/arrayops, stew/endians2, ../../common/common, ../../db/ledger, ../../transaction/call_evm, ../../transaction/call_common, ../../transaction, ../../evm/state, ../../evm/types, ../../constants, ../eip4844, ../validate # ------------------------------------------------------------------------------ # Private functions # ------------------------------------------------------------------------------ proc eip1559BaseFee(header: BlockHeader; fork: EVMFork): 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.baseFeePerGas.get(0.u256) proc commitOrRollbackDependingOnGasUsed( vmState: BaseVMState; accTx: LedgerSpRef; header: BlockHeader; tx: Transaction; gasBurned: GasInt; priorityFee: GasInt; ): Result[GasInt, string] = # 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) err(&"invalid tx: block header gasLimit reached. gasLimit={header.gasLimit}, gasUsed={vmState.cumulativeGasUsed}, addition={gasBurned}") else: # Accept transaction and collect mining fee. vmState.stateDB.commit(accTx) vmState.stateDB.addBalance(vmState.coinbase(), gasBurned.u256 * priorityFee.u256) vmState.cumulativeGasUsed += gasBurned # Return remaining gas to the block gas counter so it is # available for the next transaction. vmState.gasPool += tx.gasLimit - gasBurned ok(gasBurned) 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 ): Result[GasInt, string] = ## Modelled after `https://eips.ethereum.org/EIPS/eip-1559#specification`_ ## which provides a backward compatible framwork for EIP1559. let fork = vmState.fork roDB = vmState.readOnlyStateDB baseFee256 = header.eip1559BaseFee(fork) baseFee = baseFee256.truncate(GasInt) priorityFee = min(tx.maxPriorityFeePerGasNorm(), tx.maxFeePerGasNorm() - baseFee) excessBlobGas = header.excessBlobGas.get(0'u64) # buy gas, then the gas goes into gasMeter if vmState.gasPool < tx.gasLimit: return err("gas limit reached. gasLimit=" & $vmState.gasPool & ", gasNeeded=" & $tx.gasLimit) vmState.gasPool -= tx.gasLimit let blobGasUsed = tx.getTotalBlobGas if vmState.blobGasUsed + blobGasUsed > MAX_BLOB_GAS_PER_BLOCK: return err("blobGasUsed " & $blobGasUsed & " exceeds maximum allowance " & $MAX_BLOB_GAS_PER_BLOCK) vmState.blobGasUsed += blobGasUsed # 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. let txRes = roDB.validateTransaction(tx, sender, header.gasLimit, baseFee256, excessBlobGas, fork) res = if txRes.isOk: # EIP-1153 vmState.stateDB.clearTransientStorage() # Execute the transaction. vmState.captureTxStart(tx.gasLimit) let accTx = vmState.stateDB.beginSavepoint gasBurned = tx.txCallEvm(sender, vmState, baseFee) vmState.captureTxEnd(tx.gasLimit - gasBurned) commitOrRollbackDependingOnGasUsed(vmState, accTx, header, tx, gasBurned, priorityFee) else: err(txRes.error) if vmState.collectWitnessData: vmState.stateDB.collectWitnessData() vmState.stateDB.persist(clearEmptyAccount = fork >= FkSpurious) res # ------------------------------------------------------------------------------ # Public functions # ------------------------------------------------------------------------------ proc processBeaconBlockRoot*(vmState: BaseVMState, beaconRoot: Hash256): Result[void, string] = ## processBeaconBlockRoot applies the EIP-4788 system call to the ## beacon block root contract. This method is exported to be used in tests. ## If EIP-4788 is enabled, we need to invoke the beaconroot storage ## contract with the new root. let statedb = vmState.stateDB call = CallParams( vmState : vmState, sender : SYSTEM_ADDRESS, gasLimit : 30_000_000.GasInt, gasPrice : 0.GasInt, to : BEACON_ROOTS_ADDRESS, input : @(beaconRoot.data), # It's a systemCall, no need for other knicks knacks sysCall : true, noAccessList: true, noIntrinsic : true, noGasCharge : true, noRefund : true, ) # runComputation a.k.a syscall/evm.call let res = call.runComputation(string) if res.len > 0: return err("processBeaconBlockRoot: " & res) statedb.persist(clearEmptyAccount = true) ok() proc processParentBlockHash*(vmState: BaseVMState, prevHash: Hash256): Result[void, string] = ## processParentBlockHash stores the parent block hash in the ## history storage contract as per EIP-2935. let statedb = vmState.stateDB call = CallParams( vmState : vmState, sender : SYSTEM_ADDRESS, gasLimit : 30_000_000.GasInt, gasPrice : 0.GasInt, to : HISTORY_STORAGE_ADDRESS, input : @(prevHash.data), # It's a systemCall, no need for other knicks knacks sysCall : true, noAccessList: true, noIntrinsic : true, noGasCharge : true, noRefund : true, ) # runComputation a.k.a syscall/evm.call let res = call.runComputation(string) if res.len > 0: return err("processParentBlockHash: " & res) statedb.persist(clearEmptyAccount = true) ok() func parseWithdrawalRequest(data: openArray[byte]): WithdrawalRequest = template copyFrom(T: type, input, a, b): auto = T.initCopyFrom(input.toOpenArray(a, b)) WithdrawalRequest( sourceAddress: array[20, byte].copyFrom(data, 0, 19), validatorPubkey: array[48, byte].copyFrom(data, 20, 20+47), amount: uint64.fromBytesLE(data.toOpenArray(68, 68+7)), ) proc processDequeueWithdrawalRequests*(vmState: BaseVMState): seq[Request] = ## processDequeueWithdrawalRequests applies the EIP-7002 system call ## to the withdrawal requests contract. let statedb = vmState.stateDB call = CallParams( vmState : vmState, sender : SYSTEM_ADDRESS, gasLimit : 30_000_000.GasInt, gasPrice : 0.GasInt, to : WITHDRAWAL_REQUEST_ADDRESS, # It's a systemCall, no need for other knicks knacks sysCall : true, noAccessList: true, noIntrinsic : true, noGasCharge : true, noRefund : true, ) # runComputation a.k.a syscall/evm.call let res = call.runComputation(Blob) statedb.persist(clearEmptyAccount = true) for i in 0..