a69a5cf4a0
* Require only 3 out of 5 signatures for old InstantSend in regtest mode * Use LLMQs of size 5 with threshold of 3 for regtest * Fix wrong check for out-of-range bits in CFixedBitSet * Reduce number of masternodes in masternode/LLMQ tests * Add missing \n to LogPrintf call * Use correct indexes for isolated/receiver/sender nodes The way it was before resulted in nodes 1-3 being unused and 6-8 being used for these 3 special nodes even though these are masternodes. * Avoid stopping/starting isolated node in p2p-instantsend.py It's enough to disable networking for this node.
394 lines
12 KiB
C++
394 lines
12 KiB
C++
// Copyright (c) 2014-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 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 = 70213;
|
|
|
|
/// 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
|
|
{
|
|
public:
|
|
/// Automatic locks of "simple" transactions are only allowed
|
|
/// when mempool usage is lower than this threshold
|
|
static const double AUTO_IX_MEMPOOL_THRESHOLD;
|
|
private:
|
|
static const std::string SERIALIZATION_VERSION_STRING;
|
|
|
|
// Keep track of current block height
|
|
int nCachedBlockHeight;
|
|
|
|
// maps for AlreadyHave
|
|
std::map<uint256, CTxLockRequest> mapLockRequestAccepted; ///< Tx hash - Tx
|
|
std::map<uint256, CTxLockRequest> mapLockRequestRejected; ///< Tx hash - Tx
|
|
std::map<uint256, CTxLockVote> mapTxLockVotes; ///< Vote hash - Vote
|
|
std::map<uint256, CTxLockVote> mapTxLockVotesOrphan; ///< Vote hash - Vote
|
|
|
|
std::map<uint256, CTxLockCandidate> mapTxLockCandidates; ///< Tx hash - Lock candidate
|
|
|
|
std::map<COutPoint, std::set<uint256> > mapVotedOutpoints; ///< UTXO - Tx hash set
|
|
std::map<COutPoint, uint256> mapLockedOutpoints; ///< UTXO - Tx hash
|
|
|
|
/// Track masternodes who voted with no txlockrequest (for DOS protection)
|
|
std::map<COutPoint, int64_t> 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 <typename Stream, typename Operation>
|
|
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<bool> 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 <typename Stream, typename Operation>
|
|
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<unsigned char> 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 <typename Stream, typename Operation>
|
|
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<COutPoint, CTxLockVote> mapMasternodeVotes; ///< Masternode outpoint - vote
|
|
bool fAttacked = false;
|
|
|
|
public:
|
|
COutPointLock() {}
|
|
|
|
COutPointLock(const COutPoint& outpointIn) :
|
|
outpoint(outpointIn),
|
|
mapMasternodeVotes()
|
|
{}
|
|
|
|
COutPoint GetOutpoint() const { return outpoint; }
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
template <typename Stream, typename Operation>
|
|
inline void SerializationOp(Stream& s, Operation ser_action) {
|
|
READWRITE(outpoint);
|
|
READWRITE(mapMasternodeVotes);
|
|
READWRITE(fAttacked);
|
|
}
|
|
|
|
bool AddVote(const CTxLockVote& vote);
|
|
std::vector<CTxLockVote> GetVotes() const;
|
|
bool HasMasternodeVoted(const COutPoint& outpointMasternodeIn) const;
|
|
int CountVotes() const { return fAttacked ? 0 : mapMasternodeVotes.size(); }
|
|
bool IsReady() const;
|
|
void MarkAsAttacked() { fAttacked = true; }
|
|
|
|
void Relay(CConnman& connman) const;
|
|
};
|
|
|
|
/**
|
|
* An InstantSend transaction lock candidate.
|
|
*/
|
|
class CTxLockCandidate
|
|
{
|
|
private:
|
|
int nConfirmedHeight; ///<When corresponding tx is 0-confirmed or conflicted, nConfirmedHeight is -1
|
|
int64_t nTimeCreated;
|
|
|
|
public:
|
|
CTxLockCandidate() :
|
|
nConfirmedHeight(-1),
|
|
nTimeCreated(GetTime())
|
|
{}
|
|
|
|
CTxLockCandidate(const CTxLockRequest& txLockRequestIn) :
|
|
nConfirmedHeight(-1),
|
|
nTimeCreated(GetTime()),
|
|
txLockRequest(txLockRequestIn),
|
|
mapOutPointLocks()
|
|
{}
|
|
|
|
CTxLockRequest txLockRequest;
|
|
std::map<COutPoint, COutPointLock> mapOutPointLocks;
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
template <typename Stream, typename Operation>
|
|
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
|