// Copyright (c) 2014-2017 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 INSTANTX_H #define INSTANTX_H #include "chain.h" #include "net.h" #include "primitives/transaction.h" #include "evo/deterministicmns.h" class CTxLockVote; class COutPointLock; class CTxLockRequest; class CTxLockCandidate; class CInstantSend; extern CInstantSend instantsend; /* At 15 signatures, 1/2 of the masternode network can be owned by one party without compromising the security of InstantSend (1000/2150.0)**10 = 0.00047382219560689856 (1000/2900.0)**10 = 2.3769498616783657e-05 ### getting 5 of 10 signatures w/ 1000 nodes of 2900 (1000/2900.0)**5 = 0.004875397277841433 */ static const int MIN_INSTANTSEND_PROTO_VERSION = 70210; /// For how long we are going to accept votes/locks /// after we saw the first one for a specific transaction static const int INSTANTSEND_LOCK_TIMEOUT_SECONDS = 15; /// For how long we are going to keep invalid votes and votes for failed lock attempts, /// must be greater than INSTANTSEND_LOCK_TIMEOUT_SECONDS static const int INSTANTSEND_FAILED_TIMEOUT_SECONDS = 60; extern bool fEnableInstantSend; extern int nCompleteTXLocks; /** * Manages InstantSend. Processes lock requests, candidates, and votes. */ class CInstantSend { private: static const std::string SERIALIZATION_VERSION_STRING; /// Automatic locks of "simple" transactions are only allowed /// when mempool usage is lower than this threshold static const double AUTO_IX_MEMPOOL_THRESHOLD; // Keep track of current block height int nCachedBlockHeight; // maps for AlreadyHave std::map mapLockRequestAccepted; ///< Tx hash - Tx std::map mapLockRequestRejected; ///< Tx hash - Tx std::map mapTxLockVotes; ///< Vote hash - Vote std::map mapTxLockVotesOrphan; ///< Vote hash - Vote std::map mapTxLockCandidates; ///< Tx hash - Lock candidate std::map > mapVotedOutpoints; ///< UTXO - Tx hash set std::map mapLockedOutpoints; ///< UTXO - Tx hash /// Track masternodes who voted with no txlockrequest (for DOS protection) std::map mapMasternodeOrphanVotes; ///< MN outpoint - Time bool CreateTxLockCandidate(const CTxLockRequest& txLockRequest); void CreateEmptyTxLockCandidate(const uint256& txHash); void Vote(CTxLockCandidate& txLockCandidate, CConnman& connman); /// Process consensus vote message bool ProcessNewTxLockVote(CNode* pfrom, const CTxLockVote& vote, CConnman& connman); void UpdateVotedOutpoints(const CTxLockVote& vote, CTxLockCandidate& txLockCandidate); bool ProcessOrphanTxLockVote(const CTxLockVote& vote); void ProcessOrphanTxLockVotes(); int64_t GetAverageMasternodeOrphanVoteTime(); void TryToFinalizeLockCandidate(const CTxLockCandidate& txLockCandidate); void LockTransactionInputs(const CTxLockCandidate& txLockCandidate); /// Update UI and notify external script if any void UpdateLockedTransaction(const CTxLockCandidate& txLockCandidate); bool ResolveConflicts(const CTxLockCandidate& txLockCandidate); public: mutable CCriticalSection cs_instantsend; ADD_SERIALIZE_METHODS; template inline void SerializationOp(Stream& s, Operation ser_action) { std::string strVersion; if(ser_action.ForRead()) { READWRITE(strVersion); } else { strVersion = SERIALIZATION_VERSION_STRING; READWRITE(strVersion); } READWRITE(mapLockRequestAccepted); READWRITE(mapLockRequestRejected); READWRITE(mapTxLockVotes); READWRITE(mapTxLockVotesOrphan); READWRITE(mapTxLockCandidates); READWRITE(mapVotedOutpoints); READWRITE(mapLockedOutpoints); READWRITE(mapMasternodeOrphanVotes); READWRITE(nCachedBlockHeight); if(ser_action.ForRead() && (strVersion != SERIALIZATION_VERSION_STRING)) { Clear(); } } void Clear(); void ProcessMessage(CNode* pfrom, const std::string& strCommand, CDataStream& vRecv, CConnman& connman); bool ProcessTxLockRequest(const CTxLockRequest& txLockRequest, CConnman& connman); void Vote(const uint256& txHash, CConnman& connman); bool AlreadyHave(const uint256& hash); void AcceptLockRequest(const CTxLockRequest& txLockRequest); void RejectLockRequest(const CTxLockRequest& txLockRequest); bool HasTxLockRequest(const uint256& txHash); bool GetTxLockRequest(const uint256& txHash, CTxLockRequest& txLockRequestRet); bool GetTxLockVote(const uint256& hash, CTxLockVote& txLockVoteRet); bool GetLockedOutPointTxHash(const COutPoint& outpoint, uint256& hashRet); /// Verify if transaction is currently locked bool IsLockedInstantSendTransaction(const uint256& txHash); /// Get the actual number of accepted lock signatures int GetTransactionLockSignatures(const uint256& txHash); /// Remove expired entries from maps void CheckAndRemove(); /// Verify if transaction lock timed out bool IsTxLockCandidateTimedOut(const uint256& txHash); void Relay(const uint256& txHash, CConnman& connman); void UpdatedBlockTip(const CBlockIndex *pindex); void SyncTransaction(const CTransaction& tx, const CBlockIndex *pindex, int posInBlock); std::string ToString() const; void DoMaintenance(); /// checks if we can automatically lock "simple" transactions static bool CanAutoLock(); /// flag of the AutoLock Bip9 activation static std::atomic isAutoLockBip9Active; }; /** * An InstantSend transaction lock request. */ class CTxLockRequest { private: static const CAmount MIN_FEE = 0.0001 * COIN; /// If transaction has less or equal inputs than MAX_INPUTS_FOR_AUTO_IX, /// it will be automatically locked static const int MAX_INPUTS_FOR_AUTO_IX = 4; public: /// Warn for a large number of inputs to an IS tx - fees could be substantial /// and the number txlvote responses requested large (10 * # of inputs) static const int WARN_MANY_INPUTS = 100; CTransactionRef tx; CTxLockRequest() : tx(MakeTransactionRef()) {} CTxLockRequest(const CTransaction& _tx) : tx(MakeTransactionRef(_tx)) {}; CTxLockRequest(const CTransactionRef& _tx) : tx(_tx) {}; ADD_SERIALIZE_METHODS; template inline void SerializationOp(Stream& s, Operation ser_action) { READWRITE(tx); } bool IsValid() const; CAmount GetMinFee(bool fForceMinFee) const; int GetMaxSignatures() const; // checks if related transaction is "simple" to lock it automatically bool IsSimple() const; const uint256 &GetHash() const { return tx->GetHash(); } std::string ToString() const { return tx->ToString(); } friend bool operator==(const CTxLockRequest& a, const CTxLockRequest& b) { return *a.tx == *b.tx; } friend bool operator!=(const CTxLockRequest& a, const CTxLockRequest& b) { return *a.tx != *b.tx; } explicit operator bool() const { return *this != CTxLockRequest(); } }; /** * An InstantSend transaction lock vote. Sent by a masternode in response to a * transaction lock request (ix message) to indicate the transaction input can * be locked. Contains the proposed transaction's hash and the outpoint being * locked along with the masternodes outpoint and signature. * @see CTxLockRequest */ class CTxLockVote { private: uint256 txHash; COutPoint outpoint; // TODO remove this member (not needed anymore after DIP3 has been deployed) COutPoint outpointMasternode; uint256 quorumModifierHash; uint256 masternodeProTxHash; std::vector vchMasternodeSignature; // local memory only int nConfirmedHeight; ///< When corresponding tx is 0-confirmed or conflicted, nConfirmedHeight is -1 int64_t nTimeCreated; public: CTxLockVote() : txHash(), outpoint(), outpointMasternode(), quorumModifierHash(), masternodeProTxHash(), vchMasternodeSignature(), nConfirmedHeight(-1), nTimeCreated(GetTime()) {} CTxLockVote(const uint256& txHashIn, const COutPoint& outpointIn, const COutPoint& outpointMasternodeIn, const uint256& quorumModifierHashIn, const uint256& masternodeProTxHashIn) : txHash(txHashIn), outpoint(outpointIn), outpointMasternode(outpointMasternodeIn), quorumModifierHash(quorumModifierHashIn), masternodeProTxHash(masternodeProTxHashIn), vchMasternodeSignature(), nConfirmedHeight(-1), nTimeCreated(GetTime()) {} ADD_SERIALIZE_METHODS; template inline void SerializationOp(Stream& s, Operation ser_action) { READWRITE(txHash); READWRITE(outpoint); READWRITE(outpointMasternode); READWRITE(quorumModifierHash); READWRITE(masternodeProTxHash); if (!(s.GetType() & SER_GETHASH)) { READWRITE(vchMasternodeSignature); } } uint256 GetHash() const; uint256 GetSignatureHash() const; uint256 GetTxHash() const { return txHash; } COutPoint GetOutpoint() const { return outpoint; } COutPoint GetMasternodeOutpoint() const { return outpointMasternode; } bool IsValid(CNode* pnode, CConnman& connman) const; void SetConfirmedHeight(int nConfirmedHeightIn) { nConfirmedHeight = nConfirmedHeightIn; } bool IsExpired(int nHeight) const; bool IsTimedOut() const; bool IsFailed() const; bool Sign(); bool CheckSignature() const; void Relay(CConnman& connman) const; }; /** * An InstantSend OutpointLock. */ class COutPointLock { private: COutPoint outpoint; ///< UTXO std::map mapMasternodeVotes; ///< Masternode outpoint - vote bool fAttacked = false; public: static const int SIGNATURES_REQUIRED = 6; static const int SIGNATURES_TOTAL = 10; COutPointLock() {} COutPointLock(const COutPoint& outpointIn) : outpoint(outpointIn), mapMasternodeVotes() {} COutPoint GetOutpoint() const { return outpoint; } ADD_SERIALIZE_METHODS; template inline void SerializationOp(Stream& s, Operation ser_action) { READWRITE(outpoint); READWRITE(mapMasternodeVotes); READWRITE(fAttacked); } bool AddVote(const CTxLockVote& vote); std::vector GetVotes() const; bool HasMasternodeVoted(const COutPoint& outpointMasternodeIn) const; int CountVotes() const { return fAttacked ? 0 : mapMasternodeVotes.size(); } bool IsReady() const { return !fAttacked && CountVotes() >= SIGNATURES_REQUIRED; } void MarkAsAttacked() { fAttacked = true; } void Relay(CConnman& connman) const; }; /** * An InstantSend transaction lock candidate. */ class CTxLockCandidate { private: int nConfirmedHeight; /// mapOutPointLocks; ADD_SERIALIZE_METHODS; template inline void SerializationOp(Stream& s, Operation ser_action) { READWRITE(txLockRequest); READWRITE(mapOutPointLocks); READWRITE(nTimeCreated); READWRITE(nConfirmedHeight); } uint256 GetHash() const { return txLockRequest.GetHash(); } void AddOutPointLock(const COutPoint& outpoint); void MarkOutpointAsAttacked(const COutPoint& outpoint); bool AddVote(const CTxLockVote& vote); bool IsAllOutPointsReady() const; bool HasMasternodeVoted(const COutPoint& outpointIn, const COutPoint& outpointMasternodeIn); int CountVotes() const; void SetConfirmedHeight(int nConfirmedHeightIn) { nConfirmedHeight = nConfirmedHeightIn; } bool IsExpired(int nHeight) const; bool IsTimedOut() const; void Relay(CConnman& connman) const; }; #endif