Allow process_blocktree() to start processing from any root (#5484)

* Remove unnecessary entry_height from BankInfo

* Refactor process_blocktree to support process_blocktree_from_root

* Refactor to process blocktree after loading from snapshot

* On restart make sure bank_forks contains all the banks between the root and the tip of each fork, not just the head of each fork

* Account for 1 tick_per_slot in bank 0 so that blockhash of bank0 matches the tick
This commit is contained in:
carllin 2019-08-13 17:20:14 -07:00 committed by GitHub
parent 58d4e32c97
commit cd14a940d8
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 452 additions and 293 deletions

View File

@ -1,19 +1,14 @@
//! The `bank_forks` module implments BankForks a DAG of checkpointed Banks //! The `bank_forks` module implments BankForks a DAG of checkpointed Banks
use crate::result::{Error, Result}; use crate::result::Result;
use crate::snapshot_package::SnapshotPackageSender; use crate::snapshot_package::SnapshotPackageSender;
use crate::snapshot_package::{TAR_ACCOUNTS_DIR, TAR_SNAPSHOTS_DIR};
use crate::snapshot_utils; use crate::snapshot_utils;
use crate::snapshot_utils::untar_snapshot_in;
use fs_extra::dir::CopyOptions;
use solana_measure::measure::Measure; use solana_measure::measure::Measure;
use solana_metrics::inc_new_counter_info; use solana_metrics::inc_new_counter_info;
use solana_runtime::bank::Bank; use solana_runtime::bank::Bank;
use solana_runtime::status_cache::MAX_CACHE_ENTRIES; use solana_runtime::status_cache::MAX_CACHE_ENTRIES;
use solana_sdk::timing; use solana_sdk::timing;
use std::collections::{HashMap, HashSet}; use std::collections::{HashMap, HashSet};
use std::fs;
use std::io::{Error as IOError, ErrorKind};
use std::ops::Index; use std::ops::Index;
use std::path::{Path, PathBuf}; use std::path::{Path, PathBuf};
use std::sync::Arc; use std::sync::Arc;
@ -164,18 +159,29 @@ impl BankForks {
self.banks.get(&bank_slot) self.banks.get(&bank_slot)
} }
pub fn new_from_banks(initial_banks: &[Arc<Bank>], root: u64) -> Self { pub fn new_from_banks(initial_forks: &[Arc<Bank>], rooted_path: Vec<u64>) -> Self {
let mut banks = HashMap::new(); let mut banks = HashMap::new();
let working_bank = initial_banks[0].clone(); let working_bank = initial_forks[0].clone();
for bank in initial_banks {
// Iterate through the heads of all the different forks
for bank in initial_forks {
banks.insert(bank.slot(), bank.clone()); banks.insert(bank.slot(), bank.clone());
let parents = bank.parents();
for parent in parents {
if banks.contains_key(&parent.slot()) {
// All ancestors have already been inserted by another fork
break;
} }
banks.insert(parent.slot(), parent.clone());
}
}
Self { Self {
root, root: *rooted_path.last().unwrap(),
banks, banks,
working_bank, working_bank,
snapshot_config: None, snapshot_config: None,
slots_since_snapshot: vec![], slots_since_snapshot: rooted_path,
confidence: HashMap::new(), confidence: HashMap::new(),
} }
} }
@ -367,54 +373,6 @@ impl BankForks {
pub fn snapshot_config(&self) -> &Option<SnapshotConfig> { pub fn snapshot_config(&self) -> &Option<SnapshotConfig> {
&self.snapshot_config &self.snapshot_config
} }
pub fn load_from_snapshot<P: AsRef<Path>>(
account_paths: String,
snapshot_config: &SnapshotConfig,
snapshot_tar: P,
) -> Result<Self> {
// Untar the snapshot into a temp directory under `snapshot_config.snapshot_path()`
let unpack_dir = tempfile::tempdir_in(snapshot_config.snapshot_path())?;
untar_snapshot_in(&snapshot_tar, &unpack_dir)?;
let unpacked_accounts_dir = unpack_dir.as_ref().join(TAR_ACCOUNTS_DIR);
let unpacked_snapshots_dir = unpack_dir.as_ref().join(TAR_SNAPSHOTS_DIR);
let snapshot_paths = snapshot_utils::get_snapshot_paths(&unpacked_snapshots_dir);
let bank = snapshot_utils::bank_from_snapshots(
account_paths,
&snapshot_paths,
unpacked_accounts_dir,
)?;
let bank = Arc::new(bank);
// Move the unpacked snapshots into `snapshot_config.snapshot_path()`
let dir_files = fs::read_dir(unpacked_snapshots_dir).expect("Invalid snapshot path");
let paths: Vec<PathBuf> = dir_files
.filter_map(|entry| entry.ok().map(|e| e.path()))
.collect();
let mut copy_options = CopyOptions::new();
copy_options.overwrite = true;
fs_extra::move_items(&paths, snapshot_config.snapshot_path(), &copy_options)?;
let mut banks = HashMap::new();
banks.insert(bank.slot(), bank.clone());
let root = bank.slot();
let snapshot_paths = snapshot_utils::get_snapshot_paths(&snapshot_config.snapshot_path);
if snapshot_paths.is_empty() {
return Err(Error::IO(IOError::new(
ErrorKind::Other,
"no snapshots found",
)));
}
Ok(BankForks {
banks,
working_bank: bank,
root,
snapshot_config: None,
slots_since_snapshot: vec![snapshot_paths.last().unwrap().slot],
confidence: HashMap::new(),
})
}
} }
#[cfg(test)] #[cfg(test)]
@ -429,6 +387,7 @@ mod tests {
use solana_sdk::pubkey::Pubkey; use solana_sdk::pubkey::Pubkey;
use solana_sdk::signature::{Keypair, KeypairUtil}; use solana_sdk::signature::{Keypair, KeypairUtil};
use solana_sdk::system_transaction; use solana_sdk::system_transaction;
use std::fs;
use std::sync::atomic::AtomicBool; use std::sync::atomic::AtomicBool;
use std::sync::mpsc::channel; use std::sync::mpsc::channel;
use tempfile::TempDir; use tempfile::TempDir;
@ -541,31 +500,31 @@ mod tests {
); );
} }
fn restore_from_snapshot(bank_forks: BankForks, account_paths: String, last_slot: u64) { fn restore_from_snapshot(old_bank_forks: BankForks, account_paths: String) {
let (snapshot_path, snapshot_package_output_path) = bank_forks let (snapshot_path, snapshot_package_output_path) = old_bank_forks
.snapshot_config .snapshot_config
.as_ref() .as_ref()
.map(|c| (&c.snapshot_path, &c.snapshot_package_output_path)) .map(|c| (&c.snapshot_path, &c.snapshot_package_output_path))
.unwrap(); .unwrap();
let new = BankForks::load_from_snapshot( let deserialized_bank = snapshot_utils::bank_from_archive(
account_paths, account_paths,
bank_forks.snapshot_config.as_ref().unwrap(), old_bank_forks.snapshot_config.as_ref().unwrap(),
snapshot_utils::get_snapshot_tar_path(snapshot_package_output_path), snapshot_utils::get_snapshot_tar_path(snapshot_package_output_path),
) )
.unwrap(); .unwrap();
for (slot, _) in new.banks.iter() { let bank = old_bank_forks
if *slot > 0 { .banks
let bank = bank_forks.banks.get(slot).unwrap().clone(); .get(&deserialized_bank.slot())
let new_bank = new.banks.get(slot).unwrap(); .unwrap()
bank.compare_bank(&new_bank); .clone();
} bank.compare_bank(&deserialized_bank);
}
assert_eq!(new.working_bank().slot(), last_slot); let slot_snapshot_paths = snapshot_utils::get_snapshot_paths(&snapshot_path);
for (slot, _) in new.banks.iter() {
snapshot_utils::remove_snapshot(*slot, snapshot_path).unwrap(); for p in slot_snapshot_paths {
snapshot_utils::remove_snapshot(p.slot, &snapshot_path).unwrap();
} }
} }
@ -627,7 +586,6 @@ mod tests {
restore_from_snapshot( restore_from_snapshot(
bank_forks, bank_forks,
accounts_dir.path().to_str().unwrap().to_string(), accounts_dir.path().to_str().unwrap().to_string(),
last_slot,
); );
} }

View File

@ -1,5 +1,5 @@
use crate::bank_forks::BankForks; use crate::bank_forks::BankForks;
use crate::blocktree::Blocktree; use crate::blocktree::{Blocktree, SlotMeta};
use crate::entry::{Entry, EntrySlice}; use crate::entry::{Entry, EntrySlice};
use crate::leader_schedule_cache::LeaderScheduleCache; use crate::leader_schedule_cache::LeaderScheduleCache;
use rayon::prelude::*; use rayon::prelude::*;
@ -8,6 +8,7 @@ use solana_metrics::{datapoint, datapoint_error, inc_new_counter_debug};
use solana_runtime::bank::Bank; use solana_runtime::bank::Bank;
use solana_runtime::locked_accounts_results::LockedAccountsResults; use solana_runtime::locked_accounts_results::LockedAccountsResults;
use solana_sdk::genesis_block::GenesisBlock; use solana_sdk::genesis_block::GenesisBlock;
use solana_sdk::hash::Hash;
use solana_sdk::timing::{duration_as_ms, Slot, MAX_RECENT_BLOCKHASHES}; use solana_sdk::timing::{duration_as_ms, Slot, MAX_RECENT_BLOCKHASHES};
use solana_sdk::transaction::Result; use solana_sdk::transaction::Result;
use std::result; use std::result;
@ -129,7 +130,6 @@ pub fn process_entries(bank: &Bank, entries: &[Entry]) -> Result<()> {
#[derive(Debug, PartialEq)] #[derive(Debug, PartialEq)]
pub struct BankForksInfo { pub struct BankForksInfo {
pub bank_slot: u64, pub bank_slot: u64,
pub entry_height: u64,
} }
#[derive(Debug)] #[derive(Debug)]
@ -144,52 +144,113 @@ pub fn process_blocktree(
verify_ledger: bool, verify_ledger: bool,
dev_halt_at_slot: Option<Slot>, dev_halt_at_slot: Option<Slot>,
) -> result::Result<(BankForks, Vec<BankForksInfo>, LeaderScheduleCache), BlocktreeProcessorError> { ) -> result::Result<(BankForks, Vec<BankForksInfo>, LeaderScheduleCache), BlocktreeProcessorError> {
let now = Instant::now(); info!("processing ledger from bank 0...");
info!("processing ledger...");
// Setup bank for slot 0 // Setup bank for slot 0
let mut pending_slots = { let bank0 = Arc::new(Bank::new_with_paths(&genesis_block, account_paths));
let slot = 0; process_bank_0(&bank0, blocktree, verify_ledger)?;
let bank = Arc::new(Bank::new_with_paths(&genesis_block, account_paths)); process_blocktree_from_root(blocktree, bank0, verify_ledger, dev_halt_at_slot)
let entry_height = 0;
let last_entry_hash = bank.last_blockhash();
// Load the metadata for this slot
let meta = blocktree
.meta(slot)
.map_err(|err| {
warn!("Failed to load meta for slot {}: {:?}", slot, err);
BlocktreeProcessorError::LedgerVerificationFailed
})?
.unwrap();
vec![(slot, meta, bank, entry_height, last_entry_hash)]
};
blocktree.set_roots(&[0]).expect("Couldn't set first root");
let leader_schedule_cache =
LeaderScheduleCache::new(*pending_slots[0].2.epoch_schedule(), &pending_slots[0].2);
let mut fork_info = vec![];
let mut last_status_report = Instant::now();
let mut root = 0;
let dev_halt_at_slot = dev_halt_at_slot.unwrap_or(std::u64::MAX);
while !pending_slots.is_empty() {
let (slot, meta, bank, mut entry_height, mut last_entry_hash) =
pending_slots.pop().unwrap();
if last_status_report.elapsed() > Duration::from_secs(2) {
info!("processing ledger...block {}", slot);
last_status_report = Instant::now();
} }
// Fetch all entries for this slot // Process blocktree from a known root bank
let mut entries = blocktree.get_slot_entries(slot, 0, None).map_err(|err| { pub fn process_blocktree_from_root(
warn!("Failed to load entries for slot {}: {:?}", slot, err); blocktree: &Blocktree,
bank: Arc<Bank>,
verify_ledger: bool,
dev_halt_at_slot: Option<Slot>,
) -> result::Result<(BankForks, Vec<BankForksInfo>, LeaderScheduleCache), BlocktreeProcessorError> {
info!("processing ledger from root: {}...", bank.slot());
// Starting slot must be a root, and thus has no parents
assert!(bank.parent().is_none());
let start_slot = bank.slot();
let now = Instant::now();
let mut rooted_path = vec![start_slot];
let dev_halt_at_slot = dev_halt_at_slot.unwrap_or(std::u64::MAX);
blocktree
.set_roots(&[start_slot])
.expect("Couldn't set root on startup");
let meta = blocktree.meta(start_slot).unwrap();
// Iterate and replay slots from blocktree starting from `start_slot`
let (bank_forks, bank_forks_info, leader_schedule_cache) = {
if let Some(meta) = meta {
let epoch_schedule = bank.epoch_schedule();
let mut leader_schedule_cache = LeaderScheduleCache::new(*epoch_schedule, &bank);
let fork_info = process_pending_slots(
&bank,
&meta,
blocktree,
&mut leader_schedule_cache,
&mut rooted_path,
verify_ledger,
dev_halt_at_slot,
)?;
let (banks, bank_forks_info): (Vec<_>, Vec<_>) = fork_info.into_iter().unzip();
let bank_forks = BankForks::new_from_banks(&banks, rooted_path);
(bank_forks, bank_forks_info, leader_schedule_cache)
} else {
// If there's no meta for the input `start_slot`, then we started from a snapshot
// and there's no point in processing the rest of blocktree and implies blocktree
// should be empty past this point.
let bfi = BankForksInfo {
bank_slot: start_slot,
};
let leader_schedule_cache = LeaderScheduleCache::new_from_bank(&bank);
let bank_forks = BankForks::new_from_banks(&[bank], rooted_path);
(bank_forks, vec![bfi], leader_schedule_cache)
}
};
info!(
"processing ledger...complete in {}ms, forks={}...",
duration_as_ms(&now.elapsed()),
bank_forks_info.len(),
);
Ok((bank_forks, bank_forks_info, leader_schedule_cache))
}
fn verify_and_process_entries(
bank: &Bank,
entries: &[Entry],
verify_ledger: bool,
last_entry_hash: Hash,
) -> result::Result<Hash, BlocktreeProcessorError> {
assert!(!entries.is_empty());
if verify_ledger && !entries.verify(&last_entry_hash) {
warn!("Ledger proof of history failed at slot: {}", bank.slot());
return Err(BlocktreeProcessorError::LedgerVerificationFailed);
}
process_entries(&bank, &entries).map_err(|err| {
warn!(
"Failed to process entries for slot {}: {:?}",
bank.slot(),
err
);
BlocktreeProcessorError::LedgerVerificationFailed
})?;
Ok(entries.last().unwrap().hash)
}
// Special handling required for processing the entries in slot 0
fn process_bank_0(
bank0: &Bank,
blocktree: &Blocktree,
verify_ledger: bool,
) -> result::Result<(), BlocktreeProcessorError> {
assert_eq!(bank0.slot(), 0);
// Fetch all entries for this slot
let mut entries = blocktree.get_slot_entries(0, 0, None).map_err(|err| {
warn!("Failed to load entries for slot 0, err: {:?}", err);
BlocktreeProcessorError::LedgerVerificationFailed BlocktreeProcessorError::LedgerVerificationFailed
})?; })?;
if slot == 0 {
// The first entry in the ledger is a pseudo-tick used only to ensure the number of ticks // The first entry in the ledger is a pseudo-tick used only to ensure the number of ticks
// in slot 0 is the same as the number of ticks in all subsequent slots. It is not // in slot 0 is the same as the number of ticks in all subsequent slots. It is not
// processed by the bank, skip over it. // processed by the bank, skip over it.
@ -198,36 +259,123 @@ pub fn process_blocktree(
return Err(BlocktreeProcessorError::LedgerVerificationFailed); return Err(BlocktreeProcessorError::LedgerVerificationFailed);
} }
let entry0 = entries.remove(0); let entry0 = entries.remove(0);
if !(entry0.is_tick() && entry0.verify(&last_entry_hash)) { if !(entry0.is_tick() && entry0.verify(&bank0.last_blockhash())) {
warn!("Ledger proof of history failed at entry0"); warn!("Ledger proof of history failed at entry0");
return Err(BlocktreeProcessorError::LedgerVerificationFailed); return Err(BlocktreeProcessorError::LedgerVerificationFailed);
} }
last_entry_hash = entry0.hash;
entry_height += 1;
}
if !entries.is_empty() { if !entries.is_empty() {
if verify_ledger && !entries.verify(&last_entry_hash) { verify_and_process_entries(bank0, &entries, verify_ledger, entry0.hash)?;
warn!( } else {
"Ledger proof of history failed at slot: {}, entry: {}", bank0.register_tick(&entry0.hash);
slot, entry_height
);
return Err(BlocktreeProcessorError::LedgerVerificationFailed);
} }
process_entries(&bank, &entries).map_err(|err| { bank0.freeze();
warn!("Failed to process entries for slot {}: {:?}", slot, err);
Ok(())
}
// Given a slot, add its children to the pending slots queue if those children slots are
// complete
fn process_next_slots(
bank: &Arc<Bank>,
meta: &SlotMeta,
blocktree: &Blocktree,
leader_schedule_cache: &LeaderScheduleCache,
pending_slots: &mut Vec<(u64, SlotMeta, Arc<Bank>, Hash)>,
fork_info: &mut Vec<(Arc<Bank>, BankForksInfo)>,
) -> result::Result<(), BlocktreeProcessorError> {
if meta.next_slots.is_empty() {
// Reached the end of this fork. Record the final entry height and last entry.hash
let bfi = BankForksInfo {
bank_slot: bank.slot(),
};
fork_info.push((bank.clone(), bfi));
return Ok(());
}
// This is a fork point if there are multiple children, create a new child bank for each fork
for next_slot in &meta.next_slots {
let next_meta = blocktree
.meta(*next_slot)
.map_err(|err| {
warn!("Failed to load meta for slot {}: {:?}", next_slot, err);
BlocktreeProcessorError::LedgerVerificationFailed
})?
.unwrap();
// Only process full slots in blocktree_processor, replay_stage
// handles any partials
if next_meta.is_full() {
let next_bank = Arc::new(Bank::new_from_parent(
&bank,
&leader_schedule_cache
.slot_leader_at(*next_slot, Some(&bank))
.unwrap(),
*next_slot,
));
trace!("Add child bank {} of slot={}", next_slot, bank.slot());
pending_slots.push((*next_slot, next_meta, next_bank, bank.last_blockhash()));
} else {
let bfi = BankForksInfo {
bank_slot: bank.slot(),
};
fork_info.push((bank.clone(), bfi));
}
}
// Reverse sort by slot, so the next slot to be processed can be popped
// TODO: remove me once leader_scheduler can hang with out-of-order slots?
pending_slots.sort_by(|a, b| b.0.cmp(&a.0));
Ok(())
}
// Iterate through blocktree processing slots starting from the root slot pointed to by the
// given `meta`
fn process_pending_slots(
root_bank: &Arc<Bank>,
root_meta: &SlotMeta,
blocktree: &Blocktree,
leader_schedule_cache: &mut LeaderScheduleCache,
rooted_path: &mut Vec<u64>,
verify_ledger: bool,
dev_halt_at_slot: Slot,
) -> result::Result<Vec<(Arc<Bank>, BankForksInfo)>, BlocktreeProcessorError> {
let mut fork_info = vec![];
let mut last_status_report = Instant::now();
let mut pending_slots = vec![];
process_next_slots(
root_bank,
root_meta,
blocktree,
leader_schedule_cache,
&mut pending_slots,
&mut fork_info,
)?;
while !pending_slots.is_empty() {
let (slot, meta, bank, last_entry_hash) = pending_slots.pop().unwrap();
if last_status_report.elapsed() > Duration::from_secs(2) {
info!("processing ledger...block {}", slot);
last_status_report = Instant::now();
}
// Fetch all entries for this slot
let entries = blocktree.get_slot_entries(slot, 0, None).map_err(|err| {
warn!("Failed to load entries for slot {}: {:?}", slot, err);
BlocktreeProcessorError::LedgerVerificationFailed BlocktreeProcessorError::LedgerVerificationFailed
})?; })?;
last_entry_hash = entries.last().unwrap().hash; verify_and_process_entries(&bank, &entries, verify_ledger, last_entry_hash)?;
entry_height += entries.len() as u64;
}
bank.freeze(); // all banks handled by this routine are created from complete slots bank.freeze(); // all banks handled by this routine are created from complete slots
if blocktree.is_root(slot) { if blocktree.is_root(slot) {
root = slot; let parents = bank.parents().into_iter().map(|b| b.slot()).rev().skip(1);
let parents: Vec<_> = parents.collect();
rooted_path.extend(parents);
rooted_path.push(slot);
leader_schedule_cache.set_root(&bank); leader_schedule_cache.set_root(&bank);
bank.squash(); bank.squash();
pending_slots.clear(); pending_slots.clear();
@ -235,76 +383,22 @@ pub fn process_blocktree(
} }
if slot >= dev_halt_at_slot { if slot >= dev_halt_at_slot {
let bfi = BankForksInfo { let bfi = BankForksInfo { bank_slot: slot };
bank_slot: slot,
entry_height,
};
fork_info.push((bank, bfi)); fork_info.push((bank, bfi));
break; break;
} }
if meta.next_slots.is_empty() { process_next_slots(
// Reached the end of this fork. Record the final entry height and last entry.hash
let bfi = BankForksInfo {
bank_slot: slot,
entry_height,
};
fork_info.push((bank, bfi));
continue;
}
// This is a fork point, create a new child bank for each fork
for next_slot in meta.next_slots {
let next_meta = blocktree
.meta(next_slot)
.map_err(|err| {
warn!("Failed to load meta for slot {}: {:?}", slot, err);
BlocktreeProcessorError::LedgerVerificationFailed
})?
.unwrap();
// only process full slots in blocktree_processor, replay_stage
// handles any partials
if next_meta.is_full() {
let next_bank = Arc::new(Bank::new_from_parent(
&bank, &bank,
&leader_schedule_cache &meta,
.slot_leader_at(next_slot, Some(&bank)) blocktree,
.unwrap(), leader_schedule_cache,
next_slot, &mut pending_slots,
)); &mut fork_info,
trace!("Add child bank for slot={}", next_slot); )?;
// bank_forks.insert(*next_slot, child_bank);
pending_slots.push((
next_slot,
next_meta,
next_bank,
entry_height,
last_entry_hash,
));
} else {
let bfi = BankForksInfo {
bank_slot: slot,
entry_height,
};
fork_info.push((bank.clone(), bfi));
}
} }
// reverse sort by slot, so the next slot to be processed can be pop()ed Ok(fork_info)
// TODO: remove me once leader_scheduler can hang with out-of-order slots?
pending_slots.sort_by(|a, b| b.0.cmp(&a.0));
}
let (banks, bank_forks_info): (Vec<_>, Vec<_>) = fork_info.into_iter().unzip();
let bank_forks = BankForks::new_from_banks(&banks, root);
info!(
"processing ledger...complete in {}ms, forks={}...",
duration_as_ms(&now.elapsed()),
bank_forks_info.len(),
);
Ok((bank_forks, bank_forks_info, leader_schedule_cache))
} }
#[cfg(test)] #[cfg(test)]
@ -393,7 +487,6 @@ pub mod tests {
bank_forks_info[0], bank_forks_info[0],
BankForksInfo { BankForksInfo {
bank_slot: 0, // slot 1 isn't "full", we stop at slot zero bank_slot: 0, // slot 1 isn't "full", we stop at slot zero
entry_height: ticks_per_slot,
} }
); );
} }
@ -453,7 +546,6 @@ pub mod tests {
bank_forks_info[0], bank_forks_info[0],
BankForksInfo { BankForksInfo {
bank_slot: 4, // Fork 2's head is slot 4 bank_slot: 4, // Fork 2's head is slot 4
entry_height: ticks_per_slot * 3,
} }
); );
assert!(&bank_forks[4] assert!(&bank_forks[4]
@ -464,10 +556,7 @@ pub mod tests {
.is_empty()); .is_empty());
// Ensure bank_forks holds the right banks // Ensure bank_forks holds the right banks
for info in bank_forks_info { verify_fork_infos(&bank_forks, &bank_forks_info);
assert_eq!(bank_forks[info.bank_slot].slot(), info.bank_slot);
assert!(bank_forks[info.bank_slot].is_frozen());
}
assert_eq!(bank_forks.root(), 4); assert_eq!(bank_forks.root(), 4);
} }
@ -526,7 +615,6 @@ pub mod tests {
bank_forks_info[0], bank_forks_info[0],
BankForksInfo { BankForksInfo {
bank_slot: 3, // Fork 1's head is slot 3 bank_slot: 3, // Fork 1's head is slot 3
entry_height: ticks_per_slot * 4,
} }
); );
assert_eq!( assert_eq!(
@ -541,7 +629,6 @@ pub mod tests {
bank_forks_info[1], bank_forks_info[1],
BankForksInfo { BankForksInfo {
bank_slot: 4, // Fork 2's head is slot 4 bank_slot: 4, // Fork 2's head is slot 4
entry_height: ticks_per_slot * 3,
} }
); );
assert_eq!( assert_eq!(
@ -556,10 +643,7 @@ pub mod tests {
assert_eq!(bank_forks.root(), 1); assert_eq!(bank_forks.root(), 1);
// Ensure bank_forks holds the right banks // Ensure bank_forks holds the right banks
for info in bank_forks_info { verify_fork_infos(&bank_forks, &bank_forks_info);
assert_eq!(bank_forks[info.bank_slot].slot(), info.bank_slot);
assert!(bank_forks[info.bank_slot].is_frozen());
}
} }
#[test] #[test]
@ -607,7 +691,6 @@ pub mod tests {
bank_forks_info[0], bank_forks_info[0],
BankForksInfo { BankForksInfo {
bank_slot: last_slot + 1, // Head is last_slot + 1 bank_slot: last_slot + 1, // Head is last_slot + 1
entry_height: ticks_per_slot * (last_slot + 2),
} }
); );
@ -734,19 +817,12 @@ pub mod tests {
blocktree blocktree
.write_entries(1, 0, 0, genesis_block.ticks_per_slot, &entries) .write_entries(1, 0, 0, genesis_block.ticks_per_slot, &entries)
.unwrap(); .unwrap();
let entry_height = genesis_block.ticks_per_slot + entries.len() as u64;
let (bank_forks, bank_forks_info, _) = let (bank_forks, bank_forks_info, _) =
process_blocktree(&genesis_block, &blocktree, None, true, None).unwrap(); process_blocktree(&genesis_block, &blocktree, None, true, None).unwrap();
assert_eq!(bank_forks_info.len(), 1); assert_eq!(bank_forks_info.len(), 1);
assert_eq!(bank_forks.root(), 0); assert_eq!(bank_forks.root(), 0);
assert_eq!( assert_eq!(bank_forks_info[0], BankForksInfo { bank_slot: 1 });
bank_forks_info[0],
BankForksInfo {
bank_slot: 1,
entry_height,
}
);
let bank = bank_forks[1].clone(); let bank = bank_forks[1].clone();
assert_eq!( assert_eq!(
@ -770,13 +846,7 @@ pub mod tests {
process_blocktree(&genesis_block, &blocktree, None, true, None).unwrap(); process_blocktree(&genesis_block, &blocktree, None, true, None).unwrap();
assert_eq!(bank_forks_info.len(), 1); assert_eq!(bank_forks_info.len(), 1);
assert_eq!( assert_eq!(bank_forks_info[0], BankForksInfo { bank_slot: 0 });
bank_forks_info[0],
BankForksInfo {
bank_slot: 0,
entry_height: 1,
}
);
let bank = bank_forks[0].clone(); let bank = bank_forks[0].clone();
assert_eq!(bank.tick_height(), 0); assert_eq!(bank.tick_height(), 0);
} }
@ -1264,6 +1334,83 @@ pub mod tests {
assert_eq!(bank.process_transaction(&fail_tx), Ok(())); assert_eq!(bank.process_transaction(&fail_tx), Ok(()));
} }
#[test]
fn test_process_blocktree_from_root() {
let GenesisBlockInfo {
mut genesis_block, ..
} = create_genesis_block(123);
let ticks_per_slot = 1;
genesis_block.ticks_per_slot = ticks_per_slot;
let (ledger_path, blockhash) = create_new_tmp_ledger!(&genesis_block);
let blocktree = Blocktree::open(&ledger_path).unwrap();
/*
Build a blocktree in the ledger with the following fork structure:
slot 0 (all ticks)
|
slot 1 (all ticks)
|
slot 2 (all ticks)
|
slot 3 (all ticks) -> root
|
slot 4 (all ticks)
|
slot 5 (all ticks) -> root
|
slot 6 (all ticks)
*/
let mut last_hash = blockhash;
for i in 0..6 {
last_hash =
fill_blocktree_slot_with_ticks(&blocktree, ticks_per_slot, i + 1, i, last_hash);
}
blocktree.set_roots(&[3, 5]).unwrap();
// Set up bank1
let bank0 = Arc::new(Bank::new(&genesis_block));
process_bank_0(&bank0, &blocktree, true).unwrap();
let bank1 = Arc::new(Bank::new_from_parent(&bank0, &Pubkey::default(), 1));
bank1.squash();
let slot1_entries = blocktree.get_slot_entries(1, 0, None).unwrap();
verify_and_process_entries(&bank1, &slot1_entries, true, bank0.last_blockhash()).unwrap();
// Test process_blocktree_from_root() from slot 1 onwards
let (bank_forks, bank_forks_info, _) =
process_blocktree_from_root(&blocktree, bank1, true, None).unwrap();
assert_eq!(bank_forks_info.len(), 1); // One fork
assert_eq!(
bank_forks_info[0],
BankForksInfo {
bank_slot: 6, // The head of the fork is slot 6
}
);
// slots_since_snapshot should contain everything on the rooted path
assert_eq!(
bank_forks.slots_since_snapshot().to_vec(),
vec![1, 2, 3, 4, 5]
);
assert_eq!(bank_forks.root(), 5);
// Verify the parents of the head of the fork
assert_eq!(
&bank_forks[6]
.parents()
.iter()
.map(|bank| bank.slot())
.collect::<Vec<_>>(),
&[5]
);
// Check that bank forks has the correct banks
verify_fork_infos(&bank_forks, &bank_forks_info);
}
#[test] #[test]
#[ignore] #[ignore]
fn test_process_entries_stress() { fn test_process_entries_stress() {
@ -1357,4 +1504,22 @@ pub mod tests {
let bank = Bank::new_with_paths(&genesis_block, account_paths); let bank = Bank::new_with_paths(&genesis_block, account_paths);
bank.epoch_schedule().clone() bank.epoch_schedule().clone()
} }
// Check that `bank_forks` contains all the ancestors and banks for each fork identified in
// `bank_forks_info`
fn verify_fork_infos(bank_forks: &BankForks, bank_forks_info: &[BankForksInfo]) {
for fork in bank_forks_info {
let head_slot = fork.bank_slot;
let head_bank = &bank_forks[head_slot];
let mut parents = head_bank.parents();
parents.push(head_bank.clone());
// Ensure the tip of each fork and all its parents are in the given bank_forks
for parent in parents {
let parent_bank = &bank_forks[parent.slot()];
assert_eq!(parent_bank.slot(), parent.slot());
assert!(parent_bank.is_frozen());
}
}
}
} }

View File

@ -1059,7 +1059,7 @@ mod test {
let arc_bank0 = Arc::new(bank0); let arc_bank0 = Arc::new(bank0);
let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks( let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(
&[arc_bank0.clone()], &[arc_bank0.clone()],
0, vec![0],
))); )));
let pubkey = Pubkey::new_rand(); let pubkey = Pubkey::new_rand();
let mut tower = Tower::new_from_forks(&bank_forks.read().unwrap(), &pubkey); let mut tower = Tower::new_from_forks(&bank_forks.read().unwrap(), &pubkey);

View File

@ -1,7 +1,10 @@
use crate::bank_forks::SnapshotConfig;
use crate::result::{Error, Result}; use crate::result::{Error, Result};
use crate::snapshot_package::SnapshotPackage; use crate::snapshot_package::SnapshotPackage;
use crate::snapshot_package::{TAR_ACCOUNTS_DIR, TAR_SNAPSHOTS_DIR};
use bincode::{deserialize_from, serialize_into}; use bincode::{deserialize_from, serialize_into};
use flate2::read::GzDecoder; use flate2::read::GzDecoder;
use fs_extra::dir::CopyOptions;
use solana_runtime::bank::Bank; use solana_runtime::bank::Bank;
use solana_runtime::status_cache::SlotDelta; use solana_runtime::status_cache::SlotDelta;
use solana_sdk::transaction; use solana_sdk::transaction;
@ -168,7 +171,48 @@ pub fn remove_snapshot<P: AsRef<Path>>(slot: u64, snapshot_path: P) -> Result<()
Ok(()) Ok(())
} }
pub fn bank_from_snapshots<P>( pub fn bank_from_archive<P: AsRef<Path>>(
account_paths: String,
snapshot_config: &SnapshotConfig,
snapshot_tar: P,
) -> Result<Bank> {
// Untar the snapshot into a temp directory under `snapshot_config.snapshot_path()`
let unpack_dir = tempfile::tempdir_in(snapshot_config.snapshot_path())?;
untar_snapshot_in(&snapshot_tar, &unpack_dir)?;
let unpacked_accounts_dir = unpack_dir.as_ref().join(TAR_ACCOUNTS_DIR);
let unpacked_snapshots_dir = unpack_dir.as_ref().join(TAR_SNAPSHOTS_DIR);
let snapshot_paths = get_snapshot_paths(&unpacked_snapshots_dir);
let bank = rebuild_bank_from_snapshots(account_paths, &snapshot_paths, unpacked_accounts_dir)?;
// Move the unpacked snapshots into `snapshot_config.snapshot_path()`
let dir_files = fs::read_dir(unpacked_snapshots_dir).expect("Invalid snapshot path");
let paths: Vec<PathBuf> = dir_files
.filter_map(|entry| entry.ok().map(|e| e.path()))
.collect();
let mut copy_options = CopyOptions::new();
copy_options.overwrite = true;
fs_extra::move_items(&paths, snapshot_config.snapshot_path(), &copy_options)?;
Ok(bank)
}
pub fn get_snapshot_tar_path<P: AsRef<Path>>(snapshot_output_dir: P) -> PathBuf {
snapshot_output_dir.as_ref().join("snapshot.tgz")
}
pub fn untar_snapshot_in<P: AsRef<Path>, Q: AsRef<Path>>(
snapshot_tar: P,
unpack_dir: Q,
) -> Result<()> {
let tar_gz = File::open(snapshot_tar)?;
let tar = GzDecoder::new(tar_gz);
let mut archive = Archive::new(tar);
archive.unpack(&unpack_dir)?;
Ok(())
}
fn rebuild_bank_from_snapshots<P>(
local_account_paths: String, local_account_paths: String,
snapshot_paths: &[SlotSnapshotPaths], snapshot_paths: &[SlotSnapshotPaths],
append_vecs_path: P, append_vecs_path: P,
@ -203,21 +247,6 @@ where
Ok(bank) Ok(bank)
} }
pub fn get_snapshot_tar_path<P: AsRef<Path>>(snapshot_output_dir: P) -> PathBuf {
snapshot_output_dir.as_ref().join("snapshot.tgz")
}
pub fn untar_snapshot_in<P: AsRef<Path>, Q: AsRef<Path>>(
snapshot_tar: P,
unpack_dir: Q,
) -> Result<()> {
let tar_gz = File::open(snapshot_tar)?;
let tar = GzDecoder::new(tar_gz);
let mut archive = Archive::new(tar);
archive.unpack(&unpack_dir)?;
Ok(())
}
fn get_snapshot_file_name(slot: u64) -> String { fn get_snapshot_file_name(slot: u64) -> String {
slot.to_string() slot.to_string()
} }

View File

@ -659,7 +659,10 @@ mod tests {
let cluster_info = test_cluster_info(&keypair.pubkey()); let cluster_info = test_cluster_info(&keypair.pubkey());
let GenesisBlockInfo { genesis_block, .. } = create_genesis_block(1000); let GenesisBlockInfo { genesis_block, .. } = create_genesis_block(1000);
let bank = Arc::new(Bank::new(&genesis_block)); let bank = Arc::new(Bank::new(&genesis_block));
let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(&[bank.clone()], 0))); let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(
&[bank.clone()],
vec![0],
)));
let (_slot_sender, slot_receiver) = channel(); let (_slot_sender, slot_receiver) = channel();
let storage_state = StorageState::new( let storage_state = StorageState::new(
&bank.last_blockhash(), &bank.last_blockhash(),
@ -699,7 +702,10 @@ mod tests {
let blocktree = Arc::new(Blocktree::open(&ledger_path).unwrap()); let blocktree = Arc::new(Blocktree::open(&ledger_path).unwrap());
let slot = 1; let slot = 1;
let bank = Arc::new(Bank::new(&genesis_block)); let bank = Arc::new(Bank::new(&genesis_block));
let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(&[bank.clone()], 0))); let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(
&[bank.clone()],
vec![0],
)));
let cluster_info = test_cluster_info(&keypair.pubkey()); let cluster_info = test_cluster_info(&keypair.pubkey());
let (bank_sender, bank_receiver) = channel(); let (bank_sender, bank_receiver) = channel();
@ -791,7 +797,10 @@ mod tests {
let bank = Bank::new(&genesis_block); let bank = Bank::new(&genesis_block);
let bank = Arc::new(bank); let bank = Arc::new(bank);
let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(&[bank.clone()], 0))); let bank_forks = Arc::new(RwLock::new(BankForks::new_from_banks(
&[bank.clone()],
vec![0],
)));
let cluster_info = test_cluster_info(&keypair.pubkey()); let cluster_info = test_cluster_info(&keypair.pubkey());
let (bank_sender, bank_receiver) = channel(); let (bank_sender, bank_receiver) = channel();

View File

@ -355,30 +355,28 @@ fn get_bank_forks(
// Check that the snapshot tar exists, try to load the snapshot if it does // Check that the snapshot tar exists, try to load the snapshot if it does
if tar.exists() { if tar.exists() {
// Fail hard here if snapshot fails to load, don't silently continue // Fail hard here if snapshot fails to load, don't silently continue
let bank_forks = BankForks::load_from_snapshot( let deserialized_bank = snapshot_utils::bank_from_archive(
//&genesis_block,
account_paths account_paths
.clone() .clone()
.expect("Account paths not present when booting from snapshot"), .expect("Account paths not present when booting from snapshot"),
snapshot_config, snapshot_config,
tar, &tar,
) )
.expect("Load from snapshot failed"); .expect("Load from snapshot failed");
let bank = &bank_forks.working_bank(); result = Some(
let fork_info = BankForksInfo { blocktree_processor::process_blocktree_from_root(
bank_slot: bank.slot(), blocktree,
entry_height: bank.tick_height(), Arc::new(deserialized_bank),
}; verify_ledger,
result = Some(( dev_halt_at_slot,
bank_forks, )
vec![fork_info], .expect("processing blocktree after loading snapshot failed"),
LeaderScheduleCache::new_from_bank(bank), );
));
} }
} }
// If loading from a snapshot failed/snapshot didn't exist // If a snapshot doesn't exist
if result.is_none() { if result.is_none() {
result = Some( result = Some(
blocktree_processor::process_blocktree( blocktree_processor::process_blocktree(

View File

@ -724,12 +724,12 @@ impl Bank {
// TODO: put this assert back in // TODO: put this assert back in
// assert!(!self.is_frozen()); // assert!(!self.is_frozen());
if self.ticks_per_slot() != 1 || self.slot() != 0 {
let current_tick_height = {
self.tick_height.fetch_add(1, Ordering::Relaxed); self.tick_height.fetch_add(1, Ordering::Relaxed);
self.tick_height.load(Ordering::Relaxed) as u64
};
inc_new_counter_debug!("bank-register_tick-registered", 1); inc_new_counter_debug!("bank-register_tick-registered", 1);
}
let current_tick_height = self.tick_height.load(Ordering::Relaxed) as u64;
// Register a new block hash if at the last tick in the slot // Register a new block hash if at the last tick in the slot
if current_tick_height % self.ticks_per_slot == self.ticks_per_slot - 1 { if current_tick_height % self.ticks_per_slot == self.ticks_per_slot - 1 {