mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-17 15:28:45 +00:00
6a2ef13b87
* add metrics for synced tips * Update beacon-chain/forkchoice/protoarray/metrics.go Co-authored-by: terence tsao <terence@prysmaticlabs.com> Co-authored-by: prylabs-bulldozer[bot] <58059840+prylabs-bulldozer[bot]@users.noreply.github.com>
323 lines
8.6 KiB
Go
323 lines
8.6 KiB
Go
package protoarray
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
|
|
types "github.com/prysmaticlabs/eth2-types"
|
|
"github.com/prysmaticlabs/prysm/config/params"
|
|
"github.com/prysmaticlabs/prysm/encoding/bytesutil"
|
|
)
|
|
|
|
// This returns the minimum and maximum slot of the synced_tips tree
|
|
func (f *ForkChoice) boundarySyncedTips() (types.Slot, types.Slot) {
|
|
f.syncedTips.RLock()
|
|
defer f.syncedTips.RUnlock()
|
|
|
|
min := params.BeaconConfig().FarFutureSlot
|
|
max := types.Slot(0)
|
|
for _, slot := range f.syncedTips.validatedTips {
|
|
if slot > max {
|
|
max = slot
|
|
}
|
|
if slot < min {
|
|
min = slot
|
|
}
|
|
}
|
|
return min, max
|
|
}
|
|
|
|
// Optimistic returns true if this node is optimistically synced
|
|
// A optimistically synced block is synced as usual, but its
|
|
// execution payload is not validated, while the EL is still syncing.
|
|
// WARNING: this function does not check if slot corresponds to the
|
|
// block with the given root. An incorrect response may be
|
|
// returned when requesting earlier than finalized epoch due
|
|
// to pruning of non-canonical branches. A requests for a
|
|
// combination root/slot of an available block is guaranteed
|
|
// to yield the correct result. The caller is responsible for
|
|
// checking the block's availability. A consensus bug could be
|
|
// a cause of getting this wrong, so think twice before passing
|
|
// a wrong pair.
|
|
func (f *ForkChoice) Optimistic(ctx context.Context, root [32]byte, slot types.Slot) (bool, error) {
|
|
if ctx.Err() != nil {
|
|
return false, ctx.Err()
|
|
}
|
|
// If the node is a synced tip, then it's fully validated
|
|
f.syncedTips.RLock()
|
|
_, ok := f.syncedTips.validatedTips[root]
|
|
if ok {
|
|
return false, nil
|
|
}
|
|
f.syncedTips.RUnlock()
|
|
|
|
// If the slot is higher than the max synced tip, it's optimistic
|
|
min, max := f.boundarySyncedTips()
|
|
if slot > max {
|
|
return true, nil
|
|
}
|
|
|
|
// If the slot is lower than the min synced tip, it's fully validated
|
|
if slot <= min {
|
|
return false, nil
|
|
}
|
|
|
|
// If we reached this point then the block has to be in the Fork Choice
|
|
// Store!
|
|
f.store.nodesLock.RLock()
|
|
index, ok := f.store.nodesIndices[root]
|
|
if !ok {
|
|
// This should not happen
|
|
f.store.nodesLock.RUnlock()
|
|
return false, fmt.Errorf("invalid root, slot combination, got %#x, %d",
|
|
bytesutil.Trunc(root[:]), slot)
|
|
}
|
|
node := f.store.nodes[index]
|
|
|
|
// if the node is a leaf of the Fork Choice tree, then it's
|
|
// optimistic
|
|
childIndex := node.BestChild()
|
|
if childIndex == NonExistentNode {
|
|
return true, nil
|
|
}
|
|
|
|
// recurse to the child
|
|
child := f.store.nodes[childIndex]
|
|
root = child.root
|
|
slot = child.slot
|
|
f.store.nodesLock.RUnlock()
|
|
return f.Optimistic(ctx, root, slot)
|
|
}
|
|
|
|
// This function returns the index of sync tip node that's ancestor to the input node.
|
|
// In the event of none, `NonExistentNode` is returned.
|
|
// This internal method assumes the caller holds a lock on syncedTips and s.nodesLock
|
|
func (s *Store) findSyncedTip(ctx context.Context, node *Node, syncedTips *optimisticStore) (uint64, error) {
|
|
for {
|
|
if ctx.Err() != nil {
|
|
return 0, ctx.Err()
|
|
}
|
|
if _, ok := syncedTips.validatedTips[node.root]; ok {
|
|
return s.nodesIndices[node.root], nil
|
|
}
|
|
if node.parent == NonExistentNode {
|
|
return NonExistentNode, nil
|
|
}
|
|
node = s.nodes[node.parent]
|
|
}
|
|
}
|
|
|
|
// UpdateSyncedTipsWithValidRoot is called with the root of a block that was returned as
|
|
// VALID by the EL. This routine recomputes and updates the synced_tips map to
|
|
// account for this new tip.
|
|
func (f *ForkChoice) UpdateSyncedTipsWithValidRoot(ctx context.Context, root [32]byte) error {
|
|
f.store.nodesLock.RLock()
|
|
defer f.store.nodesLock.RUnlock()
|
|
// We can only update if given root is in Fork Choice
|
|
index, ok := f.store.nodesIndices[root]
|
|
if !ok {
|
|
return errInvalidNodeIndex
|
|
}
|
|
|
|
// We can only update if root is a leaf in Fork Choice
|
|
node := f.store.nodes[index]
|
|
if node.bestChild != NonExistentNode {
|
|
return errInvalidBestChildIndex
|
|
}
|
|
|
|
// Stop early if the root is part of validated tips
|
|
f.syncedTips.Lock()
|
|
defer f.syncedTips.Unlock()
|
|
_, ok = f.syncedTips.validatedTips[root]
|
|
if ok {
|
|
return nil
|
|
}
|
|
|
|
// Cache root and slot to validated tips
|
|
newTips := make(map[[32]byte]types.Slot)
|
|
newValidSlot := node.slot
|
|
newTips[root] = newValidSlot
|
|
|
|
// Compute the full valid path from the given node to its previous synced tip
|
|
// This path will now consist of fully validated blocks. Notice that
|
|
// the previous tip may have been outside the Fork Choice store.
|
|
// In this case, only one block can be in syncedTips as the whole
|
|
// Fork Choice would be a descendant of this block.
|
|
validPath := make(map[uint64]bool)
|
|
validPath[index] = true
|
|
for {
|
|
if ctx.Err() != nil {
|
|
return ctx.Err()
|
|
}
|
|
|
|
parentIndex := node.parent
|
|
if parentIndex == NonExistentNode {
|
|
break
|
|
}
|
|
if parentIndex >= uint64(len(f.store.nodes)) {
|
|
return errInvalidNodeIndex
|
|
}
|
|
node = f.store.nodes[parentIndex]
|
|
_, ok = f.syncedTips.validatedTips[node.root]
|
|
if ok {
|
|
break
|
|
}
|
|
validPath[parentIndex] = true
|
|
}
|
|
|
|
// Retrieve the list of leaves in the Fork Choice
|
|
// These are all the nodes that have NonExistentNode as best child.
|
|
leaves, err := f.store.leaves()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// For each leaf, recompute the new tip.
|
|
for _, i := range leaves {
|
|
node = f.store.nodes[i]
|
|
j := i
|
|
for {
|
|
if ctx.Err() != nil {
|
|
return ctx.Err()
|
|
}
|
|
// Stop if we reached the previous tip
|
|
_, ok = f.syncedTips.validatedTips[node.root]
|
|
if ok {
|
|
newTips[node.root] = node.slot
|
|
break
|
|
}
|
|
|
|
// Stop if we reach valid path
|
|
_, ok = validPath[j]
|
|
if ok {
|
|
newTips[node.root] = node.slot
|
|
break
|
|
}
|
|
|
|
j = node.parent
|
|
if j == NonExistentNode {
|
|
break
|
|
}
|
|
if j >= uint64(len(f.store.nodes)) {
|
|
return errInvalidNodeIndex
|
|
}
|
|
node = f.store.nodes[j]
|
|
}
|
|
}
|
|
|
|
f.syncedTips.validatedTips = newTips
|
|
lastSyncedTipSlot.Set(float64(newValidSlot))
|
|
syncedTipsCount.Set(float64(len(newTips)))
|
|
return nil
|
|
}
|
|
|
|
// UpdateSyncedTipsWithInvalidRoot updates the synced_tips map when the block with the given root becomes INVALID.
|
|
func (f *ForkChoice) UpdateSyncedTipsWithInvalidRoot(ctx context.Context, root [32]byte) error {
|
|
f.store.nodesLock.Lock()
|
|
defer f.store.nodesLock.Unlock()
|
|
idx, ok := f.store.nodesIndices[root]
|
|
if !ok {
|
|
return errInvalidNodeIndex
|
|
}
|
|
node := f.store.nodes[idx]
|
|
// We only support changing status for the tips in Fork Choice store.
|
|
if node.bestChild != NonExistentNode {
|
|
return errInvalidNodeIndex
|
|
}
|
|
|
|
parentIndex := node.parent
|
|
// This should not happen
|
|
if parentIndex == NonExistentNode {
|
|
return errInvalidNodeIndex
|
|
}
|
|
// Update the weights of the nodes subtracting the INVALID node's weight
|
|
weight := node.weight
|
|
node = f.store.nodes[parentIndex]
|
|
for {
|
|
if ctx.Err() != nil {
|
|
return ctx.Err()
|
|
}
|
|
node.weight -= weight
|
|
if node.parent == NonExistentNode {
|
|
break
|
|
}
|
|
node = f.store.nodes[node.parent]
|
|
}
|
|
parent := copyNode(f.store.nodes[parentIndex])
|
|
|
|
// delete the invalid node, order is important
|
|
f.store.nodes = append(f.store.nodes[:idx], f.store.nodes[idx+1:]...)
|
|
delete(f.store.nodesIndices, root)
|
|
// Fix parent and best child for each node
|
|
for _, node := range f.store.nodes {
|
|
if node.parent == NonExistentNode {
|
|
node.parent = NonExistentNode
|
|
} else if node.parent > idx {
|
|
node.parent -= 1
|
|
}
|
|
if node.bestChild == NonExistentNode || node.bestChild == idx {
|
|
node.bestChild = NonExistentNode
|
|
} else if node.bestChild > idx {
|
|
node.bestChild -= 1
|
|
}
|
|
if node.bestDescendant == NonExistentNode || node.bestDescendant == idx {
|
|
node.bestDescendant = NonExistentNode
|
|
} else if node.bestDescendant > idx {
|
|
node.bestDescendant -= 1
|
|
}
|
|
}
|
|
|
|
// Update the parent's best child and best descendant if necessary.
|
|
if parent.bestChild == idx || parent.bestDescendant == idx {
|
|
for childIndex, child := range f.store.nodes {
|
|
if child.parent == parentIndex {
|
|
err := f.store.updateBestChildAndDescendant(
|
|
parentIndex, uint64(childIndex))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
// Return early if the parent is not a synced_tip.
|
|
f.syncedTips.Lock()
|
|
defer f.syncedTips.Unlock()
|
|
parentRoot := parent.root
|
|
_, ok = f.syncedTips.validatedTips[parentRoot]
|
|
if !ok {
|
|
return nil
|
|
}
|
|
|
|
leaves, err := f.store.leaves()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
for _, i := range leaves {
|
|
node = f.store.nodes[i]
|
|
for {
|
|
if ctx.Err() != nil {
|
|
return ctx.Err()
|
|
}
|
|
|
|
// Return early if the parent is still a synced tip
|
|
if node.root == parentRoot {
|
|
return nil
|
|
}
|
|
_, ok = f.syncedTips.validatedTips[node.root]
|
|
if ok {
|
|
break
|
|
}
|
|
if node.parent == NonExistentNode {
|
|
break
|
|
}
|
|
node = f.store.nodes[node.parent]
|
|
}
|
|
}
|
|
delete(f.syncedTips.validatedTips, parentRoot)
|
|
syncedTipsCount.Set(float64(len(f.syncedTips.validatedTips)))
|
|
return nil
|
|
}
|