mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-28 04:55:33 +00:00
a3c8a5c3f3
This patch reduces stack space used with EVM in ENABLE_EVMC=1 mode, from 13 MB
worst case to 550 kB, a 24x reduction.
This completes fixing the "stack problem" and closes #575 (`EVM: Different
segmentation faults when running the test suite with EVMC`).
It also closes #256 (`recursive EVM call trigger unrecoverable stack overflow`).
After this patch, it is possible to re-enable the CI targets which had to be
disabled due to #575.
This change is also a required precursor for switching over to "nearly EVMC" as
the clean and focused Nimbus-internal API between EVM and sync/database
processes, and is also key to the use of Chronos `async` in those processes
when calling the EVM.
(The motivation is the internal interface has to be substantially changed
_anyway_ for the parallel sync and database processes, and EVMC turns out to be
well-designed and well-suited for this. It provides good separation between
modules, and suits our needs better than our other current interface. Might as
well use a good one designed by someone else. EVMC is 98% done in Nimbus
thanks to great work done before by @jangko, and we can use Nimbus-specific
extensions where we need flexibility, including for performance. Being aligned
with the ecosystem is a useful bonus feature.)
All tests below were run on Ubuntu 20.04 LTS server, x86-64. This matches one
of the targets that has been disabled for a while in CI in EVMC mode due to
stack overflow crashing the tests, so it's a good choice.
Measurements before
===================
Testing commit `e76e0144 2021-04-22 11:29:42 +0700 add submodules: graphql and
toml-serialization`.
$ rm -f build/all_tests && make ENABLE_EVMC=1 test
$ ulimit -S -s 16384 # Requires larger stack than default to avoid crash.
$ ./build/all_tests 9 | tee tlog
[Suite] persist block json tests
...
Stack range 38416 depthHigh 3
...
Stack range 13074720 depthHigh 1024
[OK] tests/fixtures/PersistBlockTests/block1431916.json
These tests use 13.07 MB of stack to run, and so crash with the default stack
limit on Ubuntu Server 20.04 (8MB). Exactly 12768 bytes per EVM call stack
frame.
$ rm -f build/all_tests && make ENABLE_EVMC=1 test
$ ulimit -S -s 16384 # Requires larger stack than default.
$ ./build/all_tests 7 | tee tlog
[Suite] new generalstate json tests
...
Stack range 14384 depthHigh 2
...
Stack range 3495456 depthHigh 457
[OK] tests/fixtures/eth_tests/GeneralStateTests/stRandom2/randomStatetest639.json
...
Stack range 3709600 depthHigh 485
[OK] tests/fixtures/eth_tests/GeneralStateTests/stRandom2/randomStatetest458.json
...
Stack range 7831600 depthHigh 1024
[OK] tests/fixtures/eth_tests/GeneralStateTests/stCreate2/Create2OnDepth1024.json
These tests use 7.83MB of stack to run. About 7648 bytes per EVM call stack
frame. It _only just_ avoids crashing with the default Ubuntu Server stack
limit of 8 MB. However, it still crashes on Windows x86-64, which is why the
Windows CI EVMC target is currently disabled.
On Linux where this passes, this is so borderline that it affects work and
testing of the complex storage code, because that's called from the EVM.
Also, this greatly exceeds the default thread stack size.
Measurements after
==================
$ rm -f build/all_tests && make ENABLE_EVMC=1 test
$ ulimit -S -s 600 # Because we can! 600k stack.
$ ./build/all_tests 9 | tee tlog
[Suite] persist block json tests
...
Stack range 1936 depthHigh 3
...
Stack range 556272 depthHigh 1022
Stack range 556512 depthHigh 1023
Stack range 556816 depthHigh 1023
Stack range 557056 depthHigh 1024
Stack range 557360 depthHigh 1024
[OK] tests/fixtures/PersistBlockTests/block1431916.json
$ rm -f build/all_tests && make ENABLE_EVMC=1 test
$ ulimit -S -s 600 # Because we can! 600k stack.
$ ./build/all_tests 7 | tee tlog
[Suite] new generalstate json tests
...
Stack range 1392 depthHigh 2
...
Stack range 248912 depthHigh 457
[OK] tests/fixtures/eth_tests/GeneralStateTests/stRandom2/randomStatetest639.json
...
Stack range 264144 depthHigh 485
[OK] tests/fixtures/eth_tests/GeneralStateTests/stRandom2/randomStatetest458.json
...
Stack range 557360 depthHigh 1024
[OK] tests/fixtures/eth_tests/GeneralStateTests/stStaticCall/static_CallRecursiveBombPreCall.json
For both tests, a satisfying *544 bytes* per EVM call stack frame, and EVM
takes less than 600 kB total. With other overheads, both tests run in 600 kB
stack total at maximum EVM depth.
We must add some headroom on this for database activity called from the EVM,
and different compile targets. But it means the EVM itself is no longer a
stack burden.
This is much smaller than the default thread stack size on Linux (2MB), with
plenty of margin. (Just fyi, it isn't smaller than a _small_ thread stack on
Linux from a long time ago (128kB), and some small embedded C targets.)
This size is well suited to running EVMs in threads.
Further reduction
=================
This patch solves the stack problem. Windows and Linux 64-bit EVMC CI targets
can be re-enabled, and there is no longer a problem with stack usage.
We can reduce further to ~340 bytes per frame and 350 kB total, while still
complying with EVMC. But as this involves changing how errors are handled to
comply fully with EVMC, and removing `dispose` calls, it's not worth doing now
while there are other EVMC changes in progress that will have the same effect.
A Nimbus-specific extension will allow us to avoid recursion with EVMC anyway,
bringing bytes per frame to zero. We need the extension anyway, to support
Chronos `async` which parallel transaction processing is built around.
Interop with non-Nimbus over EVMC won't let us avoid recursion, but then we
can't control the stack frame size either. To prevent stack overflow in
interop I anticipate using (this method in Aleth)
[6e96ce34e3/libethereum/ExtVM.cpp (L61)
].
Smoke test other versions of GCC and Clang/LLVM
===============================================
As all builds including Windows use GCC or Apple's Clang/LLVM, this is just to
verify we're in the right ballpark on all targets. I've only checked `x86_64`
though, not 32-bit, and not ARM.
It's interesting to see GCC 10 uses less stack. This is because it optimises
`struct` returns better, sometimes skipping an intermediate copy. Here it
benefits the EVMC API, but I found GCC 10 also improves the larger stack usage
of the rest of `nimbus-eth1` as well.
Apple clang 12.0.0 (clang-1200.0.26.2) on MacOS 10.15:
- 544 bytes per EVM call stack frame
GCC 10.3.0 (Ubuntu 10.3.0-1ubuntu1) on Ubuntu 21.04:
- 464 bytes per EVM call stack frame
GCC 10.2.0 (Ubuntu 10.2.0-5ubuntu1~20.04) on Ubuntu 20.04 LTS:
- 464 bytes per EVM call stack frame
GCC 11.0.1 20210417 (experimental; Ubuntu 11-20210417-1ubuntu1) on Ubuntu 21.04:
- 8 bytes per EVM call stack frame
GCC 9.3.0 (Ubuntu 9.3.0-17ubuntu1~20.04) on Ubuntu 20.04 LTS:
- 544 bytes per EVM call stack frame
GCC 8.4.0 (Ubuntu 8.4.0-3ubuntu2) on Ubuntu 20.04 LTS:
- 544 bytes per EVM call stack frame
GCC 7.5.0 (Ubuntu 7.5.0-6ubuntu2) on Ubuntu 20.04 LTS:
- 544 bytes per EVM call stack frame
GCC 9.2.1 20191008 (Ubuntu 9.2.1-9ubuntu2) on Ubuntu 19.10:
- 528 bytes per EVM call stack frame
Signed-off-by: Jamie Lokier <jamie@shareable.org>
1065 lines
33 KiB
Nim
1065 lines
33 KiB
Nim
# Nimbus
|
||
# Copyright (c) 2018-2019 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
|
||
strformat, times, sets, sequtils, options,
|
||
chronicles, stint, nimcrypto, stew/ranges/ptr_arith, eth/common,
|
||
./utils/[macros_procs_opcodes, utils_numeric],
|
||
./gas_meter, ./gas_costs, ./opcode_values, ./vm_forks,
|
||
../memory, ../stack, ../code_stream, ../computation, ../state, ../types,
|
||
../../errors, ../../constants,
|
||
../../db/[db_chain, accounts_cache]
|
||
|
||
when defined(evmc_enabled):
|
||
import ../evmc_api, ../evmc_helpers, evmc/evmc
|
||
|
||
logScope:
|
||
topics = "opcode impl"
|
||
|
||
# ##################################
|
||
# Syntactic sugar
|
||
|
||
proc gasEip2929AccountCheck(c: Computation, address: EthAddress, prevCost = 0.GasInt) =
|
||
c.vmState.mutateStateDB:
|
||
let gasCost = if not db.inAccessList(address):
|
||
db.accessList(address)
|
||
ColdAccountAccessCost
|
||
else:
|
||
WarmStorageReadCost
|
||
|
||
c.gasMeter.consumeGas(gasCost - prevCost, reason = "gasEIP2929AccountCheck")
|
||
|
||
template push(x: typed) {.dirty.} =
|
||
## Push an expression on the computation stack
|
||
c.stack.push x
|
||
|
||
# ##################################
|
||
# 0s: Stop and Arithmetic Operations
|
||
|
||
op add, inline = true, lhs, rhs:
|
||
## 0x01, Addition
|
||
push: lhs + rhs
|
||
|
||
op mul, inline = true, lhs, rhs:
|
||
## 0x02, Multiplication
|
||
push: lhs * rhs
|
||
|
||
op sub, inline = true, lhs, rhs:
|
||
## 0x03, Substraction
|
||
push: lhs - rhs
|
||
|
||
op divide, inline = true, lhs, rhs:
|
||
## 0x04, Division
|
||
push:
|
||
if rhs == 0: zero(Uint256) # EVM special casing of div by 0
|
||
else: lhs div rhs
|
||
|
||
op sdiv, inline = true, lhs, rhs:
|
||
## 0x05, Signed division
|
||
var r: UInt256
|
||
if rhs != 0:
|
||
var a = lhs
|
||
var b = rhs
|
||
var signA, signB: bool
|
||
extractSign(a, signA)
|
||
extractSign(b, signB)
|
||
r = a div b
|
||
setSign(r, signA xor signB)
|
||
push(r)
|
||
|
||
op modulo, inline = true, lhs, rhs:
|
||
## 0x06, Modulo
|
||
push:
|
||
if rhs == 0: zero(Uint256)
|
||
else: lhs mod rhs
|
||
|
||
op smod, inline = true, lhs, rhs:
|
||
## 0x07, Signed modulo
|
||
var r: UInt256
|
||
if rhs != 0:
|
||
var sign: bool
|
||
var v = lhs
|
||
var m = rhs
|
||
extractSign(m, sign)
|
||
extractSign(v, sign)
|
||
r = v mod m
|
||
setSign(r, sign)
|
||
|
||
push(r)
|
||
|
||
op addmod, inline = true, lhs, rhs, modulus:
|
||
## 0x08, Modulo addition
|
||
## Intermediate computations do not roll over at 2^256
|
||
push:
|
||
if modulus == 0: zero(UInt256) # EVM special casing of div by 0
|
||
else: addmod(lhs, rhs, modulus)
|
||
|
||
op mulmod, inline = true, lhs, rhs, modulus:
|
||
## 0x09, Modulo multiplication
|
||
## Intermediate computations do not roll over at 2^256
|
||
push:
|
||
if modulus == 0: zero(UInt256) # EVM special casing of div by 0
|
||
else: mulmod(lhs, rhs, modulus)
|
||
|
||
op exp, inline = true, base, exponent:
|
||
## 0x0A, Exponentiation
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[Exp].d_handler(exponent),
|
||
reason="EXP: exponent bytes"
|
||
)
|
||
push:
|
||
if base.isZero:
|
||
if exponent.isZero:
|
||
# https://github.com/ethereum/yellowpaper/issues/257
|
||
# https://github.com/ethereum/tests/pull/460
|
||
# https://github.com/ewasm/evm2wasm/issues/137
|
||
1.u256
|
||
else:
|
||
zero(UInt256)
|
||
else:
|
||
base.pow(exponent)
|
||
|
||
op signExtend, inline = false, bits, value:
|
||
## 0x0B, Sign extend
|
||
## Extend length of two’s complement signed integer.
|
||
|
||
var res: UInt256
|
||
if bits <= 31.u256:
|
||
let
|
||
one = 1.u256
|
||
testBit = bits.truncate(int) * 8 + 7
|
||
bitPos = one shl testBit
|
||
mask = bitPos - one
|
||
if not isZero(value and bitPos):
|
||
res = value or (not mask)
|
||
else:
|
||
res = value and mask
|
||
else:
|
||
res = value
|
||
|
||
push: res
|
||
|
||
# ##########################################
|
||
# 10s: Comparison & Bitwise Logic Operations
|
||
|
||
op lt, inline = true, lhs, rhs:
|
||
## 0x10, Less-than comparison
|
||
push: (lhs < rhs).uint.u256
|
||
|
||
op gt, inline = true, lhs, rhs:
|
||
## 0x11, Greater-than comparison
|
||
push: (lhs > rhs).uint.u256
|
||
|
||
op slt, inline = true, lhs, rhs:
|
||
## 0x12, Signed less-than comparison
|
||
push: (cast[Int256](lhs) < cast[Int256](rhs)).uint.u256
|
||
|
||
op sgt, inline = true, lhs, rhs:
|
||
## 0x13, Signed greater-than comparison
|
||
push: (cast[Int256](lhs) > cast[Int256](rhs)).uint.u256
|
||
|
||
op eq, inline = true, lhs, rhs:
|
||
## 0x14, Signed greater-than comparison
|
||
push: (lhs == rhs).uint.u256
|
||
|
||
op isZero, inline = true, value:
|
||
## 0x15, Check if zero
|
||
push: value.isZero.uint.u256
|
||
|
||
op andOp, inline = true, lhs, rhs:
|
||
## 0x16, Bitwise AND
|
||
push: lhs and rhs
|
||
|
||
op orOp, inline = true, lhs, rhs:
|
||
## 0x17, Bitwise AND
|
||
push: lhs or rhs
|
||
|
||
op xorOp, inline = true, lhs, rhs:
|
||
## 0x18, Bitwise AND
|
||
push: lhs xor rhs
|
||
|
||
op notOp, inline = true, value:
|
||
## 0x19, Check if zero
|
||
push: value.not
|
||
|
||
op byteOp, inline = true, position, value:
|
||
## 0x20, Retrieve single byte from word.
|
||
|
||
let pos = position.truncate(int)
|
||
|
||
push:
|
||
if pos >= 32 or pos < 0: zero(Uint256)
|
||
else:
|
||
when system.cpuEndian == bigEndian:
|
||
cast[array[32, byte]](value)[pos].u256
|
||
else:
|
||
cast[array[32, byte]](value)[31 - pos].u256
|
||
|
||
# ##########################################
|
||
# 20s: SHA3
|
||
|
||
op sha3, inline = true, startPos, length:
|
||
## 0x20, Compute Keccak-256 hash.
|
||
let (pos, len) = (startPos.safeInt, length.safeInt)
|
||
|
||
if pos < 0 or len < 0 or pos > 2147483648:
|
||
raise newException(OutOfBoundsRead, "Out of bounds memory access")
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[Op.Sha3].m_handler(c.memory.len, pos, len),
|
||
reason="SHA3: word gas cost"
|
||
)
|
||
|
||
c.memory.extend(pos, len)
|
||
let endRange = min(pos + len, c.memory.len) - 1
|
||
if endRange == -1 or pos >= c.memory.len:
|
||
push(EMPTY_SHA3)
|
||
else:
|
||
push:
|
||
keccak256.digest c.memory.bytes.toOpenArray(pos, endRange)
|
||
|
||
# ##########################################
|
||
# 30s: Environmental Information
|
||
|
||
proc writePaddedResult(mem: var Memory,
|
||
data: openarray[byte],
|
||
memPos, dataPos, len: Natural,
|
||
paddingValue = 0.byte) =
|
||
|
||
mem.extend(memPos, len)
|
||
let dataEndPosition = dataPos.int64 + len - 1
|
||
let sourceBytes = data[min(dataPos, data.len) .. min(data.len - 1, dataEndPosition)]
|
||
mem.write(memPos, sourceBytes)
|
||
|
||
# Don't duplicate zero-padding of mem.extend
|
||
let paddingOffset = min(memPos + sourceBytes.len, mem.len)
|
||
let numPaddingBytes = min(mem.len - paddingOffset, len - sourceBytes.len)
|
||
if numPaddingBytes > 0:
|
||
# TODO: avoid unnecessary memory allocation
|
||
mem.write(paddingOffset, repeat(paddingValue, numPaddingBytes))
|
||
|
||
op address, inline = true:
|
||
## 0x30, Get address of currently executing account.
|
||
push: c.msg.contractAddress
|
||
|
||
op balance, inline = true:
|
||
## 0x31, Get balance of the given account.
|
||
let address = c.stack.popAddress()
|
||
push: c.getBalance(address)
|
||
|
||
op origin, inline = true:
|
||
## 0x32, Get execution origination address.
|
||
push: c.getOrigin()
|
||
|
||
op caller, inline = true:
|
||
## 0x33, Get caller address.
|
||
push: c.msg.sender
|
||
|
||
op callValue, inline = true:
|
||
## 0x34, Get deposited value by the instruction/transaction
|
||
## responsible for this execution
|
||
push: c.msg.value
|
||
|
||
op callDataLoad, inline = false, startPos:
|
||
## 0x35, Get input data of current environment
|
||
let start = startPos.cleanMemRef
|
||
if start >= c.msg.data.len:
|
||
push: 0
|
||
return
|
||
|
||
# If the data does not take 32 bytes, pad with zeros
|
||
let endRange = min(c.msg.data.len - 1, start + 31)
|
||
let presentBytes = endRange - start
|
||
# We rely on value being initialized with 0 by default
|
||
var value: array[32, byte]
|
||
value[0 .. presentBytes] = c.msg.data.toOpenArray(start, endRange)
|
||
|
||
push: value
|
||
|
||
op callDataSize, inline = true:
|
||
## 0x36, Get size of input data in current environment.
|
||
push: c.msg.data.len.u256
|
||
|
||
op callDataCopy, inline = false, memStartPos, copyStartPos, size:
|
||
## 0x37, Copy input data in current environment to memory.
|
||
# TODO tests: https://github.com/status-im/nimbus/issues/67
|
||
|
||
let (memPos, copyPos, len) = (memStartPos.cleanMemRef, copyStartPos.cleanMemRef, size.cleanMemRef)
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[CallDataCopy].m_handler(c.memory.len, memPos, len),
|
||
reason="CallDataCopy fee")
|
||
|
||
c.memory.writePaddedResult(c.msg.data, memPos, copyPos, len)
|
||
|
||
op codeSize, inline = true:
|
||
## 0x38, Get size of code running in current environment.
|
||
push: c.code.len
|
||
|
||
op codeCopy, inline = false, memStartPos, copyStartPos, size:
|
||
## 0x39, Copy code running in current environment to memory.
|
||
# TODO tests: https://github.com/status-im/nimbus/issues/67
|
||
|
||
let (memPos, copyPos, len) = (memStartPos.cleanMemRef, copyStartPos.cleanMemRef, size.cleanMemRef)
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[CodeCopy].m_handler(c.memory.len, memPos, len),
|
||
reason="CodeCopy fee")
|
||
|
||
c.memory.writePaddedResult(c.code.bytes, memPos, copyPos, len)
|
||
|
||
op gasprice, inline = true:
|
||
## 0x3A, Get price of gas in current environment.
|
||
push: c.getGasPrice()
|
||
|
||
op extCodeSize, inline = true:
|
||
## 0x3b, Get size of an account's code
|
||
let address = c.stack.popAddress()
|
||
push: c.getCodeSize(address)
|
||
|
||
op extCodeCopy, inline = true:
|
||
## 0x3c, Copy an account's code to memory.
|
||
let address = c.stack.popAddress()
|
||
let (memStartPos, codeStartPos, size) = c.stack.popInt(3)
|
||
let (memPos, codePos, len) = (memStartPos.cleanMemRef, codeStartPos.cleanMemRef, size.cleanMemRef)
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[ExtCodeCopy].m_handler(c.memory.len, memPos, len),
|
||
reason="ExtCodeCopy fee")
|
||
|
||
let codeBytes = c.getCode(address)
|
||
c.memory.writePaddedResult(codeBytes, memPos, codePos, len)
|
||
|
||
op returnDataSize, inline = true:
|
||
## 0x3d, Get size of output data from the previous call from the current environment.
|
||
push: c.returnData.len
|
||
|
||
op returnDataCopy, inline = false, memStartPos, copyStartPos, size:
|
||
## 0x3e, Copy output data from the previous call to memory.
|
||
let (memPos, copyPos, len) = (memStartPos.cleanMemRef, copyStartPos.cleanMemRef, size.cleanMemRef)
|
||
let gasCost = c.gasCosts[ReturnDataCopy].m_handler(c.memory.len, memPos, len)
|
||
c.gasMeter.consumeGas(
|
||
gasCost,
|
||
reason="returnDataCopy fee")
|
||
|
||
if copyPos + len > c.returnData.len:
|
||
# TODO Geth additionally checks copyPos + len < 64
|
||
# Parity uses a saturating addition
|
||
# Yellow paper mentions μs[1] + i are not subject to the 2^256 modulo.
|
||
raise newException(OutOfBoundsRead,
|
||
"Return data length is not sufficient to satisfy request. Asked \n" &
|
||
&"for data from index {copyStartPos} to {copyStartPos + size}. Return data is {c.returnData.len} in \n" &
|
||
"length")
|
||
|
||
c.memory.writePaddedResult(c.returnData, memPos, copyPos, len)
|
||
|
||
# ##########################################
|
||
# 40s: Block Information
|
||
|
||
op blockhash, inline = true, blockNumber:
|
||
## 0x40, Get the hash of one of the 256 most recent complete blocks.
|
||
push: c.getBlockHash(blockNumber)
|
||
|
||
op coinbase, inline = true:
|
||
## 0x41, Get the block's beneficiary address.
|
||
push: c.getCoinbase()
|
||
|
||
op timestamp, inline = true:
|
||
## 0x42, Get the block's timestamp.
|
||
push: c.getTimestamp()
|
||
|
||
op blocknumber, inline = true:
|
||
## 0x43, Get the block's number.
|
||
push: c.getBlockNumber()
|
||
|
||
op difficulty, inline = true:
|
||
## 0x44, Get the block's difficulty
|
||
push: c.getDifficulty()
|
||
|
||
op gasLimit, inline = true:
|
||
## 0x45, Get the block's gas limit
|
||
push: c.getGasLimit()
|
||
|
||
op chainId, inline = true:
|
||
## 0x46, Get current chain’s EIP-155 unique identifier.
|
||
push: c.getChainId()
|
||
|
||
op selfBalance, inline = true:
|
||
## 0x47, Get current contract's balance.
|
||
push: c.getBalance(c.msg.contractAddress)
|
||
|
||
# ##########################################
|
||
# 50s: Stack, Memory, Storage and Flow Operations
|
||
|
||
op pop, inline = true:
|
||
## 0x50, Remove item from stack.
|
||
discard c.stack.popInt()
|
||
|
||
op mload, inline = true, memStartPos:
|
||
## 0x51, Load word from memory
|
||
let memPos = memStartPos.cleanMemRef
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[MLoad].m_handler(c.memory.len, memPos, 32),
|
||
reason="MLOAD: GasVeryLow + memory expansion"
|
||
)
|
||
c.memory.extend(memPos, 32)
|
||
|
||
push: c.memory.read(memPos, 32) # TODO, should we convert to native endianness?
|
||
|
||
op mstore, inline = true, memStartPos, value:
|
||
## 0x52, Save word to memory
|
||
let memPos = memStartPos.cleanMemRef
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[MStore].m_handler(c.memory.len, memPos, 32),
|
||
reason="MSTORE: GasVeryLow + memory expansion"
|
||
)
|
||
|
||
c.memory.extend(memPos, 32)
|
||
c.memory.write(memPos, value.toByteArrayBE) # is big-endian correct? Parity/Geth do convert
|
||
|
||
op mstore8, inline = true, memStartPos, value:
|
||
## 0x53, Save byte to memory
|
||
let memPos = memStartPos.cleanMemRef
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[MStore].m_handler(c.memory.len, memPos, 1),
|
||
reason="MSTORE8: GasVeryLow + memory expansion"
|
||
)
|
||
|
||
c.memory.extend(memPos, 1)
|
||
c.memory.write(memPos, [value.toByteArrayBE[31]])
|
||
|
||
op sload, inline = true, slot:
|
||
## 0x54, Load word from storage.
|
||
push: c.getStorage(slot)
|
||
|
||
when not evmc_enabled:
|
||
template sstoreImpl(c: Computation, slot, newValue: Uint256) =
|
||
let currentValue {.inject.} = c.getStorage(slot)
|
||
|
||
let
|
||
gasParam = GasParams(kind: Op.Sstore, s_currentValue: currentValue)
|
||
(gasCost, gasRefund) = c.gasCosts[Sstore].c_handler(newValue, gasParam)
|
||
|
||
c.gasMeter.consumeGas(gasCost, &"SSTORE: {c.msg.contractAddress}[{slot}] -> {newValue} ({currentValue})")
|
||
|
||
if gasRefund > 0:
|
||
c.gasMeter.refundGas(gasRefund)
|
||
|
||
c.vmState.mutateStateDB:
|
||
db.setStorage(c.msg.contractAddress, slot, newValue)
|
||
|
||
when evmc_enabled:
|
||
template sstoreEvmc(c: Computation, slot, newValue: Uint256) =
|
||
let
|
||
currentValue {.inject.} = c.getStorage(slot)
|
||
status = c.host.setStorage(c.msg.contractAddress, slot, newValue)
|
||
gasParam = GasParams(kind: Op.Sstore, s_status: status)
|
||
gasCost = c.gasCosts[Sstore].c_handler(newValue, gasParam)[0]
|
||
|
||
c.gasMeter.consumeGas(gasCost, &"SSTORE: {c.msg.contractAddress}[{slot}] -> {newValue} ({currentValue})")
|
||
|
||
op sstore, inline = false, slot, newValue:
|
||
## 0x55, Save word to storage.
|
||
checkInStaticContext(c)
|
||
|
||
when evmc_enabled:
|
||
sstoreEvmc(c, slot, newValue)
|
||
else:
|
||
sstoreImpl(c, slot, newValue)
|
||
|
||
when not evmc_enabled:
|
||
template sstoreNetGasMeteringImpl(c: Computation, slot, newValue: Uint256) =
|
||
let stateDB = c.vmState.readOnlyStateDB
|
||
let currentValue {.inject.} = c.getStorage(slot)
|
||
|
||
let
|
||
gasParam = GasParams(kind: Op.Sstore,
|
||
s_currentValue: currentValue,
|
||
s_originalValue: stateDB.getCommittedStorage(c.msg.contractAddress, slot)
|
||
)
|
||
(gasCost, gasRefund) = c.gasCosts[Sstore].c_handler(newValue, gasParam)
|
||
|
||
c.gasMeter.consumeGas(gasCost, &"SSTORE EIP2200: {c.msg.contractAddress}[{slot}] -> {newValue} ({currentValue})")
|
||
|
||
if gasRefund != 0:
|
||
c.gasMeter.refundGas(gasRefund)
|
||
|
||
c.vmState.mutateStateDB:
|
||
db.setStorage(c.msg.contractAddress, slot, newValue)
|
||
|
||
op sstoreEIP2200, inline = false, slot, newValue:
|
||
checkInStaticContext(c)
|
||
const SentryGasEIP2200 = 2300 # Minimum gas required to be present for an SSTORE call, not consumed
|
||
|
||
if c.gasMeter.gasRemaining <= SentryGasEIP2200:
|
||
raise newException(OutOfGas, "Gas not enough to perform EIP2200 SSTORE")
|
||
|
||
when evmc_enabled:
|
||
sstoreEvmc(c, slot, newValue)
|
||
else:
|
||
sstoreNetGasMeteringImpl(c, slot, newValue)
|
||
|
||
op sstoreEIP1283, inline = false, slot, newValue:
|
||
checkInStaticContext(c)
|
||
|
||
when evmc_enabled:
|
||
sstoreEvmc(c, slot, newValue)
|
||
else:
|
||
sstoreNetGasMeteringImpl(c, slot, newValue)
|
||
|
||
proc jumpImpl(c: Computation, jumpTarget: UInt256) =
|
||
if jumpTarget >= c.code.len.u256:
|
||
raise newException(InvalidJumpDestination, "Invalid Jump Destination")
|
||
|
||
let jt = jumpTarget.truncate(int)
|
||
c.code.pc = jt
|
||
|
||
let nextOpcode = c.code.peek
|
||
if nextOpcode != JUMPDEST:
|
||
raise newException(InvalidJumpDestination, "Invalid Jump Destination")
|
||
# TODO: next check seems redundant
|
||
if not c.code.isValidOpcode(jt):
|
||
raise newException(InvalidInstruction, "Jump resulted in invalid instruction")
|
||
|
||
op jump, inline = true, jumpTarget:
|
||
## 0x56, Alter the program counter
|
||
jumpImpl(c, jumpTarget)
|
||
|
||
op jumpI, inline = true, jumpTarget, testedValue:
|
||
## 0x57, Conditionally alter the program counter.
|
||
if testedValue != 0:
|
||
jumpImpl(c, jumpTarget)
|
||
|
||
op pc, inline = true:
|
||
## 0x58, Get the value of the program counter prior to the increment corresponding to this instruction.
|
||
push: max(c.code.pc - 1, 0)
|
||
|
||
op msize, inline = true:
|
||
## 0x59, Get the size of active memory in bytes.
|
||
push: c.memory.len
|
||
|
||
op gas, inline = true:
|
||
## 0x5a, Get the amount of available gas, including the corresponding reduction for the cost of this instruction.
|
||
push: c.gasMeter.gasRemaining
|
||
|
||
op jumpDest, inline = true:
|
||
## 0x5b, Mark a valid destination for jumps. This operation has no effect on machine state during execution.
|
||
discard
|
||
|
||
op beginSub, inline = true:
|
||
## 0x5c, Marks the entry point to a subroutine
|
||
raise newException(OutOfGas, "Abort: Attempt to execute BeginSub opcode")
|
||
|
||
op returnSub, inline = true:
|
||
## 0x5d, Returns control to the caller of a subroutine.
|
||
if c.returnStack.len == 0:
|
||
raise newException(OutOfGas, "Abort: invalid returnStack during ReturnSub")
|
||
# Other than the check that the return stack is not empty, there is no
|
||
# need to validate the pc from 'returns', since we only ever push valid
|
||
# values onto it via jumpsub.
|
||
c.code.pc = c.returnStack.pop()
|
||
|
||
op jumpSub, inline = true, jumpTarget:
|
||
## 0x5e, Transfers control to a subroutine.
|
||
if jumpTarget >= c.code.len.u256:
|
||
raise newException(InvalidJumpDestination, "JumpSub destination exceeds code len")
|
||
|
||
let returnPC = c.code.pc
|
||
let jt = jumpTarget.truncate(int)
|
||
c.code.pc = jt
|
||
|
||
let nextOpcode = c.code.peek
|
||
if nextOpcode != BeginSub:
|
||
raise newException(InvalidJumpDestination, "Invalid JumpSub destination")
|
||
|
||
if c.returnStack.len == 1023:
|
||
raise newException(FullStack, "Out of returnStack")
|
||
|
||
c.returnStack.add returnPC
|
||
inc c.code.pc
|
||
|
||
# ##########################################
|
||
# 60s & 70s: Push Operations.
|
||
# 80s: Duplication Operations
|
||
# 90s: Exchange Operations
|
||
# a0s: Logging Operations
|
||
|
||
genPush()
|
||
genDup()
|
||
genSwap()
|
||
genLog()
|
||
|
||
# ##########################################
|
||
# f0s: System operations.
|
||
template genCreate(callName: untyped, opCode: Op): untyped =
|
||
op callName, inline = false:
|
||
checkInStaticContext(c)
|
||
let
|
||
endowment = c.stack.popInt()
|
||
memPos = c.stack.popInt().safeInt
|
||
|
||
when opCode == Create:
|
||
const callKind = evmcCreate
|
||
let memLen {.inject.} = c.stack.peekInt().safeInt
|
||
let salt = 0.u256
|
||
else:
|
||
const callKind = evmcCreate2
|
||
let memLen {.inject.} = c.stack.popInt().safeInt
|
||
let salt = c.stack.peekInt()
|
||
|
||
c.stack.top(0)
|
||
|
||
let gasParams = GasParams(kind: Create,
|
||
cr_currentMemSize: c.memory.len,
|
||
cr_memOffset: memPos,
|
||
cr_memLength: memLen
|
||
)
|
||
var gasCost = c.gasCosts[Create].c_handler(1.u256, gasParams).gasCost
|
||
when opCode == Create2:
|
||
gasCost = gasCost + c.gasCosts[Create2].m_handler(0, 0, memLen)
|
||
|
||
let reason = &"CREATE: GasCreate + {memLen} * memory expansion"
|
||
c.gasMeter.consumeGas(gasCost, reason = reason)
|
||
c.memory.extend(memPos, memLen)
|
||
c.returnData.setLen(0)
|
||
|
||
if c.msg.depth >= MaxCallDepth:
|
||
debug "Computation Failure", reason = "Stack too deep", maxDepth = MaxCallDepth, depth = c.msg.depth
|
||
return
|
||
|
||
if endowment != 0:
|
||
let senderBalance = c.getBalance(c.msg.contractAddress)
|
||
if senderBalance < endowment:
|
||
debug "Computation Failure", reason = "Insufficient funds available to transfer", required = endowment, balance = senderBalance
|
||
return
|
||
|
||
var createMsgGas = c.gasMeter.gasRemaining
|
||
if c.fork >= FkTangerine:
|
||
createMsgGas -= createMsgGas div 64
|
||
c.gasMeter.consumeGas(createMsgGas, reason="CREATE")
|
||
|
||
when evmc_enabled:
|
||
let msg = new(nimbus_message)
|
||
msg[] = nimbus_message(
|
||
kind: callKind.evmc_call_kind,
|
||
depth: (c.msg.depth + 1).int32,
|
||
gas: createMsgGas,
|
||
sender: c.msg.contractAddress,
|
||
input_data: c.memory.readPtr(memPos),
|
||
input_size: memLen.uint,
|
||
value: toEvmc(endowment),
|
||
create2_salt: toEvmc(salt)
|
||
)
|
||
|
||
c.chainTo(msg):
|
||
c.returnData = @(makeOpenArray(c.res.outputData, c.res.outputSize.int))
|
||
c.gasMeter.returnGas(c.res.gas_left)
|
||
|
||
if c.res.status_code == EVMC_SUCCESS:
|
||
c.stack.top(c.res.create_address)
|
||
|
||
if not c.res.release.isNil:
|
||
c.res.release(c.res)
|
||
else:
|
||
let childMsg = Message(
|
||
kind: callKind,
|
||
depth: c.msg.depth + 1,
|
||
gas: createMsgGas,
|
||
sender: c.msg.contractAddress,
|
||
value: endowment,
|
||
data: c.memory.read(memPos, memLen)
|
||
)
|
||
|
||
var child = newComputation(c.vmState, childMsg, salt)
|
||
c.chainTo(child):
|
||
if not child.shouldBurnGas:
|
||
c.gasMeter.returnGas(child.gasMeter.gasRemaining)
|
||
|
||
if child.isSuccess:
|
||
c.merge(child)
|
||
c.stack.top child.msg.contractAddress
|
||
else:
|
||
c.returnData = child.output
|
||
|
||
genCreate(create, Create)
|
||
genCreate(create2, Create2)
|
||
|
||
proc callParams(c: Computation): (UInt256, UInt256, EthAddress, EthAddress, CallKind, int, int, int, int, MsgFlags) =
|
||
let gas = c.stack.popInt()
|
||
let destination = c.stack.popAddress()
|
||
let value = c.stack.popInt()
|
||
|
||
result = (gas,
|
||
value,
|
||
destination,
|
||
c.msg.contractAddress, # sender
|
||
evmcCall,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.msg.flags)
|
||
|
||
proc callCodeParams(c: Computation): (UInt256, UInt256, EthAddress, EthAddress, CallKind, int, int, int, int, MsgFlags) =
|
||
let gas = c.stack.popInt()
|
||
let destination = c.stack.popAddress()
|
||
let value = c.stack.popInt()
|
||
|
||
result = (gas,
|
||
value,
|
||
destination,
|
||
c.msg.contractAddress, # sender
|
||
evmcCallCode,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.msg.flags)
|
||
|
||
proc delegateCallParams(c: Computation): (UInt256, UInt256, EthAddress, EthAddress, CallKind, int, int, int, int, MsgFlags) =
|
||
let gas = c.stack.popInt()
|
||
let destination = c.stack.popAddress()
|
||
|
||
result = (gas,
|
||
c.msg.value, # value
|
||
destination,
|
||
c.msg.sender, # sender
|
||
evmcDelegateCall,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.msg.flags)
|
||
|
||
proc staticCallParams(c: Computation): (UInt256, UInt256, EthAddress, EthAddress, CallKind, int, int, int, int, MsgFlags) =
|
||
let gas = c.stack.popInt()
|
||
let destination = c.stack.popAddress()
|
||
|
||
result = (gas,
|
||
0.u256, # value
|
||
destination,
|
||
c.msg.contractAddress, # sender
|
||
evmcCall,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
c.stack.popInt().cleanMemRef,
|
||
emvcStatic) # is_static
|
||
|
||
template genCall(callName: untyped, opCode: Op): untyped =
|
||
op callName, inline = false:
|
||
## CALL, 0xf1, Message-Call into an account
|
||
## CALLCODE, 0xf2, Message-call into this account with an alternative account's code.
|
||
## DELEGATECALL, 0xf4, Message-call into this account with an alternative account's code, but persisting the current values for sender and value.
|
||
## STATICCALL, 0xfa, Static message-call into an account.
|
||
when opCode == Call:
|
||
if emvcStatic == c.msg.flags and c.stack[^3, Uint256] > 0.u256:
|
||
raise newException(StaticContextError, "Cannot modify state while inside of a STATICCALL context")
|
||
|
||
let (gas, value, destination, sender, callKind,
|
||
memInPos, memInLen, memOutPos, memOutLen, flags) = `callName Params`(c)
|
||
|
||
push: 0
|
||
|
||
let (memOffset, memLength) = if calcMemSize(memInPos, memInLen) > calcMemSize(memOutPos, memOutLen):
|
||
(memInPos, memInLen)
|
||
else:
|
||
(memOutPos, memOutLen)
|
||
|
||
# EIP2929
|
||
# This came before old gas calculator
|
||
# because it will affect `c.gasMeter.gasRemaining`
|
||
# and further `childGasLimit`
|
||
if c.fork >= FkBerlin:
|
||
c.vmState.mutateStateDB:
|
||
if not db.inAccessList(destination):
|
||
db.accessList(destination)
|
||
# The WarmStorageReadCostEIP2929 (100) is already deducted in the form of a constant `gasCall`
|
||
c.gasMeter.consumeGas(ColdAccountAccessCost - WarmStorageReadCost, reason = "EIP2929 gasCall")
|
||
|
||
let contractAddress = when opCode in {Call, StaticCall}: destination else: c.msg.contractAddress
|
||
var (gasCost, childGasLimit) = c.gasCosts[opCode].c_handler(
|
||
value,
|
||
GasParams(kind: opCode,
|
||
c_isNewAccount: not c.accountExists(contractAddress),
|
||
c_gasBalance: c.gasMeter.gasRemaining,
|
||
c_contractGas: gas,
|
||
c_currentMemSize: c.memory.len,
|
||
c_memOffset: memOffset,
|
||
c_memLength: memLength
|
||
))
|
||
|
||
# EIP 2046: temporary disabled
|
||
# reduce gas fee for precompiles
|
||
# from 700 to 40
|
||
#when opCode == StaticCall:
|
||
# if c.fork >= FkBerlin and destination.toInt <= MaxPrecompilesAddr:
|
||
# gasCost = gasCost - 660.GasInt
|
||
|
||
if gasCost >= 0:
|
||
c.gasMeter.consumeGas(gasCost, reason = $opCode)
|
||
|
||
c.returnData.setLen(0)
|
||
|
||
if c.msg.depth >= MaxCallDepth:
|
||
debug "Computation Failure", reason = "Stack too deep", maximumDepth = MaxCallDepth, depth = c.msg.depth
|
||
# return unused gas
|
||
c.gasMeter.returnGas(childGasLimit)
|
||
return
|
||
|
||
if gasCost < 0 and childGasLimit <= 0:
|
||
raise newException(OutOfGas, "Gas not enough to perform calculation (" & callName.astToStr & ")")
|
||
|
||
c.memory.extend(memInPos, memInLen)
|
||
c.memory.extend(memOutPos, memOutLen)
|
||
|
||
when opCode in {CallCode, Call}:
|
||
let senderBalance = c.getBalance(sender)
|
||
if senderBalance < value:
|
||
debug "Insufficient funds", available = senderBalance, needed = c.msg.value
|
||
# return unused gas
|
||
c.gasMeter.returnGas(childGasLimit)
|
||
return
|
||
|
||
when evmc_enabled:
|
||
let msg = new(nimbus_message)
|
||
msg[] = nimbus_message(
|
||
kind: callKind.evmc_call_kind,
|
||
depth: (c.msg.depth + 1).int32,
|
||
gas: childGasLimit,
|
||
sender: sender,
|
||
destination: destination,
|
||
input_data: c.memory.readPtr(memInPos),
|
||
input_size: memInLen.uint,
|
||
value: toEvmc(value),
|
||
flags: flags.uint32
|
||
)
|
||
|
||
c.chainTo(msg):
|
||
c.returnData = @(makeOpenArray(c.res.outputData, c.res.outputSize.int))
|
||
|
||
let actualOutputSize = min(memOutLen, c.returnData.len)
|
||
if actualOutputSize > 0:
|
||
c.memory.write(memOutPos,
|
||
c.returnData.toOpenArray(0, actualOutputSize - 1))
|
||
|
||
c.gasMeter.returnGas(c.res.gas_left)
|
||
|
||
if c.res.status_code == EVMC_SUCCESS:
|
||
c.stack.top(1)
|
||
|
||
if not c.res.release.isNil:
|
||
c.res.release(c.res)
|
||
else:
|
||
let msg = Message(
|
||
kind: callKind,
|
||
depth: c.msg.depth + 1,
|
||
gas: childGasLimit,
|
||
sender: sender,
|
||
contractAddress: contractAddress,
|
||
codeAddress: destination,
|
||
value: value,
|
||
data: c.memory.read(memInPos, memInLen),
|
||
flags: flags)
|
||
|
||
var child = newComputation(c.vmState, msg)
|
||
c.chainTo(child):
|
||
if not child.shouldBurnGas:
|
||
c.gasMeter.returnGas(child.gasMeter.gasRemaining)
|
||
|
||
if child.isSuccess:
|
||
c.merge(child)
|
||
c.stack.top(1)
|
||
|
||
c.returnData = child.output
|
||
let actualOutputSize = min(memOutLen, child.output.len)
|
||
if actualOutputSize > 0:
|
||
c.memory.write(memOutPos,
|
||
child.output.toOpenArray(0, actualOutputSize - 1))
|
||
|
||
genCall(call, Call)
|
||
genCall(callCode, CallCode)
|
||
genCall(delegateCall, DelegateCall)
|
||
genCall(staticCall, StaticCall)
|
||
|
||
op returnOp, inline = false, startPos, size:
|
||
## 0xf3, Halt execution returning output data.
|
||
let (pos, len) = (startPos.cleanMemRef, size.cleanMemRef)
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[Return].m_handler(c.memory.len, pos, len),
|
||
reason = "RETURN"
|
||
)
|
||
|
||
c.memory.extend(pos, len)
|
||
c.output = c.memory.read(pos, len)
|
||
|
||
op revert, inline = false, startPos, size:
|
||
## 0xfd, Halt execution reverting state changes but returning data and remaining gas.
|
||
let (pos, len) = (startPos.cleanMemRef, size.cleanMemRef)
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[Revert].m_handler(c.memory.len, pos, len),
|
||
reason = "REVERT"
|
||
)
|
||
|
||
c.memory.extend(pos, len)
|
||
c.output = c.memory.read(pos, len)
|
||
# setError(msg, false) will signal cheap revert
|
||
c.setError("REVERT opcode executed", false)
|
||
|
||
op selfDestruct, inline = false:
|
||
## 0xff Halt execution and register account for later deletion.
|
||
let beneficiary = c.stack.popAddress()
|
||
c.selfDestruct(beneficiary)
|
||
|
||
op selfDestructEip150, inline = false:
|
||
let beneficiary = c.stack.popAddress()
|
||
|
||
let gasParams = GasParams(kind: SelfDestruct,
|
||
sd_condition: not c.accountExists(beneficiary)
|
||
)
|
||
|
||
let gasCost = c.gasCosts[SelfDestruct].c_handler(0.u256, gasParams).gasCost
|
||
c.gasMeter.consumeGas(gasCost, reason = "SELFDESTRUCT EIP150")
|
||
c.selfDestruct(beneficiary)
|
||
|
||
op selfDestructEip161, inline = false:
|
||
checkInStaticContext(c)
|
||
|
||
let
|
||
beneficiary = c.stack.popAddress()
|
||
isDead = not c.accountExists(beneficiary)
|
||
balance = c.getBalance(c.msg.contractAddress)
|
||
|
||
let gasParams = GasParams(kind: SelfDestruct,
|
||
sd_condition: isDead and not balance.isZero
|
||
)
|
||
|
||
let gasCost = c.gasCosts[SelfDestruct].c_handler(0.u256, gasParams).gasCost
|
||
c.gasMeter.consumeGas(gasCost, reason = "SELFDESTRUCT EIP161")
|
||
c.selfDestruct(beneficiary)
|
||
|
||
# Constantinople's new opcodes
|
||
op shlOp, inline = true, shift, num:
|
||
let shiftLen = shift.safeInt
|
||
if shiftLen >= 256:
|
||
push: 0
|
||
else:
|
||
push: num shl shiftLen
|
||
|
||
op shrOp, inline = true, shift, num:
|
||
let shiftLen = shift.safeInt
|
||
if shiftLen >= 256:
|
||
push: 0
|
||
else:
|
||
# uint version of `shr`
|
||
push: num shr shiftLen
|
||
|
||
op sarOp, inline = true:
|
||
let shiftLen = c.stack.popInt().safeInt
|
||
let num = cast[Int256](c.stack.popInt())
|
||
if shiftLen >= 256:
|
||
if num.isNegative:
|
||
push: cast[Uint256]((-1).i256)
|
||
else:
|
||
push: 0
|
||
else:
|
||
# int version of `shr` then force the result
|
||
# into uint256
|
||
push: cast[Uint256](num shr shiftLen)
|
||
|
||
op extCodeHash, inline = true:
|
||
let address = c.stack.popAddress()
|
||
push: c.getCodeHash(address)
|
||
|
||
op balanceEIP2929, inline = true:
|
||
## 0x31, Get balance of the given account.
|
||
let address = c.stack.popAddress()
|
||
|
||
c.gasEip2929AccountCheck(address, gasFees[c.fork][GasBalance])
|
||
push: c.getBalance(address)
|
||
|
||
op extCodeHashEIP2929, inline = true:
|
||
let address = c.stack.popAddress()
|
||
c.gasEip2929AccountCheck(address, gasFees[c.fork][GasExtCodeHash])
|
||
push: c.getCodeHash(address)
|
||
|
||
op extCodeSizeEIP2929, inline = true:
|
||
## 0x3b, Get size of an account's code
|
||
let address = c.stack.popAddress()
|
||
c.gasEip2929AccountCheck(address, gasFees[c.fork][GasExtCode])
|
||
push: c.getCodeSize(address)
|
||
|
||
op extCodeCopyEIP2929, inline = true:
|
||
## 0x3c, Copy an account's code to memory.
|
||
let address = c.stack.popAddress()
|
||
let (memStartPos, codeStartPos, size) = c.stack.popInt(3)
|
||
let (memPos, codePos, len) = (memStartPos.cleanMemRef, codeStartPos.cleanMemRef, size.cleanMemRef)
|
||
|
||
c.gasMeter.consumeGas(
|
||
c.gasCosts[ExtCodeCopy].m_handler(c.memory.len, memPos, len),
|
||
reason="ExtCodeCopy fee")
|
||
|
||
c.gasEip2929AccountCheck(address, gasFees[c.fork][GasExtCode])
|
||
|
||
let codeBytes = c.getCode(address)
|
||
c.memory.writePaddedResult(codeBytes, memPos, codePos, len)
|
||
|
||
op selfDestructEIP2929, inline = false:
|
||
checkInStaticContext(c)
|
||
|
||
let
|
||
beneficiary = c.stack.popAddress()
|
||
isDead = not c.accountExists(beneficiary)
|
||
balance = c.getBalance(c.msg.contractAddress)
|
||
|
||
let gasParams = GasParams(kind: SelfDestruct,
|
||
sd_condition: isDead and not balance.isZero
|
||
)
|
||
|
||
var gasCost = c.gasCosts[SelfDestruct].c_handler(0.u256, gasParams).gasCost
|
||
|
||
c.vmState.mutateStateDB:
|
||
if not db.inAccessList(beneficiary):
|
||
db.accessList(beneficiary)
|
||
gasCost = gasCost + ColdAccountAccessCost
|
||
|
||
c.gasMeter.consumeGas(gasCost, reason = "SELFDESTRUCT EIP161")
|
||
c.selfDestruct(beneficiary)
|
||
|
||
op sloadEIP2929, inline = true, slot:
|
||
## 0x54, Load word from storage.
|
||
c.vmState.mutateStateDB:
|
||
let gasCost = if not db.inAccessList(c.msg.contractAddress, slot):
|
||
db.accessList(c.msg.contractAddress, slot)
|
||
ColdSloadCost
|
||
else:
|
||
WarmStorageReadCost
|
||
c.gasMeter.consumeGas(gasCost, reason = "sloadEIP2929")
|
||
|
||
push: c.getStorage(slot)
|
||
|
||
op sstoreEIP2929, inline = false, slot, newValue:
|
||
checkInStaticContext(c)
|
||
const SentryGasEIP2200 = 2300 # Minimum gas required to be present for an SSTORE call, not consumed
|
||
|
||
if c.gasMeter.gasRemaining <= SentryGasEIP2200:
|
||
raise newException(OutOfGas, "Gas not enough to perform EIP2200 SSTORE")
|
||
|
||
c.vmState.mutateStateDB:
|
||
if not db.inAccessList(c.msg.contractAddress, slot):
|
||
db.accessList(c.msg.contractAddress, slot)
|
||
c.gasMeter.consumeGas(ColdSloadCost, reason = "sstoreEIP2929")
|
||
|
||
when evmc_enabled:
|
||
sstoreEvmc(c, slot, newValue)
|
||
else:
|
||
sstoreNetGasMeteringImpl(c, slot, newValue)
|