mirror of
https://github.com/sigp/lighthouse.git
synced 2026-05-07 00:42:42 +00:00
Prevent duplicate effective balance processing (#7209)
This commit is contained in:
@@ -175,6 +175,7 @@ pub fn process_epoch_single_pass<E: EthSpec>(
|
|||||||
|
|
||||||
let mut earliest_exit_epoch = state.earliest_exit_epoch().ok();
|
let mut earliest_exit_epoch = state.earliest_exit_epoch().ok();
|
||||||
let mut exit_balance_to_consume = state.exit_balance_to_consume().ok();
|
let mut exit_balance_to_consume = state.exit_balance_to_consume().ok();
|
||||||
|
let validators_in_consolidations = get_validators_in_consolidations(state);
|
||||||
|
|
||||||
// Split the state into several disjoint mutable borrows.
|
// Split the state into several disjoint mutable borrows.
|
||||||
let (
|
let (
|
||||||
@@ -317,17 +318,26 @@ pub fn process_epoch_single_pass<E: EthSpec>(
|
|||||||
|
|
||||||
// `process_effective_balance_updates`
|
// `process_effective_balance_updates`
|
||||||
if conf.effective_balance_updates {
|
if conf.effective_balance_updates {
|
||||||
process_single_effective_balance_update(
|
if validators_in_consolidations.contains(&validator_info.index) {
|
||||||
validator_info.index,
|
process_single_dummy_effective_balance_update(
|
||||||
*balance,
|
validator_info.index,
|
||||||
&mut validator,
|
&validator,
|
||||||
validator_info.current_epoch_participation,
|
&mut next_epoch_cache,
|
||||||
&mut next_epoch_cache,
|
state_ctxt,
|
||||||
progressive_balances,
|
)?;
|
||||||
effective_balances_ctxt,
|
} else {
|
||||||
state_ctxt,
|
process_single_effective_balance_update(
|
||||||
spec,
|
validator_info.index,
|
||||||
)?;
|
*balance,
|
||||||
|
&mut validator,
|
||||||
|
validator_info.current_epoch_participation,
|
||||||
|
&mut next_epoch_cache,
|
||||||
|
progressive_balances,
|
||||||
|
effective_balances_ctxt,
|
||||||
|
state_ctxt,
|
||||||
|
spec,
|
||||||
|
)?;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -430,6 +440,7 @@ pub fn process_epoch_single_pass<E: EthSpec>(
|
|||||||
if fork_name.electra_enabled() && conf.pending_consolidations {
|
if fork_name.electra_enabled() && conf.pending_consolidations {
|
||||||
process_pending_consolidations(
|
process_pending_consolidations(
|
||||||
state,
|
state,
|
||||||
|
&validators_in_consolidations,
|
||||||
&mut next_epoch_cache,
|
&mut next_epoch_cache,
|
||||||
effective_balances_ctxt,
|
effective_balances_ctxt,
|
||||||
conf.effective_balance_updates,
|
conf.effective_balance_updates,
|
||||||
@@ -1026,12 +1037,38 @@ fn process_pending_deposits_for_validator(
|
|||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Return the set of validators referenced by consolidations, either as source or target.
|
||||||
|
///
|
||||||
|
/// This function is blind to whether the consolidations are valid and capable of being processed,
|
||||||
|
/// it just returns the set of all indices present in consolidations. This is *sufficient* to
|
||||||
|
/// make consolidations play nicely with effective balance updates. The algorithm used is:
|
||||||
|
///
|
||||||
|
/// - In the single pass: apply effective balance updates for all validators *not* referenced by
|
||||||
|
/// consolidations.
|
||||||
|
/// - Apply consolidations.
|
||||||
|
/// - Apply effective balance updates for all validators previously skipped.
|
||||||
|
///
|
||||||
|
/// Prior to Electra, the empty set is returned.
|
||||||
|
fn get_validators_in_consolidations<E: EthSpec>(state: &BeaconState<E>) -> BTreeSet<usize> {
|
||||||
|
let mut referenced_validators = BTreeSet::new();
|
||||||
|
|
||||||
|
if let Ok(pending_consolidations) = state.pending_consolidations() {
|
||||||
|
for pending_consolidation in pending_consolidations {
|
||||||
|
referenced_validators.insert(pending_consolidation.source_index as usize);
|
||||||
|
referenced_validators.insert(pending_consolidation.target_index as usize);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
referenced_validators
|
||||||
|
}
|
||||||
|
|
||||||
/// We process pending consolidations after all of single-pass epoch processing, and then patch up
|
/// We process pending consolidations after all of single-pass epoch processing, and then patch up
|
||||||
/// the effective balances for affected validators.
|
/// the effective balances for affected validators.
|
||||||
///
|
///
|
||||||
/// This is safe because processing consolidations does not depend on the `effective_balance`.
|
/// This is safe because processing consolidations does not depend on the `effective_balance`.
|
||||||
fn process_pending_consolidations<E: EthSpec>(
|
fn process_pending_consolidations<E: EthSpec>(
|
||||||
state: &mut BeaconState<E>,
|
state: &mut BeaconState<E>,
|
||||||
|
validators_in_consolidations: &BTreeSet<usize>,
|
||||||
next_epoch_cache: &mut PreEpochCache,
|
next_epoch_cache: &mut PreEpochCache,
|
||||||
effective_balances_ctxt: &EffectiveBalancesContext,
|
effective_balances_ctxt: &EffectiveBalancesContext,
|
||||||
perform_effective_balance_updates: bool,
|
perform_effective_balance_updates: bool,
|
||||||
@@ -1042,8 +1079,6 @@ fn process_pending_consolidations<E: EthSpec>(
|
|||||||
let next_epoch = state.next_epoch()?;
|
let next_epoch = state.next_epoch()?;
|
||||||
let pending_consolidations = state.pending_consolidations()?.clone();
|
let pending_consolidations = state.pending_consolidations()?.clone();
|
||||||
|
|
||||||
let mut affected_validators = BTreeSet::new();
|
|
||||||
|
|
||||||
for pending_consolidation in &pending_consolidations {
|
for pending_consolidation in &pending_consolidations {
|
||||||
let source_index = pending_consolidation.source_index as usize;
|
let source_index = pending_consolidation.source_index as usize;
|
||||||
let target_index = pending_consolidation.target_index as usize;
|
let target_index = pending_consolidation.target_index as usize;
|
||||||
@@ -1069,9 +1104,6 @@ fn process_pending_consolidations<E: EthSpec>(
|
|||||||
decrease_balance(state, source_index, source_effective_balance)?;
|
decrease_balance(state, source_index, source_effective_balance)?;
|
||||||
increase_balance(state, target_index, source_effective_balance)?;
|
increase_balance(state, target_index, source_effective_balance)?;
|
||||||
|
|
||||||
affected_validators.insert(source_index);
|
|
||||||
affected_validators.insert(target_index);
|
|
||||||
|
|
||||||
next_pending_consolidation.safe_add_assign(1)?;
|
next_pending_consolidation.safe_add_assign(1)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1087,7 +1119,7 @@ fn process_pending_consolidations<E: EthSpec>(
|
|||||||
// Re-process effective balance updates for validators affected by consolidations.
|
// Re-process effective balance updates for validators affected by consolidations.
|
||||||
let (validators, balances, _, current_epoch_participation, _, progressive_balances, _, _) =
|
let (validators, balances, _, current_epoch_participation, _, progressive_balances, _, _) =
|
||||||
state.mutable_validator_fields()?;
|
state.mutable_validator_fields()?;
|
||||||
for validator_index in affected_validators {
|
for &validator_index in validators_in_consolidations {
|
||||||
let balance = *balances
|
let balance = *balances
|
||||||
.get(validator_index)
|
.get(validator_index)
|
||||||
.ok_or(BeaconStateError::UnknownValidator(validator_index))?;
|
.ok_or(BeaconStateError::UnknownValidator(validator_index))?;
|
||||||
@@ -1129,6 +1161,28 @@ impl EffectiveBalancesContext {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// This function is called for validators that do not have their effective balance updated as
|
||||||
|
/// part of the single-pass loop. For these validators we compute their true effective balance
|
||||||
|
/// update after processing consolidations. However, to maintain the invariants of the
|
||||||
|
/// `PreEpochCache` we must register _some_ effective balance for them immediately.
|
||||||
|
fn process_single_dummy_effective_balance_update(
|
||||||
|
validator_index: usize,
|
||||||
|
validator: &Cow<Validator>,
|
||||||
|
next_epoch_cache: &mut PreEpochCache,
|
||||||
|
state_ctxt: &StateContext,
|
||||||
|
) -> Result<(), Error> {
|
||||||
|
// Populate the effective balance cache with the current effective balance. This will be
|
||||||
|
// overriden when `process_single_effective_balance_update` is called.
|
||||||
|
let is_active_next_epoch = validator.is_active_at(state_ctxt.next_epoch);
|
||||||
|
let temporary_effective_balance = validator.effective_balance;
|
||||||
|
next_epoch_cache.update_effective_balance(
|
||||||
|
validator_index,
|
||||||
|
temporary_effective_balance,
|
||||||
|
is_active_next_epoch,
|
||||||
|
)?;
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
/// This function abstracts over phase0 and Electra effective balance processing.
|
/// This function abstracts over phase0 and Electra effective balance processing.
|
||||||
#[allow(clippy::too_many_arguments)]
|
#[allow(clippy::too_many_arguments)]
|
||||||
fn process_single_effective_balance_update(
|
fn process_single_effective_balance_update(
|
||||||
|
|||||||
Reference in New Issue
Block a user