mirror of
https://gitlab.com/pulsechaincom/prysm-pulse.git
synced 2025-01-10 03:31:20 +00:00
44973b0bb3
* in progress... * in progress... * remove log * log root * Revert "Auxiliary commit to revert individual files from f12a609ea2a2f1e87e97321f3a717cd324b5ae97" This reverts commit 5ae35edb6477d8d0ea4e94b273efc6590484da85. * cleanup * remove log * remove whitespace * remove logs * more stuff * copy * always rebuild trie * revert * add state * init state * fix all * uintptr * move slice to new package * lock in `Detach` * remove constraint * reorder * blockroots and stateroots * fill roots in empty() * fix hasher * implement slice for balances and inactivity scores * detach in setters * Revert "implement slice for balances and inactivity scores" This reverts commit 59eb9df8d766cb1c44a7eb5b3f5e3c042249943d. # Conflicts: # beacon-chain/state/state-native/setters_validator.go * use counter to track states * typos * rename interface * balances * gauge * some improvements * first try with map * fix * inactivity scores in progress * fix test # Conflicts: # beacon-chain/state/state-native/helpers_test.go * test fixes * ToProto fix * copy roots * validators * build fixes * fix bug in `ToProto` * fix fuzz test * fix bug in slice getters * fix state equality checks * make tests pass * make tests pass * more test updates * Revert "Auxiliary commit to revert individual files from 34e7344bff08a589e6341bb1829e3cb74159e878" This reverts commit ecd64efa8917f37ca41460e0356ff007fe55dd9d. * Revert "make tests pass" This reverts commit 0cf00f19eecf4678cd2b866dd107f3179d0426ef. * Revert "make tests pass" This reverts commit 521b65e1d2e13be3d720f333008b6838a8e78878. * pass tests * deepequal identifiable types * Deflake `cloners_test.go` * feature flag for block roots * feature flag * remove recursive locks * reduce complexity of rootSelector * fix randao mixes root * some fixes * revisit tests * revert change to field trie helpers * initialize field map for tests * remove whitespace * initialize roots with proper length * more fixes * out of bounds message fix * optimize length calculation * remove call to Len in PubkeyAtIndex * don't log deposits * unit tests * unit tests * fix * comments * test fixes * id * remove Enumerator interface * review feedback * simplify field trie * bring back fieldtrie package * fix bazel file * use handle32ByteArrays for root computation * fix locks * metrics * bzl * simplify some things * use htr in state test * remove code from require package * gzl * more htr * Fuzzing of the multi-value slice * assert values * getter optimizations * use At when reading from validators * Nishant's review * restore safe copy * remove empty line * build fix * restore how we get root at index for deafult mode * more review comments * optimize default behavior * simplify Slice calls * test fix * remove unnecessary package * remove unused setter * make fieldMap unexported * some improvements in state package * call `Slice` instead of manually copying * unlock in ReadFromEveryValidator * Potuz's comments * lock the state when reading from all validators # Conflicts: # beacon-chain/state/state-native/getters_validator.go * add back preston's changes * add index --------- Co-authored-by: Potuz <potuz@prysmaticlabs.com> Co-authored-by: nisdas <nishdas93@gmail.com> Co-authored-by: Preston Van Loon <pvanloon@offchainlabs.com>
289 lines
9.8 KiB
Go
289 lines
9.8 KiB
Go
package transition
|
|
|
|
import (
|
|
"context"
|
|
|
|
"github.com/pkg/errors"
|
|
b "github.com/prysmaticlabs/prysm/v4/beacon-chain/core/blocks"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/core/helpers"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state"
|
|
state_native "github.com/prysmaticlabs/prysm/v4/beacon-chain/state/state-native"
|
|
"github.com/prysmaticlabs/prysm/v4/beacon-chain/state/stateutil"
|
|
fieldparams "github.com/prysmaticlabs/prysm/v4/config/fieldparams"
|
|
"github.com/prysmaticlabs/prysm/v4/config/params"
|
|
"github.com/prysmaticlabs/prysm/v4/container/trie"
|
|
ethpb "github.com/prysmaticlabs/prysm/v4/proto/prysm/v1alpha1"
|
|
)
|
|
|
|
// GenesisBeaconState gets called when MinGenesisActiveValidatorCount count of
|
|
// full deposits were made to the deposit contract and the ChainStart log gets emitted.
|
|
//
|
|
// Spec pseudocode definition:
|
|
//
|
|
// def initialize_beacon_state_from_eth1(eth1_block_hash: Bytes32,
|
|
// eth1_timestamp: uint64,
|
|
// deposits: Sequence[Deposit]) -> BeaconState:
|
|
// fork = Fork(
|
|
// previous_version=GENESIS_FORK_VERSION,
|
|
// current_version=GENESIS_FORK_VERSION,
|
|
// epoch=GENESIS_EPOCH,
|
|
// )
|
|
// state = BeaconState(
|
|
// genesis_time=eth1_timestamp + GENESIS_DELAY,
|
|
// fork=fork,
|
|
// eth1_data=Eth1Data(block_hash=eth1_block_hash, deposit_count=uint64(len(deposits))),
|
|
// latest_block_header=BeaconBlockHeader(body_root=hash_tree_root(BeaconBlockBody())),
|
|
// randao_mixes=[eth1_block_hash] * EPOCHS_PER_HISTORICAL_VECTOR, # Seed RANDAO with Eth1 entropy
|
|
// )
|
|
//
|
|
// # Process deposits
|
|
// leaves = list(map(lambda deposit: deposit.data, deposits))
|
|
// for index, deposit in enumerate(deposits):
|
|
// deposit_data_list = List[DepositData, 2**DEPOSIT_CONTRACT_TREE_DEPTH](*leaves[:index + 1])
|
|
// state.eth1_data.deposit_root = hash_tree_root(deposit_data_list)
|
|
// process_deposit(state, deposit)
|
|
//
|
|
// # Process activations
|
|
// for index, validator in enumerate(state.validators):
|
|
// balance = state.balances[index]
|
|
// validator.effective_balance = min(balance - balance % EFFECTIVE_BALANCE_INCREMENT, MAX_EFFECTIVE_BALANCE)
|
|
// if validator.effective_balance == MAX_EFFECTIVE_BALANCE:
|
|
// validator.activation_eligibility_epoch = GENESIS_EPOCH
|
|
// validator.activation_epoch = GENESIS_EPOCH
|
|
//
|
|
// # Set genesis validators root for domain separation and chain versioning
|
|
// state.genesis_validators_root = hash_tree_root(state.validators)
|
|
//
|
|
// return state
|
|
//
|
|
// This method differs from the spec so as to process deposits beforehand instead of the end of the function.
|
|
func GenesisBeaconState(ctx context.Context, deposits []*ethpb.Deposit, genesisTime uint64, eth1Data *ethpb.Eth1Data) (state.BeaconState, error) {
|
|
st, err := EmptyGenesisState()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Process initial deposits.
|
|
st, err = helpers.UpdateGenesisEth1Data(st, deposits, eth1Data)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
st, err = b.ProcessPreGenesisDeposits(ctx, st, deposits)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not process validator deposits")
|
|
}
|
|
|
|
return OptimizedGenesisBeaconState(genesisTime, st, st.Eth1Data())
|
|
}
|
|
|
|
// PreminedGenesisBeaconState works almost exactly like GenesisBeaconState, except that it assumes that genesis deposits
|
|
// are not represented in the deposit contract and are only found in the genesis state validator registry. In order
|
|
// to ensure the deposit root and count match the empty deposit contract deployed in a testnet genesis block, the root
|
|
// of an empty deposit trie is computed and used as Eth1Data.deposit_root, and the deposit count is set to 0.
|
|
func PreminedGenesisBeaconState(ctx context.Context, deposits []*ethpb.Deposit, genesisTime uint64, eth1Data *ethpb.Eth1Data) (state.BeaconState, error) {
|
|
st, err := EmptyGenesisState()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
|
|
// Process initial deposits.
|
|
st, err = helpers.UpdateGenesisEth1Data(st, deposits, eth1Data)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
st, err = b.ProcessPreGenesisDeposits(ctx, st, deposits)
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not process validator deposits")
|
|
}
|
|
|
|
t, err := trie.NewTrie(params.BeaconConfig().DepositContractTreeDepth)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
dr, err := t.HashTreeRoot()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
if err := st.SetEth1Data(ðpb.Eth1Data{DepositRoot: dr[:], BlockHash: eth1Data.BlockHash}); err != nil {
|
|
return nil, err
|
|
}
|
|
if err := st.SetEth1DepositIndex(0); err != nil {
|
|
return nil, err
|
|
}
|
|
return OptimizedGenesisBeaconState(genesisTime, st, st.Eth1Data())
|
|
}
|
|
|
|
// OptimizedGenesisBeaconState is used to create a state that has already processed deposits. This is to efficiently
|
|
// create a mainnet state at chainstart.
|
|
func OptimizedGenesisBeaconState(genesisTime uint64, preState state.BeaconState, eth1Data *ethpb.Eth1Data) (state.BeaconState, error) {
|
|
if eth1Data == nil {
|
|
return nil, errors.New("no eth1data provided for genesis state")
|
|
}
|
|
|
|
randaoMixes := make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector)
|
|
for i := 0; i < len(randaoMixes); i++ {
|
|
h := make([]byte, 32)
|
|
copy(h, eth1Data.BlockHash)
|
|
randaoMixes[i] = h
|
|
}
|
|
|
|
zeroHash := params.BeaconConfig().ZeroHash[:]
|
|
|
|
activeIndexRoots := make([][]byte, params.BeaconConfig().EpochsPerHistoricalVector)
|
|
for i := 0; i < len(activeIndexRoots); i++ {
|
|
activeIndexRoots[i] = zeroHash
|
|
}
|
|
|
|
blockRoots := make([][]byte, params.BeaconConfig().SlotsPerHistoricalRoot)
|
|
for i := 0; i < len(blockRoots); i++ {
|
|
blockRoots[i] = zeroHash
|
|
}
|
|
|
|
stateRoots := make([][]byte, params.BeaconConfig().SlotsPerHistoricalRoot)
|
|
for i := 0; i < len(stateRoots); i++ {
|
|
stateRoots[i] = zeroHash
|
|
}
|
|
|
|
slashings := make([]uint64, params.BeaconConfig().EpochsPerSlashingsVector)
|
|
|
|
genesisValidatorsRoot, err := stateutil.ValidatorRegistryRoot(preState.Validators())
|
|
if err != nil {
|
|
return nil, errors.Wrapf(err, "could not hash tree root genesis validators %v", err)
|
|
}
|
|
|
|
st := ðpb.BeaconState{
|
|
// Misc fields.
|
|
Slot: 0,
|
|
GenesisTime: genesisTime,
|
|
GenesisValidatorsRoot: genesisValidatorsRoot[:],
|
|
|
|
Fork: ðpb.Fork{
|
|
PreviousVersion: params.BeaconConfig().GenesisForkVersion,
|
|
CurrentVersion: params.BeaconConfig().GenesisForkVersion,
|
|
Epoch: 0,
|
|
},
|
|
|
|
// Validator registry fields.
|
|
Validators: preState.Validators(),
|
|
Balances: preState.Balances(),
|
|
|
|
// Randomness and committees.
|
|
RandaoMixes: randaoMixes,
|
|
|
|
// Finality.
|
|
PreviousJustifiedCheckpoint: ðpb.Checkpoint{
|
|
Epoch: 0,
|
|
Root: params.BeaconConfig().ZeroHash[:],
|
|
},
|
|
CurrentJustifiedCheckpoint: ðpb.Checkpoint{
|
|
Epoch: 0,
|
|
Root: params.BeaconConfig().ZeroHash[:],
|
|
},
|
|
JustificationBits: []byte{0},
|
|
FinalizedCheckpoint: ðpb.Checkpoint{
|
|
Epoch: 0,
|
|
Root: params.BeaconConfig().ZeroHash[:],
|
|
},
|
|
|
|
HistoricalRoots: [][]byte{},
|
|
BlockRoots: blockRoots,
|
|
StateRoots: stateRoots,
|
|
Slashings: slashings,
|
|
CurrentEpochAttestations: []*ethpb.PendingAttestation{},
|
|
PreviousEpochAttestations: []*ethpb.PendingAttestation{},
|
|
|
|
// Eth1 data.
|
|
Eth1Data: eth1Data,
|
|
Eth1DataVotes: []*ethpb.Eth1Data{},
|
|
Eth1DepositIndex: preState.Eth1DepositIndex(),
|
|
}
|
|
|
|
bodyRoot, err := (ðpb.BeaconBlockBody{
|
|
RandaoReveal: make([]byte, 96),
|
|
Eth1Data: ðpb.Eth1Data{
|
|
DepositRoot: make([]byte, 32),
|
|
BlockHash: make([]byte, 32),
|
|
},
|
|
Graffiti: make([]byte, 32),
|
|
}).HashTreeRoot()
|
|
if err != nil {
|
|
return nil, errors.Wrap(err, "could not hash tree root empty block body")
|
|
}
|
|
|
|
st.LatestBlockHeader = ðpb.BeaconBlockHeader{
|
|
ParentRoot: zeroHash,
|
|
StateRoot: zeroHash,
|
|
BodyRoot: bodyRoot[:],
|
|
}
|
|
|
|
return state_native.InitializeFromProtoPhase0(st)
|
|
}
|
|
|
|
// EmptyGenesisState returns an empty beacon state object.
|
|
func EmptyGenesisState() (state.BeaconState, error) {
|
|
blockRoots := make([][]byte, fieldparams.BlockRootsLength)
|
|
for i := range blockRoots {
|
|
blockRoots[i] = make([]byte, fieldparams.RootLength)
|
|
}
|
|
stateRoots := make([][]byte, fieldparams.StateRootsLength)
|
|
for i := range stateRoots {
|
|
stateRoots[i] = make([]byte, fieldparams.RootLength)
|
|
}
|
|
mixes := make([][]byte, fieldparams.RandaoMixesLength)
|
|
for i := range mixes {
|
|
mixes[i] = make([]byte, fieldparams.RootLength)
|
|
}
|
|
st := ðpb.BeaconState{
|
|
// Misc fields.
|
|
Slot: 0,
|
|
Fork: ðpb.Fork{
|
|
PreviousVersion: params.BeaconConfig().GenesisForkVersion,
|
|
CurrentVersion: params.BeaconConfig().GenesisForkVersion,
|
|
Epoch: 0,
|
|
},
|
|
BlockRoots: blockRoots,
|
|
StateRoots: stateRoots,
|
|
RandaoMixes: mixes,
|
|
// Validator registry fields.
|
|
Validators: []*ethpb.Validator{},
|
|
Balances: []uint64{},
|
|
|
|
JustificationBits: []byte{0},
|
|
HistoricalRoots: [][]byte{},
|
|
CurrentEpochAttestations: []*ethpb.PendingAttestation{},
|
|
PreviousEpochAttestations: []*ethpb.PendingAttestation{},
|
|
|
|
// Eth1 data.
|
|
Eth1Data: ðpb.Eth1Data{},
|
|
Eth1DataVotes: []*ethpb.Eth1Data{},
|
|
Eth1DepositIndex: 0,
|
|
}
|
|
return state_native.InitializeFromProtoPhase0(st)
|
|
}
|
|
|
|
// IsValidGenesisState gets called whenever there's a deposit event,
|
|
// it checks whether there's enough effective balance to trigger and
|
|
// if the minimum genesis time arrived already.
|
|
//
|
|
// Spec pseudocode definition:
|
|
//
|
|
// def is_valid_genesis_state(state: BeaconState) -> bool:
|
|
// if state.genesis_time < MIN_GENESIS_TIME:
|
|
// return False
|
|
// if len(get_active_validator_indices(state, GENESIS_EPOCH)) < MIN_GENESIS_ACTIVE_VALIDATOR_COUNT:
|
|
// return False
|
|
// return True
|
|
//
|
|
// This method has been modified from the spec to allow whole states not to be saved
|
|
// but instead only cache the relevant information.
|
|
func IsValidGenesisState(chainStartDepositCount, currentTime uint64) bool {
|
|
if currentTime < params.BeaconConfig().MinGenesisTime {
|
|
return false
|
|
}
|
|
if chainStartDepositCount < params.BeaconConfig().MinGenesisActiveValidatorCount {
|
|
return false
|
|
}
|
|
return true
|
|
}
|