Less cs_main locks in quorums (#2702)

* Drop cs_main from CQuorumManager::UpdatedBlockTip()

* CLLMQUtils::IsQuorumActive() shouldn't require cs_main to be held

* Revert comment deletion
This commit is contained in:
UdjinM6 2019-02-15 17:11:35 +03:00 committed by Alexander Block
parent 3bbc75fc48
commit 3e4286a584
6 changed files with 107 additions and 111 deletions

View File

@ -166,8 +166,6 @@ void CQuorumManager::UpdatedBlockTip(const CBlockIndex* pindexNew, const CBlockI
return; return;
} }
LOCK(cs_main);
for (auto& p : Params().GetConsensus().llmqs) { for (auto& p : Params().GetConsensus().llmqs) {
EnsureQuorumConnections(p.first, pindexNew); EnsureQuorumConnections(p.first, pindexNew);
} }
@ -175,18 +173,16 @@ void CQuorumManager::UpdatedBlockTip(const CBlockIndex* pindexNew, const CBlockI
void CQuorumManager::EnsureQuorumConnections(Consensus::LLMQType llmqType, const CBlockIndex* pindexNew) void CQuorumManager::EnsureQuorumConnections(Consensus::LLMQType llmqType, const CBlockIndex* pindexNew)
{ {
AssertLockHeld(cs_main);
const auto& params = Params().GetConsensus().llmqs.at(llmqType); const auto& params = Params().GetConsensus().llmqs.at(llmqType);
auto myProTxHash = activeMasternodeInfo.proTxHash; auto myProTxHash = activeMasternodeInfo.proTxHash;
auto lastQuorums = ScanQuorums(llmqType, (size_t)params.keepOldConnections); auto lastQuorums = ScanQuorums(llmqType, pindexNew, (size_t)params.keepOldConnections);
auto connmanQuorumsToDelete = g_connman->GetMasternodeQuorums(llmqType); auto connmanQuorumsToDelete = g_connman->GetMasternodeQuorums(llmqType);
// don't remove connections for the currently in-progress DKG round // don't remove connections for the currently in-progress DKG round
int curDkgHeight = pindexNew->nHeight - (pindexNew->nHeight % params.dkgInterval); int curDkgHeight = pindexNew->nHeight - (pindexNew->nHeight % params.dkgInterval);
auto curDkgBlock = chainActive[curDkgHeight]->GetBlockHash(); auto curDkgBlock = pindexNew->GetAncestor(curDkgHeight)->GetBlockHash();
connmanQuorumsToDelete.erase(curDkgBlock); connmanQuorumsToDelete.erase(curDkgBlock);
for (auto& quorum : lastQuorums) { for (auto& quorum : lastQuorums) {
@ -222,19 +218,14 @@ void CQuorumManager::EnsureQuorumConnections(Consensus::LLMQType llmqType, const
} }
} }
bool CQuorumManager::BuildQuorumFromCommitment(const CFinalCommitment& qc, std::shared_ptr<CQuorum>& quorum) const bool CQuorumManager::BuildQuorumFromCommitment(const CFinalCommitment& qc, const CBlockIndex* pindexQuorum, std::shared_ptr<CQuorum>& quorum) const
{ {
AssertLockHeld(cs_main); assert(pindexQuorum);
assert(qc.quorumHash == pindexQuorum->GetBlockHash());
if (!mapBlockIndex.count(qc.quorumHash)) {
LogPrintf("CQuorumManager::%s -- block %s not found", __func__, qc.quorumHash.ToString());
return false;
}
auto& params = Params().GetConsensus().llmqs.at((Consensus::LLMQType)qc.llmqType);
auto quorumIndex = mapBlockIndex[qc.quorumHash];
auto members = CLLMQUtils::GetAllQuorumMembers((Consensus::LLMQType)qc.llmqType, qc.quorumHash); auto members = CLLMQUtils::GetAllQuorumMembers((Consensus::LLMQType)qc.llmqType, qc.quorumHash);
quorum->Init(qc.quorumHash, quorumIndex->nHeight, members, qc.validMembers, qc.quorumPublicKey); quorum->Init(qc.quorumHash, pindexQuorum->nHeight, members, qc.validMembers, qc.quorumPublicKey);
bool hasValidVvec = false; bool hasValidVvec = false;
if (quorum->ReadContributions(evoDb)) { if (quorum->ReadContributions(evoDb)) {
@ -301,11 +292,15 @@ bool CQuorumManager::HasQuorum(Consensus::LLMQType llmqType, const uint256& quor
std::vector<CQuorumCPtr> CQuorumManager::ScanQuorums(Consensus::LLMQType llmqType, size_t maxCount) std::vector<CQuorumCPtr> CQuorumManager::ScanQuorums(Consensus::LLMQType llmqType, size_t maxCount)
{ {
LOCK(cs_main); const CBlockIndex* pindex;
return ScanQuorums(llmqType, chainActive.Tip()->GetBlockHash(), maxCount); {
LOCK(cs_main);
pindex = chainActive.Tip();
}
return ScanQuorums(llmqType, pindex, maxCount);
} }
std::vector<CQuorumCPtr> CQuorumManager::ScanQuorums(Consensus::LLMQType llmqType, const uint256& startBlock, size_t maxCount) std::vector<CQuorumCPtr> CQuorumManager::ScanQuorums(Consensus::LLMQType llmqType, const CBlockIndex* pindexStart, size_t maxCount)
{ {
std::vector<CQuorumCPtr> result; std::vector<CQuorumCPtr> result;
result.reserve(maxCount); result.reserve(maxCount);
@ -318,19 +313,13 @@ std::vector<CQuorumCPtr> CQuorumManager::ScanQuorums(Consensus::LLMQType llmqTyp
return result; return result;
} }
LOCK(cs_main); auto pindex = pindexStart->GetAncestor(pindexStart->nHeight - (pindexStart->nHeight % params.dkgInterval));
if (!mapBlockIndex.count(startBlock)) {
return result;
}
CBlockIndex* pindex = mapBlockIndex[startBlock];
pindex = chainActive[pindex->nHeight - (pindex->nHeight % params.dkgInterval)];
while (pindex != nullptr while (pindex != nullptr
&& pindex->nHeight >= params.dkgInterval && pindex->nHeight >= params.dkgInterval
&& result.size() < maxCount && result.size() < maxCount
&& deterministicMNManager->IsDIP3Enforced(pindex->nHeight)) { && deterministicMNManager->IsDIP3Enforced(pindex->nHeight)) {
auto quorum = GetQuorum(llmqType, pindex->GetBlockHash()); auto quorum = GetQuorum(llmqType, pindex);
if (quorum) { if (quorum) {
result.emplace_back(quorum); result.emplace_back(quorum);
} }
@ -348,7 +337,25 @@ std::vector<CQuorumCPtr> CQuorumManager::ScanQuorums(Consensus::LLMQType llmqTyp
CQuorumCPtr CQuorumManager::GetQuorum(Consensus::LLMQType llmqType, const uint256& quorumHash) CQuorumCPtr CQuorumManager::GetQuorum(Consensus::LLMQType llmqType, const uint256& quorumHash)
{ {
AssertLockHeld(cs_main); CBlockIndex* pindexQuorum;
{
LOCK(cs_main);
auto quorumIt = mapBlockIndex.find(quorumHash);
if (quorumIt == mapBlockIndex.end()) {
LogPrintf("CQuorumManager::%s -- block %s not found", __func__, quorumHash.ToString());
return nullptr;
}
pindexQuorum = quorumIt->second;
}
return GetQuorum(llmqType, pindexQuorum);
}
CQuorumCPtr CQuorumManager::GetQuorum(Consensus::LLMQType llmqType, const CBlockIndex* pindexQuorum)
{
assert(pindexQuorum);
auto quorumHash = pindexQuorum->GetBlockHash();
// we must check this before we look into the cache. Reorgs might have happened which would mean we might have // we must check this before we look into the cache. Reorgs might have happened which would mean we might have
// cached quorums which are not in the active chain anymore // cached quorums which are not in the active chain anymore
@ -371,7 +378,8 @@ CQuorumCPtr CQuorumManager::GetQuorum(Consensus::LLMQType llmqType, const uint25
auto& params = Params().GetConsensus().llmqs.at(llmqType); auto& params = Params().GetConsensus().llmqs.at(llmqType);
auto quorum = std::make_shared<CQuorum>(params, blsWorker); auto quorum = std::make_shared<CQuorum>(params, blsWorker);
if (!BuildQuorumFromCommitment(qc, quorum)) {
if (!BuildQuorumFromCommitment(qc, pindexQuorum, quorum)) {
return nullptr; return nullptr;
} }

View File

@ -91,20 +91,24 @@ public:
void UpdatedBlockTip(const CBlockIndex *pindexNew, const CBlockIndex *pindexFork, bool fInitialDownload); void UpdatedBlockTip(const CBlockIndex *pindexNew, const CBlockIndex *pindexFork, bool fInitialDownload);
public:
bool HasQuorum(Consensus::LLMQType llmqType, const uint256& quorumHash); bool HasQuorum(Consensus::LLMQType llmqType, const uint256& quorumHash);
// all these methods will lock cs_main // all these methods will lock cs_main for a short period of time
CQuorumCPtr GetQuorum(Consensus::LLMQType llmqType,const uint256& quorumHash); CQuorumCPtr GetQuorum(Consensus::LLMQType llmqType, const uint256& quorumHash);
CQuorumCPtr GetNewestQuorum(Consensus::LLMQType llmqType); CQuorumCPtr GetNewestQuorum(Consensus::LLMQType llmqType);
std::vector<CQuorumCPtr> ScanQuorums(Consensus::LLMQType llmqType, size_t maxCount); std::vector<CQuorumCPtr> ScanQuorums(Consensus::LLMQType llmqType, size_t maxCount);
std::vector<CQuorumCPtr> ScanQuorums(Consensus::LLMQType llmqType, const uint256& startBlock, size_t maxCount);
// this one is cs_main-free
std::vector<CQuorumCPtr> ScanQuorums(Consensus::LLMQType llmqType, const CBlockIndex* pindexStart, size_t maxCount);
private: private:
// all private methods here are cs_main-free
void EnsureQuorumConnections(Consensus::LLMQType llmqType, const CBlockIndex *pindexNew); void EnsureQuorumConnections(Consensus::LLMQType llmqType, const CBlockIndex *pindexNew);
bool BuildQuorumFromCommitment(const CFinalCommitment& qc, std::shared_ptr<CQuorum>& quorum) const; bool BuildQuorumFromCommitment(const CFinalCommitment& qc, const CBlockIndex* pindexQuorum, std::shared_ptr<CQuorum>& quorum) const;
bool BuildQuorumContributions(const CFinalCommitment& fqc, std::shared_ptr<CQuorum>& quorum) const; bool BuildQuorumContributions(const CFinalCommitment& fqc, std::shared_ptr<CQuorum>& quorum) const;
CQuorumCPtr GetQuorum(Consensus::LLMQType llmqType, const CBlockIndex* pindex);
}; };
extern CQuorumManager* quorumManager; extern CQuorumManager* quorumManager;

View File

@ -259,19 +259,15 @@ bool CSigningManager::PreVerifyRecoveredSig(NodeId nodeId, const CRecoveredSig&
return false; return false;
} }
CQuorumCPtr quorum; CQuorumCPtr quorum = quorumManager->GetQuorum(llmqType, recoveredSig.quorumHash);
{
LOCK(cs_main);
quorum = quorumManager->GetQuorum(llmqType, recoveredSig.quorumHash); if (!quorum) {
if (!quorum) { LogPrintf("CSigningManager::%s -- quorum %s not found, node=%d\n", __func__,
LogPrintf("CSigningManager::%s -- quorum %s not found, node=%d\n", __func__, recoveredSig.quorumHash.ToString(), nodeId);
recoveredSig.quorumHash.ToString(), nodeId); return false;
return false; }
} if (!CLLMQUtils::IsQuorumActive(llmqType, quorum->quorumHash)) {
if (!CLLMQUtils::IsQuorumActive(llmqType, quorum->quorumHash)) { return false;
return false;
}
} }
if (!recoveredSig.sig.IsValid()) { if (!recoveredSig.sig.IsValid()) {
@ -316,37 +312,34 @@ void CSigningManager::CollectPendingRecoveredSigsToVerify(
} }
} }
{ for (auto& p : retSigShares) {
LOCK(cs_main); NodeId nodeId = p.first;
for (auto& p : retSigShares) { auto& v = p.second;
NodeId nodeId = p.first;
auto& v = p.second;
for (auto it = v.begin(); it != v.end();) { for (auto it = v.begin(); it != v.end();) {
auto& recSig = *it; auto& recSig = *it;
Consensus::LLMQType llmqType = (Consensus::LLMQType) recSig.llmqType; Consensus::LLMQType llmqType = (Consensus::LLMQType) recSig.llmqType;
auto quorumKey = std::make_pair((Consensus::LLMQType)recSig.llmqType, recSig.quorumHash); auto quorumKey = std::make_pair((Consensus::LLMQType)recSig.llmqType, recSig.quorumHash);
if (!retQuorums.count(quorumKey)) { if (!retQuorums.count(quorumKey)) {
CQuorumCPtr quorum = quorumManager->GetQuorum(llmqType, recSig.quorumHash); CQuorumCPtr quorum = quorumManager->GetQuorum(llmqType, recSig.quorumHash);
if (!quorum) { if (!quorum) {
LogPrintf("CSigningManager::%s -- quorum %s not found, node=%d\n", __func__, LogPrintf("CSigningManager::%s -- quorum %s not found, node=%d\n", __func__,
recSig.quorumHash.ToString(), nodeId); recSig.quorumHash.ToString(), nodeId);
it = v.erase(it); it = v.erase(it);
continue; continue;
} }
if (!CLLMQUtils::IsQuorumActive(llmqType, quorum->quorumHash)) { if (!CLLMQUtils::IsQuorumActive(llmqType, quorum->quorumHash)) {
LogPrintf("CSigningManager::%s -- quorum %s not active anymore, node=%d\n", __func__, LogPrintf("CSigningManager::%s -- quorum %s not active anymore, node=%d\n", __func__,
recSig.quorumHash.ToString(), nodeId); recSig.quorumHash.ToString(), nodeId);
it = v.erase(it); it = v.erase(it);
continue; continue;
}
retQuorums.emplace(quorumKey, quorum);
} }
++it; retQuorums.emplace(quorumKey, quorum);
} }
++it;
} }
} }
} }
@ -561,17 +554,17 @@ CQuorumCPtr CSigningManager::SelectQuorumForSigning(Consensus::LLMQType llmqType
auto& llmqParams = Params().GetConsensus().llmqs.at(llmqType); auto& llmqParams = Params().GetConsensus().llmqs.at(llmqType);
size_t poolSize = (size_t)llmqParams.signingActiveQuorumCount; size_t poolSize = (size_t)llmqParams.signingActiveQuorumCount;
uint256 startBlock; CBlockIndex* pindexStart;
{ {
LOCK(cs_main); LOCK(cs_main);
int startBlockHeight = signHeight - SIGN_HEIGHT_OFFSET; int startBlockHeight = signHeight - SIGN_HEIGHT_OFFSET;
if (startBlockHeight > chainActive.Height()) { if (startBlockHeight > chainActive.Height()) {
return nullptr; return nullptr;
} }
startBlock = chainActive[startBlockHeight]->GetBlockHash(); pindexStart = chainActive[startBlockHeight];
} }
auto quorums = quorumManager->ScanQuorums(llmqType, startBlock, poolSize); auto quorums = quorumManager->ScanQuorums(llmqType, pindexStart, poolSize);
if (quorums.empty()) { if (quorums.empty()) {
return nullptr; return nullptr;
} }

View File

@ -347,31 +347,26 @@ bool CSigSharesManager::PreVerifyBatchedSigShares(NodeId nodeId, const CBatchedS
return false; return false;
} }
CQuorumCPtr quorum; CQuorumCPtr quorum = quorumManager->GetQuorum(llmqType, batchedSigShares.quorumHash);
{ if (!quorum) {
LOCK(cs_main); // TODO should we ban here?
LogPrintf("CSigSharesManager::%s -- quorum %s not found, node=%d\n", __func__,
quorum = quorumManager->GetQuorum(llmqType, batchedSigShares.quorumHash); batchedSigShares.quorumHash.ToString(), nodeId);
if (!quorum) { return false;
// TODO should we ban here? }
LogPrintf("CSigSharesManager::%s -- quorum %s not found, node=%d\n", __func__, if (!CLLMQUtils::IsQuorumActive(llmqType, quorum->quorumHash)) {
batchedSigShares.quorumHash.ToString(), nodeId); // quorum is too old
return false; return false;
} }
if (!CLLMQUtils::IsQuorumActive(llmqType, quorum->quorumHash)) { if (!quorum->IsMember(activeMasternodeInfo.proTxHash)) {
// quorum is too old // we're not a member so we can't verify it (we actually shouldn't have received it)
return false; return false;
} }
if (!quorum->IsMember(activeMasternodeInfo.proTxHash)) { if (quorum->quorumVvec == nullptr) {
// we're not a member so we can't verify it (we actually shouldn't have received it) // TODO we should allow to ask other nodes for the quorum vvec if we missed it in the DKG
return false; LogPrintf("CSigSharesManager::%s -- we don't have the quorum vvec for %s, no verification possible. node=%d\n", __func__,
} batchedSigShares.quorumHash.ToString(), nodeId);
if (quorum->quorumVvec == nullptr) { return false;
// TODO we should allow to ask other nodes for the quorum vvec if we missed it in the DKG
LogPrintf("CSigSharesManager::%s -- we don't have the quorum vvec for %s, no verification possible. node=%d\n", __func__,
batchedSigShares.quorumHash.ToString(), nodeId);
return false;
}
} }
std::set<uint16_t> dupMembers; std::set<uint16_t> dupMembers;
@ -994,17 +989,15 @@ void CSigSharesManager::Cleanup()
} }
} }
{ // Find quorums which became inactive
// Find quorums which became inactive for (auto it = quorumsToCheck.begin(); it != quorumsToCheck.end(); ) {
LOCK(cs_main); if (CLLMQUtils::IsQuorumActive(it->first, it->second)) {
for (auto it = quorumsToCheck.begin(); it != quorumsToCheck.end(); ) { it = quorumsToCheck.erase(it);
if (CLLMQUtils::IsQuorumActive(it->first, it->second)) { } else {
it = quorumsToCheck.erase(it); ++it;
} else {
++it;
}
} }
} }
{ {
// Now delete sessions which are for inactive quorums // Now delete sessions which are for inactive quorums
LOCK(cs); LOCK(cs);

View File

@ -99,8 +99,6 @@ std::set<size_t> CLLMQUtils::CalcDeterministicWatchConnections(Consensus::LLMQTy
bool CLLMQUtils::IsQuorumActive(Consensus::LLMQType llmqType, const uint256& quorumHash) bool CLLMQUtils::IsQuorumActive(Consensus::LLMQType llmqType, const uint256& quorumHash)
{ {
AssertLockHeld(cs_main);
auto& params = Params().GetConsensus().llmqs.at(llmqType); auto& params = Params().GetConsensus().llmqs.at(llmqType);
// sig shares and recovered sigs are only accepted from recent/active quorums // sig shares and recovered sigs are only accepted from recent/active quorums

View File

@ -38,7 +38,7 @@ UniValue quorum_list(const JSONRPCRequest& request)
for (auto& p : Params().GetConsensus().llmqs) { for (auto& p : Params().GetConsensus().llmqs) {
UniValue v(UniValue::VARR); UniValue v(UniValue::VARR);
auto quorums = llmq::quorumManager->ScanQuorums(p.first, chainActive.Tip()->GetBlockHash(), count); auto quorums = llmq::quorumManager->ScanQuorums(p.first, chainActive.Tip(), count);
for (auto& q : quorums) { for (auto& q : quorums) {
v.push_back(q->quorumHash.ToString()); v.push_back(q->quorumHash.ToString());
} }