|
|
@ -37,10 +37,10 @@ const char* BlockQueueChannel::name() { return EthOrange "[]>"; } |
|
|
|
const char* BlockQueueChannel::name() { return EthOrange "▣┅▶"; } |
|
|
|
#endif |
|
|
|
|
|
|
|
size_t const c_maxKnownCount = 100000; ///< M
|
|
|
|
size_t const c_maxKnownCount = 100000; |
|
|
|
size_t const c_maxKnownSize = 128 * 1024 * 1024; |
|
|
|
size_t const c_maxUnknownCount = 100000; |
|
|
|
size_t const c_maxUnknownSize = 128 * 1024 * 1024; |
|
|
|
size_t const c_maxUnknownSize = 512 * 1024 * 1024; // Block size can be ~50kb
|
|
|
|
|
|
|
|
BlockQueue::BlockQueue(): |
|
|
|
m_unknownSize(0), |
|
|
@ -87,7 +87,7 @@ void BlockQueue::verifierBody() |
|
|
|
{ |
|
|
|
while (!m_deleting) |
|
|
|
{ |
|
|
|
std::pair<h256, bytes> work; |
|
|
|
UnverifiedBlock work; |
|
|
|
|
|
|
|
{ |
|
|
|
unique_lock<Mutex> l(m_verification); |
|
|
@ -97,12 +97,13 @@ void BlockQueue::verifierBody() |
|
|
|
swap(work, m_unverified.front()); |
|
|
|
m_unverified.pop_front(); |
|
|
|
BlockInfo bi; |
|
|
|
bi.mixHash = work.first; |
|
|
|
bi.mixHash = work.hash; |
|
|
|
bi.parentHash = work.parentHash; |
|
|
|
m_verifying.push_back(VerifiedBlock { VerifiedBlockRef { bytesConstRef(), move(bi), Transactions() }, bytes() }); |
|
|
|
} |
|
|
|
|
|
|
|
VerifiedBlock res; |
|
|
|
swap(work.second, res.blockData); |
|
|
|
swap(work.block, res.blockData); |
|
|
|
try |
|
|
|
{ |
|
|
|
res.verified = BlockChain::verifyBlock(res.blockData, m_onBad); |
|
|
@ -114,13 +115,13 @@ void BlockQueue::verifierBody() |
|
|
|
// has to be this order as that's how invariants() assumes.
|
|
|
|
WriteGuard l2(m_lock); |
|
|
|
unique_lock<Mutex> l(m_verification); |
|
|
|
m_readySet.erase(work.first); |
|
|
|
m_knownBad.insert(work.first); |
|
|
|
m_readySet.erase(work.hash); |
|
|
|
m_knownBad.insert(work.hash); |
|
|
|
} |
|
|
|
|
|
|
|
unique_lock<Mutex> l(m_verification); |
|
|
|
for (auto it = m_verifying.begin(); it != m_verifying.end(); ++it) |
|
|
|
if (it->verified.info.mixHash == work.first) |
|
|
|
if (it->verified.info.mixHash == work.hash) |
|
|
|
{ |
|
|
|
m_verifying.erase(it); |
|
|
|
goto OK1; |
|
|
@ -132,12 +133,13 @@ void BlockQueue::verifierBody() |
|
|
|
|
|
|
|
bool ready = false; |
|
|
|
{ |
|
|
|
WriteGuard l2(m_lock); |
|
|
|
unique_lock<Mutex> l(m_verification); |
|
|
|
if (!m_verifying.empty() && m_verifying.front().verified.info.mixHash == work.first) |
|
|
|
if (!m_verifying.empty() && m_verifying.front().verified.info.mixHash == work.hash) |
|
|
|
{ |
|
|
|
// we're next!
|
|
|
|
m_verifying.pop_front(); |
|
|
|
if (m_knownBad.count(res.verified.info.hash())) |
|
|
|
if (m_knownBad.count(res.verified.info.parentHash)) |
|
|
|
{ |
|
|
|
m_readySet.erase(res.verified.info.hash()); |
|
|
|
m_knownBad.insert(res.verified.info.hash()); |
|
|
@ -146,7 +148,7 @@ void BlockQueue::verifierBody() |
|
|
|
m_verified.push_back(move(res)); |
|
|
|
while (m_verifying.size() && !m_verifying.front().blockData.empty()) |
|
|
|
{ |
|
|
|
if (m_knownBad.count(m_verifying.front().verified.info.hash())) |
|
|
|
if (m_knownBad.count(m_verifying.front().verified.info.parentHash)) |
|
|
|
{ |
|
|
|
m_readySet.erase(m_verifying.front().verified.info.hash()); |
|
|
|
m_knownBad.insert(res.verified.info.hash()); |
|
|
@ -160,7 +162,7 @@ void BlockQueue::verifierBody() |
|
|
|
else |
|
|
|
{ |
|
|
|
for (auto& i: m_verifying) |
|
|
|
if (i.verified.info.mixHash == work.first) |
|
|
|
if (i.verified.info.mixHash == work.hash) |
|
|
|
{ |
|
|
|
i = move(res); |
|
|
|
goto OK; |
|
|
@ -235,6 +237,7 @@ ImportResult BlockQueue::import(bytesConstRef _block, BlockChain const& _bc, boo |
|
|
|
if (m_knownBad.count(bi.parentHash)) |
|
|
|
{ |
|
|
|
m_knownBad.insert(bi.hash()); |
|
|
|
updateBad(bi.hash()); |
|
|
|
// bad parent; this is bad too, note it as such
|
|
|
|
return ImportResult::BadChain; |
|
|
|
} |
|
|
@ -254,7 +257,7 @@ ImportResult BlockQueue::import(bytesConstRef _block, BlockChain const& _bc, boo |
|
|
|
// If valid, append to blocks.
|
|
|
|
cblockq << "OK - ready for chain insertion."; |
|
|
|
DEV_GUARDED(m_verification) |
|
|
|
m_unverified.push_back(make_pair(h, _block.toBytes())); |
|
|
|
m_unverified.push_back(UnverifiedBlock { h, bi.parentHash, _block.toBytes() }); |
|
|
|
m_moreToVerify.notify_one(); |
|
|
|
m_readySet.insert(h); |
|
|
|
m_knownSize += _block.size(); |
|
|
@ -267,39 +270,93 @@ ImportResult BlockQueue::import(bytesConstRef _block, BlockChain const& _bc, boo |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
bool BlockQueue::doneDrain(h256s const& _bad) |
|
|
|
void BlockQueue::updateBad(h256 const& _bad) |
|
|
|
{ |
|
|
|
WriteGuard l(m_lock); |
|
|
|
DEV_INVARIANT_CHECK; |
|
|
|
m_drainingSet.clear(); |
|
|
|
if (_bad.size()) |
|
|
|
{ |
|
|
|
// at least one of them was bad.
|
|
|
|
m_knownBad += _bad; |
|
|
|
DEV_GUARDED(m_verification) |
|
|
|
{ |
|
|
|
collectUnknownBad(_bad); |
|
|
|
bool moreBad = true; |
|
|
|
while (moreBad) |
|
|
|
{ |
|
|
|
moreBad = false; |
|
|
|
std::vector<VerifiedBlock> oldVerified; |
|
|
|
swap(m_verified, oldVerified); |
|
|
|
for (auto& b: oldVerified) |
|
|
|
if (m_knownBad.count(b.verified.info.parentHash)) |
|
|
|
if (m_knownBad.count(b.verified.info.parentHash) || m_knownBad.count(b.verified.info.hash())) |
|
|
|
{ |
|
|
|
m_knownBad.insert(b.verified.info.hash()); |
|
|
|
m_readySet.erase(b.verified.info.hash()); |
|
|
|
collectUnknownBad(b.verified.info.hash()); |
|
|
|
moreBad = true; |
|
|
|
} |
|
|
|
else |
|
|
|
m_verified.push_back(std::move(b)); |
|
|
|
|
|
|
|
std::deque<UnverifiedBlock> oldUnverified; |
|
|
|
swap(m_unverified, oldUnverified); |
|
|
|
for (auto& b: oldUnverified) |
|
|
|
if (m_knownBad.count(b.parentHash) || m_knownBad.count(b.hash)) |
|
|
|
{ |
|
|
|
m_knownBad.insert(b.hash); |
|
|
|
m_readySet.erase(b.hash); |
|
|
|
collectUnknownBad(b.hash); |
|
|
|
moreBad = true; |
|
|
|
} |
|
|
|
else |
|
|
|
m_unverified.push_back(std::move(b)); |
|
|
|
|
|
|
|
std::deque<VerifiedBlock> oldVerifying; |
|
|
|
swap(m_verifying, oldVerifying); |
|
|
|
for (auto& b: oldVerifying) |
|
|
|
if (m_knownBad.count(b.verified.info.parentHash) || m_knownBad.count(b.verified.info.mixHash)) |
|
|
|
{ |
|
|
|
h256 const& h = b.blockData.size() != 0 ? b.verified.info.hash() : b.verified.info.mixHash; |
|
|
|
m_knownBad.insert(h); |
|
|
|
m_readySet.erase(h); |
|
|
|
collectUnknownBad(h); |
|
|
|
moreBad = true; |
|
|
|
} |
|
|
|
else |
|
|
|
m_verifying.push_back(std::move(b)); |
|
|
|
} |
|
|
|
} |
|
|
|
DEV_INVARIANT_CHECK; |
|
|
|
} |
|
|
|
|
|
|
|
void BlockQueue::collectUnknownBad(h256 const& _bad) |
|
|
|
{ |
|
|
|
list<h256> badQueue(1, _bad); |
|
|
|
while (!badQueue.empty()) |
|
|
|
{ |
|
|
|
auto r = m_unknown.equal_range(badQueue.front()); |
|
|
|
badQueue.pop_front(); |
|
|
|
for (auto it = r.first; it != r.second; ++it) |
|
|
|
{ |
|
|
|
m_unknownSize -= it->second.second.size(); |
|
|
|
m_unknownCount--; |
|
|
|
auto newBad = it->second.first; |
|
|
|
m_unknownSet.erase(newBad); |
|
|
|
m_knownBad.insert(newBad); |
|
|
|
badQueue.push_back(newBad); |
|
|
|
} |
|
|
|
m_unknown.erase(r.first, r.second); |
|
|
|
} |
|
|
|
/* DEV_GUARDED(m_verification)
|
|
|
|
|
|
|
|
} |
|
|
|
|
|
|
|
bool BlockQueue::doneDrain(h256s const& _bad) |
|
|
|
{ |
|
|
|
WriteGuard l(m_lock); |
|
|
|
DEV_INVARIANT_CHECK; |
|
|
|
m_drainingSet.clear(); |
|
|
|
if (_bad.size()) |
|
|
|
{ |
|
|
|
// at least one of them was bad.
|
|
|
|
m_knownBad += _bad; |
|
|
|
m_knownBad += m_readySet; |
|
|
|
m_readySet.clear(); |
|
|
|
m_verified.clear(); |
|
|
|
m_verifying.clear(); |
|
|
|
m_unverified.clear(); |
|
|
|
}*/ |
|
|
|
return !m_readySet.empty(); |
|
|
|
for (h256 const& b : _bad) |
|
|
|
updateBad(b); |
|
|
|
} return !m_readySet.empty(); |
|
|
|
} |
|
|
|
|
|
|
|
void BlockQueue::tick(BlockChain const& _bc) |
|
|
@ -416,7 +473,7 @@ void BlockQueue::noteReady_WITH_LOCK(h256 const& _good) |
|
|
|
for (auto it = r.first; it != r.second; ++it) |
|
|
|
{ |
|
|
|
DEV_GUARDED(m_verification) |
|
|
|
m_unverified.push_back(it->second); |
|
|
|
m_unverified.push_back(UnverifiedBlock { it->second.first, it->first, it->second.second }); |
|
|
|
m_knownSize += it->second.second.size(); |
|
|
|
m_knownCount++; |
|
|
|
m_unknownSize -= it->second.second.size(); |
|
|
@ -431,6 +488,7 @@ void BlockQueue::noteReady_WITH_LOCK(h256 const& _good) |
|
|
|
} |
|
|
|
if (notify) |
|
|
|
m_moreToVerify.notify_all(); |
|
|
|
DEV_INVARIANT_CHECK; |
|
|
|
} |
|
|
|
|
|
|
|
void BlockQueue::retryAllUnknown() |
|
|
@ -440,7 +498,7 @@ void BlockQueue::retryAllUnknown() |
|
|
|
for (auto it = m_unknown.begin(); it != m_unknown.end(); ++it) |
|
|
|
{ |
|
|
|
DEV_GUARDED(m_verification) |
|
|
|
m_unverified.push_back(it->second); |
|
|
|
m_unverified.push_back(UnverifiedBlock { it->second.first, it->first, it->second.second }); |
|
|
|
auto newReady = it->second.first; |
|
|
|
m_unknownSet.erase(newReady); |
|
|
|
m_readySet.insert(newReady); |
|
|
|