mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-10 19:51:20 +00:00
9e3598334d
* Avoid double state copy in `latestAncestor` * remove trailing newline --------- Co-authored-by: Preston Van Loon <preston@prysmaticlabs.com>
340 lines
11 KiB
Go
340 lines
11 KiB
Go
package stategen
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/helpers"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/time"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"github.com/prysmaticlabs/prysm/v4/encoding/bytesutil"
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
var ErrNoDataForSlot = errors.New("cannot retrieve data for slot")
|
|
|
|
// HasState returns true if the state exists in cache or in DB.
|
|
func (s *State) HasState(ctx context.Context, blockRoot [32]byte) (bool, error) {
|
|
has, err := s.hasStateInCache(ctx, blockRoot)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
if has {
|
|
return true, nil
|
|
}
|
|
return s.beaconDB.HasState(ctx, blockRoot), nil
|
|
}
|
|
|
|
// hasStateInCache returns true if the state exists in cache.
|
|
func (s *State) hasStateInCache(_ context.Context, blockRoot [32]byte) (bool, error) {
|
|
if s.hotStateCache.has(blockRoot) {
|
|
return true, nil
|
|
}
|
|
_, has, err := s.epochBoundaryStateCache.getByBlockRoot(blockRoot)
|
|
if err != nil {
|
|
return false, err
|
|
}
|
|
return has, nil
|
|
}
|
|
|
|
// StateByRootIfCachedNoCopy retrieves a state using the input block root only if the state is already in the cache.
|
|
func (s *State) StateByRootIfCachedNoCopy(blockRoot [32]byte) state.BeaconState {
|
|
if !s.hotStateCache.has(blockRoot) {
|
|
return nil
|
|
}
|
|
return s.hotStateCache.getWithoutCopy(blockRoot)
|
|
}
|
|
|
|
// StateByRoot retrieves the state using input block root.
|
|
func (s *State) StateByRoot(ctx context.Context, blockRoot [32]byte) (state.BeaconState, error) {
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.StateByRoot")
|
|
defer span.End()
|
|
|
|
// Genesis case. If block root is zero hash, short circuit to use genesis state stored in DB.
|
|
if blockRoot == params.BeaconConfig().ZeroHash {
|
|
root, err := s.beaconDB.GenesisBlockRoot(ctx)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get genesis block root")
|
|
}
|
|
blockRoot = root
|
|
}
|
|
return s.loadStateByRoot(ctx, blockRoot)
|
|
}
|
|
|
|
// ActiveNonSlashedBalancesByRoot retrieves the effective balances of all active and non-slashed validators at the
|
|
// state with a given root
|
|
func (s *State) ActiveNonSlashedBalancesByRoot(ctx context.Context, blockRoot [32]byte) ([]uint64, error) {
|
|
st, err := s.StateByRoot(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if st == nil || st.IsNil() {
|
|
return nil, errNilState
|
|
}
|
|
epoch := time.CurrentEpoch(st)
|
|
|
|
balances := make([]uint64, st.NumValidators())
|
|
var balanceAccretor = func(idx int, val state.ReadOnlyValidator) error {
|
|
if helpers.IsActiveNonSlashedValidatorUsingTrie(val, epoch) {
|
|
balances[idx] = val.EffectiveBalance()
|
|
} else {
|
|
balances[idx] = 0
|
|
}
|
|
return nil
|
|
}
|
|
if err := st.ReadFromEveryValidator(balanceAccretor); err != nil {
|
|
return nil, err
|
|
}
|
|
return balances, nil
|
|
}
|
|
|
|
// StateByRootInitialSync retrieves the state from the DB for the initial syncing phase.
|
|
// It assumes initial syncing using a block list rather than a block tree hence the returned
|
|
// state is not copied (block batches returned from initial sync are linear).
|
|
// It invalidates cache for parent root because pre-state will get mutated.
|
|
//
|
|
// WARNING: Do not use this method for anything other than initial syncing purpose or block tree is applied.
|
|
func (s *State) StateByRootInitialSync(ctx context.Context, blockRoot [32]byte) (state.BeaconState, error) {
|
|
// Genesis case. If block root is zero hash, short circuit to use genesis state stored in DB.
|
|
if blockRoot == params.BeaconConfig().ZeroHash {
|
|
return s.beaconDB.GenesisState(ctx)
|
|
}
|
|
|
|
// To invalidate cache for parent root because pre-state will get mutated.
|
|
// It is a parent root because StateByRootInitialSync is always used to fetch the block's parent state.
|
|
defer s.hotStateCache.delete(blockRoot)
|
|
|
|
if s.hotStateCache.has(blockRoot) {
|
|
return s.hotStateCache.getWithoutCopy(blockRoot), nil
|
|
}
|
|
|
|
cachedInfo, ok, err := s.epochBoundaryStateCache.getByBlockRoot(blockRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if ok {
|
|
return cachedInfo.state, nil
|
|
}
|
|
|
|
startState, err := s.latestAncestor(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get ancestor state")
|
|
}
|
|
if startState == nil || startState.IsNil() {
|
|
return nil, errUnknownState
|
|
}
|
|
summary, err := s.stateSummary(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get state summary")
|
|
}
|
|
if startState.Slot() == summary.Slot {
|
|
return startState, nil
|
|
}
|
|
|
|
blks, err := s.loadBlocks(ctx, startState.Slot()+1, summary.Slot, bytesutil.ToBytes32(summary.Root))
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not load blocks")
|
|
}
|
|
startState, err = s.replayBlocks(ctx, startState, blks, summary.Slot)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not replay blocks")
|
|
}
|
|
|
|
return startState, nil
|
|
}
|
|
|
|
// This returns the state summary object of a given block root. It first checks the cache, then checks the DB.
|
|
func (s *State) stateSummary(ctx context.Context, blockRoot [32]byte) (*ethpb.StateSummary, error) {
|
|
var summary *ethpb.StateSummary
|
|
var err error
|
|
|
|
summary, err = s.beaconDB.StateSummary(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
if summary == nil {
|
|
return s.recoverStateSummary(ctx, blockRoot)
|
|
}
|
|
return summary, nil
|
|
}
|
|
|
|
// RecoverStateSummary recovers state summary object of a given block root by using the saved block in DB.
|
|
func (s *State) recoverStateSummary(ctx context.Context, blockRoot [32]byte) (*ethpb.StateSummary, error) {
|
|
if s.beaconDB.HasBlock(ctx, blockRoot) {
|
|
b, err := s.beaconDB.Block(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
summary := ðpb.StateSummary{Slot: b.Block().Slot(), Root: blockRoot[:]}
|
|
if err := s.beaconDB.SaveStateSummary(ctx, summary); err != nil {
|
|
return nil, err
|
|
}
|
|
return summary, nil
|
|
}
|
|
return nil, errors.New("could not find block in DB")
|
|
}
|
|
|
|
// DeleteStateFromCaches deletes the state from the caches.
|
|
func (s *State) DeleteStateFromCaches(_ context.Context, blockRoot [32]byte) error {
|
|
s.hotStateCache.delete(blockRoot)
|
|
return s.epochBoundaryStateCache.delete(blockRoot)
|
|
}
|
|
|
|
// This loads a beacon state from either the cache or DB, then replays blocks up the slot of the requested block root.
|
|
func (s *State) loadStateByRoot(ctx context.Context, blockRoot [32]byte) (state.BeaconState, error) {
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.loadStateByRoot")
|
|
defer span.End()
|
|
|
|
// First, it checks if the state exists in hot state cache.
|
|
cachedState := s.hotStateCache.get(blockRoot)
|
|
if cachedState != nil && !cachedState.IsNil() {
|
|
return cachedState, nil
|
|
}
|
|
|
|
// Second, it checks if the state exists in epoch boundary state cache.
|
|
cachedInfo, ok, err := s.epochBoundaryStateCache.getByBlockRoot(blockRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if ok {
|
|
return cachedInfo.state, nil
|
|
}
|
|
|
|
// Short circuit if the state is already in the DB.
|
|
if s.beaconDB.HasState(ctx, blockRoot) {
|
|
return s.beaconDB.State(ctx, blockRoot)
|
|
}
|
|
|
|
summary, err := s.stateSummary(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get state summary")
|
|
}
|
|
targetSlot := summary.Slot
|
|
|
|
// Since the requested state is not in caches or DB, start replaying using the last
|
|
// available ancestor state which is retrieved using input block's root.
|
|
startState, err := s.latestAncestor(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not get ancestor state")
|
|
}
|
|
if startState == nil || startState.IsNil() {
|
|
return nil, errUnknownBoundaryState
|
|
}
|
|
|
|
if startState.Slot() == targetSlot {
|
|
return startState, nil
|
|
}
|
|
|
|
blks, err := s.loadBlocks(ctx, startState.Slot()+1, targetSlot, bytesutil.ToBytes32(summary.Root))
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not load blocks for hot state using root")
|
|
}
|
|
|
|
replayBlockCount.Observe(float64(len(blks)))
|
|
|
|
return s.replayBlocks(ctx, startState, blks, targetSlot)
|
|
}
|
|
|
|
// latestAncestor returns the highest available ancestor state of the input block root.
|
|
// It recursively looks up block's parent until a corresponding state of the block root
|
|
// is found in the caches or DB.
|
|
//
|
|
// There's three ways to derive block parent state:
|
|
// 1) block parent state is the last finalized state
|
|
// 2) block parent state is the epoch boundary state and exists in epoch boundary cache
|
|
// 3) block parent state is in DB
|
|
func (s *State) latestAncestor(ctx context.Context, blockRoot [32]byte) (state.BeaconState, error) {
|
|
ctx, span := trace.StartSpan(ctx, "stateGen.latestAncestor")
|
|
defer span.End()
|
|
|
|
if s.isFinalizedRoot(blockRoot) {
|
|
finalizedState := s.finalizedState()
|
|
if finalizedState != nil {
|
|
return finalizedState, nil
|
|
}
|
|
}
|
|
|
|
b, err := s.beaconDB.Block(ctx, blockRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err := blocks.BeaconBlockIsNil(b); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
for {
|
|
if ctx.Err() != nil {
|
|
return nil, ctx.Err()
|
|
}
|
|
|
|
// Is the state the genesis state.
|
|
parentRoot := b.Block().ParentRoot()
|
|
if parentRoot == params.BeaconConfig().ZeroHash {
|
|
s, err := s.beaconDB.GenesisState(ctx)
|
|
return s, errors.Wrap(err, "could not get genesis state")
|
|
}
|
|
|
|
// Return an error if slot hasn't been covered by checkpoint sync.
|
|
ps := b.Block().Slot() - 1
|
|
if !s.slotAvailable(ps) {
|
|
return nil, errors.Wrapf(ErrNoDataForSlot, "slot %d not in db due to checkpoint sync", ps)
|
|
}
|
|
// Does the state exist in the hot state cache.
|
|
if s.hotStateCache.has(parentRoot) {
|
|
return s.hotStateCache.get(parentRoot), nil
|
|
}
|
|
|
|
// Does the state exist in finalized info cache.
|
|
if s.isFinalizedRoot(parentRoot) {
|
|
return s.finalizedState(), nil
|
|
}
|
|
|
|
// Does the state exist in epoch boundary cache.
|
|
cachedInfo, ok, err := s.epochBoundaryStateCache.getByBlockRoot(parentRoot)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if ok {
|
|
return cachedInfo.state, nil
|
|
}
|
|
|
|
// Does the state exists in DB.
|
|
if s.beaconDB.HasState(ctx, parentRoot) {
|
|
s, err := s.beaconDB.State(ctx, parentRoot)
|
|
return s, errors.Wrap(err, "failed to retrieve state from db")
|
|
}
|
|
|
|
b, err = s.beaconDB.Block(ctx, parentRoot)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "failed to retrieve block from db")
|
|
}
|
|
if b == nil || b.IsNil() {
|
|
return nil, errUnknownBlock
|
|
}
|
|
}
|
|
}
|
|
|
|
func (s *State) CombinedCache() *CombinedCache {
|
|
getters := make([]CachedGetter, 0)
|
|
if s.hotStateCache != nil {
|
|
getters = append(getters, s.hotStateCache)
|
|
}
|
|
if s.epochBoundaryStateCache != nil {
|
|
getters = append(getters, s.epochBoundaryStateCache)
|
|
}
|
|
return &CombinedCache{getters: getters}
|
|
}
|
|
|
|
func (s *State) slotAvailable(slot primitives.Slot) bool {
|
|
// default to assuming node was initialized from genesis - backfill only needs to be specified for checkpoint sync
|
|
if s.backfillStatus == nil {
|
|
return true
|
|
}
|
|
return s.backfillStatus.SlotCovered(slot)
|
|
}
|