From 72c526b2bbc364f8344cfdff9ee5d6c604aa9fca Mon Sep 17 00:00:00 2001 From: Tyera Date: Wed, 3 Apr 2024 14:01:00 -0600 Subject: [PATCH] Move calculation methods; partitioned epoch-rewards reorg, 4 of 5 (#529) * Add calculation sub-submodule * Move calculation methods into sub-submodule * Move unit tests into calculation sub-submodule --- runtime/src/bank.rs | 308 +-------- .../partitioned_epoch_rewards/calculation.rs | 639 ++++++++++++++++++ .../src/bank/partitioned_epoch_rewards/mod.rs | 1 + runtime/src/bank/tests.rs | 282 -------- 4 files changed, 642 insertions(+), 588 deletions(-) create mode 100644 runtime/src/bank/partitioned_epoch_rewards/calculation.rs diff --git a/runtime/src/bank.rs b/runtime/src/bank.rs index 0be5848830dd15..9bbe143de22e70 100644 --- a/runtime/src/bank.rs +++ b/runtime/src/bank.rs @@ -42,13 +42,11 @@ use { builtins::{BuiltinPrototype, BUILTINS}, metrics::*, partitioned_epoch_rewards::{ - CalculateRewardsAndDistributeVoteRewardsResult, EpochRewardCalculateParamInfo, - EpochRewardStatus, PartitionedRewardsCalculation, RewardInterval, - StakeRewardCalculationPartitioned, StakeRewards, VoteRewardsAccounts, + EpochRewardCalculateParamInfo, EpochRewardStatus, PartitionedRewardsCalculation, + RewardInterval, StakeRewards, VoteRewardsAccounts, }, }, bank_forks::BankForks, - epoch_rewards_hasher::hash_rewards_into_partitions, epoch_stakes::{EpochStakes, NodeVoteAccounts}, installed_scheduler_pool::{BankWithScheduler, InstalledSchedulerRwLock}, serde_snapshot::BankIncrementalSnapshotPersistence, @@ -1477,47 +1475,6 @@ impl Bank { ); } - /// Begin the process of calculating and distributing rewards. - /// This process can take multiple slots. - fn begin_partitioned_rewards( - &mut self, - reward_calc_tracer: Option, - thread_pool: &ThreadPool, - parent_epoch: Epoch, - parent_slot: Slot, - parent_block_height: u64, - rewards_metrics: &mut RewardsMetrics, - ) { - let CalculateRewardsAndDistributeVoteRewardsResult { - total_rewards, - distributed_rewards, - stake_rewards_by_partition, - } = self.calculate_rewards_and_distribute_vote_rewards( - parent_epoch, - reward_calc_tracer, - thread_pool, - rewards_metrics, - ); - - let slot = self.slot(); - let credit_start = self.block_height() + self.get_reward_calculation_num_blocks(); - - self.set_epoch_reward_status_active(stake_rewards_by_partition); - - // create EpochRewards sysvar that holds the balance of undistributed rewards with - // (total_rewards, distributed_rewards, credit_start), total capital will increase by (total_rewards - distributed_rewards) - self.create_epoch_rewards_sysvar(total_rewards, distributed_rewards, credit_start); - - datapoint_info!( - "epoch-rewards-status-update", - ("start_slot", slot, i64), - ("start_block_height", self.block_height(), i64), - ("active", 1, i64), - ("parent_slot", parent_slot, i64), - ("parent_block_height", parent_block_height, i64), - ); - } - pub fn byte_limit_for_scans(&self) -> Option { self.rc .accounts @@ -2202,149 +2159,6 @@ impl Bank { } } - /// Calculate rewards from previous epoch to prepare for partitioned distribution. - fn calculate_rewards_for_partitioning( - &self, - prev_epoch: Epoch, - reward_calc_tracer: Option, - thread_pool: &ThreadPool, - metrics: &mut RewardsMetrics, - ) -> PartitionedRewardsCalculation { - let capitalization = self.capitalization(); - let PrevEpochInflationRewards { - validator_rewards, - prev_epoch_duration_in_years, - validator_rate, - foundation_rate, - } = self.calculate_previous_epoch_inflation_rewards(capitalization, prev_epoch); - - let old_vote_balance_and_staked = self.stakes_cache.stakes().vote_balance_and_staked(); - - let (vote_account_rewards, mut stake_rewards) = self - .calculate_validator_rewards( - prev_epoch, - validator_rewards, - reward_calc_tracer, - thread_pool, - metrics, - ) - .unwrap_or_default(); - - let num_partitions = self.get_reward_distribution_num_blocks(&stake_rewards.stake_rewards); - let parent_blockhash = self - .parent() - .expect("Partitioned rewards calculation must still have access to parent Bank.") - .last_blockhash(); - let stake_rewards_by_partition = hash_rewards_into_partitions( - std::mem::take(&mut stake_rewards.stake_rewards), - &parent_blockhash, - num_partitions as usize, - ); - - PartitionedRewardsCalculation { - vote_account_rewards, - stake_rewards_by_partition: StakeRewardCalculationPartitioned { - stake_rewards_by_partition, - total_stake_rewards_lamports: stake_rewards.total_stake_rewards_lamports, - }, - old_vote_balance_and_staked, - validator_rewards, - validator_rate, - foundation_rate, - prev_epoch_duration_in_years, - capitalization, - } - } - - // Calculate rewards from previous epoch and distribute vote rewards - fn calculate_rewards_and_distribute_vote_rewards( - &self, - prev_epoch: Epoch, - reward_calc_tracer: Option, - thread_pool: &ThreadPool, - metrics: &mut RewardsMetrics, - ) -> CalculateRewardsAndDistributeVoteRewardsResult { - let PartitionedRewardsCalculation { - vote_account_rewards, - stake_rewards_by_partition, - old_vote_balance_and_staked, - validator_rewards, - validator_rate, - foundation_rate, - prev_epoch_duration_in_years, - capitalization, - } = self.calculate_rewards_for_partitioning( - prev_epoch, - reward_calc_tracer, - thread_pool, - metrics, - ); - let vote_rewards = self.store_vote_accounts_partitioned(vote_account_rewards, metrics); - - // update reward history of JUST vote_rewards, stake_rewards is vec![] here - self.update_reward_history(vec![], vote_rewards); - - let StakeRewardCalculationPartitioned { - stake_rewards_by_partition, - total_stake_rewards_lamports, - } = stake_rewards_by_partition; - - // the remaining code mirrors `update_rewards_with_thread_pool()` - - let new_vote_balance_and_staked = self.stakes_cache.stakes().vote_balance_and_staked(); - - // This is for vote rewards only. - let validator_rewards_paid = new_vote_balance_and_staked - old_vote_balance_and_staked; - self.assert_validator_rewards_paid(validator_rewards_paid); - - // verify that we didn't pay any more than we expected to - assert!(validator_rewards >= validator_rewards_paid + total_stake_rewards_lamports); - - info!( - "distributed vote rewards: {} out of {}, remaining {}", - validator_rewards_paid, validator_rewards, total_stake_rewards_lamports - ); - - let (num_stake_accounts, num_vote_accounts) = { - let stakes = self.stakes_cache.stakes(); - ( - stakes.stake_delegations().len(), - stakes.vote_accounts().len(), - ) - }; - self.capitalization - .fetch_add(validator_rewards_paid, Relaxed); - - let active_stake = if let Some(stake_history_entry) = - self.stakes_cache.stakes().history().get(prev_epoch) - { - stake_history_entry.effective - } else { - 0 - }; - - datapoint_info!( - "epoch_rewards", - ("slot", self.slot, i64), - ("epoch", prev_epoch, i64), - ("validator_rate", validator_rate, f64), - ("foundation_rate", foundation_rate, f64), - ("epoch_duration_in_years", prev_epoch_duration_in_years, f64), - ("validator_rewards", validator_rewards_paid, i64), - ("active_stake", active_stake, i64), - ("pre_capitalization", capitalization, i64), - ("post_capitalization", self.capitalization(), i64), - ("num_stake_accounts", num_stake_accounts, i64), - ("num_vote_accounts", num_vote_accounts, i64), - ); - - CalculateRewardsAndDistributeVoteRewardsResult { - total_rewards: validator_rewards_paid + total_stake_rewards_lamports, - distributed_rewards: validator_rewards_paid, - stake_rewards_by_partition, - } - } - fn assert_validator_rewards_paid(&self, validator_rewards_paid: u64) { assert_eq!( validator_rewards_paid, @@ -2609,36 +2423,6 @@ impl Bank { } } - /// Calculate epoch reward and return vote and stake rewards. - fn calculate_validator_rewards( - &self, - rewarded_epoch: Epoch, - rewards: u64, - reward_calc_tracer: Option, - thread_pool: &ThreadPool, - metrics: &mut RewardsMetrics, - ) -> Option<(VoteRewardsAccounts, StakeRewardCalculation)> { - let stakes = self.stakes_cache.stakes(); - let reward_calculate_param = self.get_epoch_reward_calculate_param_info(&stakes); - - self.calculate_reward_points_partitioned( - &reward_calculate_param, - rewards, - thread_pool, - metrics, - ) - .map(|point_value| { - self.calculate_stake_vote_rewards( - &reward_calculate_param, - rewarded_epoch, - point_value, - thread_pool, - reward_calc_tracer, - metrics, - ) - }) - } - /// Load, calculate and payout epoch rewards for stake and vote accounts fn pay_validator_rewards_with_thread_pool( &mut self, @@ -2810,68 +2594,6 @@ impl Bank { vote_with_stake_delegations_map } - /// Calculates epoch reward points from stake/vote accounts. - /// Returns reward lamports and points for the epoch or none if points == 0. - fn calculate_reward_points_partitioned( - &self, - reward_calculate_params: &EpochRewardCalculateParamInfo, - rewards: u64, - thread_pool: &ThreadPool, - metrics: &RewardsMetrics, - ) -> Option { - let EpochRewardCalculateParamInfo { - stake_history, - stake_delegations, - cached_vote_accounts, - } = reward_calculate_params; - - let solana_vote_program: Pubkey = solana_vote_program::id(); - - let get_vote_account = |vote_pubkey: &Pubkey| -> Option { - if let Some(vote_account) = cached_vote_accounts.get(vote_pubkey) { - return Some(vote_account.clone()); - } - // If accounts-db contains a valid vote account, then it should - // already have been cached in cached_vote_accounts; so the code - // below is only for sanity checking, and can be removed once - // the cache is deemed to be reliable. - let account = self.get_account_with_fixed_root(vote_pubkey)?; - VoteAccount::try_from(account).ok() - }; - - let new_warmup_cooldown_rate_epoch = self.new_warmup_cooldown_rate_epoch(); - let (points, measure_us) = measure_us!(thread_pool.install(|| { - stake_delegations - .par_iter() - .map(|(_stake_pubkey, stake_account)| { - let delegation = stake_account.delegation(); - let vote_pubkey = delegation.voter_pubkey; - - let Some(vote_account) = get_vote_account(&vote_pubkey) else { - return 0; - }; - 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, - stake_history, - new_warmup_cooldown_rate_epoch, - ) - .unwrap_or(0) - }) - .sum::() - })); - metrics.calculate_points_us.fetch_add(measure_us, Relaxed); - - (points > 0).then_some(PointValue { rewards, points }) - } - fn calculate_reward_points( &self, vote_with_stake_delegations_map: &VoteWithStakeDelegationsMap, @@ -3167,32 +2889,6 @@ impl Bank { .fetch_add(now.elapsed().as_micros() as u64, Relaxed); } - fn store_vote_accounts_partitioned( - &self, - vote_account_rewards: VoteRewardsAccounts, - metrics: &RewardsMetrics, - ) -> Vec<(Pubkey, RewardInfo)> { - let (_, measure_us) = measure_us!({ - // reformat data to make it not sparse. - // `StorableAccounts` does not efficiently handle sparse data. - // Not all entries in `vote_account_rewards.accounts_to_store` have a Some(account) to store. - let to_store = vote_account_rewards - .accounts_to_store - .iter() - .filter_map(|account| account.as_ref()) - .enumerate() - .map(|(i, account)| (&vote_account_rewards.rewards[i].0, account)) - .collect::>(); - self.store_accounts((self.slot(), &to_store[..])); - }); - - metrics - .store_vote_accounts_us - .fetch_add(measure_us, Relaxed); - - vote_account_rewards.rewards - } - fn store_vote_accounts( &self, vote_account_rewards: VoteRewards, diff --git a/runtime/src/bank/partitioned_epoch_rewards/calculation.rs b/runtime/src/bank/partitioned_epoch_rewards/calculation.rs new file mode 100644 index 00000000000000..0921f97a2f0b39 --- /dev/null +++ b/runtime/src/bank/partitioned_epoch_rewards/calculation.rs @@ -0,0 +1,639 @@ +use { + super::{ + Bank, CalculateRewardsAndDistributeVoteRewardsResult, EpochRewardCalculateParamInfo, + PartitionedRewardsCalculation, StakeRewardCalculationPartitioned, VoteRewardsAccounts, + }, + crate::{ + bank::{ + PrevEpochInflationRewards, RewardCalcTracer, RewardCalculationEvent, RewardsMetrics, + StakeRewardCalculation, VoteAccount, + }, + epoch_rewards_hasher::hash_rewards_into_partitions, + }, + log::info, + rayon::{ + iter::{IntoParallelRefIterator, ParallelIterator}, + ThreadPool, + }, + solana_measure::measure_us, + solana_sdk::{ + clock::{Epoch, Slot}, + pubkey::Pubkey, + reward_info::RewardInfo, + }, + solana_stake_program::points::PointValue, + std::sync::atomic::Ordering::Relaxed, +}; + +impl Bank { + /// Begin the process of calculating and distributing rewards. + /// This process can take multiple slots. + pub(in crate::bank) fn begin_partitioned_rewards( + &mut self, + reward_calc_tracer: Option, + thread_pool: &ThreadPool, + parent_epoch: Epoch, + parent_slot: Slot, + parent_block_height: u64, + rewards_metrics: &mut RewardsMetrics, + ) { + let CalculateRewardsAndDistributeVoteRewardsResult { + total_rewards, + distributed_rewards, + stake_rewards_by_partition, + } = self.calculate_rewards_and_distribute_vote_rewards( + parent_epoch, + reward_calc_tracer, + thread_pool, + rewards_metrics, + ); + + let slot = self.slot(); + let credit_start = self.block_height() + self.get_reward_calculation_num_blocks(); + + self.set_epoch_reward_status_active(stake_rewards_by_partition); + + // create EpochRewards sysvar that holds the balance of undistributed rewards with + // (total_rewards, distributed_rewards, credit_start), total capital will increase by (total_rewards - distributed_rewards) + self.create_epoch_rewards_sysvar(total_rewards, distributed_rewards, credit_start); + + datapoint_info!( + "epoch-rewards-status-update", + ("start_slot", slot, i64), + ("start_block_height", self.block_height(), i64), + ("active", 1, i64), + ("parent_slot", parent_slot, i64), + ("parent_block_height", parent_block_height, i64), + ); + } + + // Calculate rewards from previous epoch and distribute vote rewards + fn calculate_rewards_and_distribute_vote_rewards( + &self, + prev_epoch: Epoch, + reward_calc_tracer: Option, + thread_pool: &ThreadPool, + metrics: &mut RewardsMetrics, + ) -> CalculateRewardsAndDistributeVoteRewardsResult { + let PartitionedRewardsCalculation { + vote_account_rewards, + stake_rewards_by_partition, + old_vote_balance_and_staked, + validator_rewards, + validator_rate, + foundation_rate, + prev_epoch_duration_in_years, + capitalization, + } = self.calculate_rewards_for_partitioning( + prev_epoch, + reward_calc_tracer, + thread_pool, + metrics, + ); + let vote_rewards = self.store_vote_accounts_partitioned(vote_account_rewards, metrics); + + // update reward history of JUST vote_rewards, stake_rewards is vec![] here + self.update_reward_history(vec![], vote_rewards); + + let StakeRewardCalculationPartitioned { + stake_rewards_by_partition, + total_stake_rewards_lamports, + } = stake_rewards_by_partition; + + // the remaining code mirrors `update_rewards_with_thread_pool()` + + let new_vote_balance_and_staked = self.stakes_cache.stakes().vote_balance_and_staked(); + + // This is for vote rewards only. + let validator_rewards_paid = new_vote_balance_and_staked - old_vote_balance_and_staked; + self.assert_validator_rewards_paid(validator_rewards_paid); + + // verify that we didn't pay any more than we expected to + assert!(validator_rewards >= validator_rewards_paid + total_stake_rewards_lamports); + + info!( + "distributed vote rewards: {} out of {}, remaining {}", + validator_rewards_paid, validator_rewards, total_stake_rewards_lamports + ); + + let (num_stake_accounts, num_vote_accounts) = { + let stakes = self.stakes_cache.stakes(); + ( + stakes.stake_delegations().len(), + stakes.vote_accounts().len(), + ) + }; + self.capitalization + .fetch_add(validator_rewards_paid, Relaxed); + + let active_stake = if let Some(stake_history_entry) = + self.stakes_cache.stakes().history().get(prev_epoch) + { + stake_history_entry.effective + } else { + 0 + }; + + datapoint_info!( + "epoch_rewards", + ("slot", self.slot, i64), + ("epoch", prev_epoch, i64), + ("validator_rate", validator_rate, f64), + ("foundation_rate", foundation_rate, f64), + ("epoch_duration_in_years", prev_epoch_duration_in_years, f64), + ("validator_rewards", validator_rewards_paid, i64), + ("active_stake", active_stake, i64), + ("pre_capitalization", capitalization, i64), + ("post_capitalization", self.capitalization(), i64), + ("num_stake_accounts", num_stake_accounts, i64), + ("num_vote_accounts", num_vote_accounts, i64), + ); + + CalculateRewardsAndDistributeVoteRewardsResult { + total_rewards: validator_rewards_paid + total_stake_rewards_lamports, + distributed_rewards: validator_rewards_paid, + stake_rewards_by_partition, + } + } + + fn store_vote_accounts_partitioned( + &self, + vote_account_rewards: VoteRewardsAccounts, + metrics: &RewardsMetrics, + ) -> Vec<(Pubkey, RewardInfo)> { + let (_, measure_us) = measure_us!({ + // reformat data to make it not sparse. + // `StorableAccounts` does not efficiently handle sparse data. + // Not all entries in `vote_account_rewards.accounts_to_store` have a Some(account) to store. + let to_store = vote_account_rewards + .accounts_to_store + .iter() + .filter_map(|account| account.as_ref()) + .enumerate() + .map(|(i, account)| (&vote_account_rewards.rewards[i].0, account)) + .collect::>(); + self.store_accounts((self.slot(), &to_store[..])); + }); + + metrics + .store_vote_accounts_us + .fetch_add(measure_us, Relaxed); + + vote_account_rewards.rewards + } + + /// Calculate rewards from previous epoch to prepare for partitioned distribution. + pub(in crate::bank) fn calculate_rewards_for_partitioning( + &self, + prev_epoch: Epoch, + reward_calc_tracer: Option, + thread_pool: &ThreadPool, + metrics: &mut RewardsMetrics, + ) -> PartitionedRewardsCalculation { + let capitalization = self.capitalization(); + let PrevEpochInflationRewards { + validator_rewards, + prev_epoch_duration_in_years, + validator_rate, + foundation_rate, + } = self.calculate_previous_epoch_inflation_rewards(capitalization, prev_epoch); + + let old_vote_balance_and_staked = self.stakes_cache.stakes().vote_balance_and_staked(); + + let (vote_account_rewards, mut stake_rewards) = self + .calculate_validator_rewards( + prev_epoch, + validator_rewards, + reward_calc_tracer, + thread_pool, + metrics, + ) + .unwrap_or_default(); + + let num_partitions = self.get_reward_distribution_num_blocks(&stake_rewards.stake_rewards); + let parent_blockhash = self + .parent() + .expect("Partitioned rewards calculation must still have access to parent Bank.") + .last_blockhash(); + let stake_rewards_by_partition = hash_rewards_into_partitions( + std::mem::take(&mut stake_rewards.stake_rewards), + &parent_blockhash, + num_partitions as usize, + ); + + PartitionedRewardsCalculation { + vote_account_rewards, + stake_rewards_by_partition: StakeRewardCalculationPartitioned { + stake_rewards_by_partition, + total_stake_rewards_lamports: stake_rewards.total_stake_rewards_lamports, + }, + old_vote_balance_and_staked, + validator_rewards, + validator_rate, + foundation_rate, + prev_epoch_duration_in_years, + capitalization, + } + } + + /// Calculate epoch reward and return vote and stake rewards. + fn calculate_validator_rewards( + &self, + rewarded_epoch: Epoch, + rewards: u64, + reward_calc_tracer: Option, + thread_pool: &ThreadPool, + metrics: &mut RewardsMetrics, + ) -> Option<(VoteRewardsAccounts, StakeRewardCalculation)> { + let stakes = self.stakes_cache.stakes(); + let reward_calculate_param = self.get_epoch_reward_calculate_param_info(&stakes); + + self.calculate_reward_points_partitioned( + &reward_calculate_param, + rewards, + thread_pool, + metrics, + ) + .map(|point_value| { + self.calculate_stake_vote_rewards( + &reward_calculate_param, + rewarded_epoch, + point_value, + thread_pool, + reward_calc_tracer, + metrics, + ) + }) + } + + /// Calculates epoch reward points from stake/vote accounts. + /// Returns reward lamports and points for the epoch or none if points == 0. + fn calculate_reward_points_partitioned( + &self, + reward_calculate_params: &EpochRewardCalculateParamInfo, + rewards: u64, + thread_pool: &ThreadPool, + metrics: &RewardsMetrics, + ) -> Option { + let EpochRewardCalculateParamInfo { + stake_history, + stake_delegations, + cached_vote_accounts, + } = reward_calculate_params; + + let solana_vote_program: Pubkey = solana_vote_program::id(); + + let get_vote_account = |vote_pubkey: &Pubkey| -> Option { + if let Some(vote_account) = cached_vote_accounts.get(vote_pubkey) { + return Some(vote_account.clone()); + } + // If accounts-db contains a valid vote account, then it should + // already have been cached in cached_vote_accounts; so the code + // below is only for sanity checking, and can be removed once + // the cache is deemed to be reliable. + let account = self.get_account_with_fixed_root(vote_pubkey)?; + VoteAccount::try_from(account).ok() + }; + + let new_warmup_cooldown_rate_epoch = self.new_warmup_cooldown_rate_epoch(); + let (points, measure_us) = measure_us!(thread_pool.install(|| { + stake_delegations + .par_iter() + .map(|(_stake_pubkey, stake_account)| { + let delegation = stake_account.delegation(); + let vote_pubkey = delegation.voter_pubkey; + + let Some(vote_account) = get_vote_account(&vote_pubkey) else { + return 0; + }; + 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, + stake_history, + new_warmup_cooldown_rate_epoch, + ) + .unwrap_or(0) + }) + .sum::() + })); + metrics.calculate_points_us.fetch_add(measure_us, Relaxed); + + (points > 0).then_some(PointValue { rewards, points }) + } +} + +#[cfg(test)] +mod tests { + use { + super::*, + crate::{ + bank::{null_tracer, tests::create_genesis_config, VoteReward}, + genesis_utils::{ + create_genesis_config_with_vote_accounts, GenesisConfigInfo, ValidatorVoteKeypairs, + }, + stake_account::StakeAccount, + stakes::Stakes, + }, + rayon::ThreadPoolBuilder, + solana_sdk::{ + account::{accounts_equal, ReadableAccount, WritableAccount}, + native_token::{sol_to_lamports, LAMPORTS_PER_SOL}, + reward_type::RewardType, + signature::Signer, + stake::state::Delegation, + vote::state::{VoteStateVersions, MAX_LOCKOUT_HISTORY}, + }, + solana_vote_program::vote_state, + std::sync::RwLockReadGuard, + }; + + /// Helper function to create a bank that pays some rewards + fn create_reward_bank(expected_num_delegations: usize) -> (Bank, Vec, Vec) { + let validator_keypairs = (0..expected_num_delegations) + .map(|_| ValidatorVoteKeypairs::new_rand()) + .collect::>(); + + let GenesisConfigInfo { genesis_config, .. } = create_genesis_config_with_vote_accounts( + 1_000_000_000, + &validator_keypairs, + vec![2_000_000_000; expected_num_delegations], + ); + + let bank = Bank::new_for_tests(&genesis_config); + + // Fill bank_forks with banks with votes landing in the next slot + // Create enough banks such that vote account will root + for validator_vote_keypairs in &validator_keypairs { + let vote_id = validator_vote_keypairs.vote_keypair.pubkey(); + let mut vote_account = bank.get_account(&vote_id).unwrap(); + // generate some rewards + let mut vote_state = Some(vote_state::from(&vote_account).unwrap()); + for i in 0..MAX_LOCKOUT_HISTORY + 42 { + if let Some(v) = vote_state.as_mut() { + vote_state::process_slot_vote_unchecked(v, i as u64) + } + let versioned = VoteStateVersions::Current(Box::new(vote_state.take().unwrap())); + vote_state::to(&versioned, &mut vote_account).unwrap(); + match versioned { + VoteStateVersions::Current(v) => { + vote_state = Some(*v); + } + _ => panic!("Has to be of type Current"), + }; + } + bank.store_account_and_update_capitalization(&vote_id, &vote_account); + } + ( + bank, + validator_keypairs + .iter() + .map(|k| k.vote_keypair.pubkey()) + .collect(), + validator_keypairs + .iter() + .map(|k| k.stake_keypair.pubkey()) + .collect(), + ) + } + + #[test] + fn test_store_vote_accounts_partitioned() { + let (genesis_config, _mint_keypair) = create_genesis_config(1_000_000 * LAMPORTS_PER_SOL); + let bank = Bank::new_for_tests(&genesis_config); + + let expected_vote_rewards_num = 100; + + let vote_rewards = (0..expected_vote_rewards_num) + .map(|_| Some((Pubkey::new_unique(), VoteReward::new_random()))) + .collect::>(); + + let mut vote_rewards_account = VoteRewardsAccounts::default(); + vote_rewards.iter().for_each(|e| { + if let Some(p) = &e { + let info = RewardInfo { + reward_type: RewardType::Voting, + lamports: p.1.vote_rewards as i64, + post_balance: p.1.vote_rewards, + commission: Some(p.1.commission), + }; + vote_rewards_account.rewards.push((p.0, info)); + vote_rewards_account + .accounts_to_store + .push(e.as_ref().map(|p| p.1.vote_account.clone())); + } + }); + + let metrics = RewardsMetrics::default(); + + let stored_vote_accounts = + bank.store_vote_accounts_partitioned(vote_rewards_account, &metrics); + assert_eq!(expected_vote_rewards_num, stored_vote_accounts.len()); + + // load accounts to make sure they were stored correctly + vote_rewards.iter().for_each(|e| { + if let Some(p) = &e { + let (k, account) = (p.0, p.1.vote_account.clone()); + let loaded_account = bank.load_slow_with_fixed_root(&bank.ancestors, &k).unwrap(); + assert!(accounts_equal(&loaded_account.0, &account)); + } + }); + } + + #[test] + fn test_store_vote_accounts_partitioned_empty() { + let (genesis_config, _mint_keypair) = create_genesis_config(1_000_000 * LAMPORTS_PER_SOL); + let bank = Bank::new_for_tests(&genesis_config); + + let expected = 0; + let vote_rewards = VoteRewardsAccounts::default(); + let metrics = RewardsMetrics::default(); + + let stored_vote_accounts = bank.store_vote_accounts_partitioned(vote_rewards, &metrics); + assert_eq!(expected, stored_vote_accounts.len()); + } + + #[test] + /// Test rewards computation and partitioned rewards distribution at the epoch boundary + fn test_rewards_computation() { + solana_logger::setup(); + + let expected_num_delegations = 100; + let bank = create_reward_bank(expected_num_delegations).0; + + // Calculate rewards + let thread_pool = ThreadPoolBuilder::new().num_threads(1).build().unwrap(); + let mut rewards_metrics = RewardsMetrics::default(); + let expected_rewards = 100_000_000_000; + + let calculated_rewards = bank.calculate_validator_rewards( + 1, + expected_rewards, + null_tracer(), + &thread_pool, + &mut rewards_metrics, + ); + + let vote_rewards = &calculated_rewards.as_ref().unwrap().0; + let stake_rewards = &calculated_rewards.as_ref().unwrap().1; + + let total_vote_rewards: u64 = vote_rewards + .rewards + .iter() + .map(|reward| reward.1.lamports) + .sum::() as u64; + + // assert that total rewards matches the sum of vote rewards and stake rewards + assert_eq!( + stake_rewards.total_stake_rewards_lamports + total_vote_rewards, + expected_rewards + ); + + // assert that number of stake rewards matches + assert_eq!(stake_rewards.stake_rewards.len(), expected_num_delegations); + } + + #[test] + fn test_rewards_point_calculation() { + solana_logger::setup(); + + let expected_num_delegations = 100; + let (bank, _, _) = create_reward_bank(expected_num_delegations); + + let thread_pool = ThreadPoolBuilder::new().num_threads(1).build().unwrap(); + let rewards_metrics = RewardsMetrics::default(); + let expected_rewards = 100_000_000_000; + + let stakes: RwLockReadGuard>> = bank.stakes_cache.stakes(); + let reward_calculate_param = bank.get_epoch_reward_calculate_param_info(&stakes); + + let point_value = bank.calculate_reward_points_partitioned( + &reward_calculate_param, + expected_rewards, + &thread_pool, + &rewards_metrics, + ); + + assert!(point_value.is_some()); + assert_eq!(point_value.as_ref().unwrap().rewards, expected_rewards); + assert_eq!(point_value.as_ref().unwrap().points, 8400000000000); + } + + #[test] + fn test_rewards_point_calculation_empty() { + solana_logger::setup(); + + // bank with no rewards to distribute + let (genesis_config, _mint_keypair) = create_genesis_config(sol_to_lamports(1.0)); + let bank = Bank::new_for_tests(&genesis_config); + + let thread_pool = ThreadPoolBuilder::new().num_threads(1).build().unwrap(); + let rewards_metrics: RewardsMetrics = RewardsMetrics::default(); + let expected_rewards = 100_000_000_000; + let stakes: RwLockReadGuard>> = bank.stakes_cache.stakes(); + let reward_calculate_param = bank.get_epoch_reward_calculate_param_info(&stakes); + + let point_value = bank.calculate_reward_points_partitioned( + &reward_calculate_param, + expected_rewards, + &thread_pool, + &rewards_metrics, + ); + + assert!(point_value.is_none()); + } + + #[test] + fn test_calculate_stake_vote_rewards() { + solana_logger::setup(); + + let expected_num_delegations = 1; + let (bank, voters, stakers) = create_reward_bank(expected_num_delegations); + + let vote_pubkey = voters.first().unwrap(); + let mut vote_account = bank + .load_slow_with_fixed_root(&bank.ancestors, vote_pubkey) + .unwrap() + .0; + + let stake_pubkey = stakers.first().unwrap(); + let stake_account = bank + .load_slow_with_fixed_root(&bank.ancestors, stake_pubkey) + .unwrap() + .0; + + let thread_pool = ThreadPoolBuilder::new().num_threads(1).build().unwrap(); + let mut rewards_metrics = RewardsMetrics::default(); + + let point_value = PointValue { + rewards: 100000, // lamports to split + points: 1000, // over these points + }; + let tracer = |_event: &RewardCalculationEvent| {}; + let reward_calc_tracer = Some(tracer); + let rewarded_epoch = bank.epoch(); + let stakes: RwLockReadGuard>> = bank.stakes_cache.stakes(); + let reward_calculate_param = bank.get_epoch_reward_calculate_param_info(&stakes); + let (vote_rewards_accounts, stake_reward_calculation) = bank.calculate_stake_vote_rewards( + &reward_calculate_param, + rewarded_epoch, + point_value, + &thread_pool, + reward_calc_tracer, + &mut rewards_metrics, + ); + + assert_eq!( + vote_rewards_accounts.rewards.len(), + vote_rewards_accounts.accounts_to_store.len() + ); + assert_eq!(vote_rewards_accounts.rewards.len(), 1); + let rewards = &vote_rewards_accounts.rewards[0]; + let account = &vote_rewards_accounts.accounts_to_store[0]; + let vote_rewards = 0; + let commision = 0; + _ = vote_account.checked_add_lamports(vote_rewards); + assert_eq!( + account.as_ref().unwrap().lamports(), + vote_account.lamports() + ); + assert!(accounts_equal(account.as_ref().unwrap(), &vote_account)); + assert_eq!( + rewards.1, + RewardInfo { + reward_type: RewardType::Voting, + lamports: vote_rewards as i64, + post_balance: vote_account.lamports(), + commission: Some(commision), + } + ); + assert_eq!(&rewards.0, vote_pubkey); + + assert_eq!(stake_reward_calculation.stake_rewards.len(), 1); + assert_eq!( + &stake_reward_calculation.stake_rewards[0].stake_pubkey, + stake_pubkey + ); + + let original_stake_lamport = stake_account.lamports(); + let rewards = stake_reward_calculation.stake_rewards[0] + .stake_reward_info + .lamports; + let expected_reward_info = RewardInfo { + reward_type: RewardType::Staking, + lamports: rewards, + post_balance: original_stake_lamport + rewards as u64, + commission: Some(commision), + }; + assert_eq!( + stake_reward_calculation.stake_rewards[0].stake_reward_info, + expected_reward_info, + ); + } +} diff --git a/runtime/src/bank/partitioned_epoch_rewards/mod.rs b/runtime/src/bank/partitioned_epoch_rewards/mod.rs index 8248f0c51bb387..e50b5217676aa0 100644 --- a/runtime/src/bank/partitioned_epoch_rewards/mod.rs +++ b/runtime/src/bank/partitioned_epoch_rewards/mod.rs @@ -1,3 +1,4 @@ +mod calculation; mod distribution; mod sysvar; diff --git a/runtime/src/bank/tests.rs b/runtime/src/bank/tests.rs index 1fb875be988751..05a0d662f9572e 100644 --- a/runtime/src/bank/tests.rs +++ b/runtime/src/bank/tests.rs @@ -12126,105 +12126,6 @@ fn test_squash_timing_add_assign() { assert!(t0 == expected); } -/// Helper function to create a bank that pays some rewards -fn create_reward_bank(expected_num_delegations: usize) -> (Bank, Vec, Vec) { - let validator_keypairs = (0..expected_num_delegations) - .map(|_| ValidatorVoteKeypairs::new_rand()) - .collect::>(); - - let GenesisConfigInfo { genesis_config, .. } = create_genesis_config_with_vote_accounts( - 1_000_000_000, - &validator_keypairs, - vec![2_000_000_000; expected_num_delegations], - ); - - let bank = Bank::new_for_tests(&genesis_config); - - // Fill bank_forks with banks with votes landing in the next slot - // Create enough banks such that vote account will root - for validator_vote_keypairs in &validator_keypairs { - let vote_id = validator_vote_keypairs.vote_keypair.pubkey(); - let mut vote_account = bank.get_account(&vote_id).unwrap(); - // generate some rewards - let mut vote_state = Some(vote_state::from(&vote_account).unwrap()); - for i in 0..MAX_LOCKOUT_HISTORY + 42 { - if let Some(v) = vote_state.as_mut() { - vote_state::process_slot_vote_unchecked(v, i as u64) - } - let versioned = VoteStateVersions::Current(Box::new(vote_state.take().unwrap())); - vote_state::to(&versioned, &mut vote_account).unwrap(); - match versioned { - VoteStateVersions::Current(v) => { - vote_state = Some(*v); - } - _ => panic!("Has to be of type Current"), - }; - } - bank.store_account_and_update_capitalization(&vote_id, &vote_account); - } - ( - bank, - validator_keypairs - .iter() - .map(|k| k.vote_keypair.pubkey()) - .collect(), - validator_keypairs - .iter() - .map(|k| k.stake_keypair.pubkey()) - .collect(), - ) -} - -#[test] -fn test_rewards_point_calculation() { - solana_logger::setup(); - - let expected_num_delegations = 100; - let (bank, _, _) = create_reward_bank(expected_num_delegations); - - let thread_pool = ThreadPoolBuilder::new().num_threads(1).build().unwrap(); - let rewards_metrics = RewardsMetrics::default(); - let expected_rewards = 100_000_000_000; - - let stakes: RwLockReadGuard>> = bank.stakes_cache.stakes(); - let reward_calculate_param = bank.get_epoch_reward_calculate_param_info(&stakes); - - let point_value = bank.calculate_reward_points_partitioned( - &reward_calculate_param, - expected_rewards, - &thread_pool, - &rewards_metrics, - ); - - assert!(point_value.is_some()); - assert_eq!(point_value.as_ref().unwrap().rewards, expected_rewards); - assert_eq!(point_value.as_ref().unwrap().points, 8400000000000); -} - -#[test] -fn test_rewards_point_calculation_empty() { - solana_logger::setup(); - - // bank with no rewards to distribute - let (genesis_config, _mint_keypair) = create_genesis_config(sol_to_lamports(1.0)); - let bank = Bank::new_for_tests(&genesis_config); - - let thread_pool = ThreadPoolBuilder::new().num_threads(1).build().unwrap(); - let rewards_metrics: RewardsMetrics = RewardsMetrics::default(); - let expected_rewards = 100_000_000_000; - let stakes: RwLockReadGuard>> = bank.stakes_cache.stakes(); - let reward_calculate_param = bank.get_epoch_reward_calculate_param_info(&stakes); - - let point_value = bank.calculate_reward_points_partitioned( - &reward_calculate_param, - expected_rewards, - &thread_pool, - &rewards_metrics, - ); - - assert!(point_value.is_none()); -} - /// Test that reward partition range panics when passing out of range partition index #[test] #[should_panic(expected = "index out of bounds: the len is 10 but the index is 15")] @@ -12248,46 +12149,6 @@ fn test_get_stake_rewards_partition_range_panic() { let _range = &stake_rewards_bucket[15]; } -#[test] -/// Test rewards computation and partitioned rewards distribution at the epoch boundary -fn test_rewards_computation() { - solana_logger::setup(); - - let expected_num_delegations = 100; - let bank = create_reward_bank(expected_num_delegations).0; - - // Calculate rewards - let thread_pool = ThreadPoolBuilder::new().num_threads(1).build().unwrap(); - let mut rewards_metrics = RewardsMetrics::default(); - let expected_rewards = 100_000_000_000; - - let calculated_rewards = bank.calculate_validator_rewards( - 1, - expected_rewards, - null_tracer(), - &thread_pool, - &mut rewards_metrics, - ); - - let vote_rewards = &calculated_rewards.as_ref().unwrap().0; - let stake_rewards = &calculated_rewards.as_ref().unwrap().1; - - let total_vote_rewards: u64 = vote_rewards - .rewards - .iter() - .map(|reward| reward.1.lamports) - .sum::() as u64; - - // assert that total rewards matches the sum of vote rewards and stake rewards - assert_eq!( - stake_rewards.total_stake_rewards_lamports + total_vote_rewards, - expected_rewards - ); - - // assert that number of stake rewards matches - assert_eq!(stake_rewards.stake_rewards.len(), expected_num_delegations); -} - /// Test rewards computation and partitioned rewards distribution at the epoch boundary (one reward distribution block) #[test] fn test_rewards_computation_and_partitioned_distribution_one_block() { @@ -12566,61 +12427,6 @@ fn test_program_execution_restricted_for_stake_account_in_reward_period() { } } -#[test] -fn test_store_vote_accounts_partitioned() { - let (genesis_config, _mint_keypair) = create_genesis_config(1_000_000 * LAMPORTS_PER_SOL); - let bank = Bank::new_for_tests(&genesis_config); - - let expected_vote_rewards_num = 100; - - let vote_rewards = (0..expected_vote_rewards_num) - .map(|_| Some((Pubkey::new_unique(), VoteReward::new_random()))) - .collect::>(); - - let mut vote_rewards_account = VoteRewardsAccounts::default(); - vote_rewards.iter().for_each(|e| { - if let Some(p) = &e { - let info = RewardInfo { - reward_type: RewardType::Voting, - lamports: p.1.vote_rewards as i64, - post_balance: p.1.vote_rewards, - commission: Some(p.1.commission), - }; - vote_rewards_account.rewards.push((p.0, info)); - vote_rewards_account - .accounts_to_store - .push(e.as_ref().map(|p| p.1.vote_account.clone())); - } - }); - - let metrics = RewardsMetrics::default(); - - let stored_vote_accounts = bank.store_vote_accounts_partitioned(vote_rewards_account, &metrics); - assert_eq!(expected_vote_rewards_num, stored_vote_accounts.len()); - - // load accounts to make sure they were stored correctly - vote_rewards.iter().for_each(|e| { - if let Some(p) = &e { - let (k, account) = (p.0, p.1.vote_account.clone()); - let loaded_account = bank.load_slow_with_fixed_root(&bank.ancestors, &k).unwrap(); - assert!(accounts_equal(&loaded_account.0, &account)); - } - }); -} - -#[test] -fn test_store_vote_accounts_partitioned_empty() { - let (genesis_config, _mint_keypair) = create_genesis_config(1_000_000 * LAMPORTS_PER_SOL); - let bank = Bank::new_for_tests(&genesis_config); - - let expected = 0; - let vote_rewards = VoteRewardsAccounts::default(); - let metrics = RewardsMetrics::default(); - - let stored_vote_accounts = bank.store_vote_accounts_partitioned(vote_rewards, &metrics); - assert_eq!(expected, stored_vote_accounts.len()); -} - #[test] fn test_system_instruction_allocate() { let (genesis_config, mint_keypair) = create_genesis_config_no_tx_fee(sol_to_lamports(1.0)); @@ -12946,94 +12752,6 @@ fn test_calc_vote_accounts_to_store_normal() { } } -#[test] -fn test_calculate_stake_vote_rewards() { - solana_logger::setup(); - - let expected_num_delegations = 1; - let (bank, voters, stakers) = create_reward_bank(expected_num_delegations); - - let vote_pubkey = voters.first().unwrap(); - let mut vote_account = bank - .load_slow_with_fixed_root(&bank.ancestors, vote_pubkey) - .unwrap() - .0; - - let stake_pubkey = stakers.first().unwrap(); - let stake_account = bank - .load_slow_with_fixed_root(&bank.ancestors, stake_pubkey) - .unwrap() - .0; - - let thread_pool = ThreadPoolBuilder::new().num_threads(1).build().unwrap(); - let mut rewards_metrics = RewardsMetrics::default(); - - let point_value = PointValue { - rewards: 100000, // lamports to split - points: 1000, // over these points - }; - let tracer = |_event: &RewardCalculationEvent| {}; - let reward_calc_tracer = Some(tracer); - let rewarded_epoch = bank.epoch(); - let stakes: RwLockReadGuard>> = bank.stakes_cache.stakes(); - let reward_calculate_param = bank.get_epoch_reward_calculate_param_info(&stakes); - let (vote_rewards_accounts, stake_reward_calculation) = bank.calculate_stake_vote_rewards( - &reward_calculate_param, - rewarded_epoch, - point_value, - &thread_pool, - reward_calc_tracer, - &mut rewards_metrics, - ); - - assert_eq!( - vote_rewards_accounts.rewards.len(), - vote_rewards_accounts.accounts_to_store.len() - ); - assert_eq!(vote_rewards_accounts.rewards.len(), 1); - let rewards = &vote_rewards_accounts.rewards[0]; - let account = &vote_rewards_accounts.accounts_to_store[0]; - let vote_rewards = 0; - let commision = 0; - _ = vote_account.checked_add_lamports(vote_rewards); - assert_eq!( - account.as_ref().unwrap().lamports(), - vote_account.lamports() - ); - assert!(accounts_equal(account.as_ref().unwrap(), &vote_account)); - assert_eq!( - rewards.1, - RewardInfo { - reward_type: RewardType::Voting, - lamports: vote_rewards as i64, - post_balance: vote_account.lamports(), - commission: Some(commision), - } - ); - assert_eq!(&rewards.0, vote_pubkey); - - assert_eq!(stake_reward_calculation.stake_rewards.len(), 1); - assert_eq!( - &stake_reward_calculation.stake_rewards[0].stake_pubkey, - stake_pubkey - ); - - let original_stake_lamport = stake_account.lamports(); - let rewards = stake_reward_calculation.stake_rewards[0] - .stake_reward_info - .lamports; - let expected_reward_info = RewardInfo { - reward_type: RewardType::Staking, - lamports: rewards, - post_balance: original_stake_lamport + rewards as u64, - commission: Some(commision), - }; - assert_eq!( - stake_reward_calculation.stake_rewards[0].stake_reward_info, - expected_reward_info, - ); -} - #[test] fn test_register_hard_fork() { fn get_hard_forks(bank: &Bank) -> Vec {