# 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: Stack, Memory, Storage And Flow Operations ## =============================================================== ## {.push raises: [].} import ../../evm_errors, ../../code_stream, ../../computation, ../../memory, ../../stack, ../../types, ../gas_meter, ../gas_costs, ../op_codes, ../utils/utils_numeric, ./oph_defs, ./oph_helpers, eth/common, stint when not defined(evmc_enabled): import ../../state, ../../../db/ledger else: import ../evmc_gas_costs # ------------------------------------------------------------------------------ # Private helpers # ------------------------------------------------------------------------------ when evmc_enabled: proc sstoreEvmc(c: Computation, slot, newValue: UInt256, coldAccess = 0.GasInt): EvmResultVoid = let status = c.host.setStorage(c.msg.contractAddress, slot, newValue) res = ForkToSstoreCost[c.fork][status] gasCost = res.gasCost + coldAccess if res.gasRefund != 0: c.gasMeter.refundGas(res.gasRefund) c.opcodeGastCost(Sstore, gasCost, "SSTORE") else: proc sstoreImpl(c: Computation, slot, newValue: UInt256): EvmResultVoid = let currentValue = c.getStorage(slot) gasParam = GasParams( kind: Op.Sstore, s_currentValue: currentValue) res = ? c.gasCosts[Sstore].c_handler(newValue, gasParam) ? c.opcodeGastCost(Sstore, res.gasCost, "SSTORE") if res.gasRefund > 0: c.gasMeter.refundGas(res.gasRefund) c.vmState.mutateStateDB: db.setStorage(c.msg.contractAddress, slot, newValue) ok() proc sstoreNetGasMeteringImpl(c: Computation; slot, newValue: UInt256, coldAccess = 0.GasInt): EvmResultVoid = let stateDB = c.vmState.readOnlyStateDB currentValue = c.getStorage(slot) gasParam = GasParams( kind: Op.Sstore, s_currentValue: currentValue, s_originalValue: stateDB.getCommittedStorage(c.msg.contractAddress, slot)) res = ? c.gasCosts[Sstore].c_handler(newValue, gasParam) ? c.opcodeGastCost(Sstore, res.gasCost + coldAccess, "SSTORE") if res.gasRefund != 0: c.gasMeter.refundGas(res.gasRefund) c.vmState.mutateStateDB: db.setStorage(c.msg.contractAddress, slot, newValue) ok() template sstoreEvmcOrSstore(cpt, slot, newValue: untyped): auto = when evmc_enabled: sstoreEvmc(cpt, slot, newValue, 0.GasInt) else: sstoreImpl(cpt, slot, newValue) template sstoreEvmcOrNetGasMetering(cpt, slot, newValue: untyped, coldAccess = 0.GasInt): auto = when evmc_enabled: sstoreEvmc(cpt, slot, newValue, coldAccess) else: sstoreNetGasMeteringImpl(cpt, slot, newValue, coldAccess) func jumpImpl(c: Computation; jumpTarget: UInt256): EvmResultVoid = if jumpTarget >= c.code.len.u256: return err(opErr(InvalidJumpDest)) let jt = jumpTarget.truncate(int) c.code.pc = jt let nextOpcode = c.code.peek if nextOpcode != JumpDest: return err(opErr(InvalidJumpDest)) # Jump destination must be a valid opcode if not c.code.isValidOpcode(jt): return err(opErr(InvalidJumpDest)) ok() # ------------------------------------------------------------------------------ # Private, op handlers implementation # ------------------------------------------------------------------------------ proc popOp(k: var VmCtx): EvmResultVoid = ## 0x50, Remove item from stack. k.cpt.stack.popInt.isOkOr: return err(error) ok() proc mloadOp (k: var VmCtx): EvmResultVoid = ## 0x51, Load word from memory let memStartPos = ? k.cpt.stack.popInt() let memPos = memStartPos.cleanMemRef ? k.cpt.opcodeGastCost(Mload, k.cpt.gasCosts[Mload].m_handler(k.cpt.memory.len, memPos, 32), reason = "MLOAD: GasVeryLow + memory expansion") k.cpt.memory.extend(memPos, 32) k.cpt.stack.push k.cpt.memory.read32Bytes(memPos) proc mstoreOp (k: var VmCtx): EvmResultVoid = ## 0x52, Save word to memory let (memStartPos, value) = ? k.cpt.stack.popInt(2) let memPos = memStartPos.cleanMemRef ? k.cpt.opcodeGastCost(Mstore, k.cpt.gasCosts[Mstore].m_handler(k.cpt.memory.len, memPos, 32), reason = "MSTORE: GasVeryLow + memory expansion") k.cpt.memory.extend(memPos, 32) k.cpt.memory.write(memPos, value.toBytesBE) proc mstore8Op (k: var VmCtx): EvmResultVoid = ## 0x53, Save byte to memory let (memStartPos, value) = ? k.cpt.stack.popInt(2) let memPos = memStartPos.cleanMemRef ? k.cpt.opcodeGastCost(Mstore8, k.cpt.gasCosts[Mstore8].m_handler(k.cpt.memory.len, memPos, 1), reason = "MSTORE8: GasVeryLow + memory expansion") k.cpt.memory.extend(memPos, 1) k.cpt.memory.write(memPos, value.toByteArrayBE[31]) # ------- proc sloadOp (k: var VmCtx): EvmResultVoid = ## 0x54, Load word from storage. let cpt = k.cpt slot = ? cpt.stack.popInt() cpt.stack.push cpt.getStorage(slot) proc sloadEIP2929Op (k: var VmCtx): EvmResultVoid = ## 0x54, EIP2929: Load word from storage for Berlin and later let cpt = k.cpt slot = ? cpt.stack.popInt() gasCost = cpt.gasEip2929AccountCheck(cpt.msg.contractAddress, slot) ? cpt.opcodeGastCost(Sload, gasCost, reason = "sloadEIP2929") cpt.stack.push cpt.getStorage(slot) # ------- proc sstoreOp (k: var VmCtx): EvmResultVoid = ## 0x55, Save word to storage. let cpt = k.cpt (slot, newValue) = ? cpt.stack.popInt(2) ? checkInStaticContext(cpt) sstoreEvmcOrSstore(cpt, slot, newValue) proc sstoreEIP1283Op (k: var VmCtx): EvmResultVoid = ## 0x55, EIP1283: sstore for Constantinople and later let cpt = k.cpt (slot, newValue) = ? cpt.stack.popInt(2) ? checkInStaticContext(cpt) sstoreEvmcOrNetGasMetering(cpt, slot, newValue) proc sstoreEIP2200Op (k: var VmCtx): EvmResultVoid = ## 0x55, EIP2200: sstore for Istanbul and later let cpt = k.cpt (slot, newValue) = ? cpt.stack.popInt(2) ? checkInStaticContext(cpt) const SentryGasEIP2200 = 2300 if cpt.gasMeter.gasRemaining <= SentryGasEIP2200: return err(opErr(OutOfGas)) sstoreEvmcOrNetGasMetering(cpt, slot, newValue) proc sstoreEIP2929Op (k: var VmCtx): EvmResultVoid = ## 0x55, EIP2929: sstore for Berlin and later let cpt = k.cpt (slot, newValue) = ? cpt.stack.popInt(2) ? checkInStaticContext(cpt) # Minimum gas required to be present for an SSTORE call, not consumed const SentryGasEIP2200 = 2300 if cpt.gasMeter.gasRemaining <= SentryGasEIP2200: return err(opErr(OutOfGas)) var coldAccessGas = 0.GasInt when evmc_enabled: if cpt.host.accessStorage(cpt.msg.contractAddress, slot) == EVMC_ACCESS_COLD: coldAccessGas = ColdSloadCost else: cpt.vmState.mutateStateDB: if not db.inAccessList(cpt.msg.contractAddress, slot): db.accessList(cpt.msg.contractAddress, slot) coldAccessGas = ColdSloadCost sstoreEvmcOrNetGasMetering(cpt, slot, newValue, coldAccessGas) # ------- proc jumpOp (k: var VmCtx): EvmResultVoid = ## 0x56, Alter the program counter let jumpTarget = ? k.cpt.stack.popInt() jumpImpl(k.cpt, jumpTarget) proc jumpIOp (k: var VmCtx): EvmResultVoid = ## 0x57, Conditionally alter the program counter. let (jumpTarget, testedValue) = ? k.cpt.stack.popInt(2) if testedValue.isZero: return ok() jumpImpl(k.cpt, jumpTarget) proc pcOp (k: var VmCtx): EvmResultVoid = ## 0x58, Get the value of the program counter prior to the increment ## corresponding to this instruction. k.cpt.stack.push max(k.cpt.code.pc - 1, 0) proc msizeOp (k: var VmCtx): EvmResultVoid = ## 0x59, Get the size of active memory in bytes. k.cpt.stack.push k.cpt.memory.len proc gasOp (k: var VmCtx): EvmResultVoid = ## 0x5a, Get the amount of available gas, including the corresponding ## reduction for the cost of this instruction. k.cpt.stack.push k.cpt.gasMeter.gasRemaining proc jumpDestOp (k: var VmCtx): EvmResultVoid = ## 0x5b, Mark a valid destination for jumps. This operation has no effect ## on machine state during execution. ok() proc tloadOp (k: var VmCtx): EvmResultVoid = ## 0x5c, Load word from transient storage. let slot = ? k.cpt.stack.popInt() val = k.cpt.getTransientStorage(slot) k.cpt.stack.push val proc tstoreOp (k: var VmCtx): EvmResultVoid = ## 0x5d, Save word to transient storage. ? checkInStaticContext(k.cpt) let slot = ? k.cpt.stack.popInt() val = ? k.cpt.stack.popInt() k.cpt.setTransientStorage(slot, val) ok() proc mCopyOp (k: var VmCtx): EvmResultVoid = ## 0x5e, Copy memory let (dst, src, size) = ? k.cpt.stack.popInt(3) let (dstPos, srcPos, len) = (dst.cleanMemRef, src.cleanMemRef, size.cleanMemRef) ? k.cpt.opcodeGastCost(Mcopy, k.cpt.gasCosts[Mcopy].m_handler(k.cpt.memory.len, max(dstPos, srcPos), len), reason = "Mcopy fee") k.cpt.memory.copy(dstPos, srcPos, len) ok() # ------------------------------------------------------------------------------ # Public, op exec table entries # ------------------------------------------------------------------------------ const VmOpExecMemory*: seq[VmOpExec] = @[ (opCode: Pop, ## x50, Remove item from stack forks: VmOpAllForks, name: "pop", info: "Remove item from stack", exec: VmOpFn popOp), (opCode: Mload, ## 0x51, Load word from memory forks: VmOpAllForks, name: "mload", info: "Load word from memory", exec: mloadOp), (opCode: Mstore, ## 0x52, Save word to memory forks: VmOpAllForks, name: "mstore", info: "Save word to memory", exec: mstoreOp), (opCode: Mstore8, ## 0x53, Save byte to memory forks: VmOpAllForks, name: "mstore8", info: "Save byte to memory", exec: mstore8Op), (opCode: Sload, ## 0x54, Load word from storage forks: VmOpAllForks - VmOpBerlinAndLater, name: "sload", info: "Load word from storage", exec: sloadOp), (opCode: Sload, ## 0x54, sload for Berlin and later forks: VmOpBerlinAndLater, name: "sloadEIP2929", info: "EIP2929: sload for Berlin and later", exec: sloadEIP2929Op), (opCode: Sstore, ## 0x55, Save word forks: VmOpAllForks - VmOpConstantinopleAndLater, name: "sstore", info: "Save word to storage", exec: sstoreOp), (opCode: Sstore, ## 0x55, sstore for Constantinople and later forks: VmOpConstantinopleAndLater - VmOpPetersburgAndLater, name: "sstoreEIP1283", info: "EIP1283: sstore for Constantinople and later", exec: sstoreEIP1283Op), (opCode: Sstore, ## 0x55, sstore for Petersburg and later forks: VmOpPetersburgAndLater - VmOpIstanbulAndLater, name: "sstore", info: "sstore for Constantinople and later", exec: sstoreOp), (opCode: Sstore, ## 0x55, sstore for Istanbul and later forks: VmOpIstanbulAndLater - VmOpBerlinAndLater, name: "sstoreEIP2200", info: "EIP2200: sstore for Istanbul and later", exec: sstoreEIP2200Op), (opCode: Sstore, ## 0x55, sstore for Berlin and later forks: VmOpBerlinAndLater, name: "sstoreEIP2929", info: "EIP2929: sstore for Istanbul and later", exec: sstoreEIP2929Op), (opCode: Jump, ## 0x56, Jump forks: VmOpAllForks, name: "jump", info: "Alter the program counter", exec: jumpOp), (opCode: JumpI, ## 0x57, Conditional jump forks: VmOpAllForks, name: "jumpI", info: "Conditionally alter the program counter", exec: jumpIOp), (opCode: Pc, ## 0x58, Program counter prior to instruction forks: VmOpAllForks, name: "pc", info: "Get the value of the program counter prior to the increment "& "corresponding to this instruction", exec: pcOp), (opCode: Msize, ## 0x59, Memory size forks: VmOpAllForks, name: "msize", info: "Get the size of active memory in bytes", exec: msizeOp), (opCode: Gas, ## 0x5a, Get available gas forks: VmOpAllForks, name: "gas", info: "Get the amount of available gas, including the corresponding "& "reduction for the cost of this instruction", exec: gasOp), (opCode: JumpDest, ## 0x5b, Mark jump target. This operation has no effect ## on machine state during execution forks: VmOpAllForks, name: "jumpDest", info: "Mark a valid destination for jumps", exec: jumpDestOp), (opCode: Tload, ## 0x5c, Load word from transient storage. forks: VmOpCancunAndLater, name: "tLoad", info: "Load word from transient storage", exec: tloadOp), (opCode: Tstore, ## 0x5d, Save word to transient storage. forks: VmOpCancunAndLater, name: "tStore", info: "Save word to transient storage", exec: tstoreOp), (opCode: Mcopy, ## 0x5e, Copy memory forks: VmOpCancunAndLater, name: "MCopy", info: "Copy memory", exec: mCopyOp)] # ------------------------------------------------------------------------------ # End # ------------------------------------------------------------------------------