use {
crate::leader_schedule::{
IdentityKeyedLeaderSchedule, LeaderSchedule, VoteKeyedLeaderSchedule,
},
solana_clock::{Epoch, Slot, NUM_CONSECUTIVE_LEADER_SLOTS},
solana_pubkey::Pubkey,
solana_runtime::bank::Bank,
std::collections::HashMap,
};
pub fn leader_schedule(epoch: Epoch, bank: &Bank) -> Option<LeaderSchedule> {
let use_new_leader_schedule = bank.should_use_vote_keyed_leader_schedule(epoch)?;
if use_new_leader_schedule {
bank.epoch_vote_accounts(epoch).map(|vote_accounts_map| {
Box::new(VoteKeyedLeaderSchedule::new(
vote_accounts_map,
epoch,
bank.get_slots_in_epoch(epoch),
NUM_CONSECUTIVE_LEADER_SLOTS,
)) as LeaderSchedule
})
} else {
bank.epoch_staked_nodes(epoch).map(|stakes| {
Box::new(IdentityKeyedLeaderSchedule::new(
&stakes,
epoch,
bank.get_slots_in_epoch(epoch),
NUM_CONSECUTIVE_LEADER_SLOTS,
)) as LeaderSchedule
})
}
}
pub type LeaderScheduleByIdentity = HashMap<String, Vec<usize>>;
pub fn leader_schedule_by_identity<'a>(
upcoming_leaders: impl Iterator<Item = (usize, &'a Pubkey)>,
) -> LeaderScheduleByIdentity {
let mut leader_schedule_by_identity = HashMap::new();
for (slot_index, identity_pubkey) in upcoming_leaders {
leader_schedule_by_identity
.entry(identity_pubkey)
.or_insert_with(Vec::new)
.push(slot_index);
}
leader_schedule_by_identity
.into_iter()
.map(|(identity_pubkey, slot_indices)| (identity_pubkey.to_string(), slot_indices))
.collect()
}
pub fn slot_leader_at(slot: Slot, bank: &Bank) -> Option<Pubkey> {
let (epoch, slot_index) = bank.get_epoch_and_slot_index(slot);
leader_schedule(epoch, bank).map(|leader_schedule| leader_schedule[slot_index])
}
pub fn num_ticks_left_in_slot(bank: &Bank, tick_height: u64) -> u64 {
bank.ticks_per_slot() - tick_height % bank.ticks_per_slot()
}
pub fn first_of_consecutive_leader_slots(slot: Slot) -> Slot {
(slot / NUM_CONSECUTIVE_LEADER_SLOTS) * NUM_CONSECUTIVE_LEADER_SLOTS
}
#[inline]
pub fn last_of_consecutive_leader_slots(slot: Slot) -> Slot {
first_of_consecutive_leader_slots(slot) + NUM_CONSECUTIVE_LEADER_SLOTS - 1
}
#[inline]
pub fn leader_slot_index(slot: Slot) -> usize {
(slot % NUM_CONSECUTIVE_LEADER_SLOTS) as usize
}
#[inline]
pub fn remaining_slots_in_window(slot: Slot) -> u64 {
NUM_CONSECUTIVE_LEADER_SLOTS
.checked_sub(leader_slot_index(slot) as u64)
.unwrap()
}
#[cfg(test)]
mod tests {
use {
super::*,
solana_runtime::genesis_utils::{
bootstrap_validator_stake_lamports, create_genesis_config_with_leader,
deactivate_features,
},
test_case::test_case,
};
#[test_case(true; "vote keyed leader schedule")]
#[test_case(false; "identity keyed leader schedule")]
fn test_leader_schedule_via_bank(use_vote_keyed_leader_schedule: bool) {
let pubkey = solana_pubkey::new_rand();
let mut genesis_config =
create_genesis_config_with_leader(0, &pubkey, bootstrap_validator_stake_lamports())
.genesis_config;
if !use_vote_keyed_leader_schedule {
deactivate_features(
&mut genesis_config,
&vec![agave_feature_set::enable_vote_address_leader_schedule::id()],
);
}
let bank = Bank::new_for_tests(&genesis_config);
let leader_schedule = leader_schedule(0, &bank).unwrap();
assert_eq!(
leader_schedule.get_vote_key_at_slot_index(0).is_some(),
use_vote_keyed_leader_schedule
);
assert_eq!(leader_schedule[0], pubkey);
assert_eq!(leader_schedule[1], pubkey);
assert_eq!(leader_schedule[2], pubkey);
}
#[test]
fn test_leader_scheduler1_basic() {
let pubkey = solana_pubkey::new_rand();
let genesis_config =
create_genesis_config_with_leader(42, &pubkey, bootstrap_validator_stake_lamports())
.genesis_config;
let bank = Bank::new_for_tests(&genesis_config);
assert_eq!(slot_leader_at(bank.slot(), &bank).unwrap(), pubkey);
}
#[test]
fn test_leader_span_math() {
assert_eq!(NUM_CONSECUTIVE_LEADER_SLOTS, 4);
assert_eq!(first_of_consecutive_leader_slots(0), 0);
assert_eq!(first_of_consecutive_leader_slots(1), 0);
assert_eq!(first_of_consecutive_leader_slots(2), 0);
assert_eq!(first_of_consecutive_leader_slots(3), 0);
assert_eq!(first_of_consecutive_leader_slots(4), 4);
assert_eq!(last_of_consecutive_leader_slots(0), 3);
assert_eq!(last_of_consecutive_leader_slots(1), 3);
assert_eq!(last_of_consecutive_leader_slots(2), 3);
assert_eq!(last_of_consecutive_leader_slots(3), 3);
assert_eq!(last_of_consecutive_leader_slots(4), 7);
assert_eq!(leader_slot_index(0), 0);
assert_eq!(leader_slot_index(1), 1);
assert_eq!(leader_slot_index(2), 2);
assert_eq!(leader_slot_index(3), 3);
assert_eq!(leader_slot_index(4), 0);
assert_eq!(leader_slot_index(5), 1);
assert_eq!(leader_slot_index(6), 2);
assert_eq!(leader_slot_index(7), 3);
assert_eq!(remaining_slots_in_window(0), 4);
assert_eq!(remaining_slots_in_window(1), 3);
assert_eq!(remaining_slots_in_window(2), 2);
assert_eq!(remaining_slots_in_window(3), 1);
assert_eq!(remaining_slots_in_window(4), 4);
}
}