diff --git a/core/Cargo.toml b/core/Cargo.toml index 4d3c59a8ada4f8..d107296bba0e6e 100644 --- a/core/Cargo.toml +++ b/core/Cargo.toml @@ -110,6 +110,7 @@ solana-stake-program = { workspace = true } solana-unified-scheduler-pool = { workspace = true, features = [ "dev-context-only-utils", ] } +solana-vote = { workspace = true, features = ["dev-context-only-utils"] } static_assertions = { workspace = true } systemstat = { workspace = true } test-case = { workspace = true } diff --git a/core/src/commitment_service.rs b/core/src/commitment_service.rs index 03c04ad9ab4c96..5cdcfa94ec593b 100644 --- a/core/src/commitment_service.rs +++ b/core/src/commitment_service.rs @@ -202,19 +202,17 @@ impl AggregateCommitmentService { } let vote_state = if pubkey == node_vote_pubkey { // Override old vote_state in bank with latest one for my own vote pubkey - Ok(node_vote_state) + node_vote_state } else { account.vote_state() }; - if let Ok(vote_state) = vote_state { - Self::aggregate_commitment_for_vote_account( - &mut commitment, - &mut rooted_stake, - vote_state, - ancestors, - *lamports, - ); - } + Self::aggregate_commitment_for_vote_account( + &mut commitment, + &mut rooted_stake, + vote_state, + ancestors, + *lamports, + ); } (commitment, rooted_stake) @@ -546,7 +544,7 @@ mod tests { fn test_highest_super_majority_root_advance() { fn get_vote_state(vote_pubkey: Pubkey, bank: &Bank) -> VoteState { let vote_account = bank.get_vote_account(&vote_pubkey).unwrap(); - vote_account.vote_state().cloned().unwrap() + vote_account.vote_state().clone() } let block_commitment_cache = RwLock::new(BlockCommitmentCache::new_for_tests()); diff --git a/core/src/consensus.rs b/core/src/consensus.rs index 93c80c554c6ab1..96fbbc6b68d0bd 100644 --- a/core/src/consensus.rs +++ b/core/src/consensus.rs @@ -379,20 +379,7 @@ impl Tower { continue; } trace!("{} {} with stake {}", vote_account_pubkey, key, voted_stake); - let mut vote_state = match account.vote_state().cloned() { - Err(_) => { - datapoint_warn!( - "tower_warn", - ( - "warn", - format!("Unable to get vote_state from account {key}"), - String - ), - ); - continue; - } - Ok(vote_state) => vote_state, - }; + let mut vote_state = account.vote_state().clone(); for vote in &vote_state.votes { lockout_intervals .entry(vote.lockout.last_locked_out_slot()) @@ -591,7 +578,7 @@ impl Tower { pub fn last_voted_slot_in_bank(bank: &Bank, vote_account_pubkey: &Pubkey) -> Option { let vote_account = bank.get_vote_account(vote_account_pubkey)?; let vote_state = vote_account.vote_state(); - vote_state.as_ref().ok()?.last_voted_slot() + vote_state.last_voted_slot() } pub fn record_bank_vote(&mut self, bank: &Bank) -> Option { @@ -1576,10 +1563,7 @@ impl Tower { bank: &Bank, ) { if let Some(vote_account) = bank.get_vote_account(vote_account_pubkey) { - self.vote_state = vote_account - .vote_state() - .cloned() - .expect("vote_account isn't a VoteState?"); + self.vote_state = vote_account.vote_state().clone(); self.initialize_root(root); self.initialize_lockouts(|v| v.slot() > root); trace!( @@ -2428,7 +2412,7 @@ pub mod test { .get_vote_account(&vote_pubkey) .unwrap(); let state = observed.vote_state(); - info!("observed tower: {:#?}", state.as_ref().unwrap().votes); + info!("observed tower: {:#?}", state.votes); let num_slots_to_try = 200; cluster_votes diff --git a/core/src/consensus/progress_map.rs b/core/src/consensus/progress_map.rs index a06f51b2001534..447ebff0f8361e 100644 --- a/core/src/consensus/progress_map.rs +++ b/core/src/consensus/progress_map.rs @@ -420,19 +420,7 @@ impl ProgressMap { #[cfg(test)] mod test { - use { - super::*, - solana_sdk::account::{Account, AccountSharedData}, - solana_vote::vote_account::VoteAccount, - }; - - fn new_test_vote_account() -> VoteAccount { - let account = AccountSharedData::from(Account { - owner: solana_vote_program::id(), - ..Account::default() - }); - VoteAccount::try_from(account).unwrap() - } + use {super::*, solana_vote::vote_account::VoteAccount}; #[test] fn test_add_vote_pubkey() { @@ -467,7 +455,7 @@ mod test { let epoch_vote_accounts: HashMap<_, _> = vote_account_pubkeys .iter() .skip(num_vote_accounts - staked_vote_accounts) - .map(|pubkey| (*pubkey, (1, new_test_vote_account()))) + .map(|pubkey| (*pubkey, (1, VoteAccount::new_random()))) .collect(); let mut stats = PropagatedStats::default(); @@ -509,7 +497,7 @@ mod test { let epoch_vote_accounts: HashMap<_, _> = vote_account_pubkeys .iter() .skip(num_vote_accounts - staked_vote_accounts) - .map(|pubkey| (*pubkey, (1, new_test_vote_account()))) + .map(|pubkey| (*pubkey, (1, VoteAccount::new_random()))) .collect(); stats.add_node_pubkey_internal(&node_pubkey, &vote_account_pubkeys, &epoch_vote_accounts); assert!(stats.propagated_node_ids.contains(&node_pubkey)); diff --git a/core/src/replay_stage.rs b/core/src/replay_stage.rs index f2cedd7d731aac..398a15601ba3ee 100644 --- a/core/src/replay_stage.rs +++ b/core/src/replay_stage.rs @@ -2485,17 +2485,6 @@ impl ReplayStage { Some(vote_account) => vote_account, }; let vote_state = vote_account.vote_state(); - let vote_state = match vote_state.as_ref() { - Err(_) => { - warn!( - "Vote account {} is unreadable. Unable to vote", - vote_account_pubkey, - ); - return GenerateVoteTxResult::Failed; - } - Ok(vote_state) => vote_state, - }; - if vote_state.node_pubkey != node_keypair.pubkey() { info!( "Vote account node_pubkey mismatch: {} (expected: {}). Unable to vote", @@ -3473,9 +3462,7 @@ impl ReplayStage { let Some(vote_account) = bank.get_vote_account(my_vote_pubkey) else { return; }; - let Ok(mut bank_vote_state) = vote_account.vote_state().cloned() else { - return; - }; + let mut bank_vote_state = vote_account.vote_state().clone(); if bank_vote_state.last_voted_slot() <= tower.vote_state.last_voted_slot() { return; } @@ -7612,10 +7599,7 @@ pub(crate) mod tests { let vote_account = expired_bank_child .get_vote_account(&my_vote_pubkey) .unwrap(); - assert_eq!( - vote_account.vote_state().as_ref().unwrap().tower(), - vec![0, 1] - ); + assert_eq!(vote_account.vote_state().tower(), vec![0, 1]); expired_bank_child.fill_bank_with_ticks_for_tests(); expired_bank_child.freeze(); diff --git a/core/src/validator.rs b/core/src/validator.rs index 016514dd817166..0f99e4c4768497 100644 --- a/core/src/validator.rs +++ b/core/src/validator.rs @@ -2488,7 +2488,7 @@ fn get_stake_percent_in_gossip(bank: &Bank, cluster_info: &ClusterInfo, log: boo if activated_stake == 0 { continue; } - let vote_state_node_pubkey = vote_account.node_pubkey().copied().unwrap_or_default(); + let vote_state_node_pubkey = *vote_account.node_pubkey(); if let Some(peer) = peers.get(&vote_state_node_pubkey) { if peer.shred_version() == my_shred_version { diff --git a/core/src/vote_simulator.rs b/core/src/vote_simulator.rs index 31395f65a42a6e..60333dae94c12d 100644 --- a/core/src/vote_simulator.rs +++ b/core/src/vote_simulator.rs @@ -104,7 +104,7 @@ impl VoteSimulator { let tower_sync = if let Some(vote_account) = parent_bank.get_vote_account(&keypairs.vote_keypair.pubkey()) { - let mut vote_state = vote_account.vote_state().unwrap().clone(); + let mut vote_state = vote_account.vote_state().clone(); process_vote_unchecked( &mut vote_state, solana_vote_program::vote_state::Vote::new( @@ -143,12 +143,7 @@ impl VoteSimulator { .get_vote_account(&keypairs.vote_keypair.pubkey()) .unwrap(); let state = vote_account.vote_state(); - assert!(state - .as_ref() - .unwrap() - .votes - .iter() - .any(|lockout| lockout.slot() == parent)); + assert!(state.votes.iter().any(|lockout| lockout.slot() == parent)); } } while new_bank.tick_height() < new_bank.max_tick_height() { diff --git a/ledger-tool/src/main.rs b/ledger-tool/src/main.rs index d9a3a60d2f4600..ac9e3fb9f929c2 100644 --- a/ledger-tool/src/main.rs +++ b/ledger-tool/src/main.rs @@ -206,7 +206,6 @@ fn graph_forks(bank_forks: &BankForks, config: &GraphConfig) -> String { // Search all forks and collect the last vote made by each validator let mut last_votes = HashMap::new(); - let default_vote_state = VoteState::default(); for fork_slot in &fork_slots { let bank = &bank_forks[*fork_slot]; @@ -217,7 +216,6 @@ fn graph_forks(bank_forks: &BankForks, config: &GraphConfig) -> String { .sum(); for (stake, vote_account) in bank.vote_accounts().values() { let vote_state = vote_account.vote_state(); - let vote_state = vote_state.unwrap_or(&default_vote_state); if let Some(last_vote) = vote_state.votes.iter().last() { let entry = last_votes.entry(vote_state.node_pubkey).or_insert(( last_vote.slot(), @@ -258,7 +256,6 @@ fn graph_forks(bank_forks: &BankForks, config: &GraphConfig) -> String { loop { for (_, vote_account) in bank.vote_accounts().values() { let vote_state = vote_account.vote_state(); - let vote_state = vote_state.unwrap_or(&default_vote_state); if let Some(last_vote) = vote_state.votes.iter().last() { let validator_votes = all_votes.entry(vote_state.node_pubkey).or_default(); validator_votes diff --git a/ledger/src/blockstore_processor.rs b/ledger/src/blockstore_processor.rs index 304cc549e57f5b..98701cf3468209 100644 --- a/ledger/src/blockstore_processor.rs +++ b/ledger/src/blockstore_processor.rs @@ -1870,7 +1870,6 @@ fn load_frozen_forks( let new_root_bank = { if bank_forks.read().unwrap().root() >= max_root { supermajority_root_from_vote_accounts( - bank.slot(), bank.total_epoch_stake(), &bank.vote_accounts(), ).and_then(|supermajority_root| { @@ -2005,27 +2004,17 @@ fn supermajority_root(roots: &[(Slot, u64)], total_epoch_stake: u64) -> Option Option { let mut roots_stakes: Vec<(Slot, u64)> = vote_accounts - .iter() - .filter_map(|(key, (stake, account))| { + .values() + .filter_map(|(stake, account)| { if *stake == 0 { return None; } - match account.vote_state().as_ref() { - Err(_) => { - warn!( - "Unable to get vote_state from account {} in bank: {}", - key, bank_slot - ); - None - } - Ok(vote_state) => Some((vote_state.root_slot?, *stake)), - } + Some((account.vote_state().root_slot?, *stake)) }) .collect(); @@ -4603,23 +4592,20 @@ pub mod tests { }; let total_stake = 10; - let slot = 100; // Supermajority root should be None - assert!( - supermajority_root_from_vote_accounts(slot, total_stake, &HashMap::default()).is_none() - ); + assert!(supermajority_root_from_vote_accounts(total_stake, &HashMap::default()).is_none()); // Supermajority root should be None let roots_stakes = vec![(8, 1), (3, 1), (4, 1), (8, 1)]; let accounts = convert_to_vote_accounts(roots_stakes); - assert!(supermajority_root_from_vote_accounts(slot, total_stake, &accounts).is_none()); + assert!(supermajority_root_from_vote_accounts(total_stake, &accounts).is_none()); // Supermajority root should be 4, has 7/10 of the stake let roots_stakes = vec![(8, 1), (3, 1), (4, 1), (8, 5)]; let accounts = convert_to_vote_accounts(roots_stakes); assert_eq!( - supermajority_root_from_vote_accounts(slot, total_stake, &accounts).unwrap(), + supermajority_root_from_vote_accounts(total_stake, &accounts).unwrap(), 4 ); @@ -4627,7 +4613,7 @@ pub mod tests { let roots_stakes = vec![(8, 1), (3, 1), (4, 1), (8, 6)]; let accounts = convert_to_vote_accounts(roots_stakes); assert_eq!( - supermajority_root_from_vote_accounts(slot, total_stake, &accounts).unwrap(), + supermajority_root_from_vote_accounts(total_stake, &accounts).unwrap(), 8 ); } diff --git a/programs/bpf_loader/Cargo.toml b/programs/bpf_loader/Cargo.toml index ace6f2ed9c0b83..4c085663513300 100644 --- a/programs/bpf_loader/Cargo.toml +++ b/programs/bpf_loader/Cargo.toml @@ -34,7 +34,7 @@ assert_matches = { workspace = true } memoffset = { workspace = true } rand = { workspace = true } solana-sdk = { workspace = true, features = ["dev-context-only-utils"] } -solana-vote = { workspace = true } +solana-vote = { workspace = true, features = ["dev-context-only-utils"] } test-case = { workspace = true } [lib] diff --git a/programs/bpf_loader/src/syscalls/mod.rs b/programs/bpf_loader/src/syscalls/mod.rs index e70a266f340917..876b734af19156 100644 --- a/programs/bpf_loader/src/syscalls/mod.rs +++ b/programs/bpf_loader/src/syscalls/mod.rs @@ -4820,15 +4820,7 @@ mod tests { let mut vote_accounts_map = HashMap::new(); vote_accounts_map.insert( vote_address, - ( - expected_epoch_stake, - VoteAccount::try_from(AccountSharedData::new( - 0, - 0, - &solana_sdk::vote::program::id(), - )) - .unwrap(), - ), + (expected_epoch_stake, VoteAccount::new_random()), ); with_mock_invoke_context!(invoke_context, transaction_context, vec![]); diff --git a/rpc/src/rpc.rs b/rpc/src/rpc.rs index db4d7bf9e69b53..0cd32ab3ff84b1 100644 --- a/rpc/src/rpc.rs +++ b/rpc/src/rpc.rs @@ -92,7 +92,7 @@ use { TransactionBinaryEncoding, TransactionConfirmationStatus, TransactionStatus, UiConfirmedBlock, UiTransactionEncoding, }, - solana_vote_program::vote_state::{VoteState, MAX_LOCKOUT_HISTORY}, + solana_vote_program::vote_state::MAX_LOCKOUT_HISTORY, spl_token_2022::{ extension::{ interest_bearing_mint::InterestBearingConfig, BaseStateWithExtensions, @@ -1004,7 +1004,6 @@ impl JsonRpcRequestProcessor { let epoch_vote_accounts = bank .epoch_vote_accounts(bank.get_epoch_and_slot_index(bank.slot()).0) .ok_or_else(Error::invalid_request)?; - let default_vote_state = VoteState::default(); let delinquent_validator_slot_distance = config .delinquent_slot_distance .unwrap_or(DELINQUENT_VALIDATOR_SLOT_DISTANCE); @@ -1021,7 +1020,6 @@ impl JsonRpcRequestProcessor { } let vote_state = account.vote_state(); - let vote_state = vote_state.unwrap_or(&default_vote_state); let last_vote = if let Some(vote) = vote_state.votes.iter().last() { vote.slot() } else { @@ -4360,6 +4358,7 @@ pub mod tests { transaction::{ self, SimpleAddressLoader, Transaction, TransactionError, TransactionVersion, }, + vote::state::VoteState, }, solana_transaction_status::{ EncodedConfirmedBlock, EncodedTransaction, EncodedTransactionWithStatusMeta, diff --git a/runtime/src/bank.rs b/runtime/src/bank.rs index 34a1cd9d70b89c..4b6377c0c53e3c 100644 --- a/runtime/src/bank.rs +++ b/runtime/src/bank.rs @@ -2282,12 +2282,8 @@ impl Bank { invalid_vote_keys.insert(vote_pubkey, InvalidCacheEntryReason::WrongOwner); return None; } - let Ok(vote_state) = vote_account.vote_state().cloned() else { - invalid_vote_keys.insert(vote_pubkey, InvalidCacheEntryReason::BadState); - return None; - }; let vote_with_stake_delegations = VoteWithStakeDelegations { - vote_state: Arc::new(vote_state), + vote_state: Arc::new(vote_account.vote_state().clone()), vote_account: AccountSharedData::from(vote_account), delegations: Vec::default(), }; @@ -2705,7 +2701,6 @@ impl Bank { let vote_accounts = self.vote_accounts(); let recent_timestamps = vote_accounts.iter().filter_map(|(pubkey, (_, account))| { let vote_state = account.vote_state(); - let vote_state = vote_state.as_ref().ok()?; let slot_delta = self.slot().checked_sub(vote_state.last_timestamp.slot)?; (slot_delta <= slots_per_epoch).then_some({ ( @@ -2881,7 +2876,7 @@ impl Bank { // up and can be used to set the collector id to the highest staked // node. If no staked nodes exist, allow fallback to an unstaked test // collector id during tests. - let collector_id = self.stakes_cache.stakes().highest_staked_node(); + let collector_id = self.stakes_cache.stakes().highest_staked_node().copied(); #[cfg(feature = "dev-context-only-utils")] let collector_id = collector_id.or(collector_id_for_tests); self.collector_id = diff --git a/runtime/src/bank/fee_distribution.rs b/runtime/src/bank/fee_distribution.rs index 4dc511a5eee95c..383521c016179f 100644 --- a/runtime/src/bank/fee_distribution.rs +++ b/runtime/src/bank/fee_distribution.rs @@ -203,7 +203,7 @@ impl Bank { None } else { total_staked += *staked; - Some((*account.node_pubkey()?, *staked)) + Some((*account.node_pubkey(), *staked)) } }) .collect::>(); diff --git a/runtime/src/bank/partitioned_epoch_rewards/calculation.rs b/runtime/src/bank/partitioned_epoch_rewards/calculation.rs index 9d929accb5cdb1..257a531f3e04d4 100644 --- a/runtime/src/bank/partitioned_epoch_rewards/calculation.rs +++ b/runtime/src/bank/partitioned_epoch_rewards/calculation.rs @@ -385,7 +385,7 @@ impl Bank { if vote_account.owner() != &solana_vote_program { return None; } - let vote_state = vote_account.vote_state().cloned().ok()?; + let vote_state = vote_account.vote_state(); let pre_lamport = stake_account.lamports(); @@ -393,7 +393,7 @@ impl Bank { rewarded_epoch, stake_state, &mut stake_account, - &vote_state, + vote_state, &point_value, stake_history, reward_calc_tracer.as_ref(), @@ -407,13 +407,14 @@ impl Bank { "calculated reward: {} {} {} {}", stake_pubkey, pre_lamport, post_lamport, stakers_reward ); + let commission = vote_state.commission; // track voter rewards let mut voters_reward_entry = vote_account_rewards .entry(vote_pubkey) .or_insert(VoteReward { + commission, vote_account: vote_account.into(), - commission: vote_state.commission, vote_rewards: 0, vote_needs_store: false, }); @@ -438,7 +439,7 @@ impl Bank { reward_type: RewardType::Staking, lamports: i64::try_from(stakers_reward).unwrap(), post_balance, - commission: Some(vote_state.commission), + commission: Some(commission), }, stake, }); @@ -508,13 +509,10 @@ impl Bank { if vote_account.owner() != &solana_vote_program { return 0; } - let Ok(vote_state) = vote_account.vote_state() else { - return 0; - }; solana_stake_program::points::calculate_points( stake_account.stake_state(), - vote_state, + vote_account.vote_state(), stake_history, new_warmup_cooldown_rate_epoch, ) diff --git a/runtime/src/bank/serde_snapshot.rs b/runtime/src/bank/serde_snapshot.rs index 258c49593f0e12..bdcf7ae6215f72 100644 --- a/runtime/src/bank/serde_snapshot.rs +++ b/runtime/src/bank/serde_snapshot.rs @@ -535,7 +535,7 @@ mod tests { #[cfg_attr( feature = "frozen-abi", derive(AbiExample), - frozen_abi(digest = "J7MnnLU99fYk2hfZPjdqyTYxgHstwRUDk2Yr8fFnXxFp") + frozen_abi(digest = "HQYDRuCaM5V1ggSuMPTKT5Mu2vE5HX4y4ZM1Xuorx6My") )] #[derive(Serialize)] pub struct BankAbiTestWrapper { diff --git a/runtime/src/bank/tests.rs b/runtime/src/bank/tests.rs index dfbd449731179a..dc8175cebd9550 100644 --- a/runtime/src/bank/tests.rs +++ b/runtime/src/bank/tests.rs @@ -4012,12 +4012,8 @@ fn test_bank_epoch_vote_accounts() { accounts .iter() .filter_map(|(pubkey, (stake, account))| { - if let Ok(vote_state) = account.vote_state().as_ref() { - if vote_state.node_pubkey == leader_pubkey { - Some((*pubkey, *stake)) - } else { - None - } + if account.node_pubkey() == &leader_pubkey { + Some((*pubkey, *stake)) } else { None } diff --git a/runtime/src/epoch_stakes.rs b/runtime/src/epoch_stakes.rs index 84b6bdc40a6345..fa4de74d8e23cb 100644 --- a/runtime/src/epoch_stakes.rs +++ b/runtime/src/epoch_stakes.rs @@ -101,20 +101,6 @@ impl EpochStakes { .iter() .filter_map(|(key, (stake, account))| { let vote_state = account.vote_state(); - let vote_state = match vote_state.as_ref() { - Err(_) => { - datapoint_warn!( - "parse_epoch_vote_accounts", - ( - "warn", - format!("Unable to get vote_state from account {key}"), - String - ), - ); - return None; - } - Ok(vote_state) => vote_state, - }; if *stake > 0 { if let Some(authorized_voter) = vote_state diff --git a/runtime/src/snapshot_minimizer.rs b/runtime/src/snapshot_minimizer.rs index b48f1832fc0256..7aeba40adea04d 100644 --- a/runtime/src/snapshot_minimizer.rs +++ b/runtime/src/snapshot_minimizer.rs @@ -158,9 +158,8 @@ impl<'a> SnapshotMinimizer<'a> { .par_iter() .for_each(|(pubkey, (_stake, vote_account))| { self.minimized_account_set.insert(*pubkey); - if let Ok(vote_state) = vote_account.vote_state().as_ref() { - self.minimized_account_set.insert(vote_state.node_pubkey); - } + self.minimized_account_set + .insert(*vote_account.node_pubkey()); }); } diff --git a/runtime/src/stakes.rs b/runtime/src/stakes.rs index d79d8e43492687..f878510402e929 100644 --- a/runtime/src/stakes.rs +++ b/runtime/src/stakes.rs @@ -98,11 +98,6 @@ impl StakesCache { if VoteStateVersions::is_correct_size_and_initialized(account.data()) { match VoteAccount::try_from(account.to_account_shared_data()) { Ok(vote_account) => { - { - // Called to eagerly deserialize vote state - let _res = vote_account.vote_state(); - } - // drop the old account after releasing the lock let _old_vote_account = { let mut stakes = self.0.write().unwrap(); @@ -428,7 +423,6 @@ impl Stakes { new_rate_activation_epoch: Option, ) -> Option { debug_assert_ne!(vote_account.lamports(), 0u64); - debug_assert!(vote_account.is_deserialized()); let stake_delegations = &self.stake_delegations; self.vote_accounts.insert(*vote_pubkey, vote_account, || { @@ -507,9 +501,9 @@ impl Stakes { &self.stake_delegations } - pub(crate) fn highest_staked_node(&self) -> Option { + pub(crate) fn highest_staked_node(&self) -> Option<&Pubkey> { let vote_account = self.vote_accounts.find_max_by_delegated_stake()?; - vote_account.node_pubkey().copied() + Some(vote_account.node_pubkey()) } } @@ -835,7 +829,7 @@ pub(crate) mod tests { let vote11_node_pubkey = vote_state::from(&vote11_account).unwrap().node_pubkey; - let highest_staked_node = stakes_cache.stakes().highest_staked_node(); + let highest_staked_node = stakes_cache.stakes().highest_staked_node().copied(); assert_eq!(highest_staked_node, Some(vote11_node_pubkey)); } diff --git a/vote/Cargo.toml b/vote/Cargo.toml index 2eb821eac407a4..89f3b5e433f49f 100644 --- a/vote/Cargo.toml +++ b/vote/Cargo.toml @@ -12,6 +12,7 @@ edition = { workspace = true } [dependencies] itertools = { workspace = true } log = { workspace = true } +rand = { workspace = true, optional = true } serde = { workspace = true, features = ["rc"] } serde_derive = { workspace = true } solana-frozen-abi = { workspace = true, optional = true } @@ -34,7 +35,7 @@ targets = ["x86_64-unknown-linux-gnu"] rustc_version = { workspace = true, optional = true } [features] -dev-context-only-utils = [] +dev-context-only-utils = ["dep:rand"] frozen-abi = [ "dep:rustc_version", "dep:solana-frozen-abi", diff --git a/vote/src/vote_account.rs b/vote/src/vote_account.rs index 14cc788cca13d9..8155d1540f04e7 100644 --- a/vote/src/vote_account.rs +++ b/vote/src/vote_account.rs @@ -1,6 +1,9 @@ use { itertools::Itertools, - serde::ser::{Serialize, Serializer}, + serde::{ + de::{MapAccess, Visitor}, + ser::{Serialize, Serializer}, + }, solana_sdk::{ account::{AccountSharedData, ReadableAccount}, instruction::InstructionError, @@ -10,6 +13,7 @@ use { std::{ cmp::Ordering, collections::{hash_map::Entry, HashMap}, + fmt, iter::FromIterator, mem, sync::{Arc, OnceLock}, @@ -18,8 +22,7 @@ use { }; #[cfg_attr(feature = "frozen-abi", derive(AbiExample))] -#[derive(Clone, Debug, PartialEq, Deserialize)] -#[serde(try_from = "AccountSharedData")] +#[derive(Clone, Debug, PartialEq)] pub struct VoteAccount(Arc); #[derive(Debug, Error)] @@ -34,17 +37,17 @@ pub enum Error { #[derive(Debug)] struct VoteAccountInner { account: AccountSharedData, - vote_state: OnceLock>, + vote_state: VoteState, } pub type VoteAccountsHashMap = HashMap; - #[cfg_attr(feature = "frozen-abi", derive(AbiExample))] -#[derive(Clone, Debug, Deserialize)] -#[serde(from = "Arc")] +#[derive(Clone, Debug, Serialize, Deserialize)] pub struct VoteAccounts { + #[serde(deserialize_with = "deserialize_accounts_hash_map")] vote_accounts: Arc, // Inner Arc is meant to implement copy-on-write semantics. + #[serde(skip)] staked_nodes: OnceLock< Arc< HashMap< @@ -68,22 +71,49 @@ impl VoteAccount { self.0.account.owner() } - pub fn vote_state(&self) -> Result<&VoteState, &Error> { - // VoteState::deserialize deserializes a VoteStateVersions and then - // calls VoteStateVersions::convert_to_current. - self.0 - .vote_state - .get_or_init(|| VoteState::deserialize(self.0.account.data()).map_err(Error::from)) - .as_ref() - } - - pub fn is_deserialized(&self) -> bool { - self.0.vote_state.get().is_some() + pub fn vote_state(&self) -> &VoteState { + &self.0.vote_state } /// VoteState.node_pubkey of this vote-account. - pub fn node_pubkey(&self) -> Option<&Pubkey> { - self.vote_state().ok().map(|s| &s.node_pubkey) + pub fn node_pubkey(&self) -> &Pubkey { + &self.0.vote_state.node_pubkey + } + + #[cfg(feature = "dev-context-only-utils")] + pub fn new_random() -> VoteAccount { + use { + rand::Rng as _, + solana_sdk::{ + clock::Clock, + vote::state::{VoteInit, VoteStateVersions}, + }, + }; + + let mut rng = rand::thread_rng(); + + let vote_init = VoteInit { + node_pubkey: Pubkey::new_unique(), + authorized_voter: Pubkey::new_unique(), + authorized_withdrawer: Pubkey::new_unique(), + commission: rng.gen(), + }; + let clock = Clock { + slot: rng.gen(), + epoch_start_timestamp: rng.gen(), + epoch: rng.gen(), + leader_schedule_epoch: rng.gen(), + unix_timestamp: rng.gen(), + }; + let vote_state = VoteState::new(&vote_init, &clock); + let account = AccountSharedData::new_data( + rng.gen(), // lamports + &VoteStateVersions::new_current(vote_state.clone()), + &solana_sdk::vote::program::id(), // owner + ) + .unwrap(); + + VoteAccount::try_from(account).unwrap() } } @@ -103,9 +133,7 @@ impl VoteAccounts { self.vote_accounts .values() .filter(|(stake, _)| *stake != 0u64) - .filter_map(|(stake, vote_account)| { - Some((*vote_account.node_pubkey()?, stake)) - }) + .map(|(stake, vote_account)| (*vote_account.node_pubkey(), stake)) .into_grouping_map() .aggregate(|acc, _node_pubkey, stake| { Some(acc.unwrap_or_default() + stake) @@ -164,7 +192,7 @@ impl VoteAccounts { // The node keys have changed, we move the stake from the old node to the // new one Self::do_sub_node_stake(staked_nodes, *stake, old_node_pubkey); - Self::do_add_node_stake(staked_nodes, *stake, new_node_pubkey.copied()); + Self::do_add_node_stake(staked_nodes, *stake, *new_node_pubkey); } } @@ -175,11 +203,7 @@ impl VoteAccounts { // This is a new vote account. We don't know the stake yet, so we need to compute it. let (stake, vote_account) = entry.insert((calculate_stake(), new_vote_account)); if let Some(staked_nodes) = self.staked_nodes.get_mut() { - Self::do_add_node_stake( - staked_nodes, - *stake, - vote_account.node_pubkey().copied(), - ); + Self::do_add_node_stake(staked_nodes, *stake, *vote_account.node_pubkey()); } None } @@ -220,24 +244,22 @@ impl VoteAccounts { return; }; - VoteAccounts::do_add_node_stake(staked_nodes, stake, vote_account.node_pubkey().copied()); + VoteAccounts::do_add_node_stake(staked_nodes, stake, *vote_account.node_pubkey()); } fn do_add_node_stake( staked_nodes: &mut Arc>, stake: u64, - node_pubkey: Option, + node_pubkey: Pubkey, ) { if stake == 0u64 { return; } - node_pubkey.map(|node_pubkey| { - Arc::make_mut(staked_nodes) - .entry(node_pubkey) - .and_modify(|s| *s += stake) - .or_insert(stake) - }); + Arc::make_mut(staked_nodes) + .entry(node_pubkey) + .and_modify(|s| *s += stake) + .or_insert(stake); } fn sub_node_stake(&mut self, stake: u64, vote_account: &VoteAccount) { @@ -251,24 +273,22 @@ impl VoteAccounts { fn do_sub_node_stake( staked_nodes: &mut Arc>, stake: u64, - node_pubkey: Option<&Pubkey>, + node_pubkey: &Pubkey, ) { if stake == 0u64 { return; } - if let Some(node_pubkey) = node_pubkey { - let staked_nodes = Arc::make_mut(staked_nodes); - let current_stake = staked_nodes - .get_mut(node_pubkey) - .expect("this should not happen"); - match (*current_stake).cmp(&stake) { - Ordering::Less => panic!("subtraction value exceeds node's stake"), - Ordering::Equal => { - staked_nodes.remove(node_pubkey); - } - Ordering::Greater => *current_stake -= stake, + let staked_nodes = Arc::make_mut(staked_nodes); + let current_stake = staked_nodes + .get_mut(node_pubkey) + .expect("this should not happen"); + match (*current_stake).cmp(&stake) { + Ordering::Less => panic!("subtraction value exceeds node's stake"), + Ordering::Equal => { + staked_nodes.remove(node_pubkey); } + Ordering::Greater => *current_stake -= stake, } } } @@ -303,8 +323,8 @@ impl TryFrom for VoteAccountInner { return Err(Error::InvalidOwner(*account.owner())); } Ok(Self { + vote_state: VoteState::deserialize(account.data()).map_err(Error::InstructionError)?, account, - vote_state: OnceLock::new(), }) } } @@ -368,13 +388,51 @@ impl FromIterator<(Pubkey, (/*stake:*/ u64, VoteAccount))> for VoteAccounts { } } -impl Serialize for VoteAccounts { - fn serialize(&self, serializer: S) -> Result - where - S: Serializer, - { - self.vote_accounts.serialize(serializer) +// This custom deserializer is needed to ensure compatibility at snapshot loading with versions +// before https://github.com/anza-xyz/agave/pull/2659 which would theoretically allow invalid vote +// accounts in VoteAccounts. +// +// In the (near) future we should remove this custom deserializer and make it a hard error when we +// find invalid vote accounts in snapshots. +fn deserialize_accounts_hash_map<'de, D>( + deserializer: D, +) -> Result, D::Error> +where + D: serde::Deserializer<'de>, +{ + struct VoteAccountsVisitor; + + impl<'de> Visitor<'de> for VoteAccountsVisitor { + type Value = Arc; + + fn expecting(&self, formatter: &mut fmt::Formatter) -> fmt::Result { + formatter.write_str("a map of vote accounts") + } + + fn visit_map(self, mut access: M) -> Result + where + M: MapAccess<'de>, + { + let mut accounts = HashMap::new(); + + while let Some((pubkey, (stake, account))) = + access.next_entry::()? + { + match VoteAccount::try_from(account) { + Ok(vote_account) => { + accounts.insert(pubkey, (stake, vote_account)); + } + Err(e) => { + log::warn!("failed to deserialize vote account: {e}"); + } + } + } + + Ok(Arc::new(accounts)) + } } + + deserializer.deserialize_map(VoteAccountsVisitor) } #[cfg(test)] @@ -441,12 +499,10 @@ mod tests { .into_iter() .filter(|(_, (stake, _))| *stake != 0) { - if let Some(node_pubkey) = vote_account.node_pubkey() { - staked_nodes - .entry(*node_pubkey) - .and_modify(|s| *s += *stake) - .or_insert(*stake); - } + staked_nodes + .entry(*vote_account.node_pubkey()) + .and_modify(|s| *s += *stake) + .or_insert(*stake); } staked_nodes } @@ -458,9 +514,7 @@ mod tests { let lamports = account.lamports(); let vote_account = VoteAccount::try_from(account).unwrap(); assert_eq!(lamports, vote_account.lamports()); - assert_eq!(vote_state, *vote_account.vote_state().unwrap()); - // 2nd call to .vote_state() should return the cached value. - assert_eq!(vote_state, *vote_account.vote_state().unwrap()); + assert_eq!(vote_state, *vote_account.vote_state()); } #[test] @@ -468,39 +522,14 @@ mod tests { let mut rng = rand::thread_rng(); let (account, vote_state) = new_rand_vote_account(&mut rng, None); let vote_account = VoteAccount::try_from(account.clone()).unwrap(); - assert_eq!(vote_state, *vote_account.vote_state().unwrap()); - // Assert than VoteAccount has the same wire format as Account. + assert_eq!(vote_state, *vote_account.vote_state()); + // Assert that VoteAccount has the same wire format as Account. assert_eq!( bincode::serialize(&account).unwrap(), bincode::serialize(&vote_account).unwrap() ); } - #[test] - fn test_vote_account_deserialize() { - let mut rng = rand::thread_rng(); - let (account, vote_state) = new_rand_vote_account(&mut rng, None); - let data = bincode::serialize(&account).unwrap(); - let vote_account = VoteAccount::try_from(account).unwrap(); - assert_eq!(vote_state, *vote_account.vote_state().unwrap()); - let other_vote_account: VoteAccount = bincode::deserialize(&data).unwrap(); - assert_eq!(vote_account, other_vote_account); - assert_eq!(vote_state, *other_vote_account.vote_state().unwrap()); - } - - #[test] - fn test_vote_account_round_trip() { - let mut rng = rand::thread_rng(); - let (account, vote_state) = new_rand_vote_account(&mut rng, None); - let vote_account = VoteAccount::try_from(account).unwrap(); - assert_eq!(vote_state, *vote_account.vote_state().unwrap()); - let data = bincode::serialize(&vote_account).unwrap(); - let other_vote_account: VoteAccount = bincode::deserialize(&data).unwrap(); - // Assert that serialize->deserialized returns the same VoteAccount. - assert_eq!(vote_account, other_vote_account); - assert_eq!(vote_state, *other_vote_account.vote_state().unwrap()); - } - #[test] fn test_vote_accounts_serialize() { let mut rng = rand::thread_rng(); @@ -536,6 +565,40 @@ mod tests { assert_eq!(*vote_accounts.vote_accounts, vote_accounts_hash_map); } + #[test] + fn test_vote_accounts_deserialize_invalid_account() { + let mut rng = rand::thread_rng(); + // we'll populate the map with 1 valid and 2 invalid accounts, then ensure that we only get + // the valid one after deserialiation + let mut vote_accounts_hash_map = HashMap::::new(); + + let (valid_account, _) = new_rand_vote_account(&mut rng, None); + vote_accounts_hash_map.insert(Pubkey::new_unique(), (0xAA, valid_account.clone())); + + // bad data + let invalid_account_data = + AccountSharedData::new_data(42, &vec![0xFF; 42], &solana_sdk::vote::program::id()) + .unwrap(); + vote_accounts_hash_map.insert(Pubkey::new_unique(), (0xBB, invalid_account_data)); + + // wrong owner + let invalid_account_key = + AccountSharedData::new_data(42, &valid_account.data().to_vec(), &Pubkey::new_unique()) + .unwrap(); + vote_accounts_hash_map.insert(Pubkey::new_unique(), (0xCC, invalid_account_key)); + + let data = bincode::serialize(&vote_accounts_hash_map).unwrap(); + let options = bincode::options() + .with_fixint_encoding() + .allow_trailing_bytes(); + let mut deserializer = bincode::de::Deserializer::from_slice(&data, options); + let vote_accounts = deserialize_accounts_hash_map(&mut deserializer).unwrap(); + + assert_eq!(vote_accounts.len(), 1); + let (stake, _account) = vote_accounts.values().next().unwrap(); + assert_eq!(*stake, 0xAA); + } + #[test] fn test_staked_nodes() { let mut rng = rand::thread_rng(); @@ -684,7 +747,7 @@ mod tests { let staked_nodes = vote_accounts.staked_nodes(); let (pubkey, (more_stake, vote_account)) = accounts.find(|(_, (stake, _))| *stake != 0).unwrap(); - let node_pubkey = *vote_account.node_pubkey().unwrap(); + let node_pubkey = *vote_account.node_pubkey(); vote_accounts.insert(pubkey, vote_account, || more_stake); assert_ne!(staked_nodes, vote_accounts.staked_nodes()); assert_eq!(