use super::per_block_processing::{errors::BlockProcessingError, process_deposit}; use tree_hash::TreeHash; use types::typenum::U4294967296; use types::*; /// Initialize a `BeaconState` from genesis data. /// /// Spec v0.9.1 // TODO: this is quite inefficient and we probably want to rethink how we do this pub fn initialize_beacon_state_from_eth1( eth1_block_hash: Hash256, eth1_timestamp: u64, deposits: Vec, spec: &ChainSpec, ) -> Result, BlockProcessingError> { let genesis_time = eth1_timestamp - eth1_timestamp % spec.seconds_per_day + 2 * spec.seconds_per_day; let eth1_data = Eth1Data { // Temporary deposit root deposit_root: Hash256::zero(), deposit_count: deposits.len() as u64, block_hash: eth1_block_hash, }; let mut state = BeaconState::new(genesis_time, eth1_data, spec); // Seed RANDAO with Eth1 entropy state.fill_randao_mixes_with(eth1_block_hash); // Process deposits let leaves: Vec<_> = deposits .iter() .map(|deposit| deposit.data.clone()) .collect(); for (index, deposit) in deposits.into_iter().enumerate() { let deposit_data_list = VariableList::<_, U4294967296>::from(leaves[..=index].to_vec()); state.eth1_data.deposit_root = Hash256::from_slice(&deposit_data_list.tree_hash_root()); process_deposit(&mut state, &deposit, spec, true)?; } process_activations(&mut state, spec); // Now that we have our validators, initialize the caches (including the committees) state.build_all_caches(spec)?; Ok(state) } /// Determine whether a candidate genesis state is suitable for starting the chain. /// /// Spec v0.9.1 pub fn is_valid_genesis_state(state: &BeaconState, spec: &ChainSpec) -> bool { state.genesis_time >= spec.min_genesis_time && state.get_active_validator_indices(T::genesis_epoch()).len() as u64 >= spec.min_genesis_active_validator_count } /// Activate genesis validators, if their balance is acceptable. /// /// Spec v0.8.0 pub fn process_activations(state: &mut BeaconState, spec: &ChainSpec) { for (index, validator) in state.validators.iter_mut().enumerate() { let balance = state.balances[index]; validator.effective_balance = std::cmp::min( balance - balance % spec.effective_balance_increment, spec.max_effective_balance, ); if validator.effective_balance == spec.max_effective_balance { validator.activation_eligibility_epoch = T::genesis_epoch(); validator.activation_epoch = T::genesis_epoch(); } } }