2023-03-21 13:27:12 +00:00
|
|
|
# Nimbus
|
2024-06-10 09:05:30 +00:00
|
|
|
# Copyright (c) 2023-2024 Status Research & Development GmbH
|
2023-03-21 13:27:12 +00:00
|
|
|
# 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 tables, stint, eth/common
|
|
|
|
|
|
|
|
type
|
|
|
|
StorageTable = ref object
|
|
|
|
map: Table[UInt256, UInt256]
|
|
|
|
|
|
|
|
TransientStorage* = object
|
|
|
|
map: Table[EthAddress, StorageTable]
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
# Private helpers
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
proc merge(a, b: StorageTable) =
|
|
|
|
for k, v in b.map:
|
|
|
|
a.map[k] = v
|
|
|
|
|
|
|
|
#######################################################################
|
|
|
|
# Public functions
|
|
|
|
#######################################################################
|
|
|
|
|
|
|
|
proc init*(ac: var TransientStorage) =
|
2024-06-10 09:05:30 +00:00
|
|
|
ac.map = Table[EthAddress, StorageTable]()
|
2023-03-21 13:27:12 +00:00
|
|
|
|
|
|
|
proc init*(_: type TransientStorage): TransientStorage {.inline.} =
|
|
|
|
result.init()
|
|
|
|
|
|
|
|
func getStorage*(
|
|
|
|
ac: TransientStorage, address: EthAddress, slot: UInt256
|
|
|
|
): (bool, UInt256) =
|
|
|
|
var table = ac.map.getOrDefault(address)
|
|
|
|
if table.isNil:
|
|
|
|
return (false, 0.u256)
|
|
|
|
|
|
|
|
table.map.withValue(slot, val):
|
|
|
|
return (true, val[])
|
|
|
|
do:
|
|
|
|
return (false, 0.u256)
|
|
|
|
|
|
|
|
proc setStorage*(ac: var TransientStorage, address: EthAddress, slot, value: UInt256) =
|
|
|
|
var table = ac.map.getOrDefault(address)
|
|
|
|
if table.isNil:
|
|
|
|
table = StorageTable()
|
|
|
|
ac.map[address] = table
|
|
|
|
|
|
|
|
table.map[slot] = value
|
|
|
|
|
|
|
|
proc merge*(ac: var TransientStorage, other: TransientStorage) =
|
|
|
|
for k, v in other.map:
|
|
|
|
ac.map.withValue(k, val):
|
|
|
|
val[].merge(v)
|
|
|
|
do:
|
|
|
|
ac.map[k] = v
|
|
|
|
|
|
|
|
proc clear*(ac: var TransientStorage) {.inline.} =
|
|
|
|
ac.map.clear()
|