mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-19 16:48:50 +00:00
7becf4e389
why: It is not safe in general to recycle vertex IDs while the `RocksDb` cache has `VertexID` rather than `RootedVertexID` where the former type seems preferable. In some fringe cases one might remove a vertex with key `(root1,vid)` and insert another vertex with key `(root2,vid)` while re-using the vertex ID `vid`. Without knowledge of `root1` and `root2`, the LRU cache will return the same vertex for `(root2,vid)` also for `(root1,vid)`.
177 lines
5.0 KiB
Nim
177 lines
5.0 KiB
Nim
# nimbus-eth1
|
|
# Copyright (c) 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.
|
|
|
|
{.push raises: [].}
|
|
|
|
import
|
|
std/sets,
|
|
eth/common,
|
|
results,
|
|
".."/[aristo_desc, aristo_get, aristo_hike, aristo_layers, aristo_utils],
|
|
#./part_debug,
|
|
./part_desc
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Private functions
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc newCtx(ps: PartStateRef; hike: Hike): Result[PartStateCtx,AristoError] =
|
|
## ..
|
|
doAssert 0 <= hike.legs[^1].nibble
|
|
|
|
let
|
|
wp = hike.legs[^1].wp
|
|
nibble = hike.legs[^1].nibble
|
|
fromVid = wp.vtx.bVid[nibble]
|
|
|
|
if not ps.isPerimeter(fromVid) or ps.isExtension(fromVid):
|
|
return err(PartCtxNotAvailable)
|
|
|
|
let
|
|
vtx2 = wp.vtx.dup
|
|
psc = PartStateCtx(
|
|
ps: ps,
|
|
location: (hike.root,wp.vid),
|
|
nibble: nibble,
|
|
fromVid: fromVid)
|
|
|
|
# Update database so that is space for adding a new sub-tree here
|
|
vtx2.bVid[nibble] = VertexID(0)
|
|
ps.db.layersPutVtx(psc.location,vtx2)
|
|
ok psc
|
|
|
|
proc removedCompletedNode(
|
|
ps: PartStateRef;
|
|
rvid: RootedVertexID;
|
|
key: HashKey;
|
|
): bool =
|
|
let vtx = ps.db.getVtx rvid
|
|
if vtx.isNil:
|
|
return false
|
|
|
|
var subVids: seq[VertexID]
|
|
for vid in vtx.subVids():
|
|
# Only accept perimeter nodes with all links on the database (i.e. links
|
|
# must nor refere t a core node.)
|
|
if not ps.db.getVtx((rvid.root, vid)).isValid or ps.isCore(vid):
|
|
return false # not complete
|
|
subVids.add vid
|
|
|
|
# No need to keep that core vertex any longer
|
|
ps.delCore(rvid.root, key)
|
|
for vid in subVids:
|
|
ps.del vid
|
|
|
|
true
|
|
|
|
proc removeCompletedNodes(ps: PartStateRef; rvid: RootedVertexID) =
|
|
let key = ps[rvid.vid]
|
|
if ps.removedCompletedNode(rvid, key):
|
|
# Rather stupid loop to clear additional nodes. Note that the set
|
|
# of core nodes is assumed to be small, i.e. not more than a hand full.
|
|
while true:
|
|
ps.core.withValue(rvid.root, coreKeys):
|
|
block removeItem:
|
|
for cKey in coreKeys[]:
|
|
let rv = ps[cKey]
|
|
if ps.removedCompletedNode(rv, cKey):
|
|
break removeItem # continue `while`
|
|
return # done
|
|
do: return # done
|
|
|
|
# -------------------
|
|
|
|
proc ctxAcceptChange(psc: PartStateCtx): Result[bool,AristoError] =
|
|
## Apply `psc` context if there was a change on the targeted vertex,
|
|
## otherwise restore. Returns `true` exactly if there was a change on
|
|
## the database which could be applied.
|
|
##
|
|
let
|
|
ps = psc.ps
|
|
db = ps.db
|
|
(vtx,_) = ? db.getVtxRc psc.location
|
|
toVid = vtx.bVid[psc.nibble]
|
|
|
|
if not toVid.isValid:
|
|
# Nothing changed, so restore
|
|
let vtx2 = vtx.dup
|
|
vtx2.bVid[psc.nibble] = psc.fromVid
|
|
db.layersPutVtx(psc.location, vtx2)
|
|
ok(false)
|
|
|
|
elif toVid != psc.fromVid:
|
|
# Replace `fromVid` by `toVid` in state descriptor `ps`
|
|
let key = ps.move(psc.fromVid, toVid)
|
|
doAssert key.isValid
|
|
ps.changed.incl key
|
|
|
|
# Remove parent vertex it it has become complete.
|
|
ps.removeCompletedNodes(psc.location)
|
|
ok(true)
|
|
|
|
else:
|
|
ok(false)
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public functions
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc ctxMergeBegin*(
|
|
ps: PartStateRef;
|
|
root: VertexID;
|
|
path: openArray[byte];
|
|
): Result[PartStateCtx,AristoError] =
|
|
## This function clears the way for mering some payload at the argument
|
|
## path `(root,path)`.
|
|
let hike = block:
|
|
let rc = NibblesBuf.fromBytes(path).hikeUp(root,ps.db)
|
|
if rc.isOk:
|
|
return ok PartStateCtx(nil) # Nothing to do
|
|
if rc.error[1] != HikeDanglingEdge:
|
|
return err( rc.error[1]) # Cannot help here
|
|
rc.to(Hike)
|
|
|
|
ps.newCtx hike
|
|
|
|
proc ctxMergeBegin*(
|
|
ps: PartStateRef;
|
|
accPath: Hash256;
|
|
): Result[PartStateCtx,AristoError] =
|
|
## Variant of `partMergeBegin()` for different path representation
|
|
ps.ctxMergeBegin(VertexID(1), accPath.data)
|
|
|
|
|
|
proc ctxMergeCommit*(psc: PartStateCtx): Result[bool,AristoError] =
|
|
##
|
|
if psc.isNil:
|
|
return ok(false) # Nothing to do
|
|
if psc.ps.isNil:
|
|
return err(PartCtxStaleDescriptor)
|
|
|
|
let yn = ? psc.ctxAcceptChange()
|
|
psc[].reset
|
|
ok(yn)
|
|
|
|
|
|
proc ctxMergeRollback*(psc: PartStateCtx): Result[void,AristoError] =
|
|
## ..
|
|
if psc.isNil:
|
|
return ok() # Nothing to do
|
|
if psc.ps.isNil:
|
|
return err(PartCtxStaleDescriptor)
|
|
|
|
let yn = ? psc.ctxAcceptChange()
|
|
psc[].reset
|
|
if yn: err(PartVtxSlotWasModified) else: ok()
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# End
|
|
# ------------------------------------------------------------------------------
|