2020-04-29 17:40:33 +00:00
|
|
|
// Package stategen defines functions to regenerate beacon chain states
|
|
|
|
// by replaying blocks from a stored state checkpoint, useful for
|
|
|
|
// optimization and reducing a beacon node's resource consumption.
|
2020-02-16 23:28:20 +00:00
|
|
|
package stategen
|
|
|
|
|
|
|
|
import (
|
2020-03-15 16:47:49 +00:00
|
|
|
"context"
|
2020-03-03 19:07:34 +00:00
|
|
|
"sync"
|
|
|
|
|
2020-03-08 06:24:57 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/cache"
|
2020-02-16 23:28:20 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/db"
|
2020-03-15 16:47:49 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/beacon-chain/state"
|
2020-05-15 19:49:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/bytesutil"
|
|
|
|
"github.com/prysmaticlabs/prysm/shared/featureconfig"
|
2020-03-06 23:06:01 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/shared/params"
|
2020-03-15 16:47:49 +00:00
|
|
|
"go.opencensus.io/trace"
|
2020-02-16 23:28:20 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// State represents a management object that handles the internal
|
|
|
|
// logic of maintaining both hot and cold states in DB.
|
|
|
|
type State struct {
|
2020-03-03 19:07:34 +00:00
|
|
|
beaconDB db.NoHeadAccessDatabase
|
2020-03-12 00:38:30 +00:00
|
|
|
slotsPerArchivedPoint uint64
|
2020-03-03 19:07:34 +00:00
|
|
|
epochBoundarySlotToRoot map[uint64][32]byte
|
|
|
|
epochBoundaryLock sync.RWMutex
|
2020-03-08 06:24:57 +00:00
|
|
|
hotStateCache *cache.HotStateCache
|
2020-03-12 02:27:16 +00:00
|
|
|
splitInfo *splitSlotAndRoot
|
2020-03-30 22:10:45 +00:00
|
|
|
stateSummaryCache *cache.StateSummaryCache
|
2020-03-12 02:27:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This tracks the split point. The point where slot and the block root of
|
|
|
|
// cold and hot sections of the DB splits.
|
|
|
|
type splitSlotAndRoot struct {
|
|
|
|
slot uint64
|
|
|
|
root [32]byte
|
2020-02-16 23:28:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// New returns a new state management object.
|
2020-03-30 22:10:45 +00:00
|
|
|
func New(db db.NoHeadAccessDatabase, stateSummaryCache *cache.StateSummaryCache) *State {
|
2020-02-16 23:28:20 +00:00
|
|
|
return &State{
|
2020-03-03 19:07:34 +00:00
|
|
|
beaconDB: db,
|
|
|
|
epochBoundarySlotToRoot: make(map[uint64][32]byte),
|
2020-03-08 06:24:57 +00:00
|
|
|
hotStateCache: cache.NewHotStateCache(),
|
2020-03-12 02:27:16 +00:00
|
|
|
splitInfo: &splitSlotAndRoot{slot: 0, root: params.BeaconConfig().ZeroHash},
|
2020-04-08 01:11:19 +00:00
|
|
|
slotsPerArchivedPoint: params.BeaconConfig().SlotsPerArchivedPoint,
|
2020-03-30 22:10:45 +00:00
|
|
|
stateSummaryCache: stateSummaryCache,
|
2020-02-16 23:28:20 +00:00
|
|
|
}
|
|
|
|
}
|
2020-03-06 23:06:01 +00:00
|
|
|
|
2020-03-15 16:47:49 +00:00
|
|
|
// Resume resumes a new state management object from previously saved finalized check point in DB.
|
2020-03-27 20:28:38 +00:00
|
|
|
func (s *State) Resume(ctx context.Context) (*state.BeaconState, error) {
|
2020-03-15 16:47:49 +00:00
|
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.Resume")
|
|
|
|
defer span.End()
|
|
|
|
|
2020-03-27 20:28:38 +00:00
|
|
|
lastArchivedRoot := s.beaconDB.LastArchivedIndexRoot(ctx)
|
|
|
|
lastArchivedState, err := s.beaconDB.State(ctx, lastArchivedRoot)
|
2020-03-15 16:47:49 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2020-03-27 20:28:38 +00:00
|
|
|
|
2020-05-15 19:49:13 +00:00
|
|
|
if featureconfig.Get().SkipRegenHistoricalStates {
|
|
|
|
// If a node doesn't want to regen historical states, the node would
|
|
|
|
// start from last finalized check point.
|
|
|
|
cp, err := s.beaconDB.FinalizedCheckpoint(ctx)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
lastArchivedState, err = s.beaconDB.State(ctx, bytesutil.ToBytes32(cp.Root))
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
lastArchivedRoot = bytesutil.ToBytes32(cp.Root)
|
|
|
|
}
|
|
|
|
|
2020-03-16 19:07:07 +00:00
|
|
|
// Resume as genesis state if there's no last archived state.
|
|
|
|
if lastArchivedState == nil {
|
|
|
|
return s.beaconDB.GenesisState(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
s.splitInfo = &splitSlotAndRoot{slot: lastArchivedState.Slot(), root: lastArchivedRoot}
|
|
|
|
|
|
|
|
return lastArchivedState, nil
|
2020-03-15 16:47:49 +00:00
|
|
|
}
|
|
|
|
|
2020-03-06 23:06:01 +00:00
|
|
|
// This verifies the archive point frequency is valid. It checks the interval
|
|
|
|
// is a divisor of the number of slots per epoch. This ensures we have at least one
|
|
|
|
// archive point within range of our state root history when iterating
|
|
|
|
// backwards. It also ensures the archive points align with hot state summaries
|
|
|
|
// which makes it quicker to migrate hot to cold.
|
|
|
|
func verifySlotsPerArchivePoint(slotsPerArchivePoint uint64) bool {
|
|
|
|
return slotsPerArchivePoint > 0 &&
|
|
|
|
slotsPerArchivePoint%params.BeaconConfig().SlotsPerEpoch == 0
|
|
|
|
}
|