2023-04-08 01:01:10 +00:00
|
|
|
package forkchoice
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2023-08-15 10:45:48 +00:00
|
|
|
|
2023-06-11 21:50:02 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/transition"
|
2023-05-14 22:12:24 +00:00
|
|
|
|
2023-06-11 21:50:02 +00:00
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
2023-05-14 22:12:24 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes/solid"
|
2023-05-13 21:44:07 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cl/phase1/core/state"
|
2023-04-17 18:06:50 +00:00
|
|
|
"github.com/ledgerwatch/log/v3"
|
2023-04-08 01:01:10 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// Slot calculates the current slot number using the time and genesis slot.
|
|
|
|
func (f *ForkChoiceStore) Slot() uint64 {
|
2023-10-21 21:10:58 +00:00
|
|
|
return f.beaconCfg.GenesisSlot + ((f.time - f.genesisTime) / f.beaconCfg.SecondsPerSlot)
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// updateCheckpoints updates the justified and finalized checkpoints if new checkpoints have higher epochs.
|
2023-05-14 22:12:24 +00:00
|
|
|
func (f *ForkChoiceStore) updateCheckpoints(justifiedCheckpoint, finalizedCheckpoint solid.Checkpoint) {
|
|
|
|
if justifiedCheckpoint.Epoch() > f.justifiedCheckpoint.Epoch() {
|
2023-04-08 01:01:10 +00:00
|
|
|
f.justifiedCheckpoint = justifiedCheckpoint
|
|
|
|
}
|
2023-05-14 22:12:24 +00:00
|
|
|
if finalizedCheckpoint.Epoch() > f.finalizedCheckpoint.Epoch() {
|
2023-10-15 16:05:13 +00:00
|
|
|
f.onNewFinalized(finalizedCheckpoint)
|
2023-04-08 01:01:10 +00:00
|
|
|
f.finalizedCheckpoint = finalizedCheckpoint
|
2023-10-15 16:05:13 +00:00
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (f *ForkChoiceStore) onNewFinalized(newFinalized solid.Checkpoint) {
|
2023-10-21 21:10:58 +00:00
|
|
|
// get rid of checkpoint states
|
2023-10-15 16:05:13 +00:00
|
|
|
for k := range f.checkpointStates {
|
|
|
|
checkpoint := solid.Checkpoint(k)
|
|
|
|
if checkpoint.Epoch() <= newFinalized.Epoch() {
|
|
|
|
delete(f.checkpointStates, k)
|
|
|
|
continue
|
|
|
|
}
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
2023-10-21 21:10:58 +00:00
|
|
|
// get rid of children
|
|
|
|
for k, children := range f.childrens {
|
|
|
|
if children.parentSlot <= newFinalized.Epoch()*f.beaconCfg.SlotsPerEpoch {
|
|
|
|
delete(f.childrens, k)
|
2024-01-06 20:49:23 +00:00
|
|
|
delete(f.headSet, k)
|
2023-10-21 21:10:58 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
}
|
|
|
|
f.forkGraph.Prune(newFinalized.Epoch() * f.beaconCfg.SlotsPerEpoch)
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// updateCheckpoints updates the justified and finalized checkpoints if new checkpoints have higher epochs.
|
2023-05-14 22:12:24 +00:00
|
|
|
func (f *ForkChoiceStore) updateUnrealizedCheckpoints(justifiedCheckpoint, finalizedCheckpoint solid.Checkpoint) {
|
|
|
|
if justifiedCheckpoint.Epoch() > f.unrealizedJustifiedCheckpoint.Epoch() {
|
2023-04-08 01:01:10 +00:00
|
|
|
f.unrealizedJustifiedCheckpoint = justifiedCheckpoint
|
|
|
|
}
|
2023-05-14 22:12:24 +00:00
|
|
|
if finalizedCheckpoint.Epoch() > f.unrealizedFinalizedCheckpoint.Epoch() {
|
2023-04-08 01:01:10 +00:00
|
|
|
f.unrealizedFinalizedCheckpoint = finalizedCheckpoint
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// computeEpochAtSlot calculates the epoch at a given slot number.
|
|
|
|
func (f *ForkChoiceStore) computeEpochAtSlot(slot uint64) uint64 {
|
2023-10-21 21:10:58 +00:00
|
|
|
return slot / f.beaconCfg.SlotsPerEpoch
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// computeStartSlotAtEpoch calculates the starting slot of a given epoch.
|
|
|
|
func (f *ForkChoiceStore) computeStartSlotAtEpoch(epoch uint64) uint64 {
|
2023-10-21 21:10:58 +00:00
|
|
|
return epoch * f.beaconCfg.SlotsPerEpoch
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// computeSlotsSinceEpochStart calculates the number of slots since the start of the epoch of a given slot.
|
|
|
|
func (f *ForkChoiceStore) computeSlotsSinceEpochStart(slot uint64) uint64 {
|
|
|
|
return slot - f.computeStartSlotAtEpoch(f.computeEpochAtSlot(slot))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ancestor returns the ancestor to the given root.
|
|
|
|
func (f *ForkChoiceStore) Ancestor(root libcommon.Hash, slot uint64) libcommon.Hash {
|
|
|
|
header, has := f.forkGraph.GetHeader(root)
|
|
|
|
if !has {
|
|
|
|
return libcommon.Hash{}
|
|
|
|
}
|
|
|
|
for header.Slot > slot {
|
|
|
|
root = header.ParentRoot
|
|
|
|
header, has = f.forkGraph.GetHeader(header.ParentRoot)
|
|
|
|
if !has {
|
|
|
|
return libcommon.Hash{}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return root
|
|
|
|
}
|
|
|
|
|
|
|
|
// getCheckpointState computes and caches checkpoint states.
|
2023-05-14 22:12:24 +00:00
|
|
|
func (f *ForkChoiceStore) getCheckpointState(checkpoint solid.Checkpoint) (*checkpointState, error) {
|
2023-04-08 01:01:10 +00:00
|
|
|
// check if it can be found in cache.
|
2023-10-15 16:05:13 +00:00
|
|
|
if state, ok := f.checkpointStates[checkpointComparable(checkpoint)]; ok {
|
2023-04-08 01:01:10 +00:00
|
|
|
return state, nil
|
|
|
|
}
|
|
|
|
// If it is not in cache compute it and then put in cache.
|
2023-10-21 21:10:58 +00:00
|
|
|
baseState, err := f.forkGraph.GetState(checkpoint.BlockRoot(), true)
|
2023-04-08 01:01:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if baseState == nil {
|
|
|
|
return nil, fmt.Errorf("getCheckpointState: baseState not found in graph")
|
|
|
|
}
|
|
|
|
// By default use the no change encoding to signal that there is no future epoch here.
|
2023-05-14 22:12:24 +00:00
|
|
|
if baseState.Slot() < f.computeStartSlotAtEpoch(checkpoint.Epoch()) {
|
2023-04-17 18:06:50 +00:00
|
|
|
log.Debug("Long checkpoint detected")
|
2023-04-08 01:01:10 +00:00
|
|
|
// If we require to change it then process the future epoch
|
2023-06-11 21:50:02 +00:00
|
|
|
if err := transition.DefaultMachine.ProcessSlots(baseState, f.computeStartSlotAtEpoch(checkpoint.Epoch())); err != nil {
|
2023-04-08 01:01:10 +00:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
}
|
2023-05-03 08:51:39 +00:00
|
|
|
mixes := baseState.RandaoMixes()
|
2023-05-10 19:37:50 +00:00
|
|
|
// TODO: make this copy smarter when validators is a smarter struct
|
2023-05-28 15:11:18 +00:00
|
|
|
validators := make([]solid.Validator, baseState.ValidatorLength())
|
|
|
|
baseState.ForEachValidator(func(v solid.Validator, idx, total int) bool {
|
2023-05-10 19:37:50 +00:00
|
|
|
validators[idx] = v
|
|
|
|
return true
|
|
|
|
})
|
2023-10-21 21:10:58 +00:00
|
|
|
checkpointState := newCheckpointState(f.beaconCfg, f.anchorPublicKeys, validators,
|
2023-05-23 18:58:34 +00:00
|
|
|
mixes, baseState.GenesisValidatorsRoot(), baseState.Fork(), baseState.GetTotalActiveBalance(), state.Epoch(baseState.BeaconState))
|
2023-04-08 01:01:10 +00:00
|
|
|
// Cache in memory what we are left with.
|
2023-10-15 16:05:13 +00:00
|
|
|
f.checkpointStates[checkpointComparable(checkpoint)] = checkpointState
|
2023-05-03 08:51:39 +00:00
|
|
|
return checkpointState, nil
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|