Adopt heaviest subtree fork choice rule (#10441)
* Add HeaviestSubtreeForkChoice * Make replay stage switch between two fork choice rules Co-authored-by: Carl <carl@solana.com>
This commit is contained in:
parent
0510b6e336
commit
2e1d59ff85
|
@ -0,0 +1,152 @@
|
||||||
|
use crate::{
|
||||||
|
consensus::{ComputedBankState, Tower},
|
||||||
|
fork_choice::ForkChoice,
|
||||||
|
progress_map::{ForkStats, ProgressMap},
|
||||||
|
};
|
||||||
|
use solana_ledger::bank_forks::BankForks;
|
||||||
|
use solana_runtime::bank::Bank;
|
||||||
|
use solana_sdk::timing;
|
||||||
|
use std::time::Instant;
|
||||||
|
use std::{
|
||||||
|
collections::{HashMap, HashSet},
|
||||||
|
sync::{Arc, RwLock},
|
||||||
|
};
|
||||||
|
|
||||||
|
#[derive(Default)]
|
||||||
|
pub struct BankWeightForkChoice {}
|
||||||
|
|
||||||
|
impl ForkChoice for BankWeightForkChoice {
|
||||||
|
fn compute_bank_stats(
|
||||||
|
&mut self,
|
||||||
|
bank: &Bank,
|
||||||
|
_tower: &Tower,
|
||||||
|
progress: &mut ProgressMap,
|
||||||
|
computed_bank_stats: &ComputedBankState,
|
||||||
|
) {
|
||||||
|
let bank_slot = bank.slot();
|
||||||
|
// Only time progress map should be missing a bank slot
|
||||||
|
// is if this node was the leader for this slot as those banks
|
||||||
|
// are not replayed in replay_active_banks()
|
||||||
|
let parent_weight = bank
|
||||||
|
.parent()
|
||||||
|
.and_then(|b| progress.get(&b.slot()))
|
||||||
|
.map(|x| x.fork_stats.fork_weight)
|
||||||
|
.unwrap_or(0);
|
||||||
|
|
||||||
|
let stats = progress
|
||||||
|
.get_fork_stats_mut(bank_slot)
|
||||||
|
.expect("All frozen banks must exist in the Progress map");
|
||||||
|
|
||||||
|
let ComputedBankState { bank_weight, .. } = computed_bank_stats;
|
||||||
|
stats.weight = *bank_weight;
|
||||||
|
stats.fork_weight = stats.weight + parent_weight;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns:
|
||||||
|
// 1) The heaviest overall bank
|
||||||
|
// 2) The heavest bank on the same fork as the last vote (doesn't require a
|
||||||
|
// switching proof to vote for)
|
||||||
|
fn select_forks(
|
||||||
|
&self,
|
||||||
|
frozen_banks: &[Arc<Bank>],
|
||||||
|
tower: &Tower,
|
||||||
|
progress: &ProgressMap,
|
||||||
|
ancestors: &HashMap<u64, HashSet<u64>>,
|
||||||
|
_bank_forks: &RwLock<BankForks>,
|
||||||
|
) -> (Arc<Bank>, Option<Arc<Bank>>) {
|
||||||
|
let tower_start = Instant::now();
|
||||||
|
assert!(!frozen_banks.is_empty());
|
||||||
|
let num_frozen_banks = frozen_banks.len();
|
||||||
|
|
||||||
|
trace!("frozen_banks {}", frozen_banks.len());
|
||||||
|
let num_old_banks = frozen_banks
|
||||||
|
.iter()
|
||||||
|
.filter(|b| b.slot() < tower.root().unwrap_or(0))
|
||||||
|
.count();
|
||||||
|
|
||||||
|
let last_vote = tower.last_vote().slots.last().cloned();
|
||||||
|
let mut heaviest_bank_on_same_fork = None;
|
||||||
|
let mut heaviest_same_fork_weight = 0;
|
||||||
|
let stats: Vec<&ForkStats> = frozen_banks
|
||||||
|
.iter()
|
||||||
|
.map(|bank| {
|
||||||
|
// Only time progress map should be missing a bank slot
|
||||||
|
// is if this node was the leader for this slot as those banks
|
||||||
|
// are not replayed in replay_active_banks()
|
||||||
|
let stats = progress
|
||||||
|
.get_fork_stats(bank.slot())
|
||||||
|
.expect("All frozen banks must exist in the Progress map");
|
||||||
|
|
||||||
|
if let Some(last_vote) = last_vote {
|
||||||
|
if ancestors
|
||||||
|
.get(&bank.slot())
|
||||||
|
.expect("Entry in frozen banks must exist in ancestors")
|
||||||
|
.contains(&last_vote)
|
||||||
|
{
|
||||||
|
// Descendant of last vote cannot be locked out
|
||||||
|
assert!(!stats.is_locked_out);
|
||||||
|
|
||||||
|
// ancestors(slot) should not contain the slot itself,
|
||||||
|
// so we should never get the same bank as the last vote
|
||||||
|
assert_ne!(bank.slot(), last_vote);
|
||||||
|
// highest weight, lowest slot first. frozen_banks is sorted
|
||||||
|
// from least slot to greatest slot, so if two banks have
|
||||||
|
// the same fork weight, the lower slot will be picked
|
||||||
|
if stats.fork_weight > heaviest_same_fork_weight {
|
||||||
|
heaviest_bank_on_same_fork = Some(bank.clone());
|
||||||
|
heaviest_same_fork_weight = stats.fork_weight;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stats
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
let num_not_recent = stats.iter().filter(|s| !s.is_recent).count();
|
||||||
|
let num_has_voted = stats.iter().filter(|s| s.has_voted).count();
|
||||||
|
let num_empty = stats.iter().filter(|s| s.is_empty).count();
|
||||||
|
let num_threshold_failure = stats.iter().filter(|s| !s.vote_threshold).count();
|
||||||
|
let num_votable_threshold_failure = stats
|
||||||
|
.iter()
|
||||||
|
.filter(|s| s.is_recent && !s.has_voted && !s.vote_threshold)
|
||||||
|
.count();
|
||||||
|
|
||||||
|
let mut candidates: Vec<_> = frozen_banks.iter().zip(stats.iter()).collect();
|
||||||
|
|
||||||
|
//highest weight, lowest slot first
|
||||||
|
candidates.sort_by_key(|b| (b.1.fork_weight, 0i64 - b.0.slot() as i64));
|
||||||
|
let rv = candidates
|
||||||
|
.last()
|
||||||
|
.expect("frozen banks was nonempty so candidates must also be nonempty");
|
||||||
|
let ms = timing::duration_as_ms(&tower_start.elapsed());
|
||||||
|
let weights: Vec<(u128, u64, u64)> = candidates
|
||||||
|
.iter()
|
||||||
|
.map(|x| (x.1.weight, x.0.slot(), x.1.block_height))
|
||||||
|
.collect();
|
||||||
|
debug!(
|
||||||
|
"@{:?} tower duration: {:?} len: {}/{} weights: {:?}",
|
||||||
|
timing::timestamp(),
|
||||||
|
ms,
|
||||||
|
candidates.len(),
|
||||||
|
stats.iter().filter(|s| !s.has_voted).count(),
|
||||||
|
weights,
|
||||||
|
);
|
||||||
|
datapoint_debug!(
|
||||||
|
"replay_stage-select_forks",
|
||||||
|
("frozen_banks", num_frozen_banks as i64, i64),
|
||||||
|
("not_recent", num_not_recent as i64, i64),
|
||||||
|
("has_voted", num_has_voted as i64, i64),
|
||||||
|
("old_banks", num_old_banks as i64, i64),
|
||||||
|
("empty_banks", num_empty as i64, i64),
|
||||||
|
("threshold_failure", num_threshold_failure as i64, i64),
|
||||||
|
(
|
||||||
|
"votable_threshold_failure",
|
||||||
|
num_votable_threshold_failure as i64,
|
||||||
|
i64
|
||||||
|
),
|
||||||
|
("tower_duration", ms as i64, i64),
|
||||||
|
);
|
||||||
|
|
||||||
|
(rv.0.clone(), heaviest_bank_on_same_fork)
|
||||||
|
}
|
||||||
|
}
|
|
@ -21,7 +21,7 @@ use solana_vote_program::{
|
||||||
use std::{
|
use std::{
|
||||||
collections::{BTreeMap, HashMap, HashSet},
|
collections::{BTreeMap, HashMap, HashSet},
|
||||||
ops::Bound::{Included, Unbounded},
|
ops::Bound::{Included, Unbounded},
|
||||||
sync::Arc,
|
sync::{Arc, RwLock},
|
||||||
};
|
};
|
||||||
|
|
||||||
#[derive(PartialEq, Clone, Debug)]
|
#[derive(PartialEq, Clone, Debug)]
|
||||||
|
@ -79,6 +79,14 @@ impl StakeLockout {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) struct ComputedBankState {
|
||||||
|
pub stake_lockouts: HashMap<Slot, StakeLockout>,
|
||||||
|
pub total_staked: u64,
|
||||||
|
pub bank_weight: u128,
|
||||||
|
pub lockout_intervals: LockoutIntervals,
|
||||||
|
pub pubkey_votes: Vec<(Pubkey, Slot)>,
|
||||||
|
}
|
||||||
|
|
||||||
pub struct Tower {
|
pub struct Tower {
|
||||||
node_pubkey: Pubkey,
|
node_pubkey: Pubkey,
|
||||||
threshold_depth: usize,
|
threshold_depth: usize,
|
||||||
|
@ -102,10 +110,14 @@ impl Default for Tower {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Tower {
|
impl Tower {
|
||||||
pub fn new(node_pubkey: &Pubkey, vote_account_pubkey: &Pubkey, bank_forks: &BankForks) -> Self {
|
pub fn new(
|
||||||
|
node_pubkey: &Pubkey,
|
||||||
|
vote_account_pubkey: &Pubkey,
|
||||||
|
root: Slot,
|
||||||
|
heaviest_bank: &Bank,
|
||||||
|
) -> Self {
|
||||||
let mut tower = Self::new_with_key(node_pubkey);
|
let mut tower = Self::new_with_key(node_pubkey);
|
||||||
|
tower.initialize_lockouts_from_bank_forks(vote_account_pubkey, root, heaviest_bank);
|
||||||
tower.initialize_lockouts_from_bank_forks(&bank_forks, vote_account_pubkey);
|
|
||||||
|
|
||||||
tower
|
tower
|
||||||
}
|
}
|
||||||
|
@ -126,27 +138,28 @@ impl Tower {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn collect_vote_lockouts<F>(
|
pub(crate) fn collect_vote_lockouts<F>(
|
||||||
&self,
|
node_pubkey: &Pubkey,
|
||||||
bank_slot: u64,
|
bank_slot: u64,
|
||||||
vote_accounts: F,
|
vote_accounts: F,
|
||||||
ancestors: &HashMap<Slot, HashSet<u64>>,
|
ancestors: &HashMap<Slot, HashSet<u64>>,
|
||||||
all_pubkeys: &mut PubkeyReferences,
|
all_pubkeys: &mut PubkeyReferences,
|
||||||
) -> (HashMap<Slot, StakeLockout>, u64, u128, LockoutIntervals)
|
) -> ComputedBankState
|
||||||
where
|
where
|
||||||
F: Iterator<Item = (Pubkey, (u64, Account))>,
|
F: Iterator<Item = (Pubkey, (u64, Account))>,
|
||||||
{
|
{
|
||||||
let mut stake_lockouts = HashMap::new();
|
let mut stake_lockouts = HashMap::new();
|
||||||
let mut total_stake = 0;
|
let mut total_staked = 0;
|
||||||
let mut total_weight = 0;
|
let mut bank_weight = 0;
|
||||||
// Tree of intervals of lockouts of the form [slot, slot + slot.lockout],
|
// Tree of intervals of lockouts of the form [slot, slot + slot.lockout],
|
||||||
// keyed by end of the range
|
// keyed by end of the range
|
||||||
let mut lockout_intervals = BTreeMap::new();
|
let mut lockout_intervals = BTreeMap::new();
|
||||||
|
let mut pubkey_votes = vec![];
|
||||||
for (key, (lamports, account)) in vote_accounts {
|
for (key, (lamports, account)) in vote_accounts {
|
||||||
if lamports == 0 {
|
if lamports == 0 {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
trace!("{} {} with stake {}", self.node_pubkey, key, lamports);
|
trace!("{} {} with stake {}", node_pubkey, key, lamports);
|
||||||
let vote_state = VoteState::from(&account);
|
let vote_state = VoteState::from(&account);
|
||||||
if vote_state.is_none() {
|
if vote_state.is_none() {
|
||||||
datapoint_warn!(
|
datapoint_warn!(
|
||||||
|
@ -169,7 +182,7 @@ impl Tower {
|
||||||
.push((vote.slot, key));
|
.push((vote.slot, key));
|
||||||
}
|
}
|
||||||
|
|
||||||
if key == self.node_pubkey || vote_state.node_pubkey == self.node_pubkey {
|
if key == *node_pubkey || vote_state.node_pubkey == *node_pubkey {
|
||||||
debug!("vote state {:?}", vote_state);
|
debug!("vote state {:?}", vote_state);
|
||||||
debug!(
|
debug!(
|
||||||
"observed slot {}",
|
"observed slot {}",
|
||||||
|
@ -188,10 +201,15 @@ impl Tower {
|
||||||
}
|
}
|
||||||
let start_root = vote_state.root_slot;
|
let start_root = vote_state.root_slot;
|
||||||
|
|
||||||
|
// Add the latest vote to update the `heaviest_subtree_fork_choice`
|
||||||
|
if let Some(latest_vote) = vote_state.votes.back() {
|
||||||
|
pubkey_votes.push((key, latest_vote.slot));
|
||||||
|
}
|
||||||
|
|
||||||
vote_state.process_slot_vote_unchecked(bank_slot);
|
vote_state.process_slot_vote_unchecked(bank_slot);
|
||||||
|
|
||||||
for vote in &vote_state.votes {
|
for vote in &vote_state.votes {
|
||||||
total_weight += vote.lockout() as u128 * lamports as u128;
|
bank_weight += vote.lockout() as u128 * lamports as u128;
|
||||||
Self::update_ancestor_lockouts(&mut stake_lockouts, &vote, ancestors);
|
Self::update_ancestor_lockouts(&mut stake_lockouts, &vote, ancestors);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -202,7 +220,7 @@ impl Tower {
|
||||||
slot: root,
|
slot: root,
|
||||||
};
|
};
|
||||||
trace!("ROOT: {}", vote.slot);
|
trace!("ROOT: {}", vote.slot);
|
||||||
total_weight += vote.lockout() as u128 * lamports as u128;
|
bank_weight += vote.lockout() as u128 * lamports as u128;
|
||||||
Self::update_ancestor_lockouts(&mut stake_lockouts, &vote, ancestors);
|
Self::update_ancestor_lockouts(&mut stake_lockouts, &vote, ancestors);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -211,7 +229,7 @@ impl Tower {
|
||||||
confirmation_count: MAX_LOCKOUT_HISTORY as u32,
|
confirmation_count: MAX_LOCKOUT_HISTORY as u32,
|
||||||
slot: root,
|
slot: root,
|
||||||
};
|
};
|
||||||
total_weight += vote.lockout() as u128 * lamports as u128;
|
bank_weight += vote.lockout() as u128 * lamports as u128;
|
||||||
Self::update_ancestor_lockouts(&mut stake_lockouts, &vote, ancestors);
|
Self::update_ancestor_lockouts(&mut stake_lockouts, &vote, ancestors);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -232,9 +250,16 @@ impl Tower {
|
||||||
// Update all the parents of this last vote with the stake of this vote account
|
// Update all the parents of this last vote with the stake of this vote account
|
||||||
Self::update_ancestor_stakes(&mut stake_lockouts, vote.slot, lamports, ancestors);
|
Self::update_ancestor_stakes(&mut stake_lockouts, vote.slot, lamports, ancestors);
|
||||||
}
|
}
|
||||||
total_stake += lamports;
|
total_staked += lamports;
|
||||||
|
}
|
||||||
|
|
||||||
|
ComputedBankState {
|
||||||
|
stake_lockouts,
|
||||||
|
total_staked,
|
||||||
|
bank_weight,
|
||||||
|
lockout_intervals,
|
||||||
|
pubkey_votes,
|
||||||
}
|
}
|
||||||
(stake_lockouts, total_stake, total_weight, lockout_intervals)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn is_slot_confirmed(
|
pub fn is_slot_confirmed(
|
||||||
|
@ -509,7 +534,7 @@ impl Tower {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Update lockouts for all the ancestors
|
/// Update lockouts for all the ancestors
|
||||||
fn update_ancestor_lockouts(
|
pub(crate) fn update_ancestor_lockouts(
|
||||||
stake_lockouts: &mut HashMap<Slot, StakeLockout>,
|
stake_lockouts: &mut HashMap<Slot, StakeLockout>,
|
||||||
vote: &Lockout,
|
vote: &Lockout,
|
||||||
ancestors: &HashMap<Slot, HashSet<Slot>>,
|
ancestors: &HashMap<Slot, HashSet<Slot>>,
|
||||||
|
@ -529,6 +554,28 @@ impl Tower {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub(crate) fn find_heaviest_bank(
|
||||||
|
bank_forks: &RwLock<BankForks>,
|
||||||
|
node_pubkey: &Pubkey,
|
||||||
|
) -> Option<Arc<Bank>> {
|
||||||
|
let ancestors = bank_forks.read().unwrap().ancestors();
|
||||||
|
let mut bank_weights: Vec<_> = bank_forks
|
||||||
|
.read()
|
||||||
|
.unwrap()
|
||||||
|
.frozen_banks()
|
||||||
|
.values()
|
||||||
|
.map(|b| {
|
||||||
|
(
|
||||||
|
Self::bank_weight(node_pubkey, b, &ancestors),
|
||||||
|
b.parents().len(),
|
||||||
|
b.clone(),
|
||||||
|
)
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
bank_weights.sort_by_key(|b| (b.0, b.1));
|
||||||
|
bank_weights.pop().map(|b| b.2)
|
||||||
|
}
|
||||||
|
|
||||||
/// Update stake for all the ancestors.
|
/// Update stake for all the ancestors.
|
||||||
/// Note, stake is the same for all the ancestor.
|
/// Note, stake is the same for all the ancestor.
|
||||||
fn update_ancestor_stakes(
|
fn update_ancestor_stakes(
|
||||||
|
@ -537,9 +584,8 @@ impl Tower {
|
||||||
lamports: u64,
|
lamports: u64,
|
||||||
ancestors: &HashMap<Slot, HashSet<Slot>>,
|
ancestors: &HashMap<Slot, HashSet<Slot>>,
|
||||||
) {
|
) {
|
||||||
// If there's no ancestors, that means this slot must be from before the current root,
|
// If there's no ancestors, that means this slot must be from
|
||||||
// in which case the lockouts won't be calculated in bank_weight anyways, so ignore
|
// before the current root, so ignore this slot
|
||||||
// this slot
|
|
||||||
let vote_slot_ancestors = ancestors.get(&slot);
|
let vote_slot_ancestors = ancestors.get(&slot);
|
||||||
if vote_slot_ancestors.is_none() {
|
if vote_slot_ancestors.is_none() {
|
||||||
return;
|
return;
|
||||||
|
@ -552,8 +598,13 @@ impl Tower {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
fn bank_weight(&self, bank: &Bank, ancestors: &HashMap<Slot, HashSet<Slot>>) -> u128 {
|
fn bank_weight(
|
||||||
let (_, _, bank_weight, _) = self.collect_vote_lockouts(
|
node_pubkey: &Pubkey,
|
||||||
|
bank: &Bank,
|
||||||
|
ancestors: &HashMap<Slot, HashSet<Slot>>,
|
||||||
|
) -> u128 {
|
||||||
|
let ComputedBankState { bank_weight, .. } = Self::collect_vote_lockouts(
|
||||||
|
node_pubkey,
|
||||||
bank.slot(),
|
bank.slot(),
|
||||||
bank.vote_accounts().into_iter(),
|
bank.vote_accounts().into_iter(),
|
||||||
ancestors,
|
ancestors,
|
||||||
|
@ -562,47 +613,28 @@ impl Tower {
|
||||||
bank_weight
|
bank_weight
|
||||||
}
|
}
|
||||||
|
|
||||||
fn find_heaviest_bank(&self, bank_forks: &BankForks) -> Option<Arc<Bank>> {
|
|
||||||
let ancestors = bank_forks.ancestors();
|
|
||||||
let mut bank_weights: Vec<_> = bank_forks
|
|
||||||
.frozen_banks()
|
|
||||||
.values()
|
|
||||||
.map(|b| {
|
|
||||||
(
|
|
||||||
self.bank_weight(b, &ancestors),
|
|
||||||
b.parents().len(),
|
|
||||||
b.clone(),
|
|
||||||
)
|
|
||||||
})
|
|
||||||
.collect();
|
|
||||||
bank_weights.sort_by_key(|b| (b.0, b.1));
|
|
||||||
bank_weights.pop().map(|b| b.2)
|
|
||||||
}
|
|
||||||
|
|
||||||
fn initialize_lockouts_from_bank_forks(
|
fn initialize_lockouts_from_bank_forks(
|
||||||
&mut self,
|
&mut self,
|
||||||
bank_forks: &BankForks,
|
|
||||||
vote_account_pubkey: &Pubkey,
|
vote_account_pubkey: &Pubkey,
|
||||||
|
root: Slot,
|
||||||
|
heaviest_bank: &Bank,
|
||||||
) {
|
) {
|
||||||
if let Some(bank) = self.find_heaviest_bank(bank_forks) {
|
if let Some((_stake, vote_account)) = heaviest_bank.vote_accounts().get(vote_account_pubkey)
|
||||||
let root = bank_forks.root();
|
{
|
||||||
if let Some((_stake, vote_account)) = bank.vote_accounts().get(vote_account_pubkey) {
|
let mut vote_state = VoteState::deserialize(&vote_account.data)
|
||||||
let mut vote_state = VoteState::deserialize(&vote_account.data)
|
.expect("vote_account isn't a VoteState?");
|
||||||
.expect("vote_account isn't a VoteState?");
|
vote_state.root_slot = Some(root);
|
||||||
vote_state.root_slot = Some(root);
|
vote_state.votes.retain(|v| v.slot > root);
|
||||||
vote_state.votes.retain(|v| v.slot > root);
|
trace!(
|
||||||
trace!(
|
"{} lockouts initialized to {:?}",
|
||||||
"{} lockouts initialized to {:?}",
|
self.node_pubkey,
|
||||||
self.node_pubkey,
|
vote_state
|
||||||
vote_state
|
);
|
||||||
);
|
assert_eq!(
|
||||||
|
vote_state.node_pubkey, self.node_pubkey,
|
||||||
assert_eq!(
|
"vote account's node_pubkey doesn't match",
|
||||||
vote_state.node_pubkey, self.node_pubkey,
|
);
|
||||||
"vote account's node_pubkey doesn't match",
|
self.lockouts = vote_state;
|
||||||
);
|
|
||||||
self.lockouts = vote_state;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -626,10 +658,13 @@ impl Tower {
|
||||||
pub mod test {
|
pub mod test {
|
||||||
use super::*;
|
use super::*;
|
||||||
use crate::{
|
use crate::{
|
||||||
|
bank_weight_fork_choice::BankWeightForkChoice,
|
||||||
cluster_info_vote_listener::VoteTracker,
|
cluster_info_vote_listener::VoteTracker,
|
||||||
cluster_slots::ClusterSlots,
|
cluster_slots::ClusterSlots,
|
||||||
|
fork_choice::SelectVoteAndResetForkResult,
|
||||||
|
heaviest_subtree_fork_choice::HeaviestSubtreeForkChoice,
|
||||||
progress_map::ForkProgress,
|
progress_map::ForkProgress,
|
||||||
replay_stage::{HeaviestForkFailures, ReplayStage, SelectVoteAndResetForkResult},
|
replay_stage::{HeaviestForkFailures, ReplayStage},
|
||||||
};
|
};
|
||||||
use solana_ledger::bank_forks::BankForks;
|
use solana_ledger::bank_forks::BankForks;
|
||||||
use solana_runtime::{
|
use solana_runtime::{
|
||||||
|
@ -645,7 +680,7 @@ pub mod test {
|
||||||
signature::{Keypair, Signer},
|
signature::{Keypair, Signer},
|
||||||
};
|
};
|
||||||
use solana_vote_program::{
|
use solana_vote_program::{
|
||||||
vote_state::{Vote, VoteStateVersions},
|
vote_state::{Vote, VoteStateVersions, MAX_LOCKOUT_HISTORY},
|
||||||
vote_transaction,
|
vote_transaction,
|
||||||
};
|
};
|
||||||
use std::{
|
use std::{
|
||||||
|
@ -662,18 +697,26 @@ pub mod test {
|
||||||
pub vote_pubkeys: Vec<Pubkey>,
|
pub vote_pubkeys: Vec<Pubkey>,
|
||||||
pub bank_forks: RwLock<BankForks>,
|
pub bank_forks: RwLock<BankForks>,
|
||||||
pub progress: ProgressMap,
|
pub progress: ProgressMap,
|
||||||
|
pub heaviest_subtree_fork_choice: HeaviestSubtreeForkChoice,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl VoteSimulator {
|
impl VoteSimulator {
|
||||||
pub(crate) fn new(num_keypairs: usize) -> Self {
|
pub(crate) fn new(num_keypairs: usize) -> Self {
|
||||||
let (validator_keypairs, node_pubkeys, vote_pubkeys, bank_forks, progress) =
|
let (
|
||||||
Self::init_state(num_keypairs);
|
validator_keypairs,
|
||||||
|
node_pubkeys,
|
||||||
|
vote_pubkeys,
|
||||||
|
bank_forks,
|
||||||
|
progress,
|
||||||
|
heaviest_subtree_fork_choice,
|
||||||
|
) = Self::init_state(num_keypairs);
|
||||||
Self {
|
Self {
|
||||||
validator_keypairs,
|
validator_keypairs,
|
||||||
node_pubkeys,
|
node_pubkeys,
|
||||||
vote_pubkeys,
|
vote_pubkeys,
|
||||||
bank_forks: RwLock::new(bank_forks),
|
bank_forks: RwLock::new(bank_forks),
|
||||||
progress,
|
progress,
|
||||||
|
heaviest_subtree_fork_choice,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
pub(crate) fn fill_bank_forks(
|
pub(crate) fn fill_bank_forks(
|
||||||
|
@ -716,6 +759,8 @@ pub mod test {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
new_bank.freeze();
|
new_bank.freeze();
|
||||||
|
self.heaviest_subtree_fork_choice
|
||||||
|
.add_new_leaf_slot(new_bank.slot(), Some(new_bank.parent_slot()));
|
||||||
self.bank_forks.write().unwrap().insert(new_bank);
|
self.bank_forks.write().unwrap().insert(new_bank);
|
||||||
walk.forward();
|
walk.forward();
|
||||||
}
|
}
|
||||||
|
@ -750,6 +795,8 @@ pub mod test {
|
||||||
&ClusterSlots::default(),
|
&ClusterSlots::default(),
|
||||||
&self.bank_forks,
|
&self.bank_forks,
|
||||||
&mut PubkeyReferences::default(),
|
&mut PubkeyReferences::default(),
|
||||||
|
&mut self.heaviest_subtree_fork_choice,
|
||||||
|
&mut BankWeightForkChoice::default(),
|
||||||
);
|
);
|
||||||
|
|
||||||
let vote_bank = self
|
let vote_bank = self
|
||||||
|
@ -766,7 +813,7 @@ pub mod test {
|
||||||
heaviest_fork_failures,
|
heaviest_fork_failures,
|
||||||
..
|
..
|
||||||
} = ReplayStage::select_vote_and_reset_forks(
|
} = ReplayStage::select_vote_and_reset_forks(
|
||||||
&Some(vote_bank.clone()),
|
&vote_bank.clone(),
|
||||||
&None,
|
&None,
|
||||||
&ancestors,
|
&ancestors,
|
||||||
&descendants,
|
&descendants,
|
||||||
|
@ -795,6 +842,7 @@ pub mod test {
|
||||||
&None,
|
&None,
|
||||||
&mut PubkeyReferences::default(),
|
&mut PubkeyReferences::default(),
|
||||||
None,
|
None,
|
||||||
|
&mut self.heaviest_subtree_fork_choice,
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -880,6 +928,7 @@ pub mod test {
|
||||||
Vec<Pubkey>,
|
Vec<Pubkey>,
|
||||||
BankForks,
|
BankForks,
|
||||||
ProgressMap,
|
ProgressMap,
|
||||||
|
HeaviestSubtreeForkChoice,
|
||||||
) {
|
) {
|
||||||
let keypairs: HashMap<_, _> = std::iter::repeat_with(|| {
|
let keypairs: HashMap<_, _> = std::iter::repeat_with(|| {
|
||||||
let node_keypair = Keypair::new();
|
let node_keypair = Keypair::new();
|
||||||
|
@ -902,8 +951,16 @@ pub mod test {
|
||||||
.map(|keys| keys.vote_keypair.pubkey())
|
.map(|keys| keys.vote_keypair.pubkey())
|
||||||
.collect();
|
.collect();
|
||||||
|
|
||||||
let (bank_forks, progress) = initialize_state(&keypairs, 10_000);
|
let (bank_forks, progress, heaviest_subtree_fork_choice) =
|
||||||
(keypairs, node_pubkeys, vote_pubkeys, bank_forks, progress)
|
initialize_state(&keypairs, 10_000);
|
||||||
|
(
|
||||||
|
keypairs,
|
||||||
|
node_pubkeys,
|
||||||
|
vote_pubkeys,
|
||||||
|
bank_forks,
|
||||||
|
progress,
|
||||||
|
heaviest_subtree_fork_choice,
|
||||||
|
)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -911,7 +968,7 @@ pub mod test {
|
||||||
pub(crate) fn initialize_state(
|
pub(crate) fn initialize_state(
|
||||||
validator_keypairs_map: &HashMap<Pubkey, ValidatorVoteKeypairs>,
|
validator_keypairs_map: &HashMap<Pubkey, ValidatorVoteKeypairs>,
|
||||||
stake: u64,
|
stake: u64,
|
||||||
) -> (BankForks, ProgressMap) {
|
) -> (BankForks, ProgressMap, HeaviestSubtreeForkChoice) {
|
||||||
let validator_keypairs: Vec<_> = validator_keypairs_map.values().collect();
|
let validator_keypairs: Vec<_> = validator_keypairs_map.values().collect();
|
||||||
let GenesisConfigInfo {
|
let GenesisConfigInfo {
|
||||||
genesis_config,
|
genesis_config,
|
||||||
|
@ -931,7 +988,10 @@ pub mod test {
|
||||||
0,
|
0,
|
||||||
ForkProgress::new(bank0.last_blockhash(), None, None, 0, 0),
|
ForkProgress::new(bank0.last_blockhash(), None, None, 0, 0),
|
||||||
);
|
);
|
||||||
(BankForks::new(0, bank0), progress)
|
let bank_forks = BankForks::new(0, bank0);
|
||||||
|
let heaviest_subtree_fork_choice =
|
||||||
|
HeaviestSubtreeForkChoice::new_from_bank_forks(&bank_forks);
|
||||||
|
(bank_forks, progress, heaviest_subtree_fork_choice)
|
||||||
}
|
}
|
||||||
|
|
||||||
fn gen_stakes(stake_votes: &[(u64, &[u64])]) -> Vec<(Pubkey, (u64, Account))> {
|
fn gen_stakes(stake_votes: &[(u64, &[u64])]) -> Vec<(Pubkey, (u64, Account))> {
|
||||||
|
@ -1291,20 +1351,32 @@ pub mod test {
|
||||||
#[test]
|
#[test]
|
||||||
fn test_collect_vote_lockouts_sums() {
|
fn test_collect_vote_lockouts_sums() {
|
||||||
//two accounts voting for slot 0 with 1 token staked
|
//two accounts voting for slot 0 with 1 token staked
|
||||||
let accounts = gen_stakes(&[(1, &[0]), (1, &[0])]);
|
let mut accounts = gen_stakes(&[(1, &[0]), (1, &[0])]);
|
||||||
let tower = Tower::new_for_tests(0, 0.67);
|
accounts.sort_by_key(|(pk, _)| *pk);
|
||||||
|
let account_latest_votes: Vec<(Pubkey, Slot)> =
|
||||||
|
accounts.iter().map(|(pubkey, _)| (*pubkey, 0)).collect();
|
||||||
|
|
||||||
let ancestors = vec![(1, vec![0].into_iter().collect()), (0, HashSet::new())]
|
let ancestors = vec![(1, vec![0].into_iter().collect()), (0, HashSet::new())]
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.collect();
|
.collect();
|
||||||
let (staked_lockouts, total_staked, bank_weight, _) = tower.collect_vote_lockouts(
|
let ComputedBankState {
|
||||||
|
stake_lockouts,
|
||||||
|
total_staked,
|
||||||
|
bank_weight,
|
||||||
|
mut pubkey_votes,
|
||||||
|
..
|
||||||
|
} = Tower::collect_vote_lockouts(
|
||||||
|
&Pubkey::default(),
|
||||||
1,
|
1,
|
||||||
accounts.into_iter(),
|
accounts.into_iter(),
|
||||||
&ancestors,
|
&ancestors,
|
||||||
&mut PubkeyReferences::default(),
|
&mut PubkeyReferences::default(),
|
||||||
);
|
);
|
||||||
assert_eq!(staked_lockouts[&0].stake, 2);
|
assert_eq!(stake_lockouts[&0].stake, 2);
|
||||||
assert_eq!(staked_lockouts[&0].lockout, 2 + 2 + 4 + 4);
|
assert_eq!(stake_lockouts[&0].lockout, 2 + 2 + 4 + 4);
|
||||||
assert_eq!(total_staked, 2);
|
assert_eq!(total_staked, 2);
|
||||||
|
pubkey_votes.sort();
|
||||||
|
assert_eq!(pubkey_votes, account_latest_votes);
|
||||||
|
|
||||||
// Each acccount has 1 vote in it. After simulating a vote in collect_vote_lockouts,
|
// Each acccount has 1 vote in it. After simulating a vote in collect_vote_lockouts,
|
||||||
// the account will have 2 votes, with lockout 2 + 4 = 6. So expected weight for
|
// the account will have 2 votes, with lockout 2 + 4 = 6. So expected weight for
|
||||||
|
@ -1316,7 +1388,12 @@ pub mod test {
|
||||||
fn test_collect_vote_lockouts_root() {
|
fn test_collect_vote_lockouts_root() {
|
||||||
let votes: Vec<u64> = (0..MAX_LOCKOUT_HISTORY as u64).collect();
|
let votes: Vec<u64> = (0..MAX_LOCKOUT_HISTORY as u64).collect();
|
||||||
//two accounts voting for slots 0..MAX_LOCKOUT_HISTORY with 1 token staked
|
//two accounts voting for slots 0..MAX_LOCKOUT_HISTORY with 1 token staked
|
||||||
let accounts = gen_stakes(&[(1, &votes), (1, &votes)]);
|
let mut accounts = gen_stakes(&[(1, &votes), (1, &votes)]);
|
||||||
|
accounts.sort_by_key(|(pk, _)| *pk);
|
||||||
|
let account_latest_votes: Vec<(Pubkey, Slot)> = accounts
|
||||||
|
.iter()
|
||||||
|
.map(|(pubkey, _)| (*pubkey, (MAX_LOCKOUT_HISTORY - 1) as Slot))
|
||||||
|
.collect();
|
||||||
let mut tower = Tower::new_for_tests(0, 0.67);
|
let mut tower = Tower::new_for_tests(0, 0.67);
|
||||||
let mut ancestors = HashMap::new();
|
let mut ancestors = HashMap::new();
|
||||||
for i in 0..(MAX_LOCKOUT_HISTORY + 1) {
|
for i in 0..(MAX_LOCKOUT_HISTORY + 1) {
|
||||||
|
@ -1337,18 +1414,27 @@ pub mod test {
|
||||||
+ root_weight;
|
+ root_weight;
|
||||||
let expected_bank_weight = 2 * vote_account_expected_weight;
|
let expected_bank_weight = 2 * vote_account_expected_weight;
|
||||||
assert_eq!(tower.lockouts.root_slot, Some(0));
|
assert_eq!(tower.lockouts.root_slot, Some(0));
|
||||||
let (staked_lockouts, _total_staked, bank_weight, _) = tower.collect_vote_lockouts(
|
let ComputedBankState {
|
||||||
|
stake_lockouts,
|
||||||
|
bank_weight,
|
||||||
|
mut pubkey_votes,
|
||||||
|
..
|
||||||
|
} = Tower::collect_vote_lockouts(
|
||||||
|
&Pubkey::default(),
|
||||||
MAX_LOCKOUT_HISTORY as u64,
|
MAX_LOCKOUT_HISTORY as u64,
|
||||||
accounts.into_iter(),
|
accounts.into_iter(),
|
||||||
&ancestors,
|
&ancestors,
|
||||||
&mut PubkeyReferences::default(),
|
&mut PubkeyReferences::default(),
|
||||||
);
|
);
|
||||||
for i in 0..MAX_LOCKOUT_HISTORY {
|
for i in 0..MAX_LOCKOUT_HISTORY {
|
||||||
assert_eq!(staked_lockouts[&(i as u64)].stake, 2);
|
assert_eq!(stake_lockouts[&(i as u64)].stake, 2);
|
||||||
}
|
}
|
||||||
|
|
||||||
// should be the sum of all the weights for root
|
// should be the sum of all the weights for root
|
||||||
assert!(staked_lockouts[&0].lockout > (2 * (1 << MAX_LOCKOUT_HISTORY)));
|
assert!(stake_lockouts[&0].lockout > (2 * (1 << MAX_LOCKOUT_HISTORY)));
|
||||||
assert_eq!(bank_weight, expected_bank_weight);
|
assert_eq!(bank_weight, expected_bank_weight);
|
||||||
|
pubkey_votes.sort();
|
||||||
|
assert_eq!(pubkey_votes, account_latest_votes);
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -1381,7 +1467,7 @@ pub mod test {
|
||||||
);
|
);
|
||||||
tower.record_vote(i, Hash::default());
|
tower.record_vote(i, Hash::default());
|
||||||
}
|
}
|
||||||
assert!(!tower.check_vote_stake_threshold(MAX_LOCKOUT_HISTORY as u64 + 1, &stakes, 2));
|
assert!(!tower.check_vote_stake_threshold(MAX_LOCKOUT_HISTORY as u64 + 1, &stakes, 2,));
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -1609,48 +1695,7 @@ pub mod test {
|
||||||
tower.record_vote(0, Hash::default());
|
tower.record_vote(0, Hash::default());
|
||||||
tower.record_vote(1, Hash::default());
|
tower.record_vote(1, Hash::default());
|
||||||
tower.record_vote(2, Hash::default());
|
tower.record_vote(2, Hash::default());
|
||||||
assert!(tower.check_vote_stake_threshold(6, &stakes, 2));
|
assert!(tower.check_vote_stake_threshold(6, &stakes, 2,));
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_lockout_is_updated_for_entire_branch() {
|
|
||||||
let mut stake_lockouts = HashMap::new();
|
|
||||||
let vote = Lockout {
|
|
||||||
slot: 2,
|
|
||||||
confirmation_count: 1,
|
|
||||||
};
|
|
||||||
let set: HashSet<u64> = vec![0u64, 1u64].into_iter().collect();
|
|
||||||
let mut ancestors = HashMap::new();
|
|
||||||
ancestors.insert(2, set);
|
|
||||||
let set: HashSet<u64> = vec![0u64].into_iter().collect();
|
|
||||||
ancestors.insert(1, set);
|
|
||||||
Tower::update_ancestor_lockouts(&mut stake_lockouts, &vote, &ancestors);
|
|
||||||
assert_eq!(stake_lockouts[&0].lockout, 2);
|
|
||||||
assert_eq!(stake_lockouts[&1].lockout, 2);
|
|
||||||
assert_eq!(stake_lockouts[&2].lockout, 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn test_lockout_is_updated_for_slot_or_lower() {
|
|
||||||
let mut stake_lockouts = HashMap::new();
|
|
||||||
let set: HashSet<u64> = vec![0u64, 1u64].into_iter().collect();
|
|
||||||
let mut ancestors = HashMap::new();
|
|
||||||
ancestors.insert(2, set);
|
|
||||||
let set: HashSet<u64> = vec![0u64].into_iter().collect();
|
|
||||||
ancestors.insert(1, set);
|
|
||||||
let vote = Lockout {
|
|
||||||
slot: 2,
|
|
||||||
confirmation_count: 1,
|
|
||||||
};
|
|
||||||
Tower::update_ancestor_lockouts(&mut stake_lockouts, &vote, &ancestors);
|
|
||||||
let vote = Lockout {
|
|
||||||
slot: 1,
|
|
||||||
confirmation_count: 2,
|
|
||||||
};
|
|
||||||
Tower::update_ancestor_lockouts(&mut stake_lockouts, &vote, &ancestors);
|
|
||||||
assert_eq!(stake_lockouts[&0].lockout, 2 + 4);
|
|
||||||
assert_eq!(stake_lockouts[&1].lockout, 2 + 4);
|
|
||||||
assert_eq!(stake_lockouts[&2].lockout, 2);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
|
@ -1729,7 +1774,7 @@ pub mod test {
|
||||||
let total_stake = 4;
|
let total_stake = 4;
|
||||||
let threshold_size = 0.67;
|
let threshold_size = 0.67;
|
||||||
let threshold_stake = (f64::ceil(total_stake as f64 * threshold_size)) as u64;
|
let threshold_stake = (f64::ceil(total_stake as f64 * threshold_size)) as u64;
|
||||||
let tower_votes: Vec<u64> = (0..VOTE_THRESHOLD_DEPTH as u64).collect();
|
let tower_votes: Vec<Slot> = (0..VOTE_THRESHOLD_DEPTH as u64).collect();
|
||||||
let accounts = gen_stakes(&[
|
let accounts = gen_stakes(&[
|
||||||
(threshold_stake, &[(VOTE_THRESHOLD_DEPTH - 2) as u64]),
|
(threshold_stake, &[(VOTE_THRESHOLD_DEPTH - 2) as u64]),
|
||||||
(total_stake - threshold_stake, &tower_votes[..]),
|
(total_stake - threshold_stake, &tower_votes[..]),
|
||||||
|
@ -1746,29 +1791,35 @@ pub mod test {
|
||||||
for vote in &tower_votes {
|
for vote in &tower_votes {
|
||||||
tower.record_vote(*vote, Hash::default());
|
tower.record_vote(*vote, Hash::default());
|
||||||
}
|
}
|
||||||
let (staked_lockouts, total_staked, _, _) = tower.collect_vote_lockouts(
|
let ComputedBankState {
|
||||||
|
stake_lockouts,
|
||||||
|
total_staked,
|
||||||
|
..
|
||||||
|
} = Tower::collect_vote_lockouts(
|
||||||
|
&Pubkey::default(),
|
||||||
vote_to_evaluate,
|
vote_to_evaluate,
|
||||||
accounts.clone().into_iter(),
|
accounts.clone().into_iter(),
|
||||||
&ancestors,
|
&ancestors,
|
||||||
&mut PubkeyReferences::default(),
|
&mut PubkeyReferences::default(),
|
||||||
);
|
);
|
||||||
assert!(tower.check_vote_stake_threshold(vote_to_evaluate, &staked_lockouts, total_staked));
|
assert!(tower.check_vote_stake_threshold(vote_to_evaluate, &stake_lockouts, total_staked,));
|
||||||
|
|
||||||
// CASE 2: Now we want to evaluate a vote for slot VOTE_THRESHOLD_DEPTH + 1. This slot
|
// CASE 2: Now we want to evaluate a vote for slot VOTE_THRESHOLD_DEPTH + 1. This slot
|
||||||
// will expire the vote in one of the vote accounts, so we should have insufficient
|
// will expire the vote in one of the vote accounts, so we should have insufficient
|
||||||
// stake to pass the threshold
|
// stake to pass the threshold
|
||||||
let vote_to_evaluate = VOTE_THRESHOLD_DEPTH as u64 + 1;
|
let vote_to_evaluate = VOTE_THRESHOLD_DEPTH as u64 + 1;
|
||||||
let (staked_lockouts, total_staked, _, _) = tower.collect_vote_lockouts(
|
let ComputedBankState {
|
||||||
|
stake_lockouts,
|
||||||
|
total_staked,
|
||||||
|
..
|
||||||
|
} = Tower::collect_vote_lockouts(
|
||||||
|
&Pubkey::default(),
|
||||||
vote_to_evaluate,
|
vote_to_evaluate,
|
||||||
accounts.into_iter(),
|
accounts.into_iter(),
|
||||||
&ancestors,
|
&ancestors,
|
||||||
&mut PubkeyReferences::default(),
|
&mut PubkeyReferences::default(),
|
||||||
);
|
);
|
||||||
assert!(!tower.check_vote_stake_threshold(
|
assert!(!tower.check_vote_stake_threshold(vote_to_evaluate, &stake_lockouts, total_staked,));
|
||||||
vote_to_evaluate,
|
|
||||||
&staked_lockouts,
|
|
||||||
total_staked
|
|
||||||
));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
fn vote_and_check_recent(num_votes: usize) {
|
fn vote_and_check_recent(num_votes: usize) {
|
||||||
|
|
|
@ -0,0 +1,40 @@
|
||||||
|
use crate::{
|
||||||
|
consensus::{ComputedBankState, SwitchForkDecision, Tower},
|
||||||
|
progress_map::ProgressMap,
|
||||||
|
replay_stage::HeaviestForkFailures,
|
||||||
|
};
|
||||||
|
use solana_ledger::bank_forks::BankForks;
|
||||||
|
use solana_runtime::bank::Bank;
|
||||||
|
use std::{
|
||||||
|
collections::{HashMap, HashSet},
|
||||||
|
sync::{Arc, RwLock},
|
||||||
|
};
|
||||||
|
|
||||||
|
pub(crate) struct SelectVoteAndResetForkResult {
|
||||||
|
pub vote_bank: Option<(Arc<Bank>, SwitchForkDecision)>,
|
||||||
|
pub reset_bank: Option<Arc<Bank>>,
|
||||||
|
pub heaviest_fork_failures: Vec<HeaviestForkFailures>,
|
||||||
|
}
|
||||||
|
|
||||||
|
pub(crate) trait ForkChoice {
|
||||||
|
fn compute_bank_stats(
|
||||||
|
&mut self,
|
||||||
|
bank: &Bank,
|
||||||
|
tower: &Tower,
|
||||||
|
progress: &mut ProgressMap,
|
||||||
|
computed_bank_stats: &ComputedBankState,
|
||||||
|
);
|
||||||
|
|
||||||
|
// Returns:
|
||||||
|
// 1) The heaviest overall bbank
|
||||||
|
// 2) The heavest bank on the same fork as the last vote (doesn't require a
|
||||||
|
// switching proof to vote for)
|
||||||
|
fn select_forks(
|
||||||
|
&self,
|
||||||
|
frozen_banks: &[Arc<Bank>],
|
||||||
|
tower: &Tower,
|
||||||
|
progress: &ProgressMap,
|
||||||
|
ancestors: &HashMap<u64, HashSet<u64>>,
|
||||||
|
bank_forks: &RwLock<BankForks>,
|
||||||
|
) -> (Arc<Bank>, Option<Arc<Bank>>);
|
||||||
|
}
|
|
@ -0,0 +1,72 @@
|
||||||
|
use crate::{
|
||||||
|
consensus::{ComputedBankState, Tower},
|
||||||
|
fork_choice::ForkChoice,
|
||||||
|
heaviest_subtree_fork_choice::HeaviestSubtreeForkChoice,
|
||||||
|
progress_map::ProgressMap,
|
||||||
|
};
|
||||||
|
use solana_ledger::bank_forks::BankForks;
|
||||||
|
use solana_runtime::bank::Bank;
|
||||||
|
use std::{
|
||||||
|
collections::{HashMap, HashSet},
|
||||||
|
sync::{Arc, RwLock},
|
||||||
|
};
|
||||||
|
|
||||||
|
impl ForkChoice for HeaviestSubtreeForkChoice {
|
||||||
|
fn compute_bank_stats(
|
||||||
|
&mut self,
|
||||||
|
bank: &Bank,
|
||||||
|
_tower: &Tower,
|
||||||
|
_progress: &mut ProgressMap,
|
||||||
|
computed_bank_stats: &ComputedBankState,
|
||||||
|
) {
|
||||||
|
let ComputedBankState { pubkey_votes, .. } = computed_bank_stats;
|
||||||
|
|
||||||
|
// Update `heaviest_subtree_fork_choice` to find the best fork to build on
|
||||||
|
let best_overall_slot = self.add_votes(
|
||||||
|
&pubkey_votes,
|
||||||
|
bank.epoch_stakes_map(),
|
||||||
|
bank.epoch_schedule(),
|
||||||
|
);
|
||||||
|
|
||||||
|
datapoint_info!(
|
||||||
|
"best_slot",
|
||||||
|
("slot", bank.slot(), i64),
|
||||||
|
("best_slot", best_overall_slot, i64),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Returns:
|
||||||
|
// 1) The heaviest overall bbank
|
||||||
|
// 2) The heavest bank on the same fork as the last vote (doesn't require a
|
||||||
|
// switching proof to vote for)
|
||||||
|
fn select_forks(
|
||||||
|
&self,
|
||||||
|
_frozen_banks: &[Arc<Bank>],
|
||||||
|
tower: &Tower,
|
||||||
|
_progress: &ProgressMap,
|
||||||
|
_ancestors: &HashMap<u64, HashSet<u64>>,
|
||||||
|
bank_forks: &RwLock<BankForks>,
|
||||||
|
) -> (Arc<Bank>, Option<Arc<Bank>>) {
|
||||||
|
let last_vote = tower.last_vote().slots.last().cloned();
|
||||||
|
let heaviest_slot_on_same_voted_fork = last_vote.map(|last_vote| {
|
||||||
|
let heaviest_slot_on_same_voted_fork =
|
||||||
|
self.best_slot(last_vote).expect("last_vote is a frozen bank so must have been added to heaviest_subtree_fork_choice at time of freezing");
|
||||||
|
if heaviest_slot_on_same_voted_fork == last_vote {
|
||||||
|
None
|
||||||
|
} else {
|
||||||
|
Some(heaviest_slot_on_same_voted_fork)
|
||||||
|
}
|
||||||
|
}).unwrap_or(None);
|
||||||
|
let heaviest_slot = self.best_overall_slot();
|
||||||
|
let r_bank_forks = bank_forks.read().unwrap();
|
||||||
|
(
|
||||||
|
r_bank_forks.get(heaviest_slot).unwrap().clone(),
|
||||||
|
heaviest_slot_on_same_voted_fork.map(|heaviest_slot_on_same_voted_fork| {
|
||||||
|
r_bank_forks
|
||||||
|
.get(heaviest_slot_on_same_voted_fork)
|
||||||
|
.unwrap()
|
||||||
|
.clone()
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
File diff suppressed because it is too large
Load Diff
|
@ -15,6 +15,7 @@ mod deprecated;
|
||||||
pub mod shred_fetch_stage;
|
pub mod shred_fetch_stage;
|
||||||
#[macro_use]
|
#[macro_use]
|
||||||
pub mod contact_info;
|
pub mod contact_info;
|
||||||
|
pub mod bank_weight_fork_choice;
|
||||||
pub mod cluster_info;
|
pub mod cluster_info;
|
||||||
pub mod cluster_slots;
|
pub mod cluster_slots;
|
||||||
pub mod consensus;
|
pub mod consensus;
|
||||||
|
@ -26,8 +27,10 @@ pub mod crds_gossip_push;
|
||||||
pub mod crds_value;
|
pub mod crds_value;
|
||||||
pub mod epoch_slots;
|
pub mod epoch_slots;
|
||||||
pub mod fetch_stage;
|
pub mod fetch_stage;
|
||||||
|
pub mod fork_choice;
|
||||||
pub mod gen_keys;
|
pub mod gen_keys;
|
||||||
pub mod gossip_service;
|
pub mod gossip_service;
|
||||||
|
pub mod heaviest_subtree_fork_choice;
|
||||||
pub mod ledger_cleanup_service;
|
pub mod ledger_cleanup_service;
|
||||||
pub mod local_vote_signer_service;
|
pub mod local_vote_signer_service;
|
||||||
pub mod non_circulating_supply;
|
pub mod non_circulating_supply;
|
||||||
|
|
File diff suppressed because it is too large
Load Diff
|
@ -357,20 +357,18 @@ fn test_cluster_partition_1_1_1() {
|
||||||
|
|
||||||
#[test]
|
#[test]
|
||||||
#[serial]
|
#[serial]
|
||||||
fn test_kill_partition() {
|
fn test_kill_heaviest_partition() {
|
||||||
// This test:
|
// This test:
|
||||||
// 1) Spins up three partitions
|
// 1) Spins up four partitions, the heaviest being the first with more stake
|
||||||
// 2) Forces more slots in the leader schedule for the first partition so
|
// 2) Schedules the other validators for sufficient slots in the schedule
|
||||||
// that this partition will be the heaviiest
|
|
||||||
// 3) Schedules the other validators for sufficient slots in the schedule
|
|
||||||
// so that they will still be locked out of voting for the major partitoin
|
// so that they will still be locked out of voting for the major partitoin
|
||||||
// when the partition resolves
|
// when the partition resolves
|
||||||
// 4) Kills the major partition. Validators are locked out, but should be
|
// 3) Kills the most staked partition. Validators are locked out, but should all
|
||||||
// able to reset to the major partition
|
// eventually choose the major partition
|
||||||
// 5) Check for recovery
|
// 4) Check for recovery
|
||||||
let mut leader_schedule = vec![];
|
let mut leader_schedule = vec![];
|
||||||
let num_slots_per_validator = 8;
|
let num_slots_per_validator = 8;
|
||||||
let partitions: [&[usize]; 3] = [&[9], &[10], &[10]];
|
let partitions: [&[usize]; 4] = [&[11], &[10], &[10], &[10]];
|
||||||
let validator_keys: Vec<_> = iter::repeat_with(|| Arc::new(Keypair::new()))
|
let validator_keys: Vec<_> = iter::repeat_with(|| Arc::new(Keypair::new()))
|
||||||
.take(partitions.len())
|
.take(partitions.len())
|
||||||
.collect();
|
.collect();
|
||||||
|
|
|
@ -2411,6 +2411,10 @@ impl Bank {
|
||||||
self.epoch_stakes.get(&epoch)
|
self.epoch_stakes.get(&epoch)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn epoch_stakes_map(&self) -> &HashMap<Epoch, EpochStakes> {
|
||||||
|
&self.epoch_stakes
|
||||||
|
}
|
||||||
|
|
||||||
/// vote accounts for the specific epoch along with the stake
|
/// vote accounts for the specific epoch along with the stake
|
||||||
/// attributed to each account
|
/// attributed to each account
|
||||||
pub fn epoch_vote_accounts(&self, epoch: Epoch) -> Option<&HashMap<Pubkey, (u64, Account)>> {
|
pub fn epoch_vote_accounts(&self, epoch: Epoch) -> Option<&HashMap<Pubkey, (u64, Account)>> {
|
||||||
|
@ -2448,11 +2452,11 @@ impl Bank {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Get the fixed stake of the given vote account for the current epoch
|
/// Get the fixed stake of the given vote account for the current epoch
|
||||||
pub fn epoch_vote_account_stake(&self, voting_pubkey: &Pubkey) -> u64 {
|
pub fn epoch_vote_account_stake(&self, vote_account: &Pubkey) -> u64 {
|
||||||
*self
|
*self
|
||||||
.epoch_vote_accounts(self.epoch())
|
.epoch_vote_accounts(self.epoch())
|
||||||
.expect("Bank epoch vote accounts must contain entry for the bank's own epoch")
|
.expect("Bank epoch vote accounts must contain entry for the bank's own epoch")
|
||||||
.get(voting_pubkey)
|
.get(vote_account)
|
||||||
.map(|(stake, _)| stake)
|
.map(|(stake, _)| stake)
|
||||||
.unwrap_or(&0)
|
.unwrap_or(&0)
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,6 +50,13 @@ impl EpochStakes {
|
||||||
&self.epoch_authorized_voters
|
&self.epoch_authorized_voters
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub fn vote_account_stake(&self, vote_account: &Pubkey) -> u64 {
|
||||||
|
Stakes::vote_accounts(&self.stakes)
|
||||||
|
.get(vote_account)
|
||||||
|
.map(|(stake, _)| *stake)
|
||||||
|
.unwrap_or(0)
|
||||||
|
}
|
||||||
|
|
||||||
fn parse_epoch_vote_accounts(
|
fn parse_epoch_vote_accounts(
|
||||||
epoch_vote_accounts: &HashMap<Pubkey, (u64, Account)>,
|
epoch_vote_accounts: &HashMap<Pubkey, (u64, Account)>,
|
||||||
leader_schedule_epoch: Epoch,
|
leader_schedule_epoch: Epoch,
|
||||||
|
|
Loading…
Reference in New Issue