# Nimbus # Copyright (c) 2021-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. ## EVM Opcode Handlers: Call Operations ## ==================================== ## {.push raises: [].} import ../../../constants, ../../evm_errors, ../../../common/evmforks, ../../../core/eip7702, ../../computation, ../../memory, ../../precompiles, ../../stack, ../../types, ../gas_costs, ../gas_meter, ../op_codes, ../utils/utils_numeric, ./oph_defs, chronicles, eth/common, eth/common/eth_types, stew/assign2, stint when not defined(evmc_enabled): import ../../state, ../../../db/ledger else: import stew/saturation_arith # ------------------------------------------------------------------------------ # Private # ------------------------------------------------------------------------------ type LocalParams = object gas: UInt256 value: UInt256 codeAddress: Address sender: Address memInPos: int memInLen: int memOutPos: int memOutLen: int flags: MsgFlags memOffset: int memLength: int contractAddress: Address gasCallEIPs: GasInt proc gasCallEIP2929(c: Computation, address: Address): GasInt = when evmc_enabled: if c.host.accessAccount(address) == EVMC_ACCESS_COLD: return ColdAccountAccessCost - WarmStorageReadCost else: c.vmState.mutateStateDB: if not db.inAccessList(address): db.accessList(address) # The WarmStorageReadCostEIP2929 (100) is already deducted in # the form of a constant `gasCall` return ColdAccountAccessCost - WarmStorageReadCost proc delegateResolutionCost(c: Computation, address: Address): GasInt = when evmc_enabled: if c.host.accessAccount(address) == EVMC_ACCESS_COLD: ColdAccountAccessCost else: WarmStorageReadCost else: c.vmState.mutateStateDB: if not db.inAccessList(address): db.accessList(address) return ColdAccountAccessCost else: return WarmStorageReadCost proc updateStackAndParams(q: var LocalParams; c: Computation) = c.stack.lsTop(0) let outLen = calcMemSize(q.memOutPos, q.memOutLen) inLen = calcMemSize(q.memInPos, q.memInLen) # get the bigger one if outLen < inLen: q.memOffset = q.memInPos q.memLength = q.memInLen else: q.memOffset = q.memOutPos q.memLength = q.memOutLen # EIP2929: This came before old gas calculator # because it will affect `c.gasMeter.gasRemaining` # and further `childGasLimit` if FkBerlin <= c.fork: q.gasCallEIPs = gasCallEIP2929(c, q.codeAddress) if FkPrague <= c.fork: let delegateTo = parseDelegationAddress(c.getCode(q.codeAddress)) if delegateTo.isSome: q.gasCallEIPs += delegateResolutionCost(c, delegateTo[]) proc callParams(c: Computation): EvmResult[LocalParams] = ## Helper for callOp() ? c.stack.lsCheck(7) var res = LocalParams( gas : c.stack.lsPeekInt(^1), codeAddress : c.stack.lsPeekAddress(^2), value : c.stack.lsPeekInt(^3), memInPos : c.stack.lsPeekMemRef(^4), memInLen : c.stack.lsPeekMemRef(^5), memOutPos : c.stack.lsPeekMemRef(^6), memOutLen : c.stack.lsPeekMemRef(^7), sender : c.msg.contractAddress, flags : c.msg.flags, ) c.stack.lsShrink(6) res.contractAddress = res.codeAddress res.updateStackAndParams(c) ok(res) proc callCodeParams(c: Computation): EvmResult[LocalParams] = ## Helper for callCodeOp() var res = ? c.callParams() res.contractAddress = c.msg.contractAddress ok(res) proc delegateCallParams(c: Computation): EvmResult[LocalParams] = ## Helper for delegateCall() ? c.stack.lsCheck(6) var res = LocalParams( gas : c.stack.lsPeekInt(^1), codeAddress : c.stack.lsPeekAddress(^2), memInPos : c.stack.lsPeekMemRef(^3), memInLen : c.stack.lsPeekMemRef(^4), memOutPos : c.stack.lsPeekMemRef(^5), memOutLen : c.stack.lsPeekMemRef(^6), value : c.msg.value, sender : c.msg.sender, flags : c.msg.flags, contractAddress: c.msg.contractAddress, ) c.stack.lsShrink(5) res.updateStackAndParams(c) ok(res) proc staticCallParams(c: Computation): EvmResult[LocalParams] = ## Helper for staticCall() ? c.stack.lsCheck(6) var res = LocalParams( gas : c.stack.lsPeekInt(^1), codeAddress : c.stack.lsPeekAddress(^2), memInPos : c.stack.lsPeekMemRef(^3), memInLen : c.stack.lsPeekMemRef(^4), memOutPos : c.stack.lsPeekMemRef(^5), memOutLen : c.stack.lsPeekMemRef(^6), value : 0.u256, sender : c.msg.contractAddress, flags : {EVMC_STATIC}, ) c.stack.lsShrink(5) res.contractAddress = res.codeAddress res.updateStackAndParams(c) ok(res) when evmc_enabled: template execSubCall(c: Computation; msg: ref nimbus_message; p: LocalParams) = c.chainTo(msg): assign(c.returnData, makeOpenArray(c.res.output_data, c.res.output_size.int)) let actualOutputSize = min(p.memOutLen, c.returnData.len) if actualOutputSize > 0: ? c.memory.write(p.memOutPos, c.returnData.toOpenArray(0, actualOutputSize - 1)) c.gasMeter.returnGas(GasInt c.res.gas_left) c.gasMeter.refundGas(c.res.gas_refund) if c.res.status_code == EVMC_SUCCESS: c.stack.lsTop(1) if not c.res.release.isNil: c.res.release(c.res) ok() else: proc execSubCall(c: Computation; childMsg: Message; memPos, memLen: int) = ## Call new VM -- helper for `Call`-like operations # need to provide explicit and for capturing in chainTo proc() # and are provided by value and need not be captured var precompile = getPrecompile(c.fork, childMsg.codeAddress) child = newComputation( c.vmState, false, childMsg, isPrecompile = precompile.isSome(), keepStack = false) c.chainTo(child): if not child.shouldBurnGas: c.gasMeter.returnGas(child.gasMeter.gasRemaining) if child.isSuccess: c.gasMeter.refundGas(child.gasMeter.gasRefunded) c.stack.lsTop(1) let actualOutputSize = min(memLen, child.output.len) if actualOutputSize > 0: ? c.memory.write(memPos, child.output.toOpenArray(0, actualOutputSize - 1)) c.returnData = move(child.output) ok() # ------------------------------------------------------------------------------ # Private, op handlers implementation # ------------------------------------------------------------------------------ proc callOp(cpt: VmCpt): EvmResultVoid = ## 0xf1, Message-Call into an account if EVMC_STATIC in cpt.msg.flags: let val = ? cpt.stack[^3, UInt256] if val > 0.u256: return err(opErr(StaticContext)) let p = ? cpt.callParams (gasCost, childGasLimit) = ? cpt.gasCosts[Call].c_handler( p.value, GasParams( kind: Call, isNewAccount: not cpt.accountExists(p.contractAddress), gasLeft: cpt.gasMeter.gasRemaining, gasCallEIPs: p.gasCallEIPs, contractGas: p.gas, currentMemSize: cpt.memory.len, memOffset: p.memOffset, memLength: p.memLength)) ? cpt.opcodeGasCost(Call, gasCost, reason = $Call) cpt.returnData.setLen(0) if cpt.msg.depth >= MaxCallDepth: debug "Computation Failure", reason = "Stack too deep", maximumDepth = MaxCallDepth, depth = cpt.msg.depth cpt.gasMeter.returnGas(childGasLimit) return ok() cpt.memory.extend(p.memInPos, p.memInLen) cpt.memory.extend(p.memOutPos, p.memOutLen) let senderBalance = cpt.getBalance(p.sender) if senderBalance < p.value: cpt.gasMeter.returnGas(childGasLimit) return ok() when evmc_enabled: let msg = new(nimbus_message) c = cpt msg[] = nimbus_message( kind : EVMC_CALL, depth : (cpt.msg.depth + 1).int32, gas : int64.saturate(childGasLimit), sender : p.sender, recipient : p.contractAddress, code_address: p.codeAddress, input_data : cpt.memory.readPtr(p.memInPos), input_size : p.memInLen.uint, value : toEvmc(p.value), flags : p.flags ) c.execSubCall(msg, p) else: var childMsg = Message( kind: EVMC_CALL, depth: cpt.msg.depth + 1, gas: childGasLimit, sender: p.sender, contractAddress: p.contractAddress, codeAddress: p.codeAddress, value: p.value, flags: p.flags) assign(childMsg.data, cpt.memory.read(p.memInPos, p.memInLen)) cpt.execSubCall( memPos = p.memOutPos, memLen = p.memOutLen, childMsg = childMsg) ok() # --------------------- proc callCodeOp(cpt: VmCpt): EvmResultVoid = ## 0xf2, Message-call into this account with an alternative account's code. let p = ? cpt.callCodeParams (gasCost, childGasLimit) = ? cpt.gasCosts[CallCode].c_handler( p.value, GasParams( kind: CallCode, isNewAccount: not cpt.accountExists(p.contractAddress), gasLeft: cpt.gasMeter.gasRemaining, gasCallEIPs: p.gasCallEIPs, contractGas: p.gas, currentMemSize: cpt.memory.len, memOffset: p.memOffset, memLength: p.memLength)) ? cpt.opcodeGasCost(CallCode, gasCost, reason = $CallCode) cpt.returnData.setLen(0) if cpt.msg.depth >= MaxCallDepth: debug "Computation Failure", reason = "Stack too deep", maximumDepth = MaxCallDepth, depth = cpt.msg.depth cpt.gasMeter.returnGas(childGasLimit) return ok() cpt.memory.extend(p.memInPos, p.memInLen) cpt.memory.extend(p.memOutPos, p.memOutLen) let senderBalance = cpt.getBalance(p.sender) if senderBalance < p.value: cpt.gasMeter.returnGas(childGasLimit) return ok() when evmc_enabled: let msg = new(nimbus_message) c = cpt msg[] = nimbus_message( kind : EVMC_CALLCODE, depth : (cpt.msg.depth + 1).int32, gas : int64.saturate(childGasLimit), sender : p.sender, recipient : p.contractAddress, code_address: p.codeAddress, input_data : cpt.memory.readPtr(p.memInPos), input_size : p.memInLen.uint, value : toEvmc(p.value), flags : p.flags ) c.execSubCall(msg, p) else: var childMsg = Message( kind: EVMC_CALLCODE, depth: cpt.msg.depth + 1, gas: childGasLimit, sender: p.sender, contractAddress: p.contractAddress, codeAddress: p.codeAddress, value: p.value, flags: p.flags) assign(childMsg.data, cpt.memory.read(p.memInPos, p.memInLen)) cpt.execSubCall( memPos = p.memOutPos, memLen = p.memOutLen, childMsg = childMsg) ok() # --------------------- proc delegateCallOp(cpt: VmCpt): EvmResultVoid = ## 0xf4, Message-call into this account with an alternative account's ## code, but persisting the current values for sender and value. let p = ? cpt.delegateCallParams (gasCost, childGasLimit) = ? cpt.gasCosts[DelegateCall].c_handler( p.value, GasParams( kind: DelegateCall, isNewAccount: not cpt.accountExists(p.contractAddress), gasLeft: cpt.gasMeter.gasRemaining, gasCallEIPs: p.gasCallEIPs, contractGas: p.gas, currentMemSize: cpt.memory.len, memOffset: p.memOffset, memLength: p.memLength)) ? cpt.opcodeGasCost(DelegateCall, gasCost, reason = $DelegateCall) cpt.returnData.setLen(0) if cpt.msg.depth >= MaxCallDepth: debug "Computation Failure", reason = "Stack too deep", maximumDepth = MaxCallDepth, depth = cpt.msg.depth cpt.gasMeter.returnGas(childGasLimit) return ok() cpt.memory.extend(p.memInPos, p.memInLen) cpt.memory.extend(p.memOutPos, p.memOutLen) when evmc_enabled: let msg = new(nimbus_message) c = cpt msg[] = nimbus_message( kind : EVMC_DELEGATECALL, depth : (cpt.msg.depth + 1).int32, gas : int64.saturate(childGasLimit), sender : p.sender, recipient : p.contractAddress, code_address: p.codeAddress, input_data : cpt.memory.readPtr(p.memInPos), input_size : p.memInLen.uint, value : toEvmc(p.value), flags : p.flags ) c.execSubCall(msg, p) else: var childMsg = Message( kind: EVMC_DELEGATECALL, depth: cpt.msg.depth + 1, gas: childGasLimit, sender: p.sender, contractAddress: p.contractAddress, codeAddress: p.codeAddress, value: p.value, flags: p.flags) assign(childMsg.data, cpt.memory.read(p.memInPos, p.memInLen)) cpt.execSubCall( memPos = p.memOutPos, memLen = p.memOutLen, childMsg = childMsg) ok() # --------------------- proc staticCallOp(cpt: VmCpt): EvmResultVoid = ## 0xfa, Static message-call into an account. let p = ? cpt.staticCallParams (gasCost, childGasLimit) = ? cpt.gasCosts[StaticCall].c_handler( p.value, GasParams( kind: StaticCall, isNewAccount: not cpt.accountExists(p.contractAddress), gasLeft: cpt.gasMeter.gasRemaining, gasCallEIPs: p.gasCallEIPs, contractGas: p.gas, currentMemSize: cpt.memory.len, memOffset: p.memOffset, memLength: p.memLength)) ? cpt.opcodeGasCost(StaticCall, gasCost, reason = $StaticCall) cpt.returnData.setLen(0) if cpt.msg.depth >= MaxCallDepth: debug "Computation Failure", reason = "Stack too deep", maximumDepth = MaxCallDepth, depth = cpt.msg.depth cpt.gasMeter.returnGas(childGasLimit) return ok() cpt.memory.extend(p.memInPos, p.memInLen) cpt.memory.extend(p.memOutPos, p.memOutLen) when evmc_enabled: let msg = new(nimbus_message) c = cpt msg[] = nimbus_message( kind : EVMC_CALL, depth : (cpt.msg.depth + 1).int32, gas : int64.saturate(childGasLimit), sender : p.sender, recipient : p.contractAddress, code_address: p.codeAddress, input_data : cpt.memory.readPtr(p.memInPos), input_size : p.memInLen.uint, value : toEvmc(p.value), flags : p.flags ) c.execSubCall(msg, p) else: var childMsg = Message( kind: EVMC_CALL, depth: cpt.msg.depth + 1, gas: childGasLimit, sender: p.sender, contractAddress: p.contractAddress, codeAddress: p.codeAddress, value: p.value, flags: p.flags) assign(childMsg.data, cpt.memory.read(p.memInPos, p.memInLen)) cpt.execSubCall( memPos = p.memOutPos, memLen = p.memOutLen, childMsg = childMsg) ok() # ------------------------------------------------------------------------------ # Public, op exec table entries # ------------------------------------------------------------------------------ const VmOpExecCall*: seq[VmOpExec] = @[ (opCode: Call, ## 0xf1, Message-Call into an account forks: VmOpAllForks, name: "call", info: "Message-Call into an account", exec: callOp), (opCode: CallCode, ## 0xf2, Message-Call with alternative code forks: VmOpAllForks, name: "callCode", info: "Message-call into this account with alternative account's code", exec: callCodeOp), (opCode: DelegateCall, ## 0xf4, CallCode with persisting sender and value forks: VmOpHomesteadAndLater, name: "delegateCall", info: "Message-call into this account with an alternative account's " & "code but persisting the current values for sender and value.", exec: delegateCallOp), (opCode: StaticCall, ## 0xfa, Static message-call into an account forks: VmOpByzantiumAndLater, name: "staticCall", info: "Static message-call into an account", exec: staticCallOp)] # ------------------------------------------------------------------------------ # End # ------------------------------------------------------------------------------