2023-04-08 01:01:10 +00:00
|
|
|
package forkchoice
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"sort"
|
|
|
|
|
|
|
|
libcommon "github.com/ledgerwatch/erigon-lib/common"
|
|
|
|
"github.com/ledgerwatch/erigon/cl/cltypes"
|
|
|
|
)
|
|
|
|
|
|
|
|
// GetHead fetches the current head.
|
|
|
|
func (f *ForkChoiceStore) GetHead() (libcommon.Hash, uint64, error) {
|
|
|
|
f.mu.Lock()
|
|
|
|
defer f.mu.Unlock()
|
2023-05-02 14:19:22 +00:00
|
|
|
return f.getHead()
|
|
|
|
}
|
|
|
|
|
2024-01-06 20:49:23 +00:00
|
|
|
// accountWeights updates the weights of the validators, given the vote and given an head leaf.
|
|
|
|
func (f *ForkChoiceStore) accountWeights(votes, weights map[libcommon.Hash]uint64, justifedRoot, leaf libcommon.Hash) {
|
|
|
|
curr := leaf
|
|
|
|
accumulated := uint64(0)
|
|
|
|
for curr != justifedRoot {
|
|
|
|
accumulated += votes[curr]
|
|
|
|
votes[curr] = 0 // make sure we don't double count
|
|
|
|
weights[curr] += accumulated
|
|
|
|
header, has := f.forkGraph.GetHeader(curr)
|
|
|
|
if !has {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
curr = header.ParentRoot
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-05-02 14:19:22 +00:00
|
|
|
func (f *ForkChoiceStore) getHead() (libcommon.Hash, uint64, error) {
|
2023-09-29 21:42:07 +00:00
|
|
|
if f.headHash != (libcommon.Hash{}) {
|
|
|
|
return f.headHash, f.headSlot, nil
|
|
|
|
}
|
2023-04-08 01:01:10 +00:00
|
|
|
// Retrieve att
|
2023-09-29 21:42:07 +00:00
|
|
|
f.headHash = f.justifiedCheckpoint.BlockRoot()
|
|
|
|
blocks := f.getFilteredBlockTree(f.headHash)
|
2023-04-08 01:01:10 +00:00
|
|
|
// See which validators can be used for attestation score
|
2023-05-14 22:12:24 +00:00
|
|
|
justificationState, err := f.getCheckpointState(f.justifiedCheckpoint)
|
2023-04-08 01:01:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return libcommon.Hash{}, 0, err
|
|
|
|
}
|
2024-01-06 20:49:23 +00:00
|
|
|
// Do a simple scan to determine the fork votes.
|
|
|
|
votes := make(map[libcommon.Hash]uint64)
|
|
|
|
for validatorIndex, message := range f.latestMessages {
|
|
|
|
if message == (LatestMessage{}) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if !readFromBitset(justificationState.actives, validatorIndex) || readFromBitset(justificationState.slasheds, validatorIndex) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if _, hasLatestMessage := f.getLatestMessage(uint64(validatorIndex)); !hasLatestMessage {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if f.isUnequivocating(uint64(validatorIndex)) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
votes[message.Root] += justificationState.balances[validatorIndex]
|
|
|
|
}
|
|
|
|
if f.proposerBoostRoot != (libcommon.Hash{}) {
|
|
|
|
boost := justificationState.activeBalance / justificationState.beaconConfig.SlotsPerEpoch
|
|
|
|
votes[f.proposerBoostRoot] += (boost * justificationState.beaconConfig.ProposerScoreBoost) / 100
|
|
|
|
}
|
|
|
|
// Account for weights on each head fork
|
|
|
|
f.weights = make(map[libcommon.Hash]uint64)
|
|
|
|
for head := range f.headSet {
|
|
|
|
f.accountWeights(votes, f.weights, f.justifiedCheckpoint.BlockRoot(), head)
|
|
|
|
}
|
|
|
|
|
2023-04-08 01:01:10 +00:00
|
|
|
for {
|
|
|
|
// Filter out current head children.
|
2023-10-21 21:10:58 +00:00
|
|
|
unfilteredChildren := f.children(f.headHash)
|
2023-04-08 01:01:10 +00:00
|
|
|
children := []libcommon.Hash{}
|
|
|
|
for _, child := range unfilteredChildren {
|
|
|
|
if _, ok := blocks[child]; ok {
|
|
|
|
children = append(children, child)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
// Stop if we dont have any more children
|
|
|
|
if len(children) == 0 {
|
2023-09-29 21:42:07 +00:00
|
|
|
header, hasHeader := f.forkGraph.GetHeader(f.headHash)
|
2023-04-08 01:01:10 +00:00
|
|
|
if !hasHeader {
|
|
|
|
return libcommon.Hash{}, 0, fmt.Errorf("no slot for head is stored")
|
|
|
|
}
|
2023-09-29 21:42:07 +00:00
|
|
|
f.headSlot = header.Slot
|
|
|
|
return f.headHash, f.headSlot, nil
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
// Average case scenario.
|
|
|
|
if len(children) == 1 {
|
2023-09-29 21:42:07 +00:00
|
|
|
f.headHash = children[0]
|
2023-04-08 01:01:10 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
// Sort children by lexigographical order
|
|
|
|
sort.Slice(children, func(i, j int) bool {
|
|
|
|
childA := children[i]
|
|
|
|
childB := children[j]
|
|
|
|
return bytes.Compare(childA[:], childB[:]) < 0
|
|
|
|
})
|
|
|
|
|
|
|
|
// After sorting is done determine best fit.
|
2023-09-29 21:42:07 +00:00
|
|
|
f.headHash = children[0]
|
2024-01-06 20:49:23 +00:00
|
|
|
maxWeight := f.weights[children[0]]
|
2023-04-08 01:01:10 +00:00
|
|
|
for i := 1; i < len(children); i++ {
|
2024-01-06 20:49:23 +00:00
|
|
|
weight := f.weights[children[i]]
|
2023-04-08 01:01:10 +00:00
|
|
|
// Lexicographical order is king.
|
|
|
|
if weight >= maxWeight {
|
2023-09-29 21:42:07 +00:00
|
|
|
f.headHash = children[i]
|
2023-04-08 01:01:10 +00:00
|
|
|
maxWeight = weight
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// filterValidatorSetForAttestationScores preliminarly filter the validator set obliging to consensus rules.
|
2023-10-15 16:05:13 +00:00
|
|
|
func (f *ForkChoiceStore) filterValidatorSetForAttestationScores(c *checkpointState, epoch uint64) []uint64 {
|
|
|
|
filtered := make([]uint64, 0, c.validatorSetSize)
|
|
|
|
for validatorIndex := 0; validatorIndex < c.validatorSetSize; validatorIndex++ {
|
|
|
|
if !readFromBitset(c.actives, validatorIndex) || readFromBitset(c.slasheds, validatorIndex) {
|
2023-04-08 01:01:10 +00:00
|
|
|
continue
|
|
|
|
}
|
2024-01-06 20:49:23 +00:00
|
|
|
if _, hasLatestMessage := f.getLatestMessage(uint64(validatorIndex)); !hasLatestMessage {
|
2023-04-08 01:01:10 +00:00
|
|
|
continue
|
|
|
|
}
|
2024-01-06 20:49:23 +00:00
|
|
|
if f.isUnequivocating(uint64(validatorIndex)) {
|
2023-04-08 01:01:10 +00:00
|
|
|
continue
|
|
|
|
}
|
|
|
|
filtered = append(filtered, uint64(validatorIndex))
|
|
|
|
}
|
|
|
|
return filtered
|
|
|
|
}
|
|
|
|
|
|
|
|
// getWeight computes weight in head decision of canonical chain.
|
2023-05-03 08:51:39 +00:00
|
|
|
func (f *ForkChoiceStore) getWeight(root libcommon.Hash, indicies []uint64, state *checkpointState) uint64 {
|
2023-04-08 01:01:10 +00:00
|
|
|
header, has := f.forkGraph.GetHeader(root)
|
|
|
|
if !has {
|
|
|
|
return 0
|
|
|
|
}
|
|
|
|
// Compute attestation score
|
|
|
|
var attestationScore uint64
|
|
|
|
for _, validatorIndex := range indicies {
|
|
|
|
if f.Ancestor(f.latestMessages[validatorIndex].Root, header.Slot) != root {
|
|
|
|
continue
|
|
|
|
}
|
2023-10-15 16:05:13 +00:00
|
|
|
attestationScore += state.balances[validatorIndex]
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
if f.proposerBoostRoot == (libcommon.Hash{}) {
|
|
|
|
return attestationScore
|
|
|
|
}
|
|
|
|
|
|
|
|
// Boost is applied if root is an ancestor of proposer_boost_root
|
|
|
|
if f.Ancestor(f.proposerBoostRoot, header.Slot) == root {
|
2023-05-03 08:51:39 +00:00
|
|
|
committeeWeight := state.activeBalance / state.beaconConfig.SlotsPerEpoch
|
|
|
|
attestationScore += (committeeWeight * state.beaconConfig.ProposerScoreBoost) / 100
|
2023-04-08 01:01:10 +00:00
|
|
|
}
|
|
|
|
return attestationScore
|
|
|
|
}
|
|
|
|
|
|
|
|
// getFilteredBlockTree filters out dumb blocks.
|
|
|
|
func (f *ForkChoiceStore) getFilteredBlockTree(base libcommon.Hash) map[libcommon.Hash]*cltypes.BeaconBlockHeader {
|
|
|
|
blocks := make(map[libcommon.Hash]*cltypes.BeaconBlockHeader)
|
|
|
|
f.getFilterBlockTree(base, blocks)
|
|
|
|
return blocks
|
|
|
|
}
|
|
|
|
|
|
|
|
// getFilterBlockTree recursively traverses the block tree to identify viable blocks.
|
|
|
|
// It takes a block hash and a map of viable blocks as input parameters, and returns a boolean value indicating
|
|
|
|
// whether the current block is viable.
|
|
|
|
func (f *ForkChoiceStore) getFilterBlockTree(blockRoot libcommon.Hash, blocks map[libcommon.Hash]*cltypes.BeaconBlockHeader) bool {
|
|
|
|
header, has := f.forkGraph.GetHeader(blockRoot)
|
|
|
|
if !has {
|
|
|
|
return false
|
|
|
|
}
|
2023-10-21 21:10:58 +00:00
|
|
|
children := f.children(blockRoot)
|
2023-04-08 01:01:10 +00:00
|
|
|
// If there are children iterate down recursively and see which branches are viable.
|
|
|
|
if len(children) > 0 {
|
|
|
|
isAnyViable := false
|
|
|
|
for _, child := range children {
|
|
|
|
if f.getFilterBlockTree(child, blocks) {
|
|
|
|
isAnyViable = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if isAnyViable {
|
|
|
|
blocks[blockRoot] = header
|
|
|
|
}
|
|
|
|
return isAnyViable
|
|
|
|
}
|
|
|
|
currentJustifiedCheckpoint, has := f.forkGraph.GetCurrentJustifiedCheckpoint(blockRoot)
|
|
|
|
if !has {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
finalizedJustifiedCheckpoint, has := f.forkGraph.GetFinalizedCheckpoint(blockRoot)
|
|
|
|
if !has {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2023-10-21 21:10:58 +00:00
|
|
|
genesisEpoch := f.beaconCfg.GenesisEpoch
|
2023-05-14 22:12:24 +00:00
|
|
|
if (f.justifiedCheckpoint.Epoch() == genesisEpoch || currentJustifiedCheckpoint.Equal(f.justifiedCheckpoint)) &&
|
|
|
|
(f.finalizedCheckpoint.Epoch() == genesisEpoch || finalizedJustifiedCheckpoint.Equal(f.finalizedCheckpoint)) {
|
2023-04-08 01:01:10 +00:00
|
|
|
blocks[blockRoot] = header
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|