Add TreeDiff trait to reuse tree functions (#11046)

Co-authored-by: Carl <carl@solana.com>
This commit is contained in:
carllin 2020-07-14 00:38:48 -07:00 committed by GitHub
parent 356b552439
commit e9cbdf711b
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 48 additions and 35 deletions

View File

@ -2,6 +2,7 @@ use crate::{
consensus::{ComputedBankState, Tower}, consensus::{ComputedBankState, Tower},
fork_choice::ForkChoice, fork_choice::ForkChoice,
progress_map::ProgressMap, progress_map::ProgressMap,
tree_diff::TreeDiff,
}; };
use solana_runtime::{bank::Bank, bank_forks::BankForks, epoch_stakes::EpochStakes}; use solana_runtime::{bank::Bank, bank_forks::BankForks, epoch_stakes::EpochStakes};
use solana_sdk::{ use solana_sdk::{
@ -141,10 +142,6 @@ impl HeaviestSubtreeForkChoice {
.map(|fork_info| fork_info.stake_voted_subtree) .map(|fork_info| fork_info.stake_voted_subtree)
} }
pub fn contains_slot(&self, slot: Slot) -> bool {
self.fork_infos.contains_key(&slot)
}
pub fn root(&self) -> Slot { pub fn root(&self) -> Slot {
self.root self.root
} }
@ -248,30 +245,6 @@ impl HeaviestSubtreeForkChoice {
self.propagate_new_leaf(slot, parent) self.propagate_new_leaf(slot, parent)
} }
// Find all nodes reachable from `root1`, excluding subtree at `root2`
pub fn subtree_diff(&self, root1: Slot, root2: Slot) -> HashSet<Slot> {
if !self.contains_slot(root1) {
return HashSet::new();
}
let mut pending_slots = vec![root1];
let mut reachable_set = HashSet::new();
while !pending_slots.is_empty() {
let current_slot = pending_slots.pop().unwrap();
if current_slot == root2 {
continue;
}
reachable_set.insert(current_slot);
for child in self
.children(current_slot)
.expect("slot was discovered earlier, must exist")
{
pending_slots.push(*child);
}
}
reachable_set
}
// Returns if the given `maybe_best_child` is the heaviest among the children // Returns if the given `maybe_best_child` is the heaviest among the children
// it's parent // it's parent
fn is_best_child(&self, maybe_best_child: Slot) -> bool { fn is_best_child(&self, maybe_best_child: Slot) -> bool {
@ -305,12 +278,6 @@ impl HeaviestSubtreeForkChoice {
AncestorIterator::new(start_slot, &self.fork_infos).collect() AncestorIterator::new(start_slot, &self.fork_infos).collect()
} }
pub fn children(&self, slot: Slot) -> Option<&[Slot]> {
self.fork_infos
.get(&slot)
.map(|fork_info| &fork_info.children[..])
}
pub fn merge( pub fn merge(
&mut self, &mut self,
other: HeaviestSubtreeForkChoice, other: HeaviestSubtreeForkChoice,
@ -542,6 +509,18 @@ impl HeaviestSubtreeForkChoice {
} }
} }
impl TreeDiff for HeaviestSubtreeForkChoice {
fn contains_slot(&self, slot: Slot) -> bool {
self.fork_infos.contains_key(&slot)
}
fn children(&self, slot: Slot) -> Option<&[Slot]> {
self.fork_infos
.get(&slot)
.map(|fork_info| &fork_info.children[..])
}
}
impl ForkChoice for HeaviestSubtreeForkChoice { impl ForkChoice for HeaviestSubtreeForkChoice {
fn compute_bank_stats( fn compute_bank_stats(
&mut self, &mut self,

View File

@ -62,6 +62,7 @@ pub mod sigverify_stage;
pub mod snapshot_packager_service; pub mod snapshot_packager_service;
pub mod tpu; pub mod tpu;
pub mod transaction_status_service; pub mod transaction_status_service;
pub mod tree_diff;
pub mod tvu; pub mod tvu;
pub mod validator; pub mod validator;
pub mod verified_vote_packets; pub mod verified_vote_packets;

View File

@ -3,6 +3,7 @@ use crate::{
repair_service::RepairTiming, repair_service::RepairTiming,
repair_weighted_traversal::{self, Contains}, repair_weighted_traversal::{self, Contains},
serve_repair::RepairType, serve_repair::RepairType,
tree_diff::TreeDiff,
}; };
use solana_ledger::{ancestor_iterator::AncestorIterator, blockstore::Blockstore}; use solana_ledger::{ancestor_iterator::AncestorIterator, blockstore::Blockstore};
use solana_measure::measure::Measure; use solana_measure::measure::Measure;

View File

@ -1,6 +1,6 @@
use crate::{ use crate::{
heaviest_subtree_fork_choice::HeaviestSubtreeForkChoice, repair_service::RepairService, heaviest_subtree_fork_choice::HeaviestSubtreeForkChoice, repair_service::RepairService,
serve_repair::RepairType, serve_repair::RepairType, tree_diff::TreeDiff,
}; };
use solana_ledger::blockstore::Blockstore; use solana_ledger::blockstore::Blockstore;
use solana_sdk::clock::Slot; use solana_sdk::clock::Slot;

32
core/src/tree_diff.rs Normal file
View File

@ -0,0 +1,32 @@
use solana_sdk::clock::Slot;
use std::collections::HashSet;
pub trait TreeDiff {
fn children(&self, slot: Slot) -> Option<&[Slot]>;
fn contains_slot(&self, slot: Slot) -> bool;
// Find all nodes reachable from `root1`, excluding subtree at `root2`
fn subtree_diff(&self, root1: Slot, root2: Slot) -> HashSet<Slot> {
if !self.contains_slot(root1) {
return HashSet::new();
}
let mut pending_slots = vec![root1];
let mut reachable_set = HashSet::new();
while !pending_slots.is_empty() {
let current_slot = pending_slots.pop().unwrap();
if current_slot == root2 {
continue;
}
reachable_set.insert(current_slot);
for child in self
.children(current_slot)
.expect("slot was discovered earlier, must exist")
{
pending_slots.push(*child);
}
}
reachable_set
}
}