2018-05-24 16:14:55 +02:00
|
|
|
// Copyright (c) 2018 The Dash Core developers
|
|
|
|
// Distributed under the MIT/X11 software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
|
|
|
|
|
|
|
#include "quorums_dkgsessionmgr.h"
|
|
|
|
#include "quorums_blockprocessor.h"
|
2019-01-08 09:55:19 +01:00
|
|
|
#include "quorums_debug.h"
|
2018-05-24 16:14:55 +02:00
|
|
|
#include "quorums_init.h"
|
|
|
|
#include "quorums_utils.h"
|
|
|
|
|
|
|
|
#include "chainparams.h"
|
|
|
|
#include "net_processing.h"
|
|
|
|
#include "spork.h"
|
|
|
|
#include "validation.h"
|
|
|
|
|
|
|
|
namespace llmq
|
|
|
|
{
|
|
|
|
|
|
|
|
CDKGSessionManager* quorumDKGSessionManager;
|
|
|
|
|
|
|
|
static const std::string DB_VVEC = "qdkg_V";
|
|
|
|
static const std::string DB_SKCONTRIB = "qdkg_S";
|
|
|
|
|
|
|
|
CDKGSessionManager::CDKGSessionManager(CEvoDB& _evoDb, CBLSWorker& _blsWorker) :
|
|
|
|
evoDb(_evoDb),
|
|
|
|
blsWorker(_blsWorker)
|
|
|
|
{
|
|
|
|
for (auto& qt : Params().GetConsensus().llmqs) {
|
|
|
|
dkgSessionHandlers.emplace(std::piecewise_construct,
|
|
|
|
std::forward_as_tuple(qt.first),
|
|
|
|
std::forward_as_tuple(qt.second, _evoDb, messageHandlerPool, blsWorker, *this));
|
|
|
|
}
|
|
|
|
|
|
|
|
messageHandlerPool.resize(2);
|
|
|
|
RenameThreadPool(messageHandlerPool, "quorum-msg");
|
|
|
|
}
|
|
|
|
|
|
|
|
CDKGSessionManager::~CDKGSessionManager()
|
|
|
|
{
|
|
|
|
messageHandlerPool.stop(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CDKGSessionManager::UpdatedBlockTip(const CBlockIndex* pindexNew, const CBlockIndex* pindexFork, bool fInitialDownload)
|
|
|
|
{
|
|
|
|
const auto& consensus = Params().GetConsensus();
|
|
|
|
|
|
|
|
CleanupCache();
|
|
|
|
|
|
|
|
if (fInitialDownload)
|
|
|
|
return;
|
|
|
|
if (!deterministicMNManager->IsDIP3Active(pindexNew->nHeight))
|
|
|
|
return;
|
|
|
|
if (!sporkManager.IsSporkActive(SPORK_17_QUORUM_DKG_ENABLED))
|
|
|
|
return;
|
|
|
|
|
|
|
|
LOCK(cs_main);
|
|
|
|
|
|
|
|
for (auto& qt : dkgSessionHandlers) {
|
|
|
|
qt.second.UpdatedBlockTip(pindexNew, pindexFork, fInitialDownload);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void CDKGSessionManager::ProcessMessage(CNode* pfrom, const std::string& strCommand, CDataStream& vRecv, CConnman& connman)
|
|
|
|
{
|
|
|
|
if (!sporkManager.IsSporkActive(SPORK_17_QUORUM_DKG_ENABLED))
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (strCommand != NetMsgType::QCONTRIB
|
|
|
|
&& strCommand != NetMsgType::QCOMPLAINT
|
|
|
|
&& strCommand != NetMsgType::QJUSTIFICATION
|
|
|
|
&& strCommand != NetMsgType::QPCOMMITMENT
|
|
|
|
&& strCommand != NetMsgType::QWATCH) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strCommand == NetMsgType::QWATCH) {
|
|
|
|
pfrom->qwatch = true;
|
|
|
|
for (auto& p : dkgSessionHandlers) {
|
|
|
|
LOCK2(p.second.cs, p.second.curSession->invCs);
|
|
|
|
p.second.curSession->participatingNodes.emplace(pfrom->addr);
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vRecv.size() < 1) {
|
|
|
|
LOCK(cs_main);
|
|
|
|
Misbehaving(pfrom->id, 100);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// peek into the message and see which LLMQType it is. First byte of all messages is always the LLMQType
|
|
|
|
Consensus::LLMQType llmqType = (Consensus::LLMQType)*vRecv.begin();
|
|
|
|
if (!dkgSessionHandlers.count(llmqType)) {
|
|
|
|
LOCK(cs_main);
|
|
|
|
Misbehaving(pfrom->id, 100);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
dkgSessionHandlers.at(llmqType).ProcessMessage(pfrom, strCommand, vRecv, connman);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDKGSessionManager::AlreadyHave(const CInv& inv) const
|
|
|
|
{
|
|
|
|
if (!sporkManager.IsSporkActive(SPORK_17_QUORUM_DKG_ENABLED))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (auto& p : dkgSessionHandlers) {
|
|
|
|
auto& dkgType = p.second;
|
|
|
|
if (dkgType.pendingContributions.HasSeen(inv.hash)
|
|
|
|
|| dkgType.pendingComplaints.HasSeen(inv.hash)
|
|
|
|
|| dkgType.pendingJustifications.HasSeen(inv.hash)
|
|
|
|
|| dkgType.pendingPrematureCommitments.HasSeen(inv.hash)) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDKGSessionManager::GetContribution(const uint256& hash, CDKGContribution& ret) const
|
|
|
|
{
|
|
|
|
if (!sporkManager.IsSporkActive(SPORK_17_QUORUM_DKG_ENABLED))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (auto& p : dkgSessionHandlers) {
|
|
|
|
auto& dkgType = p.second;
|
|
|
|
LOCK2(dkgType.cs, dkgType.curSession->invCs);
|
|
|
|
if (dkgType.phase < QuorumPhase_Initialized || dkgType.phase > QuorumPhase_Contribute) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
auto it = dkgType.curSession->contributions.find(hash);
|
|
|
|
if (it != dkgType.curSession->contributions.end()) {
|
|
|
|
ret = it->second;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDKGSessionManager::GetComplaint(const uint256& hash, CDKGComplaint& ret) const
|
|
|
|
{
|
|
|
|
if (!sporkManager.IsSporkActive(SPORK_17_QUORUM_DKG_ENABLED))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (auto& p : dkgSessionHandlers) {
|
|
|
|
auto& dkgType = p.second;
|
|
|
|
LOCK2(dkgType.cs, dkgType.curSession->invCs);
|
|
|
|
if (dkgType.phase < QuorumPhase_Contribute || dkgType.phase > QuorumPhase_Complain) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
auto it = dkgType.curSession->complaints.find(hash);
|
|
|
|
if (it != dkgType.curSession->complaints.end()) {
|
|
|
|
ret = it->second;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDKGSessionManager::GetJustification(const uint256& hash, CDKGJustification& ret) const
|
|
|
|
{
|
|
|
|
if (!sporkManager.IsSporkActive(SPORK_17_QUORUM_DKG_ENABLED))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (auto& p : dkgSessionHandlers) {
|
|
|
|
auto& dkgType = p.second;
|
|
|
|
LOCK2(dkgType.cs, dkgType.curSession->invCs);
|
|
|
|
if (dkgType.phase < QuorumPhase_Complain || dkgType.phase > QuorumPhase_Justify) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
auto it = dkgType.curSession->justifications.find(hash);
|
|
|
|
if (it != dkgType.curSession->justifications.end()) {
|
|
|
|
ret = it->second;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDKGSessionManager::GetPrematureCommitment(const uint256& hash, CDKGPrematureCommitment& ret) const
|
|
|
|
{
|
|
|
|
if (!sporkManager.IsSporkActive(SPORK_17_QUORUM_DKG_ENABLED))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
for (auto& p : dkgSessionHandlers) {
|
|
|
|
auto& dkgType = p.second;
|
|
|
|
LOCK2(dkgType.cs, dkgType.curSession->invCs);
|
|
|
|
if (dkgType.phase < QuorumPhase_Justify || dkgType.phase > QuorumPhase_Commit) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
auto it = dkgType.curSession->prematureCommitments.find(hash);
|
|
|
|
if (it != dkgType.curSession->prematureCommitments.end() && dkgType.curSession->validCommitments.count(hash)) {
|
|
|
|
ret = it->second;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CDKGSessionManager::WriteVerifiedVvecContribution(Consensus::LLMQType llmqType, const uint256& quorumHash, const uint256& proTxHash, const BLSVerificationVectorPtr& vvec)
|
|
|
|
{
|
|
|
|
evoDb.GetRawDB().Write(std::make_tuple(DB_VVEC, (uint8_t)llmqType, quorumHash, proTxHash), *vvec);
|
|
|
|
}
|
|
|
|
|
|
|
|
void CDKGSessionManager::WriteVerifiedSkContribution(Consensus::LLMQType llmqType, const uint256& quorumHash, const uint256& proTxHash, const CBLSSecretKey& skContribution)
|
|
|
|
{
|
|
|
|
evoDb.GetRawDB().Write(std::make_tuple(DB_SKCONTRIB, (uint8_t)llmqType, quorumHash, proTxHash), skContribution);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDKGSessionManager::GetVerifiedContributions(Consensus::LLMQType llmqType, const uint256& quorumHash, const std::vector<bool>& validMembers, std::vector<uint16_t>& memberIndexesRet, std::vector<BLSVerificationVectorPtr>& vvecsRet, BLSSecretKeyVector& skContributionsRet)
|
|
|
|
{
|
|
|
|
auto members = CLLMQUtils::GetAllQuorumMembers(llmqType, quorumHash);
|
|
|
|
|
|
|
|
if (validMembers.size() != members.size()) {
|
|
|
|
// should never happen as we should always call this method with correct params
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
memberIndexesRet.clear();
|
|
|
|
vvecsRet.clear();
|
|
|
|
skContributionsRet.clear();
|
|
|
|
memberIndexesRet.reserve(members.size());
|
|
|
|
vvecsRet.reserve(members.size());
|
|
|
|
skContributionsRet.reserve(members.size());
|
|
|
|
for (size_t i = 0; i < members.size(); i++) {
|
|
|
|
if (validMembers[i]) {
|
|
|
|
BLSVerificationVectorPtr vvec;
|
|
|
|
CBLSSecretKey skContribution;
|
|
|
|
if (!GetVerifiedContribution(llmqType, quorumHash, members[i]->proTxHash, vvec, skContribution)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
memberIndexesRet.emplace_back(i);
|
|
|
|
vvecsRet.emplace_back(vvec);
|
|
|
|
skContributionsRet.emplace_back(skContribution);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool CDKGSessionManager::GetVerifiedContribution(Consensus::LLMQType llmqType, const uint256& quorumHash, const uint256& proTxHash, BLSVerificationVectorPtr& vvecRet, CBLSSecretKey& skContributionRet)
|
|
|
|
{
|
|
|
|
LOCK(contributionsCacheCs);
|
|
|
|
ContributionsCacheKey cacheKey = {llmqType, quorumHash, proTxHash};
|
|
|
|
auto it = contributionsCache.find(cacheKey);
|
|
|
|
if (it != contributionsCache.end()) {
|
|
|
|
vvecRet = it->second.vvec;
|
|
|
|
skContributionRet = it->second.skContribution;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
BLSVerificationVector vvec;
|
|
|
|
BLSVerificationVectorPtr vvecPtr;
|
|
|
|
CBLSSecretKey skContribution;
|
|
|
|
if (evoDb.GetRawDB().Read(std::make_tuple(DB_VVEC, (uint8_t)llmqType, quorumHash, proTxHash), vvec)) {
|
|
|
|
vvecPtr = std::make_shared<BLSVerificationVector>(std::move(vvec));
|
|
|
|
}
|
|
|
|
evoDb.GetRawDB().Read(std::make_tuple(DB_SKCONTRIB, (uint8_t)llmqType, quorumHash, proTxHash), skContribution);
|
|
|
|
|
|
|
|
it = contributionsCache.emplace(cacheKey, ContributionsCacheEntry{GetTimeMillis(), vvecPtr, skContribution}).first;
|
|
|
|
|
|
|
|
vvecRet = it->second.vvec;
|
|
|
|
skContributionRet = it->second.skContribution;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
void CDKGSessionManager::CleanupCache()
|
|
|
|
{
|
|
|
|
LOCK(contributionsCacheCs);
|
|
|
|
auto curTime = GetTimeMillis();
|
|
|
|
for (auto it = contributionsCache.begin(); it != contributionsCache.end(); ) {
|
|
|
|
if (curTime - it->second.entryTime > MAX_CONTRIBUTION_CACHE_TIME) {
|
|
|
|
it = contributionsCache.erase(it);
|
|
|
|
} else {
|
|
|
|
++it;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|