2023-03-29 16:16:45 +00:00
|
|
|
package forkchoice
|
2023-04-08 01:01:10 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
lru "github.com/hashicorp/golang-lru/v2"
|
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
|
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/core/state"
|
2023-04-17 18:06:50 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/execution_client"
|
2023-04-08 01:01:10 +00:00
|
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/forkchoice/fork_graph"
|
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
checkpointsPerCache = 1024
|
|
|
|
allowedCachedStates = 4
|
|
|
|
)
|
|
|
|
|
|
|
|
type ForkChoiceStore struct {
|
|
|
|
time uint64
|
2023-04-17 18:06:50 +00:00
|
|
|
highestSeen uint64
|
2023-04-08 01:01:10 +00:00
|
|
|
justifiedCheckpoint *cltypes.Checkpoint
|
|
|
|
finalizedCheckpoint *cltypes.Checkpoint
|
|
|
|
unrealizedJustifiedCheckpoint *cltypes.Checkpoint
|
|
|
|
unrealizedFinalizedCheckpoint *cltypes.Checkpoint
|
|
|
|
proposerBoostRoot libcommon.Hash
|
|
|
|
// Use go map because this is actually an unordered set
|
|
|
|
equivocatingIndicies map[uint64]struct{}
|
|
|
|
forkGraph *fork_graph.ForkGraph
|
|
|
|
// I use the cache due to the convenient auto-cleanup feauture.
|
2023-05-03 08:51:39 +00:00
|
|
|
checkpointStates *lru.Cache[cltypes.Checkpoint, *checkpointState] // We keep ssz snappy of it as the full beacon state is full of rendundant data.
|
2023-04-17 18:06:50 +00:00
|
|
|
latestMessages map[uint64]*LatestMessage
|
|
|
|
// We keep track of them so that we can forkchoice with EL.
|
|
|
|
eth2Roots *lru.Cache[libcommon.Hash, libcommon.Hash] // ETH2 root -> ETH1 hash
|
|
|
|
mu sync.Mutex
|
|
|
|
// EL
|
|
|
|
engine execution_client.ExecutionEngine
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type LatestMessage struct {
|
|
|
|
Epoch uint64
|
|
|
|
Root libcommon.Hash
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewForkChoiceStore initialize a new store from the given anchor state, either genesis or checkpoint sync state.
|
2023-04-20 20:47:58 +00:00
|
|
|
func NewForkChoiceStore(anchorState *state.BeaconState, engine execution_client.ExecutionEngine, enabledPruning bool) (*ForkChoiceStore, error) {
|
2023-04-08 01:01:10 +00:00
|
|
|
anchorRoot, err := anchorState.BlockRoot()
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
anchorCheckpoint := &cltypes.Checkpoint{
|
2023-05-04 13:18:42 +00:00
|
|
|
Epoch: state.Epoch(anchorState.BeaconState),
|
2023-04-08 01:01:10 +00:00
|
|
|
Root: anchorRoot,
|
|
|
|
}
|
2023-05-03 08:51:39 +00:00
|
|
|
checkpointStates, err := lru.New[cltypes.Checkpoint, *checkpointState](allowedCachedStates)
|
2023-04-08 01:01:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2023-04-17 18:06:50 +00:00
|
|
|
eth2Roots, err := lru.New[libcommon.Hash, libcommon.Hash](checkpointsPerCache)
|
2023-04-08 01:01:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return &ForkChoiceStore{
|
2023-04-17 18:06:50 +00:00
|
|
|
highestSeen: anchorState.Slot(),
|
2023-04-08 01:01:10 +00:00
|
|
|
time: anchorState.GenesisTime() + anchorState.BeaconConfig().SecondsPerSlot*anchorState.Slot(),
|
|
|
|
justifiedCheckpoint: anchorCheckpoint.Copy(),
|
|
|
|
finalizedCheckpoint: anchorCheckpoint.Copy(),
|
|
|
|
unrealizedJustifiedCheckpoint: anchorCheckpoint.Copy(),
|
|
|
|
unrealizedFinalizedCheckpoint: anchorCheckpoint.Copy(),
|
2023-04-20 20:47:58 +00:00
|
|
|
forkGraph: fork_graph.New(anchorState, enabledPruning),
|
2023-04-08 01:01:10 +00:00
|
|
|
equivocatingIndicies: map[uint64]struct{}{},
|
|
|
|
latestMessages: map[uint64]*LatestMessage{},
|
|
|
|
checkpointStates: checkpointStates,
|
2023-04-17 18:06:50 +00:00
|
|
|
eth2Roots: eth2Roots,
|
|
|
|
engine: engine,
|
2023-04-08 01:01:10 +00:00
|
|
|
}, nil
|
|
|
|
}
|
|
|
|
|
2023-04-17 18:06:50 +00:00
|
|
|
// Highest seen returns highest seen slot
|
|
|
|
func (f *ForkChoiceStore) HighestSeen() uint64 {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
return f.highestSeen
|
|
|
|
}
|
|
|
|
|
2023-04-08 01:01:10 +00:00
|
|
|
// Time returns current time
|
|
|
|
func (f *ForkChoiceStore) Time() uint64 {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
return f.time
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProposerBoostRoot returns proposer boost root
|
|
|
|
func (f *ForkChoiceStore) ProposerBoostRoot() libcommon.Hash {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
return f.proposerBoostRoot
|
|
|
|
}
|
|
|
|
|
|
|
|
// JustifiedCheckpoint returns justified checkpoint
|
|
|
|
func (f *ForkChoiceStore) JustifiedCheckpoint() *cltypes.Checkpoint {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
return f.justifiedCheckpoint
|
|
|
|
}
|
|
|
|
|
|
|
|
// FinalizedCheckpoint returns justified checkpoint
|
|
|
|
func (f *ForkChoiceStore) FinalizedCheckpoint() *cltypes.Checkpoint {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
return f.finalizedCheckpoint
|
|
|
|
}
|
2023-04-17 18:06:50 +00:00
|
|
|
|
|
|
|
// FinalizedCheckpoint returns justified checkpoint
|
|
|
|
func (f *ForkChoiceStore) Engine() execution_client.ExecutionEngine {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
return f.engine
|
|
|
|
}
|
|
|
|
|
|
|
|
// FinalizedCheckpoint returns justified checkpoint
|
|
|
|
func (f *ForkChoiceStore) GetEth1Hash(eth2Root libcommon.Hash) libcommon.Hash {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
ret, _ := f.eth2Roots.Get(eth2Root)
|
|
|
|
return ret
|
|
|
|
}
|
2023-04-26 13:33:21 +00:00
|
|
|
|
|
|
|
// FinalizedCheckpoint returns justified checkpoint
|
|
|
|
func (f *ForkChoiceStore) AnchorSlot() uint64 {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
|
|
|
return f.forkGraph.AnchorSlot()
|
|
|
|
}
|