// Copyright (c) 2018-2019 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. #ifndef DASH_QUORUMS_H #define DASH_QUORUMS_H #include "evo/evodb.h" #include "evo/deterministicmns.h" #include "validationinterface.h" #include "consensus/params.h" #include "bls/bls.h" #include "bls/bls_worker.h" namespace llmq { class CDKGSessionManager; /** * An object of this class represents a quorum which was mined on-chain (through a quorum commitment) * It at least contains informations about the members and the quorum public key which is needed to verify recovered * signatures from this quorum. * * In case the local node is a member of the same quorum and successfully participated in the DKG, the quorum object * will also contain the secret key share and the quorum verification vector. The quorum vvec is then used to recover * the public key shares of individual members, which are needed to verify signature shares of these members. */ class CQuorum { friend class CQuorumManager; public: const Consensus::LLMQParams& params; uint256 quorumHash; int height; std::vector members; std::vector validMembers; CBLSPublicKey quorumPublicKey; // These are only valid when we either participated in the DKG or fully watched it BLSVerificationVectorPtr quorumVvec; CBLSSecretKey skShare; private: // Recovery of public key shares is very slow, so we start a background thread that pre-populates a cache so that // the public key shares are ready when needed later mutable CBLSWorkerCache blsCache; std::atomic stopCachePopulatorThread; std::thread cachePopulatorThread; public: CQuorum(const Consensus::LLMQParams& _params, CBLSWorker& _blsWorker) : params(_params), blsCache(_blsWorker), stopCachePopulatorThread(false) {} ~CQuorum(); void Init(const uint256& quorumHash, int height, const std::vector& members, const std::vector& validMembers, const CBLSPublicKey& quorumPublicKey); bool IsMember(const uint256& proTxHash) const; bool IsValidMember(const uint256& proTxHash) const; int GetMemberIndex(const uint256& proTxHash) const; CBLSPublicKey GetPubKeyShare(size_t memberIdx) const; CBLSSecretKey GetSkShare() const; private: void WriteContributions(CEvoDB& evoDb); bool ReadContributions(CEvoDB& evoDb); static void StartCachePopulatorThread(std::shared_ptr _this); }; typedef std::shared_ptr CQuorumPtr; typedef std::shared_ptr CQuorumCPtr; /** * The quorum manager maintains quorums which were mined on chain. When a quorum is requested from the manager, * it will lookup the commitment (through CQuorumBlockProcessor) and build a CQuorum object from it. * * It is also responsible for initialization of the inter-quorum connections for new quorums. */ class CQuorumManager { private: CEvoDB& evoDb; CBLSWorker& blsWorker; CDKGSessionManager& dkgManager; CCriticalSection quorumsCacheCs; std::map, CQuorumPtr> quorumsCache; public: CQuorumManager(CEvoDB& _evoDb, CBLSWorker& _blsWorker, CDKGSessionManager& _dkgManager); void UpdatedBlockTip(const CBlockIndex *pindexNew, const CBlockIndex *pindexFork, bool fInitialDownload); public: bool HasQuorum(Consensus::LLMQType llmqType, const uint256& quorumHash); // all these methods will lock cs_main CQuorumCPtr GetQuorum(Consensus::LLMQType llmqType,const uint256& quorumHash); CQuorumCPtr GetNewestQuorum(Consensus::LLMQType llmqType); std::vector ScanQuorums(Consensus::LLMQType llmqType, size_t maxCount); std::vector ScanQuorums(Consensus::LLMQType llmqType, const uint256& startBlock, size_t maxCount); private: void EnsureQuorumConnections(Consensus::LLMQType llmqType, const CBlockIndex *pindexNew); bool BuildQuorumFromCommitment(const CFinalCommitment& qc, std::shared_ptr& quorum) const; bool BuildQuorumContributions(const CFinalCommitment& fqc, std::shared_ptr& quorum) const; }; extern CQuorumManager* quorumManager; } #endif //DASH_QUORUMS_H