mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-05 09:14:28 +00:00
2a4441762e
* add changes * fix tests * fix edge case * fix logging
195 lines
5.0 KiB
Go
195 lines
5.0 KiB
Go
package stategen
|
|
|
|
import (
|
|
"errors"
|
|
"strconv"
|
|
"sync"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
"k8s.io/client-go/tools/cache"
|
|
)
|
|
|
|
var (
|
|
// maxCacheSize is 8. That means 8 epochs and roughly an hour
|
|
// of no finality can be endured.
|
|
maxCacheSize = uint64(8)
|
|
errNotSlotRootInfo = errors.New("not slot root info type")
|
|
errNotRootStateInfo = errors.New("not root state info type")
|
|
)
|
|
|
|
// slotRootInfo specifies the slot root info in the epoch boundary state cache.
|
|
type slotRootInfo struct {
|
|
slot primitives.Slot
|
|
blockRoot [32]byte
|
|
}
|
|
|
|
// slotKeyFn takes the string representation of the slot to be used as key
|
|
// to retrieve root.
|
|
func slotKeyFn(obj interface{}) (string, error) {
|
|
s, ok := obj.(*slotRootInfo)
|
|
if !ok {
|
|
return "", errNotSlotRootInfo
|
|
}
|
|
return slotToString(s.slot), nil
|
|
}
|
|
|
|
// rootStateInfo specifies the root state info in the epoch boundary state cache.
|
|
type rootStateInfo struct {
|
|
root [32]byte
|
|
state state.BeaconState
|
|
}
|
|
|
|
// rootKeyFn takes the string representation of the block root to be used as key
|
|
// to retrieve epoch boundary state.
|
|
func rootKeyFn(obj interface{}) (string, error) {
|
|
s, ok := obj.(*rootStateInfo)
|
|
if !ok {
|
|
return "", errNotRootStateInfo
|
|
}
|
|
return string(s.root[:]), nil
|
|
}
|
|
|
|
// epochBoundaryState struct with two queues by looking up beacon state by slot or root.
|
|
type epochBoundaryState struct {
|
|
rootStateCache *cache.FIFO
|
|
slotRootCache *cache.FIFO
|
|
lock sync.RWMutex
|
|
}
|
|
|
|
// newBoundaryStateCache creates a new block newBoundaryStateCache for storing and accessing epoch boundary states from
|
|
// memory.
|
|
func newBoundaryStateCache() *epochBoundaryState {
|
|
return &epochBoundaryState{
|
|
rootStateCache: cache.NewFIFO(rootKeyFn),
|
|
slotRootCache: cache.NewFIFO(slotKeyFn),
|
|
}
|
|
}
|
|
|
|
// ByBlockRoot satisfies the CachedGetter interface
|
|
func (e *epochBoundaryState) ByBlockRoot(r [32]byte) (state.BeaconState, error) {
|
|
rsi, ok, err := e.getByBlockRoot(r)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if !ok {
|
|
return nil, ErrNotInCache
|
|
}
|
|
return rsi.state, nil
|
|
}
|
|
|
|
// get epoch boundary state by its block root. Returns copied state in state info object if exists. Otherwise returns nil.
|
|
func (e *epochBoundaryState) getByBlockRoot(r [32]byte) (*rootStateInfo, bool, error) {
|
|
e.lock.RLock()
|
|
defer e.lock.RUnlock()
|
|
|
|
return e.getByBlockRootLockFree(r)
|
|
}
|
|
|
|
func (e *epochBoundaryState) getByBlockRootLockFree(r [32]byte) (*rootStateInfo, bool, error) {
|
|
obj, exists, err := e.rootStateCache.GetByKey(string(r[:]))
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
if !exists {
|
|
return nil, false, nil
|
|
}
|
|
s, ok := obj.(*rootStateInfo)
|
|
if !ok {
|
|
return nil, false, errNotRootStateInfo
|
|
}
|
|
|
|
return &rootStateInfo{
|
|
root: r,
|
|
state: s.state.Copy(),
|
|
}, true, nil
|
|
}
|
|
|
|
// get epoch boundary state by its slot. Returns copied state in state info object if exists. Otherwise returns nil.
|
|
func (e *epochBoundaryState) getBySlot(s primitives.Slot) (*rootStateInfo, bool, error) {
|
|
e.lock.RLock()
|
|
defer e.lock.RUnlock()
|
|
|
|
obj, exists, err := e.slotRootCache.GetByKey(slotToString(s))
|
|
if err != nil {
|
|
return nil, false, err
|
|
}
|
|
if !exists {
|
|
return nil, false, nil
|
|
}
|
|
info, ok := obj.(*slotRootInfo)
|
|
if !ok {
|
|
return nil, false, errNotSlotRootInfo
|
|
}
|
|
|
|
return e.getByBlockRootLockFree(info.blockRoot)
|
|
}
|
|
|
|
// put adds a state to the epoch boundary state cache. This method also trims the
|
|
// least recently added state info if the cache size has reached the max cache
|
|
// size limit.
|
|
func (e *epochBoundaryState) put(blockRoot [32]byte, s state.BeaconState) error {
|
|
e.lock.Lock()
|
|
defer e.lock.Unlock()
|
|
|
|
if err := e.slotRootCache.AddIfNotPresent(&slotRootInfo{
|
|
slot: s.Slot(),
|
|
blockRoot: blockRoot,
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
if err := e.rootStateCache.AddIfNotPresent(&rootStateInfo{
|
|
root: blockRoot,
|
|
state: s.Copy(),
|
|
}); err != nil {
|
|
return err
|
|
}
|
|
|
|
trim(e.rootStateCache, maxCacheSize)
|
|
trim(e.slotRootCache, maxCacheSize)
|
|
|
|
return nil
|
|
}
|
|
|
|
// delete the state from the epoch boundary state cache.
|
|
func (e *epochBoundaryState) delete(blockRoot [32]byte) error {
|
|
e.lock.Lock()
|
|
defer e.lock.Unlock()
|
|
rInfo, ok, err := e.getByBlockRootLockFree(blockRoot)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if !ok {
|
|
return nil
|
|
}
|
|
slotInfo := &slotRootInfo{
|
|
slot: rInfo.state.Slot(),
|
|
blockRoot: blockRoot,
|
|
}
|
|
if err = e.slotRootCache.Delete(slotInfo); err != nil {
|
|
return err
|
|
}
|
|
return e.rootStateCache.Delete(&rootStateInfo{
|
|
root: blockRoot,
|
|
})
|
|
}
|
|
|
|
// trim the FIFO queue to the maxSize.
|
|
func trim(queue *cache.FIFO, maxSize uint64) {
|
|
for s := uint64(len(queue.ListKeys())); s > maxSize; s-- {
|
|
if _, err := queue.Pop(popProcessNoopFunc); err != nil { // This never returns an error, but we'll handle anyway for sanity.
|
|
panic(err)
|
|
}
|
|
}
|
|
}
|
|
|
|
// popProcessNoopFunc is a no-op function that never returns an error.
|
|
func popProcessNoopFunc(_ interface{}) error {
|
|
return nil
|
|
}
|
|
|
|
// Converts input uint64 to string. To be used as key for slot to get root.
|
|
func slotToString(s primitives.Slot) string {
|
|
return strconv.FormatUint(uint64(s), 10)
|
|
}
|