From bfc974ff8bebf6d7e57f07ba6d5773fe55e9e3c4 Mon Sep 17 00:00:00 2001 From: Carl Date: Wed, 1 Apr 2020 13:21:58 -0700 Subject: [PATCH] Simplify vote simulator --- core/src/consensus.rs | 633 +++++++++++++++++---------------------- core/src/replay_stage.rs | 58 ++-- 2 files changed, 303 insertions(+), 388 deletions(-) diff --git a/core/src/consensus.rs b/core/src/consensus.rs index ea71a556cf1abe..d847b37d6883eb 100644 --- a/core/src/consensus.rs +++ b/core/src/consensus.rs @@ -353,7 +353,7 @@ impl Tower { let last_vote_ancestors = ancestors.get(&last_vote).unwrap(); let switch_slot_ancestors = ancestors.get(&switch_slot).unwrap(); - if switch_slot_ancestors.contains(last_vote) { + if switch_slot == *last_vote || switch_slot_ancestors.contains(last_vote) { // If the `switch_slot is a descendant of the last vote, // no switching proof is neceessary return true; @@ -423,7 +423,7 @@ impl Tower { pub fn check_vote_stake_threshold( &self, - slot: u64, + slot: Slot, stake_lockouts: &HashMap, total_staked: u64, ) -> bool { @@ -434,11 +434,8 @@ impl Tower { if let Some(fork_stake) = stake_lockouts.get(&vote.slot) { let lockout = fork_stake.stake as f64 / total_staked as f64; trace!( - "fork_stake slot: {} lockout: {} fork_stake: {} total_stake: {}", - slot, - lockout, - fork_stake.stake, - total_staked + "fork_stake slot: {}, vote slot: {}, lockout: {} fork_stake: {} total_stake: {}", + slot, vote.slot, lockout, fork_stake.stake, total_staked ); if vote.confirmation_count as usize > self.threshold_depth { for old_vote in &self.lockouts.votes { @@ -594,46 +591,72 @@ pub mod test { vote_state::{Vote, VoteStateVersions}, vote_transaction, }; - use std::collections::{HashMap, VecDeque}; + use std::collections::HashMap; use std::sync::RwLock; use std::{thread::sleep, time::Duration}; - use trees::{tr, Node, Tree, TreeWalk}; - - pub(crate) struct VoteSimulator<'a> { - searchable_nodes: HashMap>, - } - - impl<'a> VoteSimulator<'a> { - pub(crate) fn new(forks: &'a Tree) -> Self { - let mut searchable_nodes = HashMap::new(); - let root = forks.root(); - searchable_nodes.insert(root.data, root); - Self { searchable_nodes } + use trees::{tr, Tree, TreeWalk}; + + pub(crate) struct VoteSimulator { + pub validator_keypairs: HashMap, + pub node_pubkeys: Vec, + pub vote_pubkeys: Vec, + pub bank_forks: RwLock, + pub progress: ProgressMap, + } + + impl VoteSimulator { + pub(crate) fn new(num_keypairs: usize) -> Self { + let (validator_keypairs, node_pubkeys, vote_pubkeys, bank_forks, progress) = + Self::init_state(num_keypairs); + Self { + validator_keypairs, + node_pubkeys, + vote_pubkeys, + bank_forks: RwLock::new(bank_forks), + progress, + } } - pub(crate) fn fill_bank_forks( + &mut self, forks: Tree, - bank_forks: &mut BankForks, - progress: &mut ProgressMap, + cluster_votes: &HashMap>, ) { - assert!(bank_forks.get(0).is_some()); - assert_eq!(forks.root().data, 0); + let root = forks.root().data; + assert!(self.bank_forks.read().unwrap().get(root).is_some()); let mut walk = TreeWalk::from(forks); loop { if let Some(visit) = walk.get() { let slot = visit.node().data; - progress + self.progress .entry(slot) .or_insert_with(|| ForkProgress::new(Hash::default(), None, None)); - if bank_forks.get(slot).is_some() { + if self.bank_forks.read().unwrap().get(slot).is_some() { walk.forward(); continue; } let parent = walk.get_parent().unwrap().data; - let parent_bank = bank_forks.get(parent).unwrap(); + let parent_bank = self.bank_forks.read().unwrap().get(parent).unwrap().clone(); let new_bank = Bank::new_from_parent(&parent_bank, &Pubkey::default(), slot); - bank_forks.insert(new_bank); + for (pubkey, vote) in cluster_votes.iter() { + if vote.contains(&parent) { + let keypairs = self.validator_keypairs.get(pubkey).unwrap(); + let last_blockhash = parent_bank.last_blockhash(); + let vote_tx = vote_transaction::new_vote_transaction( + // Must vote > root to be processed + vec![parent], + parent_bank.hash(), + last_blockhash, + &keypairs.node_keypair, + &keypairs.vote_keypair, + &keypairs.vote_keypair, + ); + info!("voting {} {}", parent_bank.slot(), parent_bank.hash()); + new_bank.process_transaction(&vote_tx).unwrap(); + } + } + new_bank.freeze(); + self.bank_forks.write().unwrap().insert(new_bank); walk.forward(); } else { break; @@ -644,82 +667,15 @@ pub mod test { pub(crate) fn simulate_vote( &mut self, vote_slot: Slot, - bank_forks: &RwLock, - cluster_votes: &mut HashMap>, - validator_keypairs: &HashMap, - my_keypairs: &ValidatorVoteKeypairs, - progress: &mut ProgressMap, + my_pubkey: &Pubkey, tower: &mut Tower, ) -> Vec { - let node = self - .find_node_and_update_simulation(vote_slot) - .expect("Vote to simulate must be for a slot in the tree"); - - let mut missing_nodes = VecDeque::new(); - let mut current = node; - loop { - let current_slot = current.data; - if bank_forks.read().unwrap().get(current_slot).is_some() - || tower.root().map(|r| current_slot < r).unwrap_or(false) - { - break; - } else { - missing_nodes.push_front(current); - } - - if let Some(parent) = current.parent() { - current = parent; - } else { - break; - } - } - - // Create any missing banks along the path - for missing_node in missing_nodes { - let missing_slot = missing_node.data; - let parent = missing_node.parent().unwrap().data; - let parent_bank = bank_forks - .read() - .unwrap() - .get(parent) - .expect("parent bank must exist") - .clone(); - info!("parent of {} is {}", missing_slot, parent_bank.slot(),); - progress - .entry(missing_slot) - .or_insert_with(|| ForkProgress::new(parent_bank.last_blockhash(), None, None)); - - // Create the missing bank - let new_bank = - Bank::new_from_parent(&parent_bank, &Pubkey::default(), missing_slot); - - // Simulate ingesting the cluster's votes for the parent into this bank - for (pubkey, vote) in cluster_votes.iter() { - if vote.contains(&parent_bank.slot()) { - let keypairs = validator_keypairs.get(pubkey).unwrap(); - let last_blockhash = parent_bank.last_blockhash(); - let vote_tx = vote_transaction::new_vote_transaction( - // Must vote > root to be processed - vec![parent_bank.slot()], - parent_bank.hash(), - last_blockhash, - &keypairs.node_keypair, - &keypairs.vote_keypair, - &keypairs.vote_keypair, - ); - info!("voting {} {}", parent_bank.slot(), parent_bank.hash()); - new_bank.process_transaction(&vote_tx).unwrap(); - } - } - new_bank.freeze(); - bank_forks.write().unwrap().insert(new_bank); - } - - // Now try to simulate the vote - let my_pubkey = my_keypairs.node_keypair.pubkey(); + // Try to simulate the vote + let my_keypairs = self.validator_keypairs.get(&my_pubkey).unwrap(); let my_vote_pubkey = my_keypairs.vote_keypair.pubkey(); - let ancestors = bank_forks.read().unwrap().ancestors(); - let mut frozen_banks: Vec<_> = bank_forks + let ancestors = self.bank_forks.read().unwrap().ancestors(); + let mut frozen_banks: Vec<_> = self + .bank_forks .read() .unwrap() .frozen_banks() @@ -732,14 +688,15 @@ pub mod test { &ancestors, &mut frozen_banks, tower, - progress, + &mut self.progress, &VoteTracker::default(), &ClusterSlots::default(), - bank_forks, + &self.bank_forks, &mut HashSet::new(), ); - let vote_bank = bank_forks + let vote_bank = self + .bank_forks .read() .unwrap() .get(vote_slot) @@ -747,13 +704,13 @@ pub mod test { .clone(); // Try to vote on the given slot - let descendants = bank_forks.read().unwrap().descendants(); + let descendants = self.bank_forks.read().unwrap().descendants(); let (_, _, failure_reasons) = ReplayStage::select_vote_and_reset_forks( &Some(vote_bank.clone()), &None, &ancestors, &descendants, - &progress, + &self.progress, &tower, ); @@ -766,58 +723,123 @@ pub mod test { if let Some(new_root) = tower.record_bank_vote(vote) { ReplayStage::handle_new_root( new_root, - bank_forks, - progress, + &self.bank_forks, + &mut self.progress, &None, &mut 0, &mut HashSet::new(), ); } - // Mark the vote for this bank under this node's pubkey so it will be - // integrated into any future child banks - cluster_votes.entry(my_pubkey).or_default().push(vote_slot); vec![] } - // Find a node representing the given slot - fn find_node_and_update_simulation(&mut self, slot: u64) -> Option<&'a Node> { - let mut successful_search_node: Option<&'a Node> = None; - let mut found_node = None; - for search_node in self.searchable_nodes.values() { - if let Some((target, new_searchable_nodes)) = Self::find_node(search_node, slot) { - successful_search_node = Some(search_node); - found_node = Some(target); - for node in new_searchable_nodes { - self.searchable_nodes.insert(node.data, node); + fn create_and_vote_new_branch( + &mut self, + start_slot: Slot, + end_slot: Slot, + cluster_votes: &HashMap>, + votes_to_simulate: &HashSet, + my_pubkey: &Pubkey, + tower: &mut Tower, + ) -> HashMap> { + (start_slot + 1..=end_slot) + .filter_map(|slot| { + let mut fork_tip_parent = tr(slot - 1); + fork_tip_parent.push_front(tr(slot)); + self.fill_bank_forks(fork_tip_parent, &cluster_votes); + if votes_to_simulate.contains(&slot) { + Some((slot, self.simulate_vote(slot, &my_pubkey, tower))) + } else { + None } - break; + }) + .collect() + } + + fn simulate_lockout_interval( + &mut self, + slot: Slot, + lockout_interval: (u64, u64), + vote_account_pubkey: &Pubkey, + ) { + self.progress + .entry(slot) + .or_insert_with(|| ForkProgress::new(Hash::default(), None, None)) + .fork_stats + .lockout_intervals + .entry(lockout_interval.1) + .or_default() + .push((lockout_interval.0, *vote_account_pubkey)); + } + + fn can_progress_on_fork( + &mut self, + my_pubkey: &Pubkey, + tower: &mut Tower, + start_slot: u64, + num_slots: u64, + cluster_votes: &mut HashMap>, + ) -> bool { + // Check that within some reasonable time, validator can make a new + // root on this fork + let old_root = tower.root(); + + for i in 1..num_slots { + // The parent of the tip of the fork + let mut fork_tip_parent = tr(start_slot + i - 1); + // The tip of the fork + fork_tip_parent.push_front(tr(start_slot + i)); + self.fill_bank_forks(fork_tip_parent, cluster_votes); + if self + .simulate_vote(i + start_slot, &my_pubkey, tower) + .is_empty() + { + cluster_votes + .entry(*my_pubkey) + .or_default() + .push(start_slot + i); + } + if old_root != tower.root() { + return true; } } - successful_search_node.map(|node| { - self.searchable_nodes.remove(&node.data); - }); - found_node + + false } - fn find_node( - node: &'a Node, - slot: u64, - ) -> Option<(&'a Node, Vec<&'a Node>)> { - if node.data == slot { - Some((node, node.iter().collect())) - } else { - let mut search_result: Option<(&'a Node, Vec<&'a Node>)> = None; - for child in node.iter() { - if let Some((_, ref mut new_searchable_nodes)) = search_result { - new_searchable_nodes.push(child); - continue; - } - search_result = Self::find_node(child, slot); - } + fn init_state( + num_keypairs: usize, + ) -> ( + HashMap, + Vec, + Vec, + BankForks, + ProgressMap, + ) { + let keypairs: HashMap<_, _> = std::iter::repeat_with(|| { + let node_keypair = Keypair::new(); + let vote_keypair = Keypair::new(); + let stake_keypair = Keypair::new(); + let node_pubkey = node_keypair.pubkey(); + ( + node_pubkey, + ValidatorVoteKeypairs::new(node_keypair, vote_keypair, stake_keypair), + ) + }) + .take(num_keypairs) + .collect(); + let node_pubkeys: Vec<_> = keypairs + .values() + .map(|keys| keys.node_keypair.pubkey()) + .collect(); + let vote_pubkeys: Vec<_> = keypairs + .values() + .map(|keys| keys.vote_keypair.pubkey()) + .collect(); - search_result - } + let (bank_forks, progress) = initialize_state(&keypairs, 10_000); + (keypairs, node_pubkeys, vote_pubkeys, bank_forks, progress) } } @@ -845,40 +867,6 @@ pub mod test { (BankForks::new(0, bank0), progress) } - fn init_bank_forks( - num_keypairs: usize, - ) -> ( - HashMap, - Vec, - Vec, - BankForks, - ProgressMap, - ) { - let keypairs: HashMap<_, _> = std::iter::repeat_with(|| { - let node_keypair = Keypair::new(); - let vote_keypair = Keypair::new(); - let stake_keypair = Keypair::new(); - let node_pubkey = node_keypair.pubkey(); - ( - node_pubkey, - ValidatorVoteKeypairs::new(node_keypair, vote_keypair, stake_keypair), - ) - }) - .take(num_keypairs) - .collect(); - let node_pubkeys: Vec<_> = keypairs - .values() - .map(|keys| keys.node_keypair.pubkey()) - .collect(); - let vote_pubkeys: Vec<_> = keypairs - .values() - .map(|keys| keys.vote_keypair.pubkey()) - .collect(); - - let (bank_forks, progress) = initialize_state(&keypairs, 10_000); - (keypairs, node_pubkeys, vote_pubkeys, bank_forks, progress) - } - fn gen_stakes(stake_votes: &[(u64, &[u64])]) -> Vec<(Pubkey, (u64, Account))> { let mut stakes = vec![]; for (lamports, votes) in stake_votes { @@ -899,72 +887,26 @@ pub mod test { stakes } - fn can_progress_on_fork( - my_pubkey: &Pubkey, - tower: &mut Tower, - start_slot: u64, - num_slots: u64, - bank_forks: &RwLock, - cluster_votes: &mut HashMap>, - keypairs: &HashMap, - progress: &mut ProgressMap, - ) -> bool { - // Check that within some reasonable time, validator can make a new - // root on this fork - let old_root = tower.root(); - let mut main_fork = tr(start_slot); - let mut tip = main_fork.root_mut(); - - for i in 1..num_slots { - tip.push_front(tr(start_slot + i)); - tip = tip.first_mut().unwrap(); - } - let mut voting_simulator = VoteSimulator::new(&main_fork); - for i in 1..num_slots { - voting_simulator.simulate_vote( - i + start_slot, - &bank_forks, - cluster_votes, - &keypairs, - keypairs.get(&my_pubkey).unwrap(), - progress, - tower, - ); - if old_root != tower.root() { - return true; - } - } - - false - } - #[test] fn test_simple_votes() { + // Init state + let mut vote_simulator = VoteSimulator::new(1); + let node_pubkey = vote_simulator.node_pubkeys[0]; + let mut tower = Tower::new_with_key(&node_pubkey); + // Create the tree of banks let forks = tr(0) / (tr(1) / (tr(2) / (tr(3) / (tr(4) / tr(5))))); // Set the voting behavior - let mut voting_simulator = VoteSimulator::new(&forks); + let mut cluster_votes = HashMap::new(); let votes = vec![0, 1, 2, 3, 4, 5]; + cluster_votes.insert(node_pubkey, votes.clone()); + vote_simulator.fill_bank_forks(forks, &cluster_votes); // Simulate the votes - let (keypairs, node_pubkeys, _, bank_forks, mut progress) = init_bank_forks(1); - let node_pubkey = node_pubkeys[0]; - let bank_forks = RwLock::new(bank_forks); - let mut tower = Tower::new_with_key(&node_pubkey); - - let mut cluster_votes = HashMap::new(); for vote in votes { - assert!(voting_simulator - .simulate_vote( - vote, - &bank_forks, - &mut cluster_votes, - &keypairs, - keypairs.get(&node_pubkey).unwrap(), - &mut progress, - &mut tower, - ) + assert!(vote_simulator + .simulate_vote(vote, &node_pubkey, &mut tower,) .is_empty()); } @@ -974,31 +916,26 @@ pub mod test { } } - fn simulate_lockout_interval( - progress: &mut ProgressMap, - slot: Slot, - lockout_interval: (u64, u64), - vote_account_pubkey: &Pubkey, - ) { - progress - .entry(slot) - .or_insert_with(|| ForkProgress::new(Hash::default(), None, None)) - .fork_stats - .lockout_intervals - .entry(lockout_interval.1) - .or_default() - .push((lockout_interval.0, *vote_account_pubkey)); - } - #[test] fn test_switch_threshold() { - let (keypairs, node_pubkeys, vote_pubkeys, mut bank_forks, mut progress) = - init_bank_forks(2); - let my_pubkey = node_pubkeys[0]; - let other_vote_account = vote_pubkeys[1]; - let bank0 = bank_forks.get(0).unwrap().clone(); + // Init state + let mut vote_simulator = VoteSimulator::new(2); + let my_pubkey = vote_simulator.node_pubkeys[0]; + let other_vote_account = vote_simulator.vote_pubkeys[1]; + let bank0 = vote_simulator + .bank_forks + .read() + .unwrap() + .get(0) + .unwrap() + .clone(); let total_stake = bank0.total_epoch_stake(); - assert_eq!(total_stake, keypairs.len() as u64 * 10_000); + assert_eq!( + total_stake, + vote_simulator.validator_keypairs.len() as u64 * 10_000 + ); + + // Create the tree of banks let forks = tr(0) / (tr(1) / (tr(2) @@ -1011,9 +948,9 @@ pub mod test { / (tr(110)))))); // Fill the BankForks according to the above fork structure - VoteSimulator::fill_bank_forks(forks, &mut bank_forks, &mut progress); - let ancestors = bank_forks.ancestors(); - let descendants = bank_forks.descendants(); + vote_simulator.fill_bank_forks(forks, &HashMap::new()); + let ancestors = vote_simulator.bank_forks.read().unwrap().ancestors(); + let descendants = vote_simulator.bank_forks.read().unwrap().descendants(); let mut tower = Tower::new_with_key(&my_pubkey); // Last vote is 47 @@ -1024,7 +961,7 @@ pub mod test { 48, &ancestors, &descendants, - &progress, + &vote_simulator.progress, total_stake, bank0.epoch_vote_accounts(0).unwrap(), )); @@ -1034,55 +971,55 @@ pub mod test { 110, &ancestors, &descendants, - &progress, + &vote_simulator.progress, total_stake, bank0.epoch_vote_accounts(0).unwrap(), )); // Adding another validator lockout on a descendant of last vote should // not count toward the switch threshold - simulate_lockout_interval(&mut progress, 50, (49, 100), &other_vote_account); + vote_simulator.simulate_lockout_interval(50, (49, 100), &other_vote_account); assert!(!tower.check_switch_threshold( 110, &ancestors, &descendants, - &progress, + &vote_simulator.progress, total_stake, bank0.epoch_vote_accounts(0).unwrap(), )); // Adding another validator lockout on an ancestor of last vote should // not count toward the switch threshold - simulate_lockout_interval(&mut progress, 50, (45, 100), &other_vote_account); + vote_simulator.simulate_lockout_interval(50, (45, 100), &other_vote_account); assert!(!tower.check_switch_threshold( 110, &ancestors, &descendants, - &progress, + &vote_simulator.progress, total_stake, bank0.epoch_vote_accounts(0).unwrap(), )); // Adding another validator lockout on a different fork, but the lockout // doesn't cover the last vote, should not satisfy the switch threshold - simulate_lockout_interval(&mut progress, 14, (12, 46), &other_vote_account); + vote_simulator.simulate_lockout_interval(14, (12, 46), &other_vote_account); assert!(!tower.check_switch_threshold( 110, &ancestors, &descendants, - &progress, + &vote_simulator.progress, total_stake, bank0.epoch_vote_accounts(0).unwrap(), )); // Adding another validator lockout on a different fork, and the lockout // covers the last vote, should satisfy the switch threshold - simulate_lockout_interval(&mut progress, 14, (12, 47), &other_vote_account); + vote_simulator.simulate_lockout_interval(14, (12, 47), &other_vote_account); assert!(tower.check_switch_threshold( 110, &ancestors, &descendants, - &progress, + &vote_simulator.progress, total_stake, bank0.epoch_vote_accounts(0).unwrap(), )); @@ -1090,12 +1027,16 @@ pub mod test { // If we set a root, then any lockout intervals below the root shouldn't // count toward the switch threshold. This means the other validator's // vote lockout no longer counts - bank_forks.set_root(43, &None); + vote_simulator + .bank_forks + .write() + .unwrap() + .set_root(43, &None); assert!(!tower.check_switch_threshold( 110, - &bank_forks.ancestors(), - &bank_forks.descendants(), - &progress, + &vote_simulator.bank_forks.read().unwrap().ancestors(), + &vote_simulator.bank_forks.read().unwrap().descendants(), + &vote_simulator.progress, total_stake, bank0.epoch_vote_accounts(0).unwrap(), )); @@ -1103,9 +1044,9 @@ pub mod test { #[test] fn test_switch_threshold_votes() { - let (keypairs, node_pubkeys, _, bank_forks, mut progress) = init_bank_forks(4); - let my_pubkey = node_pubkeys[0]; - let bank_forks = RwLock::new(bank_forks); + // Init state + let mut vote_simulator = VoteSimulator::new(4); + let my_pubkey = vote_simulator.node_pubkeys[0]; let mut tower = Tower::new_with_key(&my_pubkey); let forks = tr(0) / (tr(1) @@ -1115,58 +1056,56 @@ pub mod test { / (tr(43) / (tr(44) // Minor fork 2 - / (tr(45) / (tr(46) / (tr(47) / (tr(48) / (tr(49) / (tr(50))))))) - / (tr(110)))))); - - // Set the voting behavior - let mut voting_simulator = VoteSimulator::new(&forks); - let mut cluster_votes: HashMap> = HashMap::new(); + / (tr(45) / (tr(46)))) + / (tr(110))))); // Have two validators, each representing 20% of the stake vote on // minor fork 2 at slots 46 + 47 - cluster_votes.insert(node_pubkeys[1], vec![46]); - cluster_votes.insert(node_pubkeys[2], vec![47]); + let mut cluster_votes: HashMap> = HashMap::new(); + cluster_votes.insert(vote_simulator.node_pubkeys[1], vec![46]); + cluster_votes.insert(vote_simulator.node_pubkeys[2], vec![47]); + vote_simulator.fill_bank_forks(forks, &cluster_votes); // Vote on the first minor fork at slot 14, should succeed - assert!(voting_simulator - .simulate_vote( - 14, - &bank_forks, - &mut cluster_votes, - &keypairs, - keypairs.get(&my_pubkey).unwrap(), - &mut progress, - &mut tower, - ) + assert!(vote_simulator + .simulate_vote(14, &my_pubkey, &mut tower,) .is_empty()); // The other two validators voted at slots 46, 47, which // will only both show up in slot 48, at which point // 2/5 > SWITCH_FORK_THRESHOLD of the stake has voted // on another fork, so switching should suceed - for slot in 45..=48 { - let res = voting_simulator.simulate_vote( - slot, - &bank_forks, - &mut cluster_votes, - &keypairs, - keypairs.get(&my_pubkey).unwrap(), - &mut progress, - &mut tower, - ); - + let votes_to_simulate = (46..=48).into_iter().collect(); + let results = vote_simulator.create_and_vote_new_branch( + 45, + 48, + &cluster_votes, + &votes_to_simulate, + &my_pubkey, + &mut tower, + ); + for slot in 46..=48 { if slot == 48 { - assert!(res.is_empty()) + assert!(results.get(&slot).unwrap().is_empty()); } else { - assert_eq!(res[0], HeaviestForkFailures::FailedSwitchThreshold(slot)); + assert_eq!( + *results.get(&slot).unwrap(), + vec![HeaviestForkFailures::FailedSwitchThreshold(slot)] + ); } } } #[test] fn test_double_partition() { - solana_logger::setup(); - // Create the tree of banks in a BankForks object + // Init state + let mut vote_simulator = VoteSimulator::new(2); + let node_pubkey = vote_simulator.node_pubkeys[0]; + let vote_pubkey = vote_simulator.vote_pubkeys[0]; + let mut tower = Tower::new_with_key(&node_pubkey); + + let num_slots_to_try = 200; + // Create the tree of banks let forks = tr(0) / (tr(1) / (tr(2) @@ -1183,57 +1122,37 @@ pub mod test { / (tr(44) // Minor fork 2 / (tr(45) / (tr(46) / (tr(47) / (tr(48) / (tr(49) / (tr(50))))))) - / (tr(110))))))))))))); + / (tr(110) / (tr(110 + 2 * num_slots_to_try)))))))))))))); - // Set the voting behavior - let mut voting_simulator = VoteSimulator::new(&forks); - let mut votes: Vec = vec![]; + // Set the successful voting behavior + let mut cluster_votes = HashMap::new(); + let mut my_votes: Vec = vec![]; + let next_unlocked_slot = 110; // Vote on the first minor fork - votes.extend((0..=14).into_iter()); + my_votes.extend((0..=14).into_iter()); // Come back to the main fork - votes.extend((43..=44).into_iter()); + my_votes.extend((43..=44).into_iter()); // Vote on the second minor fork - votes.extend((45..=50).into_iter()); - - let (keypairs, node_pubkeys, vote_pubkeys, bank_forks, mut progress) = init_bank_forks(1); - let node_pubkey = node_pubkeys[0]; - let vote_pubkey = vote_pubkeys[0]; - let mut cluster_votes: HashMap> = HashMap::new(); - let bank_forks = RwLock::new(bank_forks); - let mut tower = Tower::new_with_key(&node_pubkey); - // Simulate the votes. Should fail on trying to come back to the main fork - // at 106 exclusively due to threshold failure - for vote in &votes { + my_votes.extend((45..=50).into_iter()); + // Vote to come back to main fork + my_votes.push(next_unlocked_slot); + cluster_votes.insert(node_pubkey, my_votes.clone()); + // Make the other validator vote fork to pass the threshold checks + let other_votes = my_votes.clone(); + cluster_votes.insert(vote_simulator.node_pubkeys[1], other_votes); + vote_simulator.fill_bank_forks(forks, &cluster_votes); + + // Simulate the votes. + for vote in &my_votes { // All these votes should be ok - assert!(voting_simulator - .simulate_vote( - *vote, - &bank_forks, - &mut cluster_votes, - &keypairs, - keypairs.get(&node_pubkey).unwrap(), - &mut progress, - &mut tower, - ) + assert!(vote_simulator + .simulate_vote(*vote, &node_pubkey, &mut tower,) .is_empty()); } - // Try to come back to main fork - let next_unlocked_slot = 110; - assert!(voting_simulator - .simulate_vote( - next_unlocked_slot, - &bank_forks, - &mut cluster_votes, - &keypairs, - keypairs.get(&node_pubkey).unwrap(), - &mut progress, - &mut tower, - ) - .is_empty()); - info!("local tower: {:#?}", tower.lockouts.votes); - let vote_accounts = bank_forks + let vote_accounts = vote_simulator + .bank_forks .read() .unwrap() .get(next_unlocked_slot) @@ -1243,15 +1162,17 @@ pub mod test { let state = VoteState::from(&observed.1).unwrap(); info!("observed tower: {:#?}", state.votes); - assert!(can_progress_on_fork( + let num_slots_to_try = 200; + cluster_votes + .get_mut(&vote_simulator.node_pubkeys[1]) + .unwrap() + .extend(next_unlocked_slot + 1..next_unlocked_slot + num_slots_to_try); + assert!(vote_simulator.can_progress_on_fork( &node_pubkey, &mut tower, next_unlocked_slot, - 200, - &bank_forks, + num_slots_to_try, &mut cluster_votes, - &keypairs, - &mut progress )); } diff --git a/core/src/replay_stage.rs b/core/src/replay_stage.rs index c307ef72e92f9d..5da3412526779b 100644 --- a/core/src/replay_stage.rs +++ b/core/src/replay_stage.rs @@ -2723,41 +2723,27 @@ pub(crate) mod tests { #[test] fn test_child_bank_heavier() { - let node_keypair = Keypair::new(); - let vote_keypair = Keypair::new(); - let stake_keypair = Keypair::new(); - let node_pubkey = node_keypair.pubkey(); - let mut keypairs = HashMap::new(); - keypairs.insert( - node_pubkey, - ValidatorVoteKeypairs::new(node_keypair, vote_keypair, stake_keypair), - ); - - let (bank_forks, mut progress) = initialize_state(&keypairs, 10_000); - let bank_forks = Arc::new(RwLock::new(bank_forks)); + // Init state + let mut vote_simulator = VoteSimulator::new(1); + let node_pubkey = vote_simulator.node_pubkeys[0]; let mut tower = Tower::new_with_key(&node_pubkey); // Create the tree of banks in a BankForks object let forks = tr(0) / (tr(1) / (tr(2) / (tr(3)))); - let mut voting_simulator = VoteSimulator::new(&forks); - let mut cluster_votes: HashMap> = HashMap::new(); - let votes: Vec = vec![0, 2]; - for vote in &votes { - assert!(voting_simulator - .simulate_vote( - *vote, - &bank_forks, - &mut cluster_votes, - &keypairs, - keypairs.get(&node_pubkey).unwrap(), - &mut progress, - &mut tower, - ) + // Set the voting behavior + let mut cluster_votes = HashMap::new(); + let votes = vec![0, 2]; + cluster_votes.insert(node_pubkey, votes.clone()); + vote_simulator.fill_bank_forks(forks, &cluster_votes); + for vote in votes { + assert!(vote_simulator + .simulate_vote(vote, &node_pubkey, &mut tower,) .is_empty()); } - let mut frozen_banks: Vec<_> = bank_forks + let mut frozen_banks: Vec<_> = vote_simulator + .bank_forks .read() .unwrap() .frozen_banks() @@ -2767,20 +2753,28 @@ pub(crate) mod tests { ReplayStage::compute_bank_stats( &Pubkey::default(), - &bank_forks.read().unwrap().ancestors(), + &vote_simulator.bank_forks.read().unwrap().ancestors(), &mut frozen_banks, &tower, - &mut progress, + &mut vote_simulator.progress, &VoteTracker::default(), &ClusterSlots::default(), - &bank_forks, + &vote_simulator.bank_forks, &mut HashSet::new(), ); frozen_banks.sort_by_key(|bank| bank.slot()); for pair in frozen_banks.windows(2) { - let first = progress.get_fork_stats(pair[0].slot()).unwrap().fork_weight; - let second = progress.get_fork_stats(pair[1].slot()).unwrap().fork_weight; + let first = vote_simulator + .progress + .get_fork_stats(pair[0].slot()) + .unwrap() + .fork_weight; + let second = vote_simulator + .progress + .get_fork_stats(pair[1].slot()) + .unwrap() + .fork_weight; assert!(second >= first); } }