diff --git a/libdevcore/Common.h b/libdevcore/Common.h index 453c17e6f..1ee83c794 100644 --- a/libdevcore/Common.h +++ b/libdevcore/Common.h @@ -181,7 +181,7 @@ private: /// Scope guard for invariant check in a class derived from HasInvariants. #if ETH_DEBUG -#define DEV_INVARIANT_CHECK ::dev::InvariantChecker __dev_invariantCheck(this) +#define DEV_INVARIANT_CHECK { ::dev::InvariantChecker __dev_invariantCheck(this); } #else #define DEV_INVARIANT_CHECK (void)0; #endif diff --git a/libethereum/BlockQueue.cpp b/libethereum/BlockQueue.cpp index ad6e62117..f142be62e 100644 --- a/libethereum/BlockQueue.cpp +++ b/libethereum/BlockQueue.cpp @@ -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 work; + UnverifiedBlock work; { unique_lock 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 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 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 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()) + DEV_GUARDED(m_verification) { - // 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 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 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 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_GUARDED(m_verification) + DEV_INVARIANT_CHECK; +} + +void BlockQueue::collectUnknownBad(h256 const& _bad) +{ + list 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_knownBad += _bad; - m_knownBad += m_readySet; - m_readySet.clear(); - m_verified.clear(); - m_verifying.clear(); - m_unverified.clear(); - }*/ - return !m_readySet.empty(); + 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); + } + +} + +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; + 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); diff --git a/libethereum/BlockQueue.h b/libethereum/BlockQueue.h index 83821a5e9..1227edbdc 100644 --- a/libethereum/BlockQueue.h +++ b/libethereum/BlockQueue.h @@ -119,11 +119,21 @@ public: bool unknownFull() const; private: + + struct UnverifiedBlock + { + h256 hash; + h256 parentHash; + bytes block; + }; + void noteReady_WITH_LOCK(h256 const& _b); bool invariants() const override; void verifierBody(); + void collectUnknownBad(h256 const& _bad); + void updateBad(h256 const& _bad); mutable boost::shared_mutex m_lock; ///< General lock for the sets, m_future and m_unknown. h256Hash m_drainingSet; ///< All blocks being imported. @@ -139,7 +149,7 @@ private: std::condition_variable m_moreToVerify; ///< Signaled when m_unverified has a new entry. std::vector m_verified; ///< List of blocks, in correct order, verified and ready for chain-import. std::deque m_verifying; ///< List of blocks being verified; as long as the block component (bytes) is empty, it's not finished. - std::deque> m_unverified; ///< List of blocks, in correct order, ready for verification. + std::deque m_unverified; ///< List of in correct order, ready for verification. std::vector m_verifiers; ///< Threads who only verify. bool m_deleting = false; ///< Exit condition for verifiers.