mirror of
https://github.com/status-im/nimbus-eth1.git
synced 2025-01-22 18:20:11 +00:00
a241050c94
* Log/trace cancellation events in scheduler * Provide `clear()` functions for explicitly flushing data objects * Renaming header cache functions why: More systematic, all functions start with prefix `dbHeader` * Remove `danglingParent` from layout why: Already provided by header cache * Remove `couplerHash` and `headHash` from layout why: No need to cache, `headHash` is unused and `couplerHash` used typically once, only. * Remove `lastLayout` from sync descriptor why: No need to compare changes, saving is always triggered after actively changing the sync layout state * Early reject unsuitable head + finalised header from CL why: The finalised header is only passed by its hash so the header must be fetched somewhere, e.g. from a peer via eth/xx. Also, finalised headers earlier than the `base` from `FC` cannot be handled due to the `Aristo` single state database architecture. Luckily, on a full node, the complete block history is available so unsuitable finalised headers are stored there already which is exploited here to avoid unnecessary network traffic. * Code cosmetics, remove cruft, prettify logging, remove `final` metrics detail: The `final` layout parameter will be deprecated and later removed * Update/re-calibrate syncer logic documentation why: The current implementation sucks if the `FC` module changes the canonical branch in the middle of completing a header chain (due to concurrent updates by the `newPayload()` logic.) * Implement according to re-calibrated syncer docu details: The implementation employs the notion of named layout states (see `SyncLayoutState` in `worker_desc.nim`) which are derived from the state parameter triple `(C,D,H)` as described in `README.md`.
130 lines
3.8 KiB
Nim
130 lines
3.8 KiB
Nim
# Nimbus
|
|
# Copyright (c) 2023-2024 Status Research & Development GmbH
|
|
# Licensed and distributed under either of
|
|
# * MIT license (license terms in the root directory or at
|
|
# https://opensource.org/licenses/MIT).
|
|
# * Apache v2 license (license terms in the root directory or at
|
|
# https://www.apache.org/licenses/LICENSE-2.0).
|
|
# at your option. This file may not be copied, modified, or distributed
|
|
# except according to those terms.
|
|
|
|
{.push raises:[].}
|
|
|
|
import
|
|
pkg/eth/[common, p2p],
|
|
pkg/results,
|
|
pkg/stew/interval_set,
|
|
../worker_desc
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# Public functions
|
|
# ------------------------------------------------------------------------------
|
|
|
|
proc headersUnprocFetch*(
|
|
ctx: BeaconCtxRef;
|
|
maxLen: uint64;
|
|
): Result[BnRange,void] =
|
|
## Fetch interval from block ranges with maximal size `maxLen`, where
|
|
## `0` is interpreted as `2^64`.
|
|
##
|
|
let
|
|
q = ctx.hdr.unprocessed
|
|
|
|
# Fetch top/right interval with largest block numbers
|
|
jv = q.le().valueOr:
|
|
return err()
|
|
|
|
# Curb interval to maximal length `maxLen`
|
|
iv = block:
|
|
if maxLen == 0 or (0 < jv.len and jv.len <= maxLen):
|
|
jv
|
|
else:
|
|
# Curb interval `jv` to length `maxLen`
|
|
#
|
|
# Note that either (fringe case):
|
|
# (`jv.len`==0) => (`jv`==`[0,high(u64)]`) => `jv.maxPt`==`high(u64)`
|
|
# or (in the non-fringe case)
|
|
# (`maxLen` < `jv.len`) => (`jv.maxPt` - `maxLen` + 1 < `jv.maxPt`)
|
|
#
|
|
BnRange.new(jv.maxPt - maxLen + 1, jv.maxPt)
|
|
|
|
discard q.reduce(iv)
|
|
ctx.hdr.borrowed += iv.len
|
|
ok(iv)
|
|
|
|
|
|
proc headersUnprocCommit*(ctx: BeaconCtxRef; borrowed: uint) =
|
|
## Commit back all processed range
|
|
ctx.hdr.borrowed -= borrowed
|
|
|
|
proc headersUnprocCommit*(ctx: BeaconCtxRef; borrowed: uint; retuor: BnRange) =
|
|
## Merge back unprocessed range `retour`
|
|
ctx.headersUnprocCommit borrowed
|
|
doAssert ctx.hdr.unprocessed.merge(retuor) == retuor.len
|
|
|
|
proc headersUnprocCommit*(
|
|
ctx: BeaconCtxRef;
|
|
borrowed: uint;
|
|
rMinPt: BlockNumber;
|
|
rMaxPt: BlockNumber) =
|
|
## Variant of `headersUnprocCommit()`
|
|
ctx.headersUnprocCommit borrowed
|
|
doAssert ctx.hdr.unprocessed.merge(rMinPt, rMaxPt) == rMaxPt - rMinPt + 1
|
|
|
|
|
|
|
|
proc headersUnprocCovered*(
|
|
ctx: BeaconCtxRef;
|
|
minPt: BlockNumber;
|
|
maxPt: BlockNumber;
|
|
): uint64 =
|
|
## Check whether range is fully contained
|
|
# Argument `maxPt` would be internally adjusted to `max(minPt,maxPt)`
|
|
if minPt <= maxPt:
|
|
return ctx.hdr.unprocessed.covered(minPt, maxPt)
|
|
|
|
proc headersUnprocCovered*(ctx: BeaconCtxRef; pt: BlockNumber): bool =
|
|
## Check whether point is contained
|
|
ctx.hdr.unprocessed.covered(pt, pt) == 1
|
|
|
|
|
|
proc headersUnprocTop*(ctx: BeaconCtxRef): BlockNumber =
|
|
let iv = ctx.hdr.unprocessed.le().valueOr:
|
|
return BlockNumber(0)
|
|
iv.maxPt
|
|
|
|
proc headersUnprocTotal*(ctx: BeaconCtxRef): uint64 =
|
|
ctx.hdr.unprocessed.total()
|
|
|
|
proc headersUnprocBorrowed*(ctx: BeaconCtxRef): uint64 =
|
|
ctx.hdr.borrowed
|
|
|
|
proc headersUnprocChunks*(ctx: BeaconCtxRef): int =
|
|
ctx.hdr.unprocessed.chunks()
|
|
|
|
proc headersUnprocIsEmpty*(ctx: BeaconCtxRef): bool =
|
|
ctx.hdr.unprocessed.chunks() == 0
|
|
|
|
# ------------
|
|
|
|
proc headersUnprocInit*(ctx: BeaconCtxRef) =
|
|
## Constructor
|
|
ctx.hdr.unprocessed = BnRangeSet.init()
|
|
|
|
|
|
proc headersUnprocClear*(ctx: BeaconCtxRef) =
|
|
## Clear
|
|
ctx.hdr.unprocessed.clear()
|
|
ctx.hdr.borrowed = 0u
|
|
|
|
proc headersUnprocSet*(ctx: BeaconCtxRef; minPt, maxPt: BlockNumber) =
|
|
## Set up new unprocessed range
|
|
ctx.headersUnprocClear()
|
|
# Argument `maxPt` would be internally adjusted to `max(minPt,maxPt)`
|
|
if minPt <= maxPt:
|
|
discard ctx.hdr.unprocessed.merge(minPt, maxPt)
|
|
|
|
# ------------------------------------------------------------------------------
|
|
# End
|
|
# ------------------------------------------------------------------------------
|