package state import ( "fmt" "github.com/ethereum/go-ethereum/common" "github.com/prysmaticlabs/prysm/beacon-chain/core/incentives" "github.com/prysmaticlabs/prysm/beacon-chain/core/types" v "github.com/prysmaticlabs/prysm/beacon-chain/core/validators" "github.com/prysmaticlabs/prysm/beacon-chain/utils" pb "github.com/prysmaticlabs/prysm/proto/beacon/p2p/v1" "github.com/prysmaticlabs/prysm/shared/params" ) // NewStateTransition computes the new beacon state, given the previous beacon state // a beacon block, and its parent slot. This method is called during a cycle transition. // We also check for validator set change transition and compute for new // committees if necessary during this transition. func NewStateTransition( st *types.BeaconState, block *types.Block, parentSlot uint64, blockVoteCache utils.BlockVoteCache, ) (*types.BeaconState, error) { var lastStateRecalculationSlotCycleBack uint64 var err error newState := st.CopyState() justifiedStreak := st.JustifiedStreak() justifiedSlot := st.LastJustifiedSlot() finalizedSlot := st.LastFinalizedSlot() timeSinceFinality := block.SlotNumber() - newState.LastFinalizedSlot() newState.SetValidators(v.CopyValidators(newState.Validators())) newState.ClearAttestations(st.LastStateRecalculationSlot()) // Derive the new set of recent block hashes. recentBlockHashes, err := st.CalculateNewBlockHashes(block, parentSlot) if err != nil { return nil, err } newState.SetLatestBlockHashes(recentBlockHashes) if err != nil { return nil, fmt.Errorf("failed to update recent block hashes: %v", err) } newRandao := createRandaoMix(block.RandaoReveal(), st.RandaoMix()) newState.SetRandaoMix(newRandao[:]) // The changes below are only applied if this is a cycle transition. if block.SlotNumber()%params.BeaconConfig().CycleLength == 0 { if st.LastStateRecalculationSlot() < params.BeaconConfig().CycleLength { lastStateRecalculationSlotCycleBack = 0 } else { lastStateRecalculationSlotCycleBack = st.LastStateRecalculationSlot() - params.BeaconConfig().CycleLength } // walk through all the slots from LastStateRecalculationSlot - cycleLength to // LastStateRecalculationSlot - 1. for i := uint64(0); i < params.BeaconConfig().CycleLength; i++ { var blockVoteBalance uint64 slot := lastStateRecalculationSlotCycleBack + i blockHash := recentBlockHashes[i] blockVoteBalance, validators := incentives.TallyVoteBalances( common.BytesToHash(blockHash), blockVoteCache, newState.Validators(), v.ActiveValidatorIndices(newState.Validators()), v.TotalActiveValidatorDeposit(newState.Validators()), timeSinceFinality, ) newState.SetValidators(validators) justifiedSlot, finalizedSlot, justifiedStreak = FinalizeAndJustifySlots( slot, justifiedSlot, finalizedSlot, justifiedStreak, blockVoteBalance, v.TotalActiveValidatorDeposit(st.Validators()), ) } crossLinks, err := crossLinkCalculations( newState, st.PendingAttestations(), block.SlotNumber(), ) if err != nil { return nil, err } newState.SetCrossLinks(crossLinks) newState.SetLastJustifiedSlot(justifiedSlot) newState.SetLastFinalizedSlot(finalizedSlot) newState.SetJustifiedStreak(justifiedStreak) // Exit the validators when their balance fall below min online deposit size. newState.SetValidators(v.CheckValidatorMinDeposit(newState.Validators(), block.SlotNumber())) // Entering new validator set change transition. if newState.IsValidatorSetChange(block.SlotNumber()) { newState.SetValidatorSetChangeSlot(newState.LastStateRecalculationSlot()) shardAndCommitteesForSlots, err := validatorSetRecalculations( newState.ShardAndCommitteesForSlots(), newState.Validators(), block.ParentHash(), ) if err != nil { return nil, err } newState.SetShardAndCommitteesForSlots(shardAndCommitteesForSlots) period := block.SlotNumber() / params.BeaconConfig().MinWithdrawalPeriod totalPenalties := newState.PenalizedETH(period) newState.SetValidators(v.ChangeValidators(block.SlotNumber(), totalPenalties, newState.Validators())) } } newState.SetLastStateRecalculationSlot(newState.LastStateRecalculationSlot() + 1) return newState, nil } // crossLinkCalculations checks if the proposed shard block has recevied // 2/3 of the votes. If yes, we update crosslink record to point to // the proposed shard block with latest beacon chain slot numbers. func crossLinkCalculations( st *types.BeaconState, pendingAttestations []*pb.AggregatedAttestation, currentSlot uint64, ) ([]*pb.CrosslinkRecord, error) { slot := st.LastStateRecalculationSlot() + params.BeaconConfig().CycleLength crossLinkRecords := st.Crosslinks() for _, attestation := range pendingAttestations { shardCommittees, err := v.GetShardAndCommitteesForSlot( st.ShardAndCommitteesForSlots(), st.LastStateRecalculationSlot(), attestation.GetSlot(), ) if err != nil { return nil, err } indices, err := v.AttesterIndices(shardCommittees, attestation) if err != nil { return nil, err } totalBalance, voteBalance, err := v.VotedBalanceInAttestation(st.Validators(), indices, attestation) if err != nil { return nil, err } newValidatorSet, err := incentives.ApplyCrosslinkRewardsAndPenalties( crossLinkRecords, currentSlot, indices, attestation, st.Validators(), v.TotalActiveValidatorDeposit(st.Validators()), totalBalance, voteBalance, ) if err != nil { return nil, err } st.SetValidators(newValidatorSet) crossLinkRecords = UpdateCrosslinks(slot, voteBalance, totalBalance, attestation, crossLinkRecords) } return crossLinkRecords, nil } // validatorSetRecalculation recomputes the validator set. func validatorSetRecalculations( shardAndCommittesForSlots []*pb.ShardAndCommitteeArray, validators []*pb.ValidatorRecord, seed [32]byte, ) ([]*pb.ShardAndCommitteeArray, error) { lastSlot := len(shardAndCommittesForSlots) - 1 lastCommitteeFromLastSlot := len(shardAndCommittesForSlots[lastSlot].ArrayShardAndCommittee) - 1 crosslinkLastShard := shardAndCommittesForSlots[lastSlot].ArrayShardAndCommittee[lastCommitteeFromLastSlot].Shard crosslinkNextShard := (crosslinkLastShard + 1) % params.BeaconConfig().ShardCount newShardCommitteeArray, err := v.ShuffleValidatorsToCommittees( seed, validators, crosslinkNextShard, ) if err != nil { return nil, err } return append(shardAndCommittesForSlots[params.BeaconConfig().CycleLength:], newShardCommitteeArray...), nil } // createRandaoMix sets the block randao seed into a beacon state randao. This function // XOR's the current state randao with the block's randao value added by the // proposer. func createRandaoMix(blockRandao [32]byte, beaconStateRandao [32]byte) [32]byte { for i, b := range blockRandao { beaconStateRandao[i] ^= b } return beaconStateRandao }