implement block witness data collection in accounts_cache

This commit is contained in:
jangko 2020-06-03 20:50:13 +07:00
parent 7d062e171b
commit 50816f2ebb
No known key found for this signature in database
GPG Key ID: 31702AE10541E6B9
2 changed files with 67 additions and 18 deletions

View File

@ -1,7 +1,8 @@
import import
tables, hashes, sets, tables, hashes, sets,
eth/[common, rlp], eth/trie/[hexary, db, trie_defs], eth/[common, rlp], eth/trie/[hexary, db, trie_defs],
../constants, ../utils, storage_types ../constants, ../utils, storage_types,
../../stateless/[witness_types, multi_keys]
type type
AccountFlag = enum AccountFlag = enum
@ -23,11 +24,15 @@ type
originalStorage: TableRef[UInt256, UInt256] originalStorage: TableRef[UInt256, UInt256]
overlayStorage: Table[UInt256, UInt256] overlayStorage: Table[UInt256, UInt256]
WitnessData* = object
storageKeys*: HashSet[UInt256]
codeTouched*: bool
AccountsCache* = ref object AccountsCache* = ref object
db: TrieDatabaseRef db: TrieDatabaseRef
trie: SecureHexaryTrie trie: SecureHexaryTrie
savePoint: SavePoint savePoint: SavePoint
unrevertablyTouched: HashSet[EthAddress] witnessCache: Table[EthAddress, WitnessData]
ReadOnlyStateDB* = distinct AccountsCache ReadOnlyStateDB* = distinct AccountsCache
@ -51,7 +56,7 @@ proc init*(x: typedesc[AccountsCache], db: TrieDatabaseRef,
new result new result
result.db = db result.db = db
result.trie = initSecureHexaryTrie(db, root, pruneTrie) result.trie = initSecureHexaryTrie(db, root, pruneTrie)
result.unrevertablyTouched = initHashSet[EthAddress]() result.witnessCache = initTable[EthAddress, WitnessData]()
discard result.beginSavepoint discard result.beginSavepoint
proc init*(x: typedesc[AccountsCache], db: TrieDatabaseRef, pruneTrie: bool = true): AccountsCache = proc init*(x: typedesc[AccountsCache], db: TrieDatabaseRef, pruneTrie: bool = true): AccountsCache =
@ -373,21 +378,6 @@ proc clearStorage*(ac: var AccountsCache, address: EthAddress) =
# there is no point to clone the storage since we want to remove it # there is no point to clone the storage since we want to remove it
ac.makeDirty(address, cloneStorage = false).account.storageRoot = emptyRlpHash ac.makeDirty(address, cloneStorage = false).account.storageRoot = emptyRlpHash
proc unrevertableTouch*(ac: var AccountsCache, address: EthAddress) =
ac.unrevertablyTouched.incl address
proc removeEmptyAccounts*(ac: var AccountsCache) =
# make sure all savepoints already committed
doAssert(ac.savePoint.parentSavePoint.isNil)
for _, acc in ac.savePoint.cache:
if IsTouched in acc.flags and acc.isEmpty:
acc.kill()
for address in ac.unrevertablyTouched:
var acc = ac.getAccount(address)
if acc.isEmpty:
acc.kill()
proc deleteAccount*(ac: var AccountsCache, address: EthAddress) = proc deleteAccount*(ac: var AccountsCache, address: EthAddress) =
# make sure all savepoints already committed # make sure all savepoints already committed
doAssert(ac.savePoint.parentSavePoint.isNil) doAssert(ac.savePoint.parentSavePoint.isNil)
@ -433,6 +423,50 @@ proc getStorageRoot*(ac: AccountsCache, address: EthAddress): Hash256 =
if acc.isNil: emptyAcc.storageRoot if acc.isNil: emptyAcc.storageRoot
else: acc.account.storageRoot else: acc.account.storageRoot
func update(wd: var WitnessData, acc: RefAccount) =
wd.codeTouched = CodeChanged in acc.flags
if not acc.originalStorage.isNil:
for k, v in acc.originalStorage:
if v == 0: continue
wd.storageKeys.incl k
for k, v in acc.overlayStorage:
if v == 0 and k notin wd.storageKeys:
continue
if v == 0 and k in wd.storageKeys:
wd.storageKeys.excl k
continue
wd.storageKeys.incl k
func witnessData(acc: RefAccount): WitnessData =
result.storageKeys = initHashSet[UInt256]()
update(result, acc)
proc collectWitnessData*(ac: var AccountsCache) =
# make sure all savepoint already committed
doAssert(ac.savePoint.parentSavePoint.isNil)
# usually witness data is collected before we call persist()
for address, acc in ac.savePoint.cache:
ac.witnessCache.withValue(address, val) do:
update(val[], acc)
do:
ac.witnessCache[address] = witnessData(acc)
func multiKeys(slots: HashSet[UInt256]): MultikeysRef =
if slots.len == 0: return
new result
for x in slots:
result.add x.toBytesBE
result.sort()
proc makeMultiKeys*(ac: AccountsCache): MultikeysRef =
# this proc is called after we done executing a block
new result
for k, v in ac.witnessCache:
result.add(k, v.codeTouched, multiKeys(v.storageKeys))
result.sort()
proc rootHash*(db: ReadOnlyStateDB): KeccakHash {.borrow.} proc rootHash*(db: ReadOnlyStateDB): KeccakHash {.borrow.}
proc getCodeHash*(db: ReadOnlyStateDB, address: EthAddress): Hash256 {.borrow.} proc getCodeHash*(db: ReadOnlyStateDB, address: EthAddress): Hash256 {.borrow.}
proc getStorageRoot*(db: ReadOnlyStateDB, address: EthAddress): Hash256 {.borrow.} proc getStorageRoot*(db: ReadOnlyStateDB, address: EthAddress): Hash256 {.borrow.}

View File

@ -76,6 +76,21 @@ proc newMultiKeys*(keys: openArray[StorageSlot]): MultikeysRef =
result.keys[i] = KeyData(storageMode: true, hash: keccak(a).data, storageSlot: a) result.keys[i] = KeyData(storageMode: true, hash: keccak(a).data, storageSlot: a)
result.keys.sort(cmpHash) result.keys.sort(cmpHash)
# never mix storageMode!
proc add*(m: MultikeysRef, address: EthAddress, codeTouched: bool, storageKeys = MultikeysRef(nil)) =
m.keys.add KeyData(
storageMode: false,
hash: keccak(address).data,
address: address,
codeTouched: codeTouched,
storageKeys: storageKeys)
proc add*(m: MultikeysRef, slot: StorageSlot) =
m.keys.add KeyData(storageMode: true, hash: keccak(slot).data, storageSlot: slot)
proc sort*(m: MultikeysRef) =
m.keys.sort(cmpHash)
func initGroup*(m: MultikeysRef): Group = func initGroup*(m: MultikeysRef): Group =
type T = type result.last type T = type result.last
result = Group(first: 0.T, last: (m.keys.len - 1).T) result = Group(first: 0.T, last: (m.keys.len - 1).T)