mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-08 02:31:19 +00:00
332 lines
13 KiB
Go
332 lines
13 KiB
Go
package altair
|
|
|
|
import (
|
|
"bytes"
|
|
"context"
|
|
"fmt"
|
|
"math/big"
|
|
|
|
"github.com/pkg/errors"
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/blocks"
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/helpers"
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/core/time"
|
|
"github.com/prysmaticlabs/prysm/v5/beacon-chain/state"
|
|
"github.com/prysmaticlabs/prysm/v5/config/params"
|
|
consensusblocks "github.com/prysmaticlabs/prysm/v5/consensus-types/blocks"
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/interfaces"
|
|
"github.com/prysmaticlabs/prysm/v5/consensus-types/primitives"
|
|
ethpb "github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1"
|
|
"github.com/prysmaticlabs/prysm/v5/proto/prysm/v1alpha1/attestation"
|
|
"go.opencensus.io/trace"
|
|
)
|
|
|
|
// ProcessAttestationsNoVerifySignature applies processing operations to a block's inner attestation
|
|
// records. The only difference would be that the attestation signature would not be verified.
|
|
func ProcessAttestationsNoVerifySignature(
|
|
ctx context.Context,
|
|
beaconState state.BeaconState,
|
|
b interfaces.ReadOnlyBeaconBlock,
|
|
) (state.BeaconState, error) {
|
|
if b == nil || b.IsNil() {
|
|
return nil, consensusblocks.ErrNilBeaconBlock
|
|
}
|
|
body := b.Body()
|
|
totalBalance, err := helpers.TotalActiveBalance(beaconState)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
for idx, att := range body.Attestations() {
|
|
beaconState, err = ProcessAttestationNoVerifySignature(ctx, beaconState, att, totalBalance)
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not verify attestation at index %d in block", idx)
|
|
}
|
|
}
|
|
return beaconState, nil
|
|
}
|
|
|
|
// ProcessAttestationNoVerifySignature processes the attestation without verifying the attestation signature. This
|
|
// method is used to validate attestations whose signatures have already been verified or will be verified later.
|
|
func ProcessAttestationNoVerifySignature(
|
|
ctx context.Context,
|
|
beaconState state.BeaconState,
|
|
att *ethpb.Attestation,
|
|
totalBalance *big.Int,
|
|
) (state.BeaconState, error) {
|
|
ctx, span := trace.StartSpan(ctx, "altair.ProcessAttestationNoVerifySignature")
|
|
defer span.End()
|
|
|
|
if err := blocks.VerifyAttestationNoVerifySignature(ctx, beaconState, att); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
delay, err := beaconState.Slot().SafeSubSlot(att.Data.Slot)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("att slot %d can't be greater than state slot %d", att.Data.Slot, beaconState.Slot())
|
|
}
|
|
participatedFlags, err := AttestationParticipationFlagIndices(beaconState, att.Data, delay)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
committee, err := helpers.BeaconCommitteeFromState(ctx, beaconState, att.Data.Slot, att.Data.CommitteeIndex)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
indices, err := attestation.AttestingIndices(att.AggregationBits, committee)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return SetParticipationAndRewardProposer(ctx, beaconState, att.Data.Target.Epoch, indices, participatedFlags, totalBalance)
|
|
}
|
|
|
|
// SetParticipationAndRewardProposer retrieves and sets the epoch participation bits in state. Based on the epoch participation, it rewards
|
|
// the proposer in state.
|
|
//
|
|
// Spec code:
|
|
//
|
|
// # Update epoch participation flags
|
|
// if data.target.epoch == get_current_epoch(state):
|
|
// epoch_participation = state.current_epoch_participation
|
|
// else:
|
|
// epoch_participation = state.previous_epoch_participation
|
|
//
|
|
// proposer_reward_numerator = 0
|
|
// for index in get_attesting_indices(state, data, attestation.aggregation_bits):
|
|
// for flag_index, weight in enumerate(PARTICIPATION_FLAG_WEIGHTS):
|
|
// if flag_index in participation_flag_indices and not has_flag(epoch_participation[index], flag_index):
|
|
// epoch_participation[index] = add_flag(epoch_participation[index], flag_index)
|
|
// proposer_reward_numerator += get_base_reward(state, index) * weight
|
|
//
|
|
// # Reward proposer
|
|
// proposer_reward_denominator = (WEIGHT_DENOMINATOR - PROPOSER_WEIGHT) * WEIGHT_DENOMINATOR // PROPOSER_WEIGHT
|
|
// proposer_reward = Gwei(proposer_reward_numerator // proposer_reward_denominator)
|
|
// increase_balance(state, get_beacon_proposer_index(state), proposer_reward)
|
|
func SetParticipationAndRewardProposer(
|
|
ctx context.Context,
|
|
beaconState state.BeaconState,
|
|
targetEpoch primitives.Epoch,
|
|
indices []uint64,
|
|
participatedFlags map[uint8]bool, totalBalance *big.Int) (state.BeaconState, error) {
|
|
var proposerRewardNumerator uint64
|
|
currentEpoch := time.CurrentEpoch(beaconState)
|
|
var stateErr error
|
|
if targetEpoch == currentEpoch {
|
|
stateErr = beaconState.ModifyCurrentParticipationBits(func(val []byte) ([]byte, error) {
|
|
propRewardNum, epochParticipation, err := EpochParticipation(beaconState, indices, val, participatedFlags, totalBalance)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
proposerRewardNumerator = propRewardNum
|
|
return epochParticipation, nil
|
|
})
|
|
} else {
|
|
stateErr = beaconState.ModifyPreviousParticipationBits(func(val []byte) ([]byte, error) {
|
|
propRewardNum, epochParticipation, err := EpochParticipation(beaconState, indices, val, participatedFlags, totalBalance)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
proposerRewardNumerator = propRewardNum
|
|
return epochParticipation, nil
|
|
})
|
|
}
|
|
if stateErr != nil {
|
|
return nil, stateErr
|
|
}
|
|
|
|
if err := RewardProposer(ctx, beaconState, proposerRewardNumerator); err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
return beaconState, nil
|
|
}
|
|
|
|
// HasValidatorFlag returns true if the flag at position has set.
|
|
func HasValidatorFlag(flag, flagPosition uint8) (bool, error) {
|
|
if flagPosition > 7 {
|
|
return false, errors.New("flag position exceeds length")
|
|
}
|
|
return ((flag >> flagPosition) & 1) == 1, nil
|
|
}
|
|
|
|
// AddValidatorFlag adds new validator flag to existing one.
|
|
func AddValidatorFlag(flag, flagPosition uint8) (uint8, error) {
|
|
if flagPosition > 7 {
|
|
return flag, errors.New("flag position exceeds length")
|
|
}
|
|
return flag | (1 << flagPosition), nil
|
|
}
|
|
|
|
// EpochParticipation sets and returns the proposer reward numerator and epoch participation.
|
|
//
|
|
// Spec code:
|
|
//
|
|
// proposer_reward_numerator = 0
|
|
// for index in get_attesting_indices(state, data, attestation.aggregation_bits):
|
|
// for flag_index, weight in enumerate(PARTICIPATION_FLAG_WEIGHTS):
|
|
// if flag_index in participation_flag_indices and not has_flag(epoch_participation[index], flag_index):
|
|
// epoch_participation[index] = add_flag(epoch_participation[index], flag_index)
|
|
// proposer_reward_numerator += get_base_reward(state, index) * weight
|
|
func EpochParticipation(beaconState state.BeaconState, indices []uint64, epochParticipation []byte, participatedFlags map[uint8]bool, totalBalance *big.Int) (uint64, []byte, error) {
|
|
cfg := params.BeaconConfig()
|
|
sourceFlagIndex := cfg.TimelySourceFlagIndex
|
|
targetFlagIndex := cfg.TimelyTargetFlagIndex
|
|
headFlagIndex := cfg.TimelyHeadFlagIndex
|
|
proposerRewardNumerator := uint64(0)
|
|
for _, index := range indices {
|
|
if index >= uint64(len(epochParticipation)) {
|
|
return 0, nil, fmt.Errorf("index %d exceeds participation length %d", index, len(epochParticipation))
|
|
}
|
|
br, err := BaseRewardWithTotalBalance(beaconState, primitives.ValidatorIndex(index), totalBalance)
|
|
if err != nil {
|
|
return 0, nil, err
|
|
}
|
|
has, err := HasValidatorFlag(epochParticipation[index], sourceFlagIndex)
|
|
if err != nil {
|
|
return 0, nil, err
|
|
}
|
|
if participatedFlags[sourceFlagIndex] && !has {
|
|
epochParticipation[index], err = AddValidatorFlag(epochParticipation[index], sourceFlagIndex)
|
|
if err != nil {
|
|
return 0, nil, err
|
|
}
|
|
proposerRewardNumerator += br * cfg.TimelySourceWeight
|
|
}
|
|
has, err = HasValidatorFlag(epochParticipation[index], targetFlagIndex)
|
|
if err != nil {
|
|
return 0, nil, err
|
|
}
|
|
if participatedFlags[targetFlagIndex] && !has {
|
|
epochParticipation[index], err = AddValidatorFlag(epochParticipation[index], targetFlagIndex)
|
|
if err != nil {
|
|
return 0, nil, err
|
|
}
|
|
proposerRewardNumerator += br * cfg.TimelyTargetWeight
|
|
}
|
|
has, err = HasValidatorFlag(epochParticipation[index], headFlagIndex)
|
|
if err != nil {
|
|
return 0, nil, err
|
|
}
|
|
if participatedFlags[headFlagIndex] && !has {
|
|
epochParticipation[index], err = AddValidatorFlag(epochParticipation[index], headFlagIndex)
|
|
if err != nil {
|
|
return 0, nil, err
|
|
}
|
|
proposerRewardNumerator += br * cfg.TimelyHeadWeight
|
|
}
|
|
}
|
|
|
|
return proposerRewardNumerator, epochParticipation, nil
|
|
}
|
|
|
|
// RewardProposer rewards proposer by increasing proposer's balance with input reward numerator and calculated reward denominator.
|
|
//
|
|
// Spec code:
|
|
//
|
|
// proposer_reward_denominator = (WEIGHT_DENOMINATOR - PROPOSER_WEIGHT) * WEIGHT_DENOMINATOR // PROPOSER_WEIGHT
|
|
// proposer_reward = Gwei(proposer_reward_numerator // proposer_reward_denominator)
|
|
// increase_balance(state, get_beacon_proposer_index(state), proposer_reward)
|
|
func RewardProposer(ctx context.Context, beaconState state.BeaconState, proposerRewardNumerator uint64) error {
|
|
cfg := params.BeaconConfig()
|
|
d := (cfg.WeightDenominator - cfg.ProposerWeight) * cfg.WeightDenominator / cfg.ProposerWeight
|
|
proposerReward := proposerRewardNumerator / d
|
|
i, err := helpers.BeaconProposerIndex(ctx, beaconState)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return helpers.IncreaseBalance(beaconState, i, proposerReward, cfg.IsPulseChain())
|
|
}
|
|
|
|
// AttestationParticipationFlagIndices retrieves a map of attestation scoring based on Altair's participation flag indices.
|
|
// This is used to facilitate process attestation during state transition and during upgrade to altair state.
|
|
//
|
|
// Spec code:
|
|
// def get_attestation_participation_flag_indices(state: BeaconState,
|
|
//
|
|
// data: AttestationData,
|
|
// inclusion_delay: uint64) -> Sequence[int]:
|
|
// """
|
|
// Return the flag indices that are satisfied by an attestation.
|
|
// """
|
|
// if data.target.epoch == get_current_epoch(state):
|
|
// justified_checkpoint = state.current_justified_checkpoint
|
|
// else:
|
|
// justified_checkpoint = state.previous_justified_checkpoint
|
|
//
|
|
// # Matching roots
|
|
// is_matching_source = data.source == justified_checkpoint
|
|
// is_matching_target = is_matching_source and data.target.root == get_block_root(state, data.target.epoch)
|
|
// is_matching_head = is_matching_target and data.beacon_block_root == get_block_root_at_slot(state, data.slot)
|
|
// assert is_matching_source
|
|
//
|
|
// participation_flag_indices = []
|
|
// if is_matching_source and inclusion_delay <= integer_squareroot(SLOTS_PER_EPOCH):
|
|
// participation_flag_indices.append(TIMELY_SOURCE_FLAG_INDEX)
|
|
// if is_matching_target and inclusion_delay <= SLOTS_PER_EPOCH:
|
|
// participation_flag_indices.append(TIMELY_TARGET_FLAG_INDEX)
|
|
// if is_matching_head and inclusion_delay == MIN_ATTESTATION_INCLUSION_DELAY:
|
|
// participation_flag_indices.append(TIMELY_HEAD_FLAG_INDEX)
|
|
//
|
|
// return participation_flag_indices
|
|
func AttestationParticipationFlagIndices(beaconState state.BeaconState, data *ethpb.AttestationData, delay primitives.Slot) (map[uint8]bool, error) {
|
|
currEpoch := time.CurrentEpoch(beaconState)
|
|
var justifiedCheckpt *ethpb.Checkpoint
|
|
if data.Target.Epoch == currEpoch {
|
|
justifiedCheckpt = beaconState.CurrentJustifiedCheckpoint()
|
|
} else {
|
|
justifiedCheckpt = beaconState.PreviousJustifiedCheckpoint()
|
|
}
|
|
|
|
matchedSrc, matchedTgt, matchedHead, err := MatchingStatus(beaconState, data, justifiedCheckpt)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if !matchedSrc {
|
|
return nil, errors.New("source epoch does not match")
|
|
}
|
|
|
|
participatedFlags := make(map[uint8]bool)
|
|
cfg := params.BeaconConfig()
|
|
sourceFlagIndex := cfg.TimelySourceFlagIndex
|
|
targetFlagIndex := cfg.TimelyTargetFlagIndex
|
|
headFlagIndex := cfg.TimelyHeadFlagIndex
|
|
sqtRootSlots := cfg.SqrRootSlotsPerEpoch
|
|
if matchedSrc && delay <= sqtRootSlots {
|
|
participatedFlags[sourceFlagIndex] = true
|
|
}
|
|
matchedSrcTgt := matchedSrc && matchedTgt
|
|
// Before Deneb no attestation should pass validation without having delay <= slotsPerEpoch.
|
|
if matchedSrcTgt {
|
|
participatedFlags[targetFlagIndex] = true
|
|
}
|
|
matchedSrcTgtHead := matchedHead && matchedSrcTgt
|
|
if matchedSrcTgtHead && delay == cfg.MinAttestationInclusionDelay {
|
|
participatedFlags[headFlagIndex] = true
|
|
}
|
|
return participatedFlags, nil
|
|
}
|
|
|
|
// MatchingStatus returns the matching statues for attestation data's source target and head.
|
|
//
|
|
// Spec code:
|
|
//
|
|
// is_matching_source = data.source == justified_checkpoint
|
|
// is_matching_target = is_matching_source and data.target.root == get_block_root(state, data.target.epoch)
|
|
// is_matching_head = is_matching_target and data.beacon_block_root == get_block_root_at_slot(state, data.slot)
|
|
func MatchingStatus(beaconState state.BeaconState, data *ethpb.AttestationData, cp *ethpb.Checkpoint) (matchedSrc, matchedTgt, matchedHead bool, err error) {
|
|
matchedSrc = attestation.CheckPointIsEqual(data.Source, cp)
|
|
|
|
r, err := helpers.BlockRoot(beaconState, data.Target.Epoch)
|
|
if err != nil {
|
|
return false, false, false, err
|
|
}
|
|
matchedTgt = bytes.Equal(r, data.Target.Root)
|
|
|
|
r, err = helpers.BlockRootAtSlot(beaconState, data.Slot)
|
|
if err != nil {
|
|
return false, false, false, err
|
|
}
|
|
matchedHead = bytes.Equal(r, data.BeaconBlockRoot)
|
|
return
|
|
}
|