Speed up getchaintips.

This commit is contained in:
mrbandrews 2016-04-18 12:10:47 -04:00
parent 1b2460bd58
commit 87049e832d

View file

@ -760,17 +760,30 @@ UniValue getchaintips(const UniValue& params, bool fHelp)
LOCK(cs_main); LOCK(cs_main);
/* Build up a list of chain tips. We start with the list of all /*
known blocks, and successively remove blocks that appear as pprev * Idea: the set of chain tips is chainActive.tip, plus orphan blocks which do not have another orphan building off of them.
of another block. */ * Algorithm:
* - Make one pass through mapBlockIndex, picking out the orphan blocks, and also storing a set of the orphan block's pprev pointers.
* - Iterate through the orphan blocks. If the block isn't pointed to by another orphan, it is a chain tip.
* - add chainActive.Tip()
*/
std::set<const CBlockIndex*, CompareBlocksByHeight> setTips; std::set<const CBlockIndex*, CompareBlocksByHeight> setTips;
BOOST_FOREACH(const PAIRTYPE(const uint256, CBlockIndex*)& item, mapBlockIndex) std::set<const CBlockIndex*> setOrphans;
setTips.insert(item.second); std::set<const CBlockIndex*> setPrevs;
BOOST_FOREACH(const PAIRTYPE(const uint256, CBlockIndex*)& item, mapBlockIndex) BOOST_FOREACH(const PAIRTYPE(const uint256, CBlockIndex*)& item, mapBlockIndex)
{ {
const CBlockIndex* pprev = item.second->pprev; if (!chainActive.Contains(item.second)) {
if (pprev) setOrphans.insert(item.second);
setTips.erase(pprev); setPrevs.insert(item.second->pprev);
}
}
for (std::set<const CBlockIndex*>::iterator it = setOrphans.begin(); it != setOrphans.end(); ++it)
{
if (setPrevs.erase(*it) == 0) {
setTips.insert(*it);
}
} }
// Always report the currently active tip. // Always report the currently active tip.