(cherry picked from commit 66946a4680)
Co-authored-by: carllin <wumu727@gmail.com>
			
			
This commit is contained in:
		| @@ -1,13 +1,13 @@ | ||||
| use crate::{ | ||||
|     cluster_info_vote_listener::SlotVoteTracker, consensus::StakeLockout, | ||||
|     replay_stage::SUPERMINORITY_THRESHOLD, | ||||
|     cluster_info_vote_listener::SlotVoteTracker, cluster_slots::SlotPubkeys, | ||||
|     consensus::StakeLockout, replay_stage::SUPERMINORITY_THRESHOLD, | ||||
| }; | ||||
| use solana_ledger::{ | ||||
|     bank_forks::BankForks, | ||||
|     blockstore_processor::{ConfirmationProgress, ConfirmationTiming}, | ||||
| }; | ||||
| use solana_runtime::bank::Bank; | ||||
| use solana_sdk::{clock::Slot, hash::Hash, pubkey::Pubkey}; | ||||
| use solana_sdk::{account::Account, clock::Slot, hash::Hash, pubkey::Pubkey}; | ||||
| use std::{ | ||||
|     collections::{HashMap, HashSet}, | ||||
|     rc::Rc, | ||||
| @@ -179,14 +179,84 @@ pub(crate) struct ForkStats { | ||||
| #[derive(Clone, Default)] | ||||
| pub(crate) struct PropagatedStats { | ||||
|     pub(crate) propagated_validators: HashSet<Rc<Pubkey>>, | ||||
|     pub(crate) propagated_node_ids: HashSet<Rc<Pubkey>>, | ||||
|     pub(crate) propagated_validators_stake: u64, | ||||
|     pub(crate) is_propagated: bool, | ||||
|     pub(crate) is_leader_slot: bool, | ||||
|     pub(crate) prev_leader_slot: Option<Slot>, | ||||
|     pub(crate) slot_vote_tracker: Option<Arc<RwLock<SlotVoteTracker>>>, | ||||
|     pub(crate) cluster_slot_pubkeys: Option<Arc<RwLock<SlotPubkeys>>>, | ||||
|     pub(crate) total_epoch_stake: u64, | ||||
| } | ||||
|  | ||||
| impl PropagatedStats { | ||||
|     pub fn add_vote_pubkey( | ||||
|         &mut self, | ||||
|         vote_pubkey: &Pubkey, | ||||
|         all_pubkeys: &mut HashSet<Rc<Pubkey>>, | ||||
|         stake: u64, | ||||
|     ) { | ||||
|         if !self.propagated_validators.contains(vote_pubkey) { | ||||
|             let mut cached_pubkey: Option<Rc<Pubkey>> = all_pubkeys.get(vote_pubkey).cloned(); | ||||
|             if cached_pubkey.is_none() { | ||||
|                 let new_pubkey = Rc::new(*vote_pubkey); | ||||
|                 all_pubkeys.insert(new_pubkey.clone()); | ||||
|                 cached_pubkey = Some(new_pubkey); | ||||
|             } | ||||
|             let vote_pubkey = cached_pubkey.unwrap(); | ||||
|             self.propagated_validators.insert(vote_pubkey); | ||||
|             self.propagated_validators_stake += stake; | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     pub fn add_node_pubkey( | ||||
|         &mut self, | ||||
|         node_pubkey: &Pubkey, | ||||
|         all_pubkeys: &mut HashSet<Rc<Pubkey>>, | ||||
|         bank: &Bank, | ||||
|     ) { | ||||
|         if !self.propagated_node_ids.contains(node_pubkey) { | ||||
|             let node_vote_accounts = bank | ||||
|                 .epoch_vote_accounts_for_node_id(&node_pubkey) | ||||
|                 .map(|v| &v.vote_accounts); | ||||
|  | ||||
|             if let Some(node_vote_accounts) = node_vote_accounts { | ||||
|                 self.add_node_pubkey_internal( | ||||
|                     node_pubkey, | ||||
|                     all_pubkeys, | ||||
|                     node_vote_accounts, | ||||
|                     bank.epoch_vote_accounts(bank.epoch()) | ||||
|                         .expect("Epoch stakes for bank's own epoch must exist"), | ||||
|                 ); | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     fn add_node_pubkey_internal( | ||||
|         &mut self, | ||||
|         node_pubkey: &Pubkey, | ||||
|         all_pubkeys: &mut HashSet<Rc<Pubkey>>, | ||||
|         vote_account_pubkeys: &[Pubkey], | ||||
|         epoch_vote_accounts: &HashMap<Pubkey, (u64, Account)>, | ||||
|     ) { | ||||
|         let mut cached_pubkey: Option<Rc<Pubkey>> = all_pubkeys.get(node_pubkey).cloned(); | ||||
|         if cached_pubkey.is_none() { | ||||
|             let new_pubkey = Rc::new(*node_pubkey); | ||||
|             all_pubkeys.insert(new_pubkey.clone()); | ||||
|             cached_pubkey = Some(new_pubkey); | ||||
|         } | ||||
|         let node_pubkey = cached_pubkey.unwrap(); | ||||
|         self.propagated_node_ids.insert(node_pubkey); | ||||
|         for vote_account_pubkey in vote_account_pubkeys.iter() { | ||||
|             let stake = epoch_vote_accounts | ||||
|                 .get(vote_account_pubkey) | ||||
|                 .map(|(stake, _)| *stake) | ||||
|                 .unwrap_or(0); | ||||
|             self.add_vote_pubkey(vote_account_pubkey, all_pubkeys, stake); | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| #[derive(Default)] | ||||
| pub(crate) struct ProgressMap { | ||||
|     progress_map: HashMap<Slot, ForkProgress>, | ||||
| @@ -288,6 +358,120 @@ impl ProgressMap { | ||||
| mod test { | ||||
|     use super::*; | ||||
|  | ||||
|     #[test] | ||||
|     fn test_add_vote_pubkey() { | ||||
|         let mut stats = PropagatedStats::default(); | ||||
|         let mut all_pubkeys = HashSet::new(); | ||||
|         let mut vote_pubkey = Pubkey::new_rand(); | ||||
|         all_pubkeys.insert(Rc::new(vote_pubkey.clone())); | ||||
|  | ||||
|         // Add a vote pubkey, the number of references in all_pubkeys | ||||
|         // should be 2 | ||||
|         stats.add_vote_pubkey(&vote_pubkey, &mut all_pubkeys, 1); | ||||
|         assert!(stats.propagated_validators.contains(&vote_pubkey)); | ||||
|         assert_eq!(stats.propagated_validators_stake, 1); | ||||
|         assert_eq!(Rc::strong_count(all_pubkeys.get(&vote_pubkey).unwrap()), 2); | ||||
|  | ||||
|         // Adding it again should change no state since the key already existed | ||||
|         stats.add_vote_pubkey(&vote_pubkey, &mut all_pubkeys, 1); | ||||
|         assert!(stats.propagated_validators.contains(&vote_pubkey)); | ||||
|         assert_eq!(stats.propagated_validators_stake, 1); | ||||
|  | ||||
|         // Addding another pubkey should succeed | ||||
|         vote_pubkey = Pubkey::new_rand(); | ||||
|         stats.add_vote_pubkey(&vote_pubkey, &mut all_pubkeys, 2); | ||||
|         assert!(stats.propagated_validators.contains(&vote_pubkey)); | ||||
|         assert_eq!(stats.propagated_validators_stake, 3); | ||||
|         assert_eq!(Rc::strong_count(all_pubkeys.get(&vote_pubkey).unwrap()), 2); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn test_add_node_pubkey_internal() { | ||||
|         let num_vote_accounts = 10; | ||||
|         let staked_vote_accounts = 5; | ||||
|         let vote_account_pubkeys: Vec<_> = std::iter::repeat_with(|| Pubkey::new_rand()) | ||||
|             .take(num_vote_accounts) | ||||
|             .collect(); | ||||
|         let epoch_vote_accounts: HashMap<_, _> = vote_account_pubkeys | ||||
|             .iter() | ||||
|             .skip(num_vote_accounts - staked_vote_accounts) | ||||
|             .map(|pubkey| (*pubkey, (1, Account::default()))) | ||||
|             .collect(); | ||||
|  | ||||
|         let mut stats = PropagatedStats::default(); | ||||
|         let mut all_pubkeys = HashSet::new(); | ||||
|         let mut node_pubkey = Pubkey::new_rand(); | ||||
|         all_pubkeys.insert(Rc::new(node_pubkey.clone())); | ||||
|  | ||||
|         // Add a vote pubkey, the number of references in all_pubkeys | ||||
|         // should be 2 | ||||
|         stats.add_node_pubkey_internal( | ||||
|             &node_pubkey, | ||||
|             &mut all_pubkeys, | ||||
|             &vote_account_pubkeys, | ||||
|             &epoch_vote_accounts, | ||||
|         ); | ||||
|         assert!(stats.propagated_node_ids.contains(&node_pubkey)); | ||||
|         assert_eq!( | ||||
|             stats.propagated_validators_stake, | ||||
|             staked_vote_accounts as u64 | ||||
|         ); | ||||
|         assert_eq!(Rc::strong_count(all_pubkeys.get(&node_pubkey).unwrap()), 2); | ||||
|  | ||||
|         // Adding it again should not change any state | ||||
|         stats.add_node_pubkey_internal( | ||||
|             &node_pubkey, | ||||
|             &mut all_pubkeys, | ||||
|             &vote_account_pubkeys, | ||||
|             &epoch_vote_accounts, | ||||
|         ); | ||||
|         assert!(stats.propagated_node_ids.contains(&node_pubkey)); | ||||
|         assert_eq!( | ||||
|             stats.propagated_validators_stake, | ||||
|             staked_vote_accounts as u64 | ||||
|         ); | ||||
|  | ||||
|         // Addding another pubkey with same vote accounts should succeed, but stake | ||||
|         // shouldn't increase | ||||
|         node_pubkey = Pubkey::new_rand(); | ||||
|         stats.add_node_pubkey_internal( | ||||
|             &node_pubkey, | ||||
|             &mut all_pubkeys, | ||||
|             &vote_account_pubkeys, | ||||
|             &epoch_vote_accounts, | ||||
|         ); | ||||
|         assert!(stats.propagated_node_ids.contains(&node_pubkey)); | ||||
|         assert_eq!( | ||||
|             stats.propagated_validators_stake, | ||||
|             staked_vote_accounts as u64 | ||||
|         ); | ||||
|         assert_eq!(Rc::strong_count(all_pubkeys.get(&node_pubkey).unwrap()), 2); | ||||
|  | ||||
|         // Addding another pubkey with different vote accounts should succeed | ||||
|         // and increase stake | ||||
|         node_pubkey = Pubkey::new_rand(); | ||||
|         let vote_account_pubkeys: Vec<_> = std::iter::repeat_with(|| Pubkey::new_rand()) | ||||
|             .take(num_vote_accounts) | ||||
|             .collect(); | ||||
|         let epoch_vote_accounts: HashMap<_, _> = vote_account_pubkeys | ||||
|             .iter() | ||||
|             .skip(num_vote_accounts - staked_vote_accounts) | ||||
|             .map(|pubkey| (*pubkey, (1, Account::default()))) | ||||
|             .collect(); | ||||
|         stats.add_node_pubkey_internal( | ||||
|             &node_pubkey, | ||||
|             &mut all_pubkeys, | ||||
|             &vote_account_pubkeys, | ||||
|             &epoch_vote_accounts, | ||||
|         ); | ||||
|         assert!(stats.propagated_node_ids.contains(&node_pubkey)); | ||||
|         assert_eq!( | ||||
|             stats.propagated_validators_stake, | ||||
|             2 * staked_vote_accounts as u64 | ||||
|         ); | ||||
|         assert_eq!(Rc::strong_count(all_pubkeys.get(&node_pubkey).unwrap()), 2); | ||||
|     } | ||||
|  | ||||
|     #[test] | ||||
|     fn test_is_propagated_status_on_construction() { | ||||
|         // If the given ValidatorStakeInfo == None, then this is not | ||||
|   | ||||
		Reference in New Issue
	
	Block a user