mirror of
https://gitlab.com/pulsechaincom/erigon-pulse.git
synced 2025-01-10 13:01:21 +00:00
245 lines
6.8 KiB
Go
245 lines
6.8 KiB
Go
package state
|
|
|
|
import (
|
|
"github.com/ledgerwatch/erigon/cmd/erigon-cl/core/state/state_encoding"
|
|
"github.com/ledgerwatch/erigon/common"
|
|
)
|
|
|
|
func (b *BeaconState) HashTreeRoot() ([32]byte, error) {
|
|
if err := b.computeDirtyLeaves(); err != nil {
|
|
return [32]byte{}, err
|
|
}
|
|
|
|
currentLayer := b.leaves
|
|
// Pad to 32 of length
|
|
for len(currentLayer) != 32 {
|
|
currentLayer = append(currentLayer, [32]byte{})
|
|
}
|
|
return state_encoding.MerkleRootFromLeaves(currentLayer)
|
|
}
|
|
|
|
func (b *BeaconState) computeDirtyLeaves() error {
|
|
// Update all dirty leafs
|
|
// ----
|
|
|
|
// Field(0): GenesisTime
|
|
if b.isLeafDirty(GenesisTimeLeafIndex) {
|
|
b.updateLeaf(GenesisTimeLeafIndex, state_encoding.Uint64Root(b.genesisTime))
|
|
}
|
|
|
|
// Field(1): GenesisValidatorsRoot
|
|
if b.isLeafDirty(GenesisValidatorsRootLeafIndex) {
|
|
b.updateLeaf(GenesisValidatorsRootLeafIndex, b.genesisValidatorsRoot)
|
|
}
|
|
|
|
// Field(2): Slot
|
|
if b.isLeafDirty(SlotLeafIndex) {
|
|
b.updateLeaf(SlotLeafIndex, state_encoding.Uint64Root(b.slot))
|
|
}
|
|
|
|
// Field(3): Fork
|
|
if b.isLeafDirty(ForkLeafIndex) {
|
|
forkRoot, err := b.fork.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(ForkLeafIndex, forkRoot)
|
|
}
|
|
|
|
// Field(4): LatestBlockHeader
|
|
if b.isLeafDirty(LatestBlockHeaderLeafIndex) {
|
|
headerRoot, err := b.latestBlockHeader.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(LatestBlockHeaderLeafIndex, headerRoot)
|
|
}
|
|
|
|
// Field(5): BlockRoots
|
|
if b.isLeafDirty(BlockRootsLeafIndex) {
|
|
blockRootsRoot, err := state_encoding.ArraysRoot(b.blockRoots, state_encoding.BlockRootsLength)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(BlockRootsLeafIndex, blockRootsRoot)
|
|
}
|
|
|
|
// Field(6): StateRoots
|
|
if b.isLeafDirty(StateRootsLeafIndex) {
|
|
stateRootsRoot, err := state_encoding.ArraysRoot(b.stateRoots, state_encoding.StateRootsLength)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(StateRootsLeafIndex, stateRootsRoot)
|
|
}
|
|
|
|
// Field(7): HistoricalRoots
|
|
if b.isLeafDirty(HistoricalRootsLeafIndex) {
|
|
historicalRootsRoot, err := state_encoding.ArraysRootWithLimit(b.historicalRoots, state_encoding.HistoricalRootsLength)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(HistoricalRootsLeafIndex, historicalRootsRoot)
|
|
}
|
|
|
|
// Field(8): Eth1Data
|
|
if b.isLeafDirty(Eth1DataLeafIndex) {
|
|
dataRoot, err := b.eth1Data.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(Eth1DataLeafIndex, dataRoot)
|
|
}
|
|
|
|
// Field(9): Eth1DataVotes
|
|
if b.isLeafDirty(Eth1DataVotesLeafIndex) {
|
|
votesRoot, err := state_encoding.Eth1DataVectorRoot(b.eth1DataVotes)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(Eth1DataVotesLeafIndex, votesRoot)
|
|
}
|
|
|
|
// Field(10): Eth1DepositIndex
|
|
if b.isLeafDirty(Eth1DepositIndexLeafIndex) {
|
|
b.updateLeaf(Eth1DepositIndexLeafIndex, state_encoding.Uint64Root(b.eth1DepositIndex))
|
|
}
|
|
|
|
// Field(11): Validators
|
|
if b.isLeafDirty(ValidatorsLeafIndex) {
|
|
vRoot, err := state_encoding.ValidatorsVectorRoot(b.validators)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(ValidatorsLeafIndex, vRoot)
|
|
}
|
|
|
|
// Field(12): Balances
|
|
if b.isLeafDirty(BalancesLeafIndex) {
|
|
balancesRoot, err := state_encoding.Uint64ListRootWithLimit(b.balances, state_encoding.ValidatorLimitForBalancesChunks())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(BalancesLeafIndex, balancesRoot)
|
|
}
|
|
|
|
// Field(13): RandaoMixes
|
|
if b.isLeafDirty(RandaoMixesLeafIndex) {
|
|
randaoRootsRoot, err := state_encoding.ArraysRoot(b.randaoMixes, state_encoding.RandaoMixesLength)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(RandaoMixesLeafIndex, randaoRootsRoot)
|
|
}
|
|
|
|
// Field(14): Slashings
|
|
if b.isLeafDirty(SlashingsLeafIndex) {
|
|
slashingsRoot, err := state_encoding.SlashingsRoot(b.slashings)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(SlashingsLeafIndex, slashingsRoot)
|
|
}
|
|
// Field(15): PreviousEpochParticipation
|
|
if b.isLeafDirty(PreviousEpochParticipationLeafIndex) {
|
|
participationRoot, err := state_encoding.ParticipationBitsRoot(b.previousEpochParticipation)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(PreviousEpochParticipationLeafIndex, participationRoot)
|
|
}
|
|
|
|
// Field(16): CurrentEpochParticipation
|
|
if b.isLeafDirty(CurrentEpochParticipationLeafIndex) {
|
|
participationRoot, err := state_encoding.ParticipationBitsRoot(b.currentEpochParticipation)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(CurrentEpochParticipationLeafIndex, participationRoot)
|
|
}
|
|
|
|
// Field(17): JustificationBits
|
|
if b.isLeafDirty(JustificationBitsLeafIndex) {
|
|
var root [32]byte
|
|
copy(root[:], b.justificationBits)
|
|
b.updateLeaf(JustificationBitsLeafIndex, root)
|
|
}
|
|
|
|
// Field(18): PreviousJustifiedCheckpoint
|
|
if b.isLeafDirty(PreviousJustifiedCheckpointLeafIndex) {
|
|
checkpointRoot, err := b.previousJustifiedCheckpoint.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(PreviousJustifiedCheckpointLeafIndex, checkpointRoot)
|
|
}
|
|
|
|
// Field(19): CurrentJustifiedCheckpoint
|
|
if b.isLeafDirty(CurrentJustifiedCheckpointLeafIndex) {
|
|
checkpointRoot, err := b.currentJustifiedCheckpoint.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(CurrentJustifiedCheckpointLeafIndex, checkpointRoot)
|
|
}
|
|
|
|
// Field(20): FinalizedCheckpoint
|
|
if b.isLeafDirty(FinalizedCheckpointLeafIndex) {
|
|
checkpointRoot, err := b.finalizedCheckpoint.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(FinalizedCheckpointLeafIndex, checkpointRoot)
|
|
}
|
|
|
|
// Field(21): Inactivity Scores
|
|
if b.isLeafDirty(InactivityScoresLeafIndex) {
|
|
scoresRoot, err := state_encoding.Uint64ListRootWithLimit(b.inactivityScores, state_encoding.ValidatorLimitForBalancesChunks())
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(InactivityScoresLeafIndex, scoresRoot)
|
|
}
|
|
|
|
// Field(22): CurrentSyncCommitte
|
|
if b.isLeafDirty(CurrentSyncCommitteeLeafIndex) {
|
|
committeeRoot, err := b.currentSyncCommittee.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(CurrentSyncCommitteeLeafIndex, committeeRoot)
|
|
}
|
|
|
|
// Field(23): NextSyncCommitte
|
|
if b.isLeafDirty(NextSyncCommitteeLeafIndex) {
|
|
committeeRoot, err := b.nextSyncCommittee.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(NextSyncCommitteeLeafIndex, committeeRoot)
|
|
}
|
|
|
|
// Field(24): LatestExecutionPayloadHeader
|
|
if b.isLeafDirty(LatestExecutionPayloadHeaderLeafIndex) {
|
|
headerRoot, err := b.latestExecutionPayloadHeader.HashTreeRoot()
|
|
if err != nil {
|
|
return err
|
|
}
|
|
b.updateLeaf(LatestExecutionPayloadHeaderLeafIndex, headerRoot)
|
|
}
|
|
return nil
|
|
}
|
|
|
|
func (b *BeaconState) updateLeaf(idx StateLeafIndex, leaf common.Hash) {
|
|
// Update leaf with new value.
|
|
b.leaves[idx] = leaf
|
|
// Now leaf is clean :).
|
|
b.touchedLeaves[idx] = false
|
|
}
|
|
|
|
func (b *BeaconState) isLeafDirty(idx StateLeafIndex) bool {
|
|
// If leaf is non-initialized or if it was touched then we change it.
|
|
touched, isInitialized := b.touchedLeaves[idx]
|
|
return !isInitialized || touched // change only if the leaf was touched or root is non-initialized.
|
|
}
|