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-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
|
|
|
|
2017-04-20 22:34:47 +02:00
|
|
|
static const CAmount 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,
|
|
|
|
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:
|
2015-03-06 08:24:34 +01:00
|
|
|
bool fHasSig; // flag to indicate if signed
|
2015-03-18 18:19:13 +01:00
|
|
|
int nSentTimes; //times we've sent this anonymously
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
CTxDSIn(const CTxIn& txin) :
|
|
|
|
CTxIn(txin),
|
|
|
|
fHasSig(false),
|
2016-08-12 07:43:18 +02:00
|
|
|
nSentTimes(0)
|
|
|
|
{}
|
2016-08-29 21:14:34 +02:00
|
|
|
|
|
|
|
CTxDSIn() :
|
|
|
|
CTxIn(),
|
|
|
|
fHasSig(false),
|
|
|
|
nSentTimes(0)
|
|
|
|
{}
|
2014-12-09 02:17:57 +01:00
|
|
|
};
|
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
/** Holds an mixing output
|
2015-03-06 23:17:51 +01:00
|
|
|
*/
|
|
|
|
class CTxDSOut : public CTxOut
|
|
|
|
{
|
|
|
|
public:
|
2015-03-18 18:19:13 +01:00
|
|
|
int nSentTimes; //times we've sent this anonymously
|
2015-03-06 23:17:51 +01:00
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
CTxDSOut(const CTxOut& out) :
|
|
|
|
CTxOut(out),
|
2016-08-12 07:43:18 +02:00
|
|
|
nSentTimes(0)
|
|
|
|
{}
|
2016-08-29 21:14:34 +02:00
|
|
|
|
|
|
|
CTxDSOut() :
|
|
|
|
CTxOut(),
|
|
|
|
nSentTimes(0)
|
|
|
|
{}
|
2015-03-06 23:17:51 +01:00
|
|
|
};
|
2015-03-02 00:09:33 +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;
|
|
|
|
std::vector<CTxDSOut> vecTxDSOut;
|
|
|
|
CTransaction 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>()),
|
|
|
|
vecTxDSOut(std::vector<CTxDSOut>()),
|
2017-07-25 12:57:26 +02:00
|
|
|
txCollateral(CTransaction()),
|
|
|
|
addr(CService())
|
2016-08-12 07:43:18 +02:00
|
|
|
{}
|
2015-03-06 23:17:51 +01:00
|
|
|
|
2017-05-05 13:26:27 +02:00
|
|
|
CDarkSendEntry(const std::vector<CTxIn>& vecTxIn, const std::vector<CTxOut>& vecTxOut, const CTransaction& txCollateral);
|
2016-08-29 21:14:34 +02:00
|
|
|
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
|
|
|
|
READWRITE(vecTxDSIn);
|
|
|
|
READWRITE(txCollateral);
|
|
|
|
READWRITE(vecTxDSOut);
|
|
|
|
}
|
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
|
|
|
/**
|
2016-08-05 21:49:45 +02:00
|
|
|
* A currently inprogress 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;
|
2016-08-05 21:49:45 +02:00
|
|
|
CTxIn vin;
|
|
|
|
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),
|
|
|
|
vin(CTxIn()),
|
|
|
|
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
|
|
|
|
2017-09-11 16:13:48 +02:00
|
|
|
CDarksendQueue(int nDenom, COutPoint outpoint, int64_t nTime, bool fReady) :
|
2016-08-05 21:49:45 +02:00
|
|
|
nDenom(nDenom),
|
2017-09-11 16:13:48 +02:00
|
|
|
vin(CTxIn(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>
|
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
|
2014-12-09 02:17:57 +01:00
|
|
|
READWRITE(nDenom);
|
|
|
|
READWRITE(vin);
|
2016-08-05 21:49:45 +02:00
|
|
|
READWRITE(nTime);
|
|
|
|
READWRITE(fReady);
|
2014-12-09 02:17:57 +01:00
|
|
|
READWRITE(vchSig);
|
2015-04-03 00:51:08 +02:00
|
|
|
}
|
2014-12-09 02:17:57 +01:00
|
|
|
|
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
|
2016-11-02 16:54:36 +01:00
|
|
|
bool CheckSignature(const CPubKey& pubKeyMasternode);
|
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
|
|
|
|
|
|
|
std::string ToString()
|
|
|
|
{
|
2016-11-07 00:27:26 +01:00
|
|
|
return strprintf("nDenom=%d, nTime=%lld, fReady=%s, fTried=%s, masternode=%s",
|
|
|
|
nDenom, nTime, fReady ? "true" : "false", fTried ? "true" : "false", vin.prevout.ToStringShort());
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator==(const CDarksendQueue& a, const CDarksendQueue& b)
|
|
|
|
{
|
|
|
|
return a.nDenom == b.nDenom && a.vin.prevout == b.vin.prevout && 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:
|
|
|
|
CTransaction tx;
|
|
|
|
CTxIn vin;
|
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),
|
|
|
|
tx(),
|
|
|
|
vin(),
|
|
|
|
vchSig(),
|
2016-08-12 07:43:18 +02:00
|
|
|
sigTime(0)
|
|
|
|
{}
|
2016-08-05 21:49:45 +02:00
|
|
|
|
2017-09-11 16:13:48 +02:00
|
|
|
CDarksendBroadcastTx(CTransaction tx, COutPoint outpoint, int64_t sigTime) :
|
2017-07-10 16:42:09 +02:00
|
|
|
nConfirmedHeight(-1),
|
2016-08-12 07:43:18 +02:00
|
|
|
tx(tx),
|
2017-09-11 16:13:48 +02:00
|
|
|
vin(CTxIn(outpoint)),
|
2017-07-10 16:42:09 +02:00
|
|
|
vchSig(),
|
2016-08-12 07:43:18 +02:00
|
|
|
sigTime(sigTime)
|
|
|
|
{}
|
2016-08-05 21:49:45 +02:00
|
|
|
|
|
|
|
ADD_SERIALIZE_METHODS;
|
|
|
|
|
|
|
|
template <typename Stream, typename Operation>
|
|
|
|
inline void SerializationOp(Stream& s, Operation ser_action, int nType, int nVersion) {
|
|
|
|
READWRITE(tx);
|
|
|
|
READWRITE(vin);
|
|
|
|
READWRITE(vchSig);
|
|
|
|
READWRITE(sigTime);
|
|
|
|
}
|
|
|
|
|
2017-06-30 20:30:16 +02:00
|
|
|
friend bool operator==(const CDarksendBroadcastTx& a, const CDarksendBroadcastTx& b)
|
|
|
|
{
|
|
|
|
return a.tx == b.tx;
|
|
|
|
}
|
|
|
|
friend bool operator!=(const CDarksendBroadcastTx& a, const CDarksendBroadcastTx& b)
|
|
|
|
{
|
|
|
|
return !(a == b);
|
|
|
|
}
|
|
|
|
explicit operator bool() const
|
|
|
|
{
|
|
|
|
return *this != CDarksendBroadcastTx();
|
|
|
|
}
|
|
|
|
|
2016-08-05 21:49:45 +02:00
|
|
|
bool Sign();
|
2016-11-02 16:54:36 +01:00
|
|
|
bool CheckSignature(const CPubKey& pubKeyMasternode);
|
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:
|
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
|
2016-11-09 23:49:57 +01:00
|
|
|
int64_t nTimeLastSuccessfulStep; // the time when last successful mixing step was performed, in UTC milliseconds
|
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();
|
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
|
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;
|
|
|
|
|
|
|
|
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
|
|
|
|
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 int GetDenominations(const std::vector<CTxDSOut>& vecTxDSOut);
|
|
|
|
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; }
|
|
|
|
|
|
|
|
static void AddDSTX(const CDarksendBroadcastTx& dstx);
|
|
|
|
static CDarksendBroadcastTx GetDSTX(const uint256& hash);
|
2017-07-10 16:42:09 +02:00
|
|
|
static void CheckDSTXes(int nHeight);
|
|
|
|
|
|
|
|
static void SyncTransaction(const CTransaction& tx, const CBlock* pblock);
|
2015-03-02 00:09:33 +01:00
|
|
|
};
|
2014-12-09 02:17:57 +01:00
|
|
|
|
2017-09-19 16:51:38 +02:00
|
|
|
void ThreadCheckPrivateSend(CConnman& connman);
|
2014-12-09 02:17:57 +01:00
|
|
|
|
|
|
|
#endif
|