2021-08-24 02:54:04 -05:00
|
|
|
package cache
|
|
|
|
|
|
|
|
import (
|
|
|
|
"sync"
|
|
|
|
|
|
|
|
lru "github.com/hashicorp/golang-lru"
|
2023-03-17 11:52:56 -07:00
|
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
|
|
lruwrpr "github.com/prysmaticlabs/prysm/v4/cache/lru"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/consensus-types/primitives"
|
|
|
|
"github.com/prysmaticlabs/prysm/v4/runtime/version"
|
2021-08-24 02:54:04 -05:00
|
|
|
)
|
|
|
|
|
|
|
|
// SyncCommitteeHeadStateCache for the latest head state requested by a sync committee participant.
|
|
|
|
type SyncCommitteeHeadStateCache struct {
|
|
|
|
cache *lru.Cache
|
|
|
|
lock sync.RWMutex
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewSyncCommitteeHeadState initializes a LRU cache for `SyncCommitteeHeadState` with size of 1.
|
2021-08-26 13:01:09 +08:00
|
|
|
func NewSyncCommitteeHeadState() *SyncCommitteeHeadStateCache {
|
2021-09-02 12:36:54 +02:00
|
|
|
c := lruwrpr.New(1) // only need size of 1 to avoid redundant state copies, hashing, and slot processing.
|
2021-08-26 13:01:09 +08:00
|
|
|
return &SyncCommitteeHeadStateCache{cache: c}
|
2021-08-24 02:54:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Put `slot` as key and `state` as value onto the cache.
|
2023-01-26 15:40:12 +01:00
|
|
|
func (c *SyncCommitteeHeadStateCache) Put(slot primitives.Slot, st state.BeaconState) error {
|
2021-08-24 02:54:04 -05:00
|
|
|
c.lock.Lock()
|
|
|
|
defer c.lock.Unlock()
|
2021-08-26 13:01:09 +08:00
|
|
|
// Make sure that the provided state is non nil
|
|
|
|
// and is of the correct type.
|
|
|
|
if st == nil || st.IsNil() {
|
|
|
|
return ErrNilValueProvided
|
|
|
|
}
|
2021-12-13 10:04:37 -08:00
|
|
|
|
2022-02-08 10:30:06 +01:00
|
|
|
if st.Version() == version.Phase0 {
|
2021-08-26 13:01:09 +08:00
|
|
|
return ErrIncorrectType
|
|
|
|
}
|
2021-12-13 10:04:37 -08:00
|
|
|
|
2021-08-24 02:54:04 -05:00
|
|
|
c.cache.Add(slot, st)
|
2021-08-26 13:01:09 +08:00
|
|
|
return nil
|
2021-08-24 02:54:04 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
// Get `state` using `slot` as key. Return nil if nothing is found.
|
2023-01-26 15:40:12 +01:00
|
|
|
func (c *SyncCommitteeHeadStateCache) Get(slot primitives.Slot) (state.BeaconState, error) {
|
2021-08-24 02:54:04 -05:00
|
|
|
c.lock.RLock()
|
|
|
|
defer c.lock.RUnlock()
|
|
|
|
val, exists := c.cache.Get(slot)
|
2021-08-26 13:01:09 +08:00
|
|
|
if !exists {
|
2021-08-24 02:54:04 -05:00
|
|
|
return nil, ErrNotFound
|
|
|
|
}
|
2022-02-03 01:51:24 +08:00
|
|
|
st, ok := val.(state.BeaconState)
|
2021-08-26 13:01:09 +08:00
|
|
|
if !ok {
|
2022-02-03 01:51:24 +08:00
|
|
|
return nil, ErrIncorrectType
|
|
|
|
}
|
2022-12-16 08:42:43 +08:00
|
|
|
// Sync committee is not supported in phase 0.
|
|
|
|
if st.Version() == version.Phase0 {
|
2022-02-03 01:51:24 +08:00
|
|
|
return nil, ErrIncorrectType
|
2021-08-26 13:01:09 +08:00
|
|
|
}
|
|
|
|
return st, nil
|
2021-08-24 02:54:04 -05:00
|
|
|
}
|