mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-05 09:14:28 +00:00
174 lines
5.6 KiB
Go
174 lines
5.6 KiB
Go
package stategen
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"math"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/helpers"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v4/time/slots"
|
|
"github.com/sirupsen/logrus"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
// SaveState saves the state in the cache and/or DB.
|
|
func (s *State) SaveState(ctx context.Context, blockRoot [32]byte, st state.BeaconState) error {
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.SaveState")
|
|
defer span.End()
|
|
|
|
return s.saveStateByRoot(ctx, blockRoot, st)
|
|
}
|
|
|
|
// ForceCheckpoint initiates a cold state save of the given block root's state. This method does not update the
|
|
// "last archived state" but simply saves the specified state from the root argument into the DB.
|
|
//
|
|
// The name "Checkpoint" isn't referring to checkpoint in the sense of our consensus type, but checkpoint for our historical states.
|
|
func (s *State) ForceCheckpoint(ctx context.Context, blockRoot []byte) error {
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.ForceCheckpoint")
|
|
defer span.End()
|
|
|
|
root32 := bytesutil.ToBytes32(blockRoot)
|
|
// Before the first finalized checkpoint, the finalized root is zero hash.
|
|
// Return early if there hasn't been a finalized checkpoint.
|
|
if root32 == params.BeaconConfig().ZeroHash {
|
|
return nil
|
|
}
|
|
|
|
fs, err := s.loadStateByRoot(ctx, root32)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return s.beaconDB.SaveState(ctx, fs, root32)
|
|
}
|
|
|
|
// This saves a post beacon state. On the epoch boundary,
|
|
// it saves a full state. On an intermediate slot, it saves a back pointer to the
|
|
// nearest epoch boundary state.
|
|
func (s *State) saveStateByRoot(ctx context.Context, blockRoot [32]byte, st state.BeaconState) error {
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.saveStateByRoot")
|
|
defer span.End()
|
|
|
|
// Duration can't be 0 to prevent panic for division.
|
|
duration := uint64(math.Max(float64(s.saveHotStateDB.duration), 1))
|
|
|
|
s.saveHotStateDB.lock.Lock()
|
|
if s.saveHotStateDB.enabled && st.Slot().Mod(duration) == 0 {
|
|
if err := s.beaconDB.SaveState(ctx, st, blockRoot); err != nil {
|
|
s.saveHotStateDB.lock.Unlock()
|
|
return err
|
|
}
|
|
s.saveHotStateDB.blockRootsOfSavedStates = append(s.saveHotStateDB.blockRootsOfSavedStates, blockRoot)
|
|
|
|
log.WithFields(logrus.Fields{
|
|
"slot": st.Slot(),
|
|
"totalHotStateSavedInDB": len(s.saveHotStateDB.blockRootsOfSavedStates),
|
|
}).Info("Saving hot state to DB")
|
|
}
|
|
s.saveHotStateDB.lock.Unlock()
|
|
|
|
// If the hot state is already in cache, one can be sure the state was processed and in the DB.
|
|
if s.hotStateCache.has(blockRoot) {
|
|
return nil
|
|
}
|
|
|
|
// Only on an epoch boundary slot, save epoch boundary state in epoch boundary root state cache.
|
|
if slots.IsEpochStart(st.Slot()) {
|
|
if err := s.epochBoundaryStateCache.put(blockRoot, st); err != nil {
|
|
return err
|
|
}
|
|
} else {
|
|
// Always check that the correct epoch boundary states have been saved
|
|
// for the current epoch.
|
|
epochStart, err := slots.EpochStart(slots.ToEpoch(st.Slot()))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
bRoot, err := helpers.BlockRootAtSlot(st, epochStart)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
_, ok, err := s.epochBoundaryStateCache.getByBlockRoot([32]byte(bRoot))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
// We would only recover the boundary states under this condition:
|
|
//
|
|
// 1) Would indicate that the epoch boundary was skipped due to a missed slot, we
|
|
// then recover by saving the state at that particular slot here.
|
|
if !ok {
|
|
// Only recover the state if it is in our hot state cache, otherwise we
|
|
// simply skip this step.
|
|
if s.hotStateCache.has([32]byte(bRoot)) {
|
|
log.WithFields(logrus.Fields{
|
|
"slot": epochStart,
|
|
"root": fmt.Sprintf("%#x", bRoot),
|
|
}).Debug("Recovering state for epoch boundary cache")
|
|
|
|
hState := s.hotStateCache.get([32]byte(bRoot))
|
|
if err := s.epochBoundaryStateCache.put([32]byte(bRoot), hState); err != nil {
|
|
return err
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if err := s.beaconDB.SaveStateSummary(ctx, ðpb.StateSummary{
|
|
Slot: st.Slot(),
|
|
Root: blockRoot[:],
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
// Store the copied state in the hot state cache.
|
|
s.hotStateCache.put(blockRoot, st)
|
|
|
|
return nil
|
|
}
|
|
|
|
// EnableSaveHotStateToDB enters the mode that saves hot beacon state to the DB.
|
|
// This usually gets triggered when there's long duration since finality.
|
|
func (s *State) EnableSaveHotStateToDB(_ context.Context) {
|
|
s.saveHotStateDB.lock.Lock()
|
|
defer s.saveHotStateDB.lock.Unlock()
|
|
if s.saveHotStateDB.enabled {
|
|
return
|
|
}
|
|
|
|
s.saveHotStateDB.enabled = true
|
|
|
|
log.WithFields(logrus.Fields{
|
|
"enabled": s.saveHotStateDB.enabled,
|
|
"slotsInterval": s.saveHotStateDB.duration,
|
|
}).Warn("Entering mode to save hot states in DB")
|
|
}
|
|
|
|
// DisableSaveHotStateToDB exits the mode that saves beacon state to DB for the hot states.
|
|
// This usually gets triggered once there's finality after long duration since finality.
|
|
func (s *State) DisableSaveHotStateToDB(ctx context.Context) error {
|
|
s.saveHotStateDB.lock.Lock()
|
|
defer s.saveHotStateDB.lock.Unlock()
|
|
if !s.saveHotStateDB.enabled {
|
|
return nil
|
|
}
|
|
|
|
log.WithFields(logrus.Fields{
|
|
"enabled": s.saveHotStateDB.enabled,
|
|
"deletedHotStates": len(s.saveHotStateDB.blockRootsOfSavedStates),
|
|
}).Warn("Exiting mode to save hot states in DB")
|
|
|
|
// Delete previous saved states in DB as we are turning this mode off.
|
|
s.saveHotStateDB.enabled = false
|
|
if err := s.beaconDB.DeleteStates(ctx, s.saveHotStateDB.blockRootsOfSavedStates); err != nil {
|
|
return err
|
|
}
|
|
s.saveHotStateDB.blockRootsOfSavedStates = nil
|
|
|
|
return nil
|
|
}
|