From df0ff5f0fb83bcbd00de0dac9ecbf57aa9ec6aff Mon Sep 17 00:00:00 2001 From: Etan Kissling Date: Wed, 3 Apr 2024 23:03:06 +0200 Subject: [PATCH] fix initialization of sync committee cache after loading non-epoch state (#6160) When initializing from a state that's not aligned to an epoch boundary, an earlier state is loaded that's epoch aligned, and subsequently topped up with the missing blocks. `dag.headSyncCommittee` is initialized prior to topping up the missing blocks, though. If the sync committee changes while applying the blocks (e.g., a sync committee period boundary hits), the cached information becomes unlinked from `dag.head`, leading to valid blocks based on that chain being rejected. To fix this, move cache initialization after the top up with blocks. This has been observed on Goerli by initializing from 7919502 and attempting to top up 7920111. The block gets rejected with an invalid state root on nodes that have restarted after setting 7920111 as head, while it gets accepted by all other nodes. Error message is `block: state root verification failed`. The incorrect initialization behaviour was introduced in #4592, before which the sync committee cache was initialized after applying blocks. --- beacon_chain/consensus_object_pools/blockchain_dag.nim | 8 ++++---- 1 file changed, 4 insertions(+), 4 deletions(-) diff --git a/beacon_chain/consensus_object_pools/blockchain_dag.nim b/beacon_chain/consensus_object_pools/blockchain_dag.nim index e9d60eb46..8fa5808a8 100644 --- a/beacon_chain/consensus_object_pools/blockchain_dag.nim +++ b/beacon_chain/consensus_object_pools/blockchain_dag.nim @@ -1131,10 +1131,6 @@ proc init*(T: type ChainDAGRef, cfg: RuntimeConfig, db: BeaconChainDB, head = shortLog(head), tail = shortLog(dag.tail) quit 1 - withState(dag.headState): - when consensusFork >= ConsensusFork.Altair: - dag.headSyncCommittees = forkyState.data.get_sync_committee_cache(cache) - block: # EpochRef needs an epoch boundary state assign(dag.epochRefState, dag.headState) @@ -1149,6 +1145,10 @@ proc init*(T: type ChainDAGRef, cfg: RuntimeConfig, db: BeaconChainDB, dag.head = headRef dag.heads = @[headRef] + withState(dag.headState): + when consensusFork >= ConsensusFork.Altair: + dag.headSyncCommittees = forkyState.data.get_sync_committee_cache(cache) + assign(dag.clearanceState, dag.headState) if dag.headState.latest_block_root == tail.root: