erigon-pulse/cmd/erigon-cl/core/state/state.go

173 lines
5.1 KiB
Go

package state
import (
"crypto/sha256"
"encoding/binary"
lru "github.com/hashicorp/golang-lru"
libcommon "github.com/ledgerwatch/erigon-lib/common"
"github.com/ledgerwatch/erigon/cl/clparams"
"github.com/ledgerwatch/erigon/cl/cltypes"
"github.com/ledgerwatch/erigon/cl/utils"
"github.com/ledgerwatch/erigon/core/types"
)
type HashFunc func([]byte) ([32]byte, error)
const (
blockRootsLength = 8192
stateRootsLength = 8192
randoMixesLength = 65536
slashingsLength = 8192
)
type BeaconState struct {
// State fields
genesisTime uint64
genesisValidatorsRoot libcommon.Hash
slot uint64
fork *cltypes.Fork
latestBlockHeader *cltypes.BeaconBlockHeader
blockRoots [blockRootsLength]libcommon.Hash
stateRoots [stateRootsLength]libcommon.Hash
historicalRoots []libcommon.Hash
eth1Data *cltypes.Eth1Data
eth1DataVotes []*cltypes.Eth1Data
eth1DepositIndex uint64
validators []*cltypes.Validator
balances []uint64
randaoMixes [randoMixesLength]libcommon.Hash
slashings [slashingsLength]uint64
previousEpochParticipation cltypes.ParticipationFlagsList
currentEpochParticipation cltypes.ParticipationFlagsList
justificationBits cltypes.JustificationBits
// Altair
previousJustifiedCheckpoint *cltypes.Checkpoint
currentJustifiedCheckpoint *cltypes.Checkpoint
finalizedCheckpoint *cltypes.Checkpoint
inactivityScores []uint64
currentSyncCommittee *cltypes.SyncCommittee
nextSyncCommittee *cltypes.SyncCommittee
// Bellatrix
latestExecutionPayloadHeader *types.Header
// Capella
nextWithdrawalIndex uint64
nextWithdrawalValidatorIndex uint64
historicalSummaries []*cltypes.HistoricalSummary
// Internals
version clparams.StateVersion // State version
leaves [32][32]byte // Pre-computed leaves.
touchedLeaves map[StateLeafIndex]bool // Maps each leaf to whether they were touched or not.
publicKeyIndicies map[[48]byte]uint64
// Caches
activeValidatorsCache *lru.Cache
committeeCache *lru.Cache
shuffledSetsCache *lru.Cache
totalActiveBalanceCache *uint64
totalActiveBalanceRootCache uint64
proposerIndex *uint64
// Configs
beaconConfig *clparams.BeaconChainConfig
}
func New(cfg *clparams.BeaconChainConfig) *BeaconState {
state := &BeaconState{
beaconConfig: cfg,
}
state.initBeaconState()
return state
}
func preparateRootsForHashing(roots []libcommon.Hash) [][32]byte {
ret := make([][32]byte, len(roots))
for i := range roots {
copy(ret[i][:], roots[i][:])
}
return ret
}
// MarshallSSZTo retrieve the SSZ encoded length of the state.
func (b *BeaconState) DecodeSSZ(buf []byte) error {
panic("not implemented")
}
// BlockRoot computes the block root for the state.
func (b *BeaconState) BlockRoot() ([32]byte, error) {
stateRoot, err := b.HashSSZ()
if err != nil {
return [32]byte{}, err
}
return (&cltypes.BeaconBlockHeader{
Slot: b.latestBlockHeader.Slot,
ProposerIndex: b.latestBlockHeader.ProposerIndex,
BodyRoot: b.latestBlockHeader.BodyRoot,
ParentRoot: b.latestBlockHeader.ParentRoot,
Root: stateRoot,
}).HashSSZ()
}
func (b *BeaconState) _refreshActiveBalances() {
epoch := b.Epoch()
b.totalActiveBalanceCache = new(uint64)
*b.totalActiveBalanceCache = 0
for _, validator := range b.validators {
if validator.Active(epoch) {
*b.totalActiveBalanceCache += validator.EffectiveBalance
}
}
*b.totalActiveBalanceCache = utils.Max64(b.beaconConfig.EffectiveBalanceIncrement, *b.totalActiveBalanceCache)
b.totalActiveBalanceRootCache = utils.IntegerSquareRoot(*b.totalActiveBalanceCache)
}
func (b *BeaconState) _updateProposerIndex() (err error) {
epoch := b.Epoch()
hash := sha256.New()
// Input for the seed hash.
input := b.GetSeed(epoch, clparams.MainnetBeaconConfig.DomainBeaconProposer)
slotByteArray := make([]byte, 8)
binary.LittleEndian.PutUint64(slotByteArray, b.slot)
// Add slot to the end of the input.
inputWithSlot := append(input[:], slotByteArray...)
// Calculate the hash.
hash.Write(inputWithSlot)
seed := hash.Sum(nil)
indices := b.GetActiveValidatorsIndices(epoch)
// Write the seed to an array.
seedArray := [32]byte{}
copy(seedArray[:], seed)
b.proposerIndex = new(uint64)
*b.proposerIndex, err = b.ComputeProposerIndex(indices, seedArray)
return
}
func (b *BeaconState) initBeaconState() error {
if b.touchedLeaves == nil {
b.touchedLeaves = make(map[StateLeafIndex]bool)
}
b.publicKeyIndicies = make(map[[48]byte]uint64)
b._refreshActiveBalances()
for i, validator := range b.validators {
b.publicKeyIndicies[validator.PublicKey] = uint64(i)
}
var err error
if b.activeValidatorsCache, err = lru.New(5); err != nil {
return err
}
if b.shuffledSetsCache, err = lru.New(25); err != nil {
return err
}
if b.committeeCache, err = lru.New(256); err != nil {
return err
}
if err := b._updateProposerIndex(); err != nil {
return err
}
return nil
}