2020-07-03 17:29:30 +00:00
|
|
|
package stategen
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"strconv"
|
|
|
|
"sync"
|
|
|
|
|
2022-08-16 12:20:13 +00:00
|
|
|
"github.com/prysmaticlabs/prysm/v3/beacon-chain/state"
|
|
|
|
types "github.com/prysmaticlabs/prysm/v3/consensus-types/primitives"
|
2020-07-03 17:29:30 +00:00
|
|
|
"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 {
|
2022-05-09 22:25:47 +00:00
|
|
|
slot types.Slot
|
|
|
|
blockRoot [32]byte
|
2020-07-03 17:29:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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
|
2021-07-23 16:11:21 +00:00
|
|
|
state state.BeaconState
|
2020-07-03 17:29:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:25:47 +00:00
|
|
|
// ByBlockRoot satisfies the CachedGetter interface
|
|
|
|
func (e *epochBoundaryState) ByBlockRoot(r [32]byte) (state.BeaconState, error) {
|
|
|
|
rsi, ok, err := e.getByBlockRoot(r)
|
2022-03-09 19:33:18 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if !ok {
|
|
|
|
return nil, ErrNotInCache
|
|
|
|
}
|
|
|
|
return rsi.state, nil
|
|
|
|
}
|
|
|
|
|
2020-07-03 17:29:30 +00:00
|
|
|
// get epoch boundary state by its block root. Returns copied state in state info object if exists. Otherwise returns nil.
|
2022-05-09 22:25:47 +00:00
|
|
|
func (e *epochBoundaryState) getByBlockRoot(r [32]byte) (*rootStateInfo, bool, error) {
|
2020-07-03 17:29:30 +00:00
|
|
|
e.lock.RLock()
|
|
|
|
defer e.lock.RUnlock()
|
|
|
|
|
2022-05-09 22:25:47 +00:00
|
|
|
return e.getByBlockRootLockFree(r)
|
2022-02-02 14:11:31 +00:00
|
|
|
}
|
|
|
|
|
2022-05-09 22:25:47 +00:00
|
|
|
func (e *epochBoundaryState) getByBlockRootLockFree(r [32]byte) (*rootStateInfo, bool, error) {
|
2020-07-03 17:29:30 +00:00
|
|
|
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.
|
2021-02-16 07:45:34 +00:00
|
|
|
func (e *epochBoundaryState) getBySlot(s types.Slot) (*rootStateInfo, bool, error) {
|
2020-07-03 17:29:30 +00:00
|
|
|
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
|
|
|
|
}
|
|
|
|
|
2022-05-09 22:25:47 +00:00
|
|
|
return e.getByBlockRootLockFree(info.blockRoot)
|
2020-07-03 17:29:30 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// 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.
|
2022-05-09 22:25:47 +00:00
|
|
|
func (e *epochBoundaryState) put(blockRoot [32]byte, s state.BeaconState) error {
|
2020-07-03 17:29:30 +00:00
|
|
|
e.lock.Lock()
|
|
|
|
defer e.lock.Unlock()
|
|
|
|
|
|
|
|
if err := e.slotRootCache.AddIfNotPresent(&slotRootInfo{
|
2022-05-09 22:25:47 +00:00
|
|
|
slot: s.Slot(),
|
|
|
|
blockRoot: blockRoot,
|
2020-07-03 17:29:30 +00:00
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if err := e.rootStateCache.AddIfNotPresent(&rootStateInfo{
|
2022-05-09 22:25:47 +00:00
|
|
|
root: blockRoot,
|
2020-07-03 17:29:30 +00:00
|
|
|
state: s.Copy(),
|
|
|
|
}); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
trim(e.rootStateCache, maxCacheSize)
|
|
|
|
trim(e.slotRootCache, maxCacheSize)
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-04-06 21:24:00 +00:00
|
|
|
// delete the state from the epoch boundary state cache.
|
2022-05-09 22:25:47 +00:00
|
|
|
func (e *epochBoundaryState) delete(blockRoot [32]byte) error {
|
2022-04-06 21:24:00 +00:00
|
|
|
e.lock.Lock()
|
|
|
|
defer e.lock.Unlock()
|
|
|
|
return e.rootStateCache.Delete(&rootStateInfo{
|
2022-05-09 22:25:47 +00:00
|
|
|
root: blockRoot,
|
2022-04-06 21:24:00 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2020-07-03 17:29:30 +00:00
|
|
|
// 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.
|
2020-10-12 08:11:05 +00:00
|
|
|
func popProcessNoopFunc(_ interface{}) error {
|
2020-07-03 17:29:30 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Converts input uint64 to string. To be used as key for slot to get root.
|
2021-02-16 07:45:34 +00:00
|
|
|
func slotToString(s types.Slot) string {
|
|
|
|
return strconv.FormatUint(uint64(s), 10)
|
2020-07-03 17:29:30 +00:00
|
|
|
}
|