mirror of
https://github.com/status-im/nimbus-eth2.git
synced 2025-01-28 07:15:57 +00:00
d028baea2a
On `ELECTRA_FORK_EPOCH`, PeerDAS is not yet activated, hence the current mechanism based on `BlobSidecar` is still in use. With EIP-7688, the generalized indices of `BeaconBlockBody` get reindexed, changing the length of the inclusion proof within the `BlobSidecar`. Because network Req/Resp operations allow responses across fork boundaries, this creates the need for a `ForkedBlobSidecar` in that layer, same as already done for `ForkedSignedBeaconBock` for similar reasons. Note: This PR is only needed if PeerDAS is adopted _after_ EIP-7688. If PeerDAS is adopted _before_ EIP-7688, a similar PR may be needed for forked columns. Coincidental `Forked` jank can only be fully avoided if both features activate at the same epoch, actual changes to blobs aside. Delaying EIP-7688 for sole purpose of epoch alignemnt is not worth it.
122 lines
4.4 KiB
Nim
122 lines
4.4 KiB
Nim
# beacon_chain
|
|
# Copyright (c) 2018-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
|
|
std/tables,
|
|
../spec/helpers
|
|
|
|
from std/sequtils import mapIt
|
|
from std/strutils import join
|
|
|
|
const
|
|
MaxBlobs = 3 * SLOTS_PER_EPOCH * MAX_BLOBS_PER_BLOCK
|
|
## Same limit as `MaxOrphans` in `block_quarantine`;
|
|
## blobs may arrive before an orphan is tagged `blobless`
|
|
|
|
type
|
|
BlobQuarantine* = object
|
|
blobs*: OrderedTable[
|
|
(Eth2Digest, BlobIndex, KzgCommitment), ForkedBlobSidecar]
|
|
onBlobSidecarCallback*: OnBlobSidecarCallback
|
|
|
|
BlobFetchRecord* = object
|
|
block_root*: Eth2Digest
|
|
indices*: seq[BlobIndex]
|
|
|
|
OnBlobSidecarCallback = proc(
|
|
data: BlobSidecarInfoObject) {.gcsafe, raises: [].}
|
|
|
|
func shortLog*(x: seq[BlobIndex]): string =
|
|
"<" & x.mapIt($it).join(", ") & ">"
|
|
|
|
func shortLog*(x: seq[BlobFetchRecord]): string =
|
|
"[" & x.mapIt(shortLog(it.block_root) & shortLog(it.indices)).join(", ") & "]"
|
|
|
|
func put*(quarantine: var BlobQuarantine, blobSidecar: ForkedBlobSidecar) =
|
|
if quarantine.blobs.lenu64 >= MaxBlobs:
|
|
# FIFO if full. For example, sync manager and request manager can race to
|
|
# put blobs in at the same time, so one gets blob insert -> block resolve
|
|
# -> blob insert sequence, which leaves garbage blobs.
|
|
#
|
|
# This also therefore automatically garbage-collects otherwise valid garbage
|
|
# blobs which are correctly signed, point to either correct block roots or a
|
|
# block root which isn't ever seen, and then are for any reason simply never
|
|
# used.
|
|
var oldest_blob_key: (Eth2Digest, BlobIndex, KzgCommitment)
|
|
for k in quarantine.blobs.keys:
|
|
oldest_blob_key = k
|
|
break
|
|
quarantine.blobs.del oldest_blob_key
|
|
withForkyBlob(blobSidecar):
|
|
let block_root = hash_tree_root(forkyBlob[].signed_block_header.message)
|
|
discard quarantine.blobs.hasKeyOrPut(
|
|
(block_root, forkyBlob[].index, forkyBlob[].kzg_commitment), blobSidecar)
|
|
|
|
func put*(quarantine: var BlobQuarantine, blobSidecar: ref ForkyBlobSidecar) =
|
|
quarantine.put(ForkedBlobSidecar.init(blobSidecar))
|
|
|
|
func hasBlob*(
|
|
quarantine: BlobQuarantine,
|
|
slot: Slot,
|
|
proposer_index: uint64,
|
|
index: BlobIndex): bool =
|
|
for blobSidecar in quarantine.blobs.values:
|
|
withForkyBlob(blobSidecar):
|
|
template block_header: untyped = forkyBlob[].signed_block_header.message
|
|
if block_header.slot == slot and
|
|
block_header.proposer_index == proposer_index and
|
|
forkyBlob[].index == index:
|
|
return true
|
|
false
|
|
|
|
func popBlobs*(
|
|
quarantine: var BlobQuarantine, digest: Eth2Digest,
|
|
blck:
|
|
deneb.SignedBeaconBlock |
|
|
electra.SignedBeaconBlock): auto =
|
|
const blobFork = blobForkAtConsensusFork(typeof(blck).kind).expect("Blobs OK")
|
|
type ResultType = blobFork.BlobSidecars
|
|
var r: ResultType = @[]
|
|
for idx, kzg_commitment in blck.message.body.blob_kzg_commitments:
|
|
var b: ForkedBlobSidecar
|
|
if quarantine.blobs.pop((digest, BlobIndex idx, kzg_commitment), b):
|
|
# It was already verified that the blob is linked to `blck`.
|
|
# Therefore, we can assume that `BlobFork` is correct.
|
|
doAssert b.kind == blobFork,
|
|
"Must verify blob inclusion proof before `BlobQuarantine.put`"
|
|
r.add(b.forky(blobFork))
|
|
r
|
|
|
|
func hasBlobs*(
|
|
quarantine: BlobQuarantine,
|
|
blck:
|
|
deneb.SignedBeaconBlock |
|
|
electra.SignedBeaconBlock): bool =
|
|
for idx, kzg_commitment in blck.message.body.blob_kzg_commitments:
|
|
if (blck.root, BlobIndex idx, kzg_commitment) notin quarantine.blobs:
|
|
return false
|
|
true
|
|
|
|
func blobFetchRecord*(
|
|
quarantine: BlobQuarantine,
|
|
blck:
|
|
deneb.SignedBeaconBlock |
|
|
electra.SignedBeaconBlock): BlobFetchRecord =
|
|
var indices: seq[BlobIndex]
|
|
for i in 0..<len(blck.message.body.blob_kzg_commitments):
|
|
let idx = BlobIndex(i)
|
|
if not quarantine.blobs.hasKey(
|
|
(blck.root, idx, blck.message.body.blob_kzg_commitments[i])):
|
|
indices.add(idx)
|
|
BlobFetchRecord(block_root: blck.root, indices: indices)
|
|
|
|
func init*(
|
|
T: type BlobQuarantine, onBlobSidecarCallback: OnBlobSidecarCallback): T =
|
|
T(onBlobSidecarCallback: onBlobSidecarCallback)
|