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:
parent
3bbc75fc48
commit
3e4286a584
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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;
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
|
@ -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
|
||||||
|
@ -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());
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user