2016-12-20 14:26:45 +01:00
|
|
|
// Copyright (c) 2014-2017 The Dash Core developers
|
2014-12-09 02:17:57 +01:00
|
|
|
// Distributed under the MIT/X11 software license, see the accompanying
|
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
2015-02-04 14:24:56 +01:00
|
|
|
|
2017-05-05 13:26:27 +02:00
|
|
|
#ifndef PRIVATESEND_H
|
|
|
|
#define PRIVATESEND_H
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-12-07 10:42:47 +01:00
|
|
|
#include "chain.h"
|
2017-05-05 13:26:27 +02:00
|
|
|
#include "chainparams.h"
|
|
|
|
#include "primitives/transaction.h"
|
|
|
|
#include "pubkey.h"
|
2017-06-30 20:30:16 +02:00
|
|
|
#include "sync.h"
|
2017-05-05 13:26:27 +02:00
|
|
|
#include "tinyformat.h"
|
2017-08-29 01:51:44 +02:00
|
|
|
#include "timedata.h"
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-05-05 13:26:27 +02:00
|
|
|
class CPrivateSend;
|
2017-09-19 16:51:38 +02:00
|
|
|
class CConnman;
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
// timeouts
|
2016-09-01 09:03:47 +02:00
|
|
|
static const int PRIVATESEND_AUTO_TIMEOUT_MIN = 5;
|
|
|
|
static const int PRIVATESEND_AUTO_TIMEOUT_MAX = 15;
|
|
|
|
static const int PRIVATESEND_QUEUE_TIMEOUT = 30;
|
|
|
|
static const int PRIVATESEND_SIGNING_TIMEOUT = 15;
|
2016-08-05 21:49:45 +02:00
|
|
|
|
2016-09-30 20:19:26 +02:00
|
|
|
//! minimum peer version accepted by mixing pool
|
2017-09-22 03:53:15 +02:00
|
|
|
static const int MIN_PRIVATESEND_PEER_PROTO_VERSION = 70208;
|
2016-08-05 21:49:45 +02:00
|
|
|
|
2018-05-26 20:03:23 +02:00
|
|
|
static const size_t PRIVATESEND_ENTRY_MAX_SIZE = 9;
|
2016-08-05 21:49:45 +02:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
// pool responses
|
|
|
|
enum PoolMessage {
|
|
|
|
ERR_ALREADY_HAVE,
|
|
|
|
ERR_DENOM,
|
|
|
|
ERR_ENTRIES_FULL,
|
|
|
|
ERR_EXISTING_TX,
|
|
|
|
ERR_FEES,
|
|
|
|
ERR_INVALID_COLLATERAL,
|
|
|
|
ERR_INVALID_INPUT,
|
|
|
|
ERR_INVALID_SCRIPT,
|
|
|
|
ERR_INVALID_TX,
|
|
|
|
ERR_MAXIMUM,
|
|
|
|
ERR_MN_LIST,
|
|
|
|
ERR_MODE,
|
|
|
|
ERR_NON_STANDARD_PUBKEY,
|
|
|
|
ERR_NOT_A_MN, // not used
|
|
|
|
ERR_QUEUE_FULL,
|
|
|
|
ERR_RECENT,
|
|
|
|
ERR_SESSION,
|
|
|
|
ERR_MISSING_TX,
|
|
|
|
ERR_VERSION,
|
|
|
|
MSG_NOERR,
|
|
|
|
MSG_SUCCESS,
|
|
|
|
MSG_ENTRIES_ADDED,
|
2018-05-26 20:03:23 +02:00
|
|
|
ERR_INVALID_INPUT_COUNT,
|
2017-06-30 20:30:16 +02:00
|
|
|
MSG_POOL_MIN = ERR_ALREADY_HAVE,
|
|
|
|
MSG_POOL_MAX = MSG_ENTRIES_ADDED
|
|
|
|
};
|
2015-04-03 00:51:08 +02:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
// pool states
|
|
|
|
enum PoolState {
|
|
|
|
POOL_STATE_IDLE,
|
|
|
|
POOL_STATE_QUEUE,
|
|
|
|
POOL_STATE_ACCEPTING_ENTRIES,
|
|
|
|
POOL_STATE_SIGNING,
|
|
|
|
POOL_STATE_ERROR,
|
|
|
|
POOL_STATE_SUCCESS,
|
|
|
|
POOL_STATE_MIN = POOL_STATE_IDLE,
|
|
|
|
POOL_STATE_MAX = POOL_STATE_SUCCESS
|
|
|
|
};
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
// status update message constants
|
|
|
|
enum PoolStatusUpdate {
|
|
|
|
STATUS_REJECTED,
|
|
|
|
STATUS_ACCEPTED
|
|
|
|
};
|
2016-08-05 21:49:45 +02:00
|
|
|
|
|
|
|
/** Holds an mixing input
|
2015-03-05 08:49:50 +01:00
|
|
|
*/
|
2015-03-02 00:09:33 +01:00
|
|
|
class CTxDSIn : public CTxIn
|
2014-12-09 02:17:57 +01:00
|
|
|
{
|
|
|
|
public:
|
2017-12-04 07:06:07 +01:00
|
|
|
// memory only
|
|
|
|
CScript prevPubKey;
|
2015-03-06 08:24:34 +01:00
|
|
|
bool fHasSig; // flag to indicate if signed
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-12-04 07:06:07 +01:00
|
|
|
CTxDSIn(const CTxIn& txin, const CScript& script) :
|
2016-08-05 21:49:45 +02:00
|
|
|
CTxIn(txin),
|
2017-12-04 07:06:07 +01:00
|
|
|
prevPubKey(script),
|
2018-02-21 17:32:08 +01:00
|
|
|
fHasSig(false)
|
2016-08-12 07:43:18 +02:00
|
|
|
{}
|
2016-08-29 21:14:34 +02:00
|
|
|
|
|
|
|
CTxDSIn() :
|
|
|
|
CTxIn(),
|
2017-12-04 07:06:07 +01:00
|
|
|
prevPubKey(),
|
2018-02-21 17:32:08 +01:00
|
|
|
fHasSig(false)
|
2016-08-29 21:14:34 +02:00
|
|
|
{}
|
2014-12-09 02:17:57 +01:00
|
|
|
};
|
|
|
|
|
2018-01-26 02:11:15 +01:00
|
|
|
class CDarksendAccept
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
int nDenom;
|
2018-05-26 20:03:23 +02:00
|
|
|
int nInputCount;
|
2018-01-26 02:11:15 +01:00
|
|
|
CMutableTransaction txCollateral;
|
|
|
|
|
|
|
|
CDarksendAccept() :
|
|
|
|
nDenom(0),
|
2018-05-26 20:03:23 +02:00
|
|
|
nInputCount(0),
|
2018-01-26 02:11:15 +01:00
|
|
|
txCollateral(CMutableTransaction())
|
|
|
|
{};
|
|
|
|
|
2018-05-26 20:03:23 +02:00
|
|
|
CDarksendAccept(int nDenom, int nInputCount, const CMutableTransaction& txCollateral) :
|
2018-01-26 02:11:15 +01:00
|
|
|
nDenom(nDenom),
|
2018-05-26 20:03:23 +02:00
|
|
|
nInputCount(nInputCount),
|
2018-01-26 02:11:15 +01:00
|
|
|
txCollateral(txCollateral)
|
|
|
|
{};
|
|
|
|
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action) {
|
|
|
|
READWRITE(nDenom);
|
2018-05-26 20:03:23 +02:00
|
|
|
int nVersion = s.GetVersion();
|
|
|
|
if (nVersion > 70208) {
|
|
|
|
READWRITE(nInputCount);
|
|
|
|
} else if (ser_action.ForRead()) {
|
|
|
|
nInputCount = 0;
|
|
|
|
}
|
2018-01-26 02:11:15 +01:00
|
|
|
READWRITE(txCollateral);
|
|
|
|
}
|
2018-02-01 02:10:52 +01:00
|
|
|
|
|
|
|
friend bool operator==(const CDarksendAccept& a, const CDarksendAccept& b)
|
|
|
|
{
|
|
|
|
return a.nDenom == b.nDenom && a.txCollateral == b.txCollateral;
|
|
|
|
}
|
2018-01-26 02:11:15 +01:00
|
|
|
};
|
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
// A clients transaction in the mixing pool
|
2014-12-09 02:17:57 +01:00
|
|
|
class CDarkSendEntry
|
|
|
|
{
|
|
|
|
public:
|
2016-08-05 21:49:45 +02:00
|
|
|
std::vector<CTxDSIn> vecTxDSIn;
|
2017-12-04 07:06:07 +01:00
|
|
|
std::vector<CTxOut> vecTxOut;
|
2017-09-20 14:02:53 +02:00
|
|
|
CTransactionRef txCollateral;
|
2017-07-25 12:57:26 +02:00
|
|
|
// memory only
|
|
|
|
CService addr;
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
CDarkSendEntry() :
|
2016-08-29 21:14:34 +02:00
|
|
|
vecTxDSIn(std::vector<CTxDSIn>()),
|
2017-12-04 07:06:07 +01:00
|
|
|
vecTxOut(std::vector<CTxOut>()),
|
2017-09-20 14:02:53 +02:00
|
|
|
txCollateral(MakeTransactionRef()),
|
2017-07-25 12:57:26 +02:00
|
|
|
addr(CService())
|
2016-08-12 07:43:18 +02:00
|
|
|
{}
|
2015-03-06 23:17:51 +01:00
|
|
|
|
2017-12-04 07:06:07 +01:00
|
|
|
CDarkSendEntry(const std::vector<CTxDSIn>& vecTxDSIn, const std::vector<CTxOut>& vecTxOut, const CTransaction& txCollateral) :
|
|
|
|
vecTxDSIn(vecTxDSIn),
|
|
|
|
vecTxOut(vecTxOut),
|
2017-09-20 14:02:53 +02:00
|
|
|
txCollateral(MakeTransactionRef(txCollateral)),
|
2017-12-04 07:06:07 +01:00
|
|
|
addr(CService())
|
|
|
|
{}
|
2016-08-29 21:14:34 +02:00
|
|
|
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
2017-09-19 21:36:55 +02:00
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action) {
|
2016-08-29 21:14:34 +02:00
|
|
|
READWRITE(vecTxDSIn);
|
|
|
|
READWRITE(txCollateral);
|
2017-12-04 07:06:07 +01:00
|
|
|
READWRITE(vecTxOut);
|
2016-08-29 21:14:34 +02:00
|
|
|
}
|
2015-03-06 23:17:51 +01:00
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
bool AddScriptSig(const CTxIn& txin);
|
2014-12-09 02:17:57 +01:00
|
|
|
};
|
|
|
|
|
2015-03-02 00:09:33 +01:00
|
|
|
|
2015-03-05 08:49:50 +01:00
|
|
|
/**
|
2018-02-08 06:46:44 +01:00
|
|
|
* A currently in progress mixing merge and denomination information
|
2015-03-05 08:49:50 +01:00
|
|
|
*/
|
2014-12-09 02:17:57 +01:00
|
|
|
class CDarksendQueue
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
int nDenom;
|
2018-05-26 20:03:23 +02:00
|
|
|
int nInputCount;
|
2018-02-15 08:29:44 +01:00
|
|
|
COutPoint masternodeOutpoint;
|
2016-08-05 21:49:45 +02:00
|
|
|
int64_t nTime;
|
|
|
|
bool fReady; //ready for submit
|
2014-12-09 02:17:57 +01:00
|
|
|
std::vector<unsigned char> vchSig;
|
2016-11-07 00:27:26 +01:00
|
|
|
// memory only
|
|
|
|
bool fTried;
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2016-08-12 07:43:18 +02:00
|
|
|
CDarksendQueue() :
|
|
|
|
nDenom(0),
|
2018-05-26 20:03:23 +02:00
|
|
|
nInputCount(0),
|
2018-02-15 08:29:44 +01:00
|
|
|
masternodeOutpoint(COutPoint()),
|
2016-08-12 07:43:18 +02:00
|
|
|
nTime(0),
|
|
|
|
fReady(false),
|
2016-11-07 00:27:26 +01:00
|
|
|
vchSig(std::vector<unsigned char>()),
|
|
|
|
fTried(false)
|
2016-08-12 07:43:18 +02:00
|
|
|
{}
|
2016-08-05 21:49:45 +02:00
|
|
|
|
2018-05-26 20:03:23 +02:00
|
|
|
CDarksendQueue(int nDenom, int nInputCount, COutPoint outpoint, int64_t nTime, bool fReady) :
|
2016-08-05 21:49:45 +02:00
|
|
|
nDenom(nDenom),
|
2018-05-26 20:03:23 +02:00
|
|
|
nInputCount(nInputCount),
|
2018-02-15 08:29:44 +01:00
|
|
|
masternodeOutpoint(outpoint),
|
2016-08-05 21:49:45 +02:00
|
|
|
nTime(nTime),
|
2016-08-12 07:43:18 +02:00
|
|
|
fReady(fReady),
|
2016-11-07 00:27:26 +01:00
|
|
|
vchSig(std::vector<unsigned char>()),
|
|
|
|
fTried(false)
|
2016-08-12 07:43:18 +02:00
|
|
|
{}
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2015-04-03 00:51:08 +02:00
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
2017-09-19 21:36:55 +02:00
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action) {
|
2014-12-09 02:17:57 +01:00
|
|
|
READWRITE(nDenom);
|
2018-02-15 08:29:44 +01:00
|
|
|
int nVersion = s.GetVersion();
|
2018-05-26 20:03:23 +02:00
|
|
|
if (nVersion > 70208) {
|
|
|
|
READWRITE(nInputCount);
|
|
|
|
} else if (ser_action.ForRead()) {
|
|
|
|
nInputCount = 0;
|
|
|
|
}
|
2018-02-16 15:54:53 +01:00
|
|
|
if (nVersion == 70208 && (s.GetType() & SER_NETWORK)) {
|
2018-02-15 08:29:44 +01:00
|
|
|
// converting from/to old format
|
|
|
|
CTxIn txin{};
|
|
|
|
if (ser_action.ForRead()) {
|
|
|
|
READWRITE(txin);
|
|
|
|
masternodeOutpoint = txin.prevout;
|
|
|
|
} else {
|
|
|
|
txin = CTxIn(masternodeOutpoint);
|
|
|
|
READWRITE(txin);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// using new format directly
|
|
|
|
READWRITE(masternodeOutpoint);
|
|
|
|
}
|
2016-08-05 21:49:45 +02:00
|
|
|
READWRITE(nTime);
|
|
|
|
READWRITE(fReady);
|
2018-02-16 15:54:53 +01:00
|
|
|
if (!(s.GetType() & SER_GETHASH)) {
|
|
|
|
READWRITE(vchSig);
|
|
|
|
}
|
2015-04-03 00:51:08 +02:00
|
|
|
}
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2018-02-16 15:54:53 +01:00
|
|
|
uint256 GetSignatureHash() const;
|
2016-08-05 21:49:45 +02:00
|
|
|
/** Sign this mixing transaction
|
2015-03-05 08:49:50 +01:00
|
|
|
* \return true if all conditions are met:
|
2015-03-05 09:10:15 +01:00
|
|
|
* 1) we have an active Masternode,
|
|
|
|
* 2) we have a valid Masternode private key,
|
2015-03-05 08:49:50 +01:00
|
|
|
* 3) we signed the message successfully, and
|
|
|
|
* 4) we verified the message successfully
|
|
|
|
*/
|
2014-12-09 02:17:57 +01:00
|
|
|
bool Sign();
|
2015-03-05 09:10:15 +01:00
|
|
|
/// Check if we have a valid Masternode address
|
2018-02-16 15:54:53 +01:00
|
|
|
bool CheckSignature(const CPubKey& pubKeyMasternode) const;
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-09-19 16:51:38 +02:00
|
|
|
bool Relay(CConnman &connman);
|
2016-08-05 21:49:45 +02:00
|
|
|
|
|
|
|
/// Is this queue expired?
|
2017-08-29 01:51:44 +02:00
|
|
|
bool IsExpired() { return GetAdjustedTime() - nTime > PRIVATESEND_QUEUE_TIMEOUT; }
|
2016-08-29 21:17:00 +02:00
|
|
|
|
2018-02-16 15:54:53 +01:00
|
|
|
std::string ToString() const
|
2016-08-29 21:17:00 +02:00
|
|
|
{
|
2018-05-26 20:03:23 +02:00
|
|
|
return strprintf("nDenom=%d, nInputCount=%d, nTime=%lld, fReady=%s, fTried=%s, masternode=%s",
|
|
|
|
nDenom, nInputCount, nTime, fReady ? "true" : "false", fTried ? "true" : "false", masternodeOutpoint.ToStringShort());
|
2016-11-07 00:27:26 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator==(const CDarksendQueue& a, const CDarksendQueue& b)
|
|
|
|
{
|
2018-05-26 20:03:23 +02:00
|
|
|
return a.nDenom == b.nDenom && a.nInputCount == b.nInputCount && a.masternodeOutpoint == b.masternodeOutpoint && a.nTime == b.nTime && a.fReady == b.fReady;
|
2016-08-29 21:17:00 +02:00
|
|
|
}
|
2014-12-09 02:17:57 +01:00
|
|
|
};
|
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
/** Helper class to store mixing transaction (tx) information.
|
2015-03-05 08:49:50 +01:00
|
|
|
*/
|
2014-12-09 02:17:57 +01:00
|
|
|
class CDarksendBroadcastTx
|
|
|
|
{
|
2017-07-10 16:42:09 +02:00
|
|
|
private:
|
|
|
|
// memory only
|
|
|
|
// when corresponding tx is 0-confirmed or conflicted, nConfirmedHeight is -1
|
|
|
|
int nConfirmedHeight;
|
|
|
|
|
2014-12-09 02:17:57 +01:00
|
|
|
public:
|
2017-09-20 14:02:53 +02:00
|
|
|
CTransactionRef tx;
|
2018-02-15 08:29:44 +01:00
|
|
|
COutPoint masternodeOutpoint;
|
2016-08-05 21:49:45 +02:00
|
|
|
std::vector<unsigned char> vchSig;
|
2014-12-09 02:17:57 +01:00
|
|
|
int64_t sigTime;
|
2016-08-05 21:49:45 +02:00
|
|
|
|
2016-08-12 07:43:18 +02:00
|
|
|
CDarksendBroadcastTx() :
|
2017-07-10 16:42:09 +02:00
|
|
|
nConfirmedHeight(-1),
|
2017-09-20 14:02:53 +02:00
|
|
|
tx(MakeTransactionRef()),
|
2018-02-15 08:29:44 +01:00
|
|
|
masternodeOutpoint(),
|
2017-07-10 16:42:09 +02:00
|
|
|
vchSig(),
|
2016-08-12 07:43:18 +02:00
|
|
|
sigTime(0)
|
|
|
|
{}
|
2016-08-05 21:49:45 +02:00
|
|
|
|
2017-01-04 12:22:49 +01:00
|
|
|
CDarksendBroadcastTx(const CTransactionRef& _tx, COutPoint _outpoint, int64_t _sigTime) :
|
2017-07-10 16:42:09 +02:00
|
|
|
nConfirmedHeight(-1),
|
2017-01-04 12:22:49 +01:00
|
|
|
tx(_tx),
|
2018-02-15 08:29:44 +01:00
|
|
|
masternodeOutpoint(_outpoint),
|
2017-07-10 16:42:09 +02:00
|
|
|
vchSig(),
|
2017-09-20 14:02:53 +02:00
|
|
|
sigTime(_sigTime)
|
2016-08-12 07:43:18 +02:00
|
|
|
{}
|
2016-08-05 21:49:45 +02:00
|
|
|
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
2017-09-19 21:36:55 +02:00
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action) {
|
2016-08-05 21:49:45 +02:00
|
|
|
READWRITE(tx);
|
2018-02-15 08:29:44 +01:00
|
|
|
int nVersion = s.GetVersion();
|
2018-02-16 15:54:53 +01:00
|
|
|
if (nVersion == 70208 && (s.GetType() & SER_NETWORK)) {
|
2018-02-15 08:29:44 +01:00
|
|
|
// converting from/to old format
|
|
|
|
CTxIn txin{};
|
|
|
|
if (ser_action.ForRead()) {
|
|
|
|
READWRITE(txin);
|
|
|
|
masternodeOutpoint = txin.prevout;
|
|
|
|
} else {
|
|
|
|
txin = CTxIn(masternodeOutpoint);
|
|
|
|
READWRITE(txin);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// using new format directly
|
|
|
|
READWRITE(masternodeOutpoint);
|
|
|
|
}
|
2018-02-16 15:54:53 +01:00
|
|
|
if (!(s.GetType() & SER_GETHASH)) {
|
|
|
|
READWRITE(vchSig);
|
|
|
|
}
|
2016-08-05 21:49:45 +02:00
|
|
|
READWRITE(sigTime);
|
|
|
|
}
|
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
friend bool operator==(const CDarksendBroadcastTx& a, const CDarksendBroadcastTx& b)
|
|
|
|
{
|
2017-09-20 14:02:53 +02:00
|
|
|
return *a.tx == *b.tx;
|
2017-06-30 20:30:16 +02:00
|
|
|
}
|
|
|
|
friend bool operator!=(const CDarksendBroadcastTx& a, const CDarksendBroadcastTx& b)
|
|
|
|
{
|
|
|
|
return !(a == b);
|
|
|
|
}
|
|
|
|
explicit operator bool() const
|
|
|
|
{
|
|
|
|
return *this != CDarksendBroadcastTx();
|
|
|
|
}
|
|
|
|
|
2018-02-16 15:54:53 +01:00
|
|
|
uint256 GetSignatureHash() const;
|
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
bool Sign();
|
2018-02-16 15:54:53 +01:00
|
|
|
bool CheckSignature(const CPubKey& pubKeyMasternode) const;
|
2017-07-10 16:42:09 +02:00
|
|
|
|
|
|
|
void SetConfirmedHeight(int nConfirmedHeightIn) { nConfirmedHeight = nConfirmedHeightIn; }
|
|
|
|
bool IsExpired(int nHeight);
|
2014-12-09 02:17:57 +01:00
|
|
|
};
|
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
// base class
|
|
|
|
class CPrivateSendBase
|
2014-12-09 02:17:57 +01:00
|
|
|
{
|
2017-05-05 13:26:27 +02:00
|
|
|
protected:
|
2017-12-04 07:06:07 +01:00
|
|
|
mutable CCriticalSection cs_darksend;
|
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
// The current mixing sessions in progress on the network
|
|
|
|
std::vector<CDarksendQueue> vecDarksendQueue;
|
2017-05-05 13:26:27 +02:00
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
std::vector<CDarkSendEntry> vecEntries; // Masternode/clients entries
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2016-10-20 23:11:57 +02:00
|
|
|
PoolState nState; // should be one of the POOL_STATE_XXX values
|
2018-02-12 13:47:53 +01:00
|
|
|
int64_t nTimeLastSuccessfulStep; // the time when last successful mixing step was performed
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2016-10-27 23:06:33 +02:00
|
|
|
int nSessionID; // 0 if no mixing session is active
|
2015-12-02 23:12:16 +01:00
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
CMutableTransaction finalMutableTransaction; // the finalized transaction ready for signing
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
void SetNull();
|
2017-12-04 07:06:07 +01:00
|
|
|
void CheckQueue();
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
public:
|
|
|
|
int nSessionDenom; //Users must submit an denom matching this
|
2018-05-26 20:03:23 +02:00
|
|
|
int nSessionInputCount; //Users must submit a count matching this
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
CPrivateSendBase() { SetNull(); }
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
int GetQueueSize() const { return vecDarksendQueue.size(); }
|
|
|
|
int GetState() const { return nState; }
|
|
|
|
std::string GetStateString() const;
|
2016-08-05 21:49:45 +02:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
int GetEntriesCount() const { return vecEntries.size(); }
|
|
|
|
};
|
2016-10-27 23:06:33 +02:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
// helper class
|
|
|
|
class CPrivateSend
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
// make constructor, destructor and copying not available
|
|
|
|
CPrivateSend() {}
|
|
|
|
~CPrivateSend() {}
|
|
|
|
CPrivateSend(CPrivateSend const&) = delete;
|
|
|
|
CPrivateSend& operator= (CPrivateSend const&) = delete;
|
|
|
|
|
|
|
|
static const CAmount COLLATERAL = 0.001 * COIN;
|
2017-05-05 13:26:27 +02:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
// static members
|
|
|
|
static std::vector<CAmount> vecStandardDenominations;
|
|
|
|
static std::map<uint256, CDarksendBroadcastTx> mapDSTX;
|
2016-07-15 12:21:20 +02:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
static CCriticalSection cs_mapdstx;
|
|
|
|
|
2017-12-07 10:42:47 +01:00
|
|
|
static void CheckDSTXes(int nHeight);
|
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
public:
|
|
|
|
static void InitStandardDenominations();
|
|
|
|
static std::vector<CAmount> GetStandardDenominations() { return vecStandardDenominations; }
|
|
|
|
static CAmount GetSmallestDenomination() { return vecStandardDenominations.back(); }
|
|
|
|
|
|
|
|
/// Get the denominations for a specific amount of dash.
|
|
|
|
static int GetDenominationsByAmounts(const std::vector<CAmount>& vecAmount);
|
2015-03-18 18:19:13 +01:00
|
|
|
|
2017-12-04 07:06:07 +01:00
|
|
|
static bool IsDenominatedAmount(CAmount nInputAmount);
|
|
|
|
|
2015-03-05 08:49:50 +01:00
|
|
|
/// Get the denominations for a list of outputs (returns a bitshifted integer)
|
2017-06-30 20:30:16 +02:00
|
|
|
static int GetDenominations(const std::vector<CTxOut>& vecTxOut, bool fSingleRandomDenom = false);
|
|
|
|
static std::string GetDenominationsToString(int nDenom);
|
|
|
|
static bool GetDenominationsBits(int nDenom, std::vector<int> &vecBitsRet);
|
2015-03-06 23:17:51 +01:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
static std::string GetMessageByID(PoolMessage nMessageID);
|
2017-04-20 22:34:47 +02:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
/// Get the maximum number of transactions for the pool
|
|
|
|
static int GetMaxPoolTransactions() { return Params().PoolMaxTransactions(); }
|
2015-03-02 00:09:33 +01:00
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
static CAmount GetMaxPoolAmount() { return vecStandardDenominations.empty() ? 0 : PRIVATESEND_ENTRY_MAX_SIZE * vecStandardDenominations.front(); }
|
|
|
|
|
|
|
|
/// If the collateral is valid given by a client
|
|
|
|
static bool IsCollateralValid(const CTransaction& txCollateral);
|
|
|
|
static CAmount GetCollateralAmount() { return COLLATERAL; }
|
|
|
|
static CAmount GetMaxCollateralAmount() { return COLLATERAL*4; }
|
|
|
|
|
2017-12-04 07:06:07 +01:00
|
|
|
static bool IsCollateralAmount(CAmount nInputAmount);
|
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
static void AddDSTX(const CDarksendBroadcastTx& dstx);
|
|
|
|
static CDarksendBroadcastTx GetDSTX(const uint256& hash);
|
2017-07-10 16:42:09 +02:00
|
|
|
|
2017-12-07 10:42:47 +01:00
|
|
|
static void UpdatedBlockTip(const CBlockIndex *pindex);
|
2018-01-08 18:41:06 +01:00
|
|
|
static void SyncTransaction(const CTransaction& tx, const CBlockIndex *pindex, int posInBlock);
|
2015-03-02 00:09:33 +01:00
|
|
|
};
|
2014-12-09 02:17:57 +01:00
|
|
|
|
|
|
|
#endif
|