2022-01-26 01:42:18 +00:00
|
|
|
package protoarray
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
|
|
|
|
"github.com/prysmaticlabs/prysm/config/params"
|
|
|
|
)
|
|
|
|
|
2022-03-09 03:05:51 +00:00
|
|
|
// IsOptimistic returns true if this node is optimistically synced
|
2022-01-26 01:42:18 +00:00
|
|
|
// A optimistically synced block is synced as usual, but its
|
|
|
|
// execution payload is not validated, while the EL is still syncing.
|
2022-03-09 03:05:51 +00:00
|
|
|
// This function returns an error if the block is not found in the fork choice
|
|
|
|
// store
|
2022-04-06 14:18:30 +00:00
|
|
|
func (f *ForkChoice) IsOptimistic(root [32]byte) (bool, error) {
|
2022-03-09 03:05:51 +00:00
|
|
|
f.store.nodesLock.RLock()
|
2022-04-06 14:18:30 +00:00
|
|
|
defer f.store.nodesLock.RUnlock()
|
2022-03-09 03:05:51 +00:00
|
|
|
index, ok := f.store.nodesIndices[root]
|
|
|
|
if !ok {
|
2022-03-15 20:52:59 +00:00
|
|
|
return false, ErrUnknownNodeRoot
|
2022-03-09 03:05:51 +00:00
|
|
|
}
|
|
|
|
node := f.store.nodes[index]
|
2022-04-06 14:18:30 +00:00
|
|
|
return node.status == syncing, nil
|
2022-02-05 20:24:02 +00:00
|
|
|
}
|
|
|
|
|
2022-03-09 03:05:51 +00:00
|
|
|
// SetOptimisticToValid is called with the root of a block that was returned as
|
2022-04-06 14:18:30 +00:00
|
|
|
// VALID by the EL.
|
2022-05-12 15:25:44 +00:00
|
|
|
//
|
2022-04-06 14:18:30 +00:00
|
|
|
// WARNING: This method returns an error if the root is not found in forkchoice
|
2022-03-09 03:05:51 +00:00
|
|
|
func (f *ForkChoice) SetOptimisticToValid(ctx context.Context, root [32]byte) error {
|
2022-04-06 14:18:30 +00:00
|
|
|
f.store.nodesLock.Lock()
|
|
|
|
defer f.store.nodesLock.Unlock()
|
2022-02-03 22:57:15 +00:00
|
|
|
// We can only update if given root is in Fork Choice
|
2022-01-31 15:05:05 +00:00
|
|
|
index, ok := f.store.nodesIndices[root]
|
|
|
|
if !ok {
|
2022-04-06 14:18:30 +00:00
|
|
|
return ErrUnknownNodeRoot
|
2022-01-31 15:05:05 +00:00
|
|
|
}
|
|
|
|
|
2022-04-06 14:18:30 +00:00
|
|
|
for node := f.store.nodes[index]; node.status == syncing; node = f.store.nodes[index] {
|
2022-01-31 15:05:05 +00:00
|
|
|
if ctx.Err() != nil {
|
|
|
|
return ctx.Err()
|
|
|
|
}
|
2022-04-06 14:18:30 +00:00
|
|
|
node.status = valid
|
|
|
|
index = node.parent
|
|
|
|
if index == NonExistentNode {
|
2022-01-31 15:05:05 +00:00
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
2022-02-03 22:57:15 +00:00
|
|
|
|
2022-04-06 14:18:30 +00:00
|
|
|
// SetOptimisticToInvalid updates the synced_tips map when the block with the given root becomes INVALID.
|
2022-05-02 13:53:22 +00:00
|
|
|
// It takes three parameters: the root of the INVALID block, its parent root and the payload Hash
|
|
|
|
// of the last valid block
|
|
|
|
func (f *ForkChoice) SetOptimisticToInvalid(ctx context.Context, root, parentRoot, payloadHash [32]byte) ([][32]byte, error) {
|
2022-02-03 22:57:15 +00:00
|
|
|
f.store.nodesLock.Lock()
|
|
|
|
defer f.store.nodesLock.Unlock()
|
2022-03-23 18:55:05 +00:00
|
|
|
invalidRoots := make([][32]byte, 0)
|
2022-04-06 14:18:30 +00:00
|
|
|
lastValidIndex, ok := f.store.payloadIndices[payloadHash]
|
|
|
|
if !ok || lastValidIndex == NonExistentNode {
|
|
|
|
return invalidRoots, errInvalidFinalizedNode
|
2022-02-03 22:57:15 +00:00
|
|
|
}
|
|
|
|
|
2022-05-02 13:53:22 +00:00
|
|
|
invalidIndex, ok := f.store.nodesIndices[root]
|
|
|
|
if !ok {
|
|
|
|
invalidIndex, ok = f.store.nodesIndices[parentRoot]
|
|
|
|
if !ok {
|
|
|
|
return invalidRoots, ErrUnknownNodeRoot
|
|
|
|
}
|
|
|
|
// return early if parent is LVH
|
|
|
|
if invalidIndex == lastValidIndex {
|
|
|
|
return invalidRoots, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
node := f.store.nodes[invalidIndex]
|
|
|
|
|
2022-04-06 14:18:30 +00:00
|
|
|
// Check if last valid hash is an ancestor of the passed node
|
|
|
|
firstInvalidIndex := node.parent
|
|
|
|
for ; firstInvalidIndex != NonExistentNode && firstInvalidIndex != lastValidIndex; firstInvalidIndex = node.parent {
|
|
|
|
node = f.store.nodes[firstInvalidIndex]
|
2022-02-03 22:57:15 +00:00
|
|
|
}
|
|
|
|
|
2022-05-02 13:53:22 +00:00
|
|
|
// Deal with the case that the last valid payload is in a different fork
|
|
|
|
// This means we are dealing with an EE that does not follow the spec
|
2022-04-06 14:18:30 +00:00
|
|
|
if node.parent != lastValidIndex {
|
|
|
|
node = f.store.nodes[invalidIndex]
|
2022-05-02 13:53:22 +00:00
|
|
|
// return early if invalid node was not imported
|
|
|
|
if node.root == parentRoot {
|
|
|
|
return invalidRoots, nil
|
|
|
|
}
|
|
|
|
|
2022-04-06 14:18:30 +00:00
|
|
|
firstInvalidIndex = invalidIndex
|
|
|
|
lastValidIndex = node.parent
|
|
|
|
if lastValidIndex == NonExistentNode {
|
|
|
|
return invalidRoots, errInvalidFinalizedNode
|
2022-02-03 22:57:15 +00:00
|
|
|
}
|
2022-04-06 14:18:30 +00:00
|
|
|
} else {
|
|
|
|
firstInvalidIndex = f.store.nodesIndices[node.root]
|
2022-02-03 22:57:15 +00:00
|
|
|
}
|
|
|
|
|
2022-04-06 14:18:30 +00:00
|
|
|
// Update the weights of the nodes subtracting the first INVALID node's weight
|
|
|
|
weight := node.weight
|
|
|
|
var validNode *Node
|
|
|
|
for index := lastValidIndex; index != NonExistentNode; index = validNode.parent {
|
|
|
|
validNode = f.store.nodes[index]
|
|
|
|
validNode.weight -= weight
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find the current proposer boost (it should be set to zero if an
|
|
|
|
// INVALID block was boosted)
|
|
|
|
f.store.proposerBoostLock.RLock()
|
|
|
|
boostRoot := f.store.proposerBoostRoot
|
|
|
|
previousBoostRoot := f.store.previousProposerBoostRoot
|
|
|
|
f.store.proposerBoostLock.RUnlock()
|
|
|
|
|
|
|
|
// Remove the invalid roots from our store maps and adjust their weight
|
|
|
|
// to zero
|
|
|
|
boosted := node.root == boostRoot
|
|
|
|
previouslyBoosted := node.root == previousBoostRoot
|
|
|
|
|
|
|
|
invalidIndices := map[uint64]bool{firstInvalidIndex: true}
|
|
|
|
node.status = invalid
|
|
|
|
node.weight = 0
|
|
|
|
delete(f.store.nodesIndices, node.root)
|
|
|
|
delete(f.store.canonicalNodes, node.root)
|
|
|
|
delete(f.store.payloadIndices, node.payloadHash)
|
|
|
|
for index := firstInvalidIndex + 1; index < uint64(len(f.store.nodes)); index++ {
|
|
|
|
invalidNode := f.store.nodes[index]
|
|
|
|
if _, ok := invalidIndices[invalidNode.parent]; !ok {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
if invalidNode.status == valid {
|
|
|
|
return invalidRoots, errInvalidOptimisticStatus
|
|
|
|
}
|
|
|
|
if !boosted && invalidNode.root == boostRoot {
|
|
|
|
boosted = true
|
|
|
|
}
|
|
|
|
if !previouslyBoosted && invalidNode.root == previousBoostRoot {
|
|
|
|
previouslyBoosted = true
|
|
|
|
}
|
|
|
|
invalidNode.status = invalid
|
|
|
|
invalidIndices[index] = true
|
|
|
|
invalidNode.weight = 0
|
|
|
|
delete(f.store.nodesIndices, invalidNode.root)
|
|
|
|
delete(f.store.canonicalNodes, invalidNode.root)
|
|
|
|
delete(f.store.payloadIndices, invalidNode.payloadHash)
|
|
|
|
}
|
|
|
|
if boosted {
|
|
|
|
if err := f.ResetBoostedProposerRoot(ctx); err != nil {
|
|
|
|
return invalidRoots, err
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if previouslyBoosted {
|
|
|
|
f.store.proposerBoostLock.Lock()
|
|
|
|
f.store.previousProposerBoostRoot = params.BeaconConfig().ZeroHash
|
|
|
|
f.store.previousProposerBoostScore = 0
|
|
|
|
f.store.proposerBoostLock.Unlock()
|
|
|
|
}
|
|
|
|
|
|
|
|
for index := range invalidIndices {
|
|
|
|
invalidRoots = append(invalidRoots, f.store.nodes[index].root)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Update the best child and descendant
|
|
|
|
for i := len(f.store.nodes) - 1; i >= 0; i-- {
|
|
|
|
n := f.store.nodes[i]
|
|
|
|
if n.parent != NonExistentNode {
|
|
|
|
if err := f.store.updateBestChildAndDescendant(n.parent, uint64(i)); err != nil {
|
|
|
|
return invalidRoots, err
|
2022-02-03 22:57:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2022-03-23 18:55:05 +00:00
|
|
|
return invalidRoots, nil
|
2022-02-03 22:57:15 +00:00
|
|
|
}
|