2022-05-21 13:27:07 +00:00
|
|
|
package doublylinkedtree
|
|
|
|
|
2022-06-05 17:48:21 +00:00
|
|
|
import (
|
|
|
|
"github.com/pkg/errors"
|
2022-08-16 12:20:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/core/epoch/precompute"
|
|
|
|
forkchoicetypes "github.com/prysmaticlabs/prysm/v3/beacon-chain/forkchoice/types"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/state"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/config/params"
|
|
|
|
types "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/encoding/bytesutil"
|
|
|
|
ethpb "github.com/prysmaticlabs/prysm/v3/proto/prysm/v1alpha1"
|
|
|
|
"github.com/prysmaticlabs/prysm/v3/time/slots"
|
2022-06-05 17:48:21 +00:00
|
|
|
)
|
2022-05-21 13:27:07 +00:00
|
|
|
|
|
|
|
func (s *Store) setUnrealizedJustifiedEpoch(root [32]byte, epoch types.Epoch) error {
|
|
|
|
s.nodesLock.Lock()
|
|
|
|
defer s.nodesLock.Unlock()
|
|
|
|
|
|
|
|
node, ok := s.nodeByRoot[root]
|
|
|
|
if !ok || node == nil {
|
2022-06-05 17:48:21 +00:00
|
|
|
return errors.Wrap(ErrNilNode, "could not set unrealized justified epoch")
|
2022-05-21 13:27:07 +00:00
|
|
|
}
|
|
|
|
if epoch < node.unrealizedJustifiedEpoch {
|
|
|
|
return errInvalidUnrealizedJustifiedEpoch
|
|
|
|
}
|
|
|
|
node.unrealizedJustifiedEpoch = epoch
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Store) setUnrealizedFinalizedEpoch(root [32]byte, epoch types.Epoch) error {
|
|
|
|
s.nodesLock.Lock()
|
|
|
|
defer s.nodesLock.Unlock()
|
|
|
|
|
|
|
|
node, ok := s.nodeByRoot[root]
|
|
|
|
if !ok || node == nil {
|
2022-06-05 17:48:21 +00:00
|
|
|
return errors.Wrap(ErrNilNode, "could not set unrealized finalized epoch")
|
2022-05-21 13:27:07 +00:00
|
|
|
}
|
|
|
|
if epoch < node.unrealizedFinalizedEpoch {
|
|
|
|
return errInvalidUnrealizedFinalizedEpoch
|
|
|
|
}
|
|
|
|
node.unrealizedFinalizedEpoch = epoch
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-07-06 16:55:17 +00:00
|
|
|
// updateUnrealizedCheckpoints "realizes" the unrealized justified and finalized
|
|
|
|
// epochs stored within nodes. It should be called at the beginning of each epoch.
|
|
|
|
func (f *ForkChoice) updateUnrealizedCheckpoints() {
|
2022-05-21 13:27:07 +00:00
|
|
|
f.store.nodesLock.Lock()
|
|
|
|
defer f.store.nodesLock.Unlock()
|
2022-08-13 12:30:08 +00:00
|
|
|
f.store.checkpointsLock.Lock()
|
|
|
|
defer f.store.checkpointsLock.Unlock()
|
2022-05-21 13:27:07 +00:00
|
|
|
for _, node := range f.store.nodeByRoot {
|
|
|
|
node.justifiedEpoch = node.unrealizedJustifiedEpoch
|
|
|
|
node.finalizedEpoch = node.unrealizedFinalizedEpoch
|
2022-06-09 22:28:30 +00:00
|
|
|
if node.justifiedEpoch > f.store.justifiedCheckpoint.Epoch {
|
2022-08-04 13:26:41 +00:00
|
|
|
f.store.prevJustifiedCheckpoint = f.store.justifiedCheckpoint
|
2022-07-06 16:55:17 +00:00
|
|
|
f.store.justifiedCheckpoint = f.store.unrealizedJustifiedCheckpoint
|
2022-06-29 23:37:21 +00:00
|
|
|
if node.justifiedEpoch > f.store.bestJustifiedCheckpoint.Epoch {
|
|
|
|
f.store.bestJustifiedCheckpoint = f.store.unrealizedJustifiedCheckpoint
|
|
|
|
}
|
2022-05-21 13:27:07 +00:00
|
|
|
}
|
2022-06-09 22:28:30 +00:00
|
|
|
if node.finalizedEpoch > f.store.finalizedCheckpoint.Epoch {
|
2022-06-29 23:37:21 +00:00
|
|
|
f.store.justifiedCheckpoint = f.store.unrealizedJustifiedCheckpoint
|
|
|
|
f.store.finalizedCheckpoint = f.store.unrealizedFinalizedCheckpoint
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-06 16:55:17 +00:00
|
|
|
func (s *Store) pullTips(state state.BeaconState, node *Node, jc, fc *ethpb.Checkpoint) (*ethpb.Checkpoint, *ethpb.Checkpoint) {
|
|
|
|
s.nodesLock.Lock()
|
|
|
|
defer s.nodesLock.Unlock()
|
|
|
|
|
|
|
|
if node.parent == nil { // Nothing to do if the parent is nil.
|
|
|
|
return jc, fc
|
|
|
|
}
|
|
|
|
|
2022-07-03 20:27:39 +00:00
|
|
|
s.checkpointsLock.Lock()
|
|
|
|
defer s.checkpointsLock.Unlock()
|
|
|
|
|
2022-07-06 16:55:17 +00:00
|
|
|
currentEpoch := slots.ToEpoch(slots.CurrentSlot(s.genesisTime))
|
2022-07-03 20:27:39 +00:00
|
|
|
stateSlot := state.Slot()
|
|
|
|
stateEpoch := slots.ToEpoch(stateSlot)
|
|
|
|
currJustified := node.parent.unrealizedJustifiedEpoch == currentEpoch
|
|
|
|
prevJustified := node.parent.unrealizedJustifiedEpoch+1 == currentEpoch
|
|
|
|
tooEarlyForCurr := slots.SinceEpochStarts(stateSlot)*3 < params.BeaconConfig().SlotsPerEpoch*2
|
2022-07-06 16:55:17 +00:00
|
|
|
// Exit early if it's justified or too early to be justified.
|
2022-07-03 20:27:39 +00:00
|
|
|
if currJustified || (stateEpoch == currentEpoch && prevJustified && tooEarlyForCurr) {
|
|
|
|
node.unrealizedJustifiedEpoch = node.parent.unrealizedJustifiedEpoch
|
|
|
|
node.unrealizedFinalizedEpoch = node.parent.unrealizedFinalizedEpoch
|
|
|
|
return jc, fc
|
|
|
|
}
|
|
|
|
|
2022-09-11 19:04:40 +00:00
|
|
|
ab, uj, uf, err := precompute.UnrealizedCheckpoints(state)
|
2022-06-29 23:37:21 +00:00
|
|
|
if err != nil {
|
|
|
|
log.WithError(err).Debug("could not compute unrealized checkpoints")
|
|
|
|
uj, uf = jc, fc
|
|
|
|
}
|
2022-07-06 16:55:17 +00:00
|
|
|
|
2022-09-11 19:04:40 +00:00
|
|
|
s.committeeBalance = ab / uint64(params.BeaconConfig().SlotsPerEpoch)
|
|
|
|
|
2022-07-06 16:55:17 +00:00
|
|
|
// Update store's unrealized checkpoints.
|
2022-06-29 23:37:21 +00:00
|
|
|
if uj.Epoch > s.unrealizedJustifiedCheckpoint.Epoch {
|
|
|
|
s.unrealizedJustifiedCheckpoint = &forkchoicetypes.Checkpoint{
|
|
|
|
Epoch: uj.Epoch, Root: bytesutil.ToBytes32(uj.Root),
|
2022-05-21 13:27:07 +00:00
|
|
|
}
|
|
|
|
}
|
2022-06-29 23:37:21 +00:00
|
|
|
if uf.Epoch > s.unrealizedFinalizedCheckpoint.Epoch {
|
|
|
|
s.unrealizedJustifiedCheckpoint = &forkchoicetypes.Checkpoint{
|
|
|
|
Epoch: uj.Epoch, Root: bytesutil.ToBytes32(uj.Root),
|
|
|
|
}
|
|
|
|
s.unrealizedFinalizedCheckpoint = &forkchoicetypes.Checkpoint{
|
|
|
|
Epoch: uf.Epoch, Root: bytesutil.ToBytes32(uf.Root),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-06 16:55:17 +00:00
|
|
|
// Update node's checkpoints.
|
|
|
|
node.unrealizedJustifiedEpoch, node.unrealizedFinalizedEpoch = uj.Epoch, uf.Epoch
|
2022-07-03 20:27:39 +00:00
|
|
|
if stateEpoch < currentEpoch {
|
2022-06-29 23:37:21 +00:00
|
|
|
jc, fc = uj, uf
|
|
|
|
node.justifiedEpoch = uj.Epoch
|
|
|
|
node.finalizedEpoch = uf.Epoch
|
|
|
|
}
|
|
|
|
return jc, fc
|
2022-05-21 13:27:07 +00:00
|
|
|
}
|