2011-08-11 18:14:53 +02:00
|
|
|
// Copyright (c) 2009-2010 Satoshi Nakamoto
|
2015-12-13 17:58:29 +01:00
|
|
|
// Copyright (c) 2009-2015 The Bitcoin Core developers
|
2014-12-13 12:09:33 +08:00
|
|
|
// Distributed under the MIT software license, see the accompanying
|
2012-05-18 22:02:28 +08:00
|
|
|
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
|
2011-08-11 18:14:53 +02:00
|
|
|
|
|
|
|
#include "protocol.h"
|
2013-04-13 00:13:08 -05:00
|
|
|
|
2011-08-11 18:40:12 +02:00
|
|
|
#include "util.h"
|
2014-09-24 23:32:36 -04:00
|
|
|
#include "utilstrencodings.h"
|
2013-04-13 00:13:08 -05:00
|
|
|
|
2011-10-07 11:02:21 -04:00
|
|
|
#ifndef WIN32
|
2011-08-11 18:40:12 +02:00
|
|
|
# include <arpa/inet.h>
|
|
|
|
#endif
|
|
|
|
|
2015-12-07 15:31:32 +01:00
|
|
|
namespace NetMsgType {
|
|
|
|
const char *VERSION="version";
|
|
|
|
const char *VERACK="verack";
|
|
|
|
const char *ADDR="addr";
|
|
|
|
const char *INV="inv";
|
|
|
|
const char *GETDATA="getdata";
|
|
|
|
const char *MERKLEBLOCK="merkleblock";
|
|
|
|
const char *GETBLOCKS="getblocks";
|
|
|
|
const char *GETHEADERS="getheaders";
|
|
|
|
const char *TX="tx";
|
|
|
|
const char *HEADERS="headers";
|
|
|
|
const char *BLOCK="block";
|
|
|
|
const char *GETADDR="getaddr";
|
|
|
|
const char *MEMPOOL="mempool";
|
|
|
|
const char *PING="ping";
|
|
|
|
const char *PONG="pong";
|
|
|
|
const char *NOTFOUND="notfound";
|
|
|
|
const char *FILTERLOAD="filterload";
|
|
|
|
const char *FILTERADD="filteradd";
|
|
|
|
const char *FILTERCLEAR="filterclear";
|
|
|
|
const char *REJECT="reject";
|
|
|
|
const char *SENDHEADERS="sendheaders";
|
2016-02-12 15:57:15 -05:00
|
|
|
const char *FEEFILTER="feefilter";
|
2016-06-06 01:26:52 -07:00
|
|
|
const char *SENDCMPCT="sendcmpct";
|
|
|
|
const char *CMPCTBLOCK="cmpctblock";
|
|
|
|
const char *GETBLOCKTXN="getblocktxn";
|
|
|
|
const char *BLOCKTXN="blocktxn";
|
2015-12-07 15:31:32 +01:00
|
|
|
};
|
|
|
|
|
|
|
|
/** All known message types. Keep this in the same order as the list of
|
|
|
|
* messages above and in protocol.h.
|
|
|
|
*/
|
|
|
|
const static std::string allNetMessageTypes[] = {
|
|
|
|
NetMsgType::VERSION,
|
|
|
|
NetMsgType::VERACK,
|
|
|
|
NetMsgType::ADDR,
|
|
|
|
NetMsgType::INV,
|
|
|
|
NetMsgType::GETDATA,
|
|
|
|
NetMsgType::MERKLEBLOCK,
|
|
|
|
NetMsgType::GETBLOCKS,
|
|
|
|
NetMsgType::GETHEADERS,
|
|
|
|
NetMsgType::TX,
|
|
|
|
NetMsgType::HEADERS,
|
|
|
|
NetMsgType::BLOCK,
|
|
|
|
NetMsgType::GETADDR,
|
|
|
|
NetMsgType::MEMPOOL,
|
|
|
|
NetMsgType::PING,
|
|
|
|
NetMsgType::PONG,
|
|
|
|
NetMsgType::NOTFOUND,
|
|
|
|
NetMsgType::FILTERLOAD,
|
|
|
|
NetMsgType::FILTERADD,
|
|
|
|
NetMsgType::FILTERCLEAR,
|
|
|
|
NetMsgType::REJECT,
|
2016-02-12 15:57:15 -05:00
|
|
|
NetMsgType::SENDHEADERS,
|
2016-06-06 01:26:52 -07:00
|
|
|
NetMsgType::FEEFILTER,
|
|
|
|
NetMsgType::SENDCMPCT,
|
|
|
|
NetMsgType::CMPCTBLOCK,
|
|
|
|
NetMsgType::GETBLOCKTXN,
|
|
|
|
NetMsgType::BLOCKTXN,
|
2011-08-11 18:49:03 +02:00
|
|
|
};
|
2015-12-07 15:31:32 +01:00
|
|
|
const static std::vector<std::string> allNetMessageTypesVec(allNetMessageTypes, allNetMessageTypes+ARRAYLEN(allNetMessageTypes));
|
2011-08-11 18:14:53 +02:00
|
|
|
|
2014-10-27 20:24:31 -04:00
|
|
|
CMessageHeader::CMessageHeader(const MessageStartChars& pchMessageStartIn)
|
2011-08-11 18:14:53 +02:00
|
|
|
{
|
2014-10-27 20:24:31 -04:00
|
|
|
memcpy(pchMessageStart, pchMessageStartIn, MESSAGE_START_SIZE);
|
2011-08-11 18:14:53 +02:00
|
|
|
memset(pchCommand, 0, sizeof(pchCommand));
|
|
|
|
nMessageSize = -1;
|
2016-09-27 14:05:24 +02:00
|
|
|
memset(pchChecksum, 0, CHECKSUM_SIZE);
|
2011-08-11 18:14:53 +02:00
|
|
|
}
|
|
|
|
|
2014-10-27 20:24:31 -04:00
|
|
|
CMessageHeader::CMessageHeader(const MessageStartChars& pchMessageStartIn, const char* pszCommand, unsigned int nMessageSizeIn)
|
2011-08-11 18:14:53 +02:00
|
|
|
{
|
2014-10-27 20:24:31 -04:00
|
|
|
memcpy(pchMessageStart, pchMessageStartIn, MESSAGE_START_SIZE);
|
2014-09-16 18:13:05 +02:00
|
|
|
memset(pchCommand, 0, sizeof(pchCommand));
|
2011-08-11 18:14:53 +02:00
|
|
|
strncpy(pchCommand, pszCommand, COMMAND_SIZE);
|
|
|
|
nMessageSize = nMessageSizeIn;
|
2016-09-27 14:05:24 +02:00
|
|
|
memset(pchChecksum, 0, CHECKSUM_SIZE);
|
2011-08-11 18:14:53 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string CMessageHeader::GetCommand() const
|
|
|
|
{
|
2014-09-16 18:13:05 +02:00
|
|
|
return std::string(pchCommand, pchCommand + strnlen(pchCommand, COMMAND_SIZE));
|
2011-08-11 18:14:53 +02:00
|
|
|
}
|
|
|
|
|
2014-10-27 20:24:31 -04:00
|
|
|
bool CMessageHeader::IsValid(const MessageStartChars& pchMessageStartIn) const
|
2011-08-11 18:14:53 +02:00
|
|
|
{
|
|
|
|
// Check start string
|
2014-10-27 20:24:31 -04:00
|
|
|
if (memcmp(pchMessageStart, pchMessageStartIn, MESSAGE_START_SIZE) != 0)
|
2011-08-11 18:14:53 +02:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// Check the command string for errors
|
|
|
|
for (const char* p1 = pchCommand; p1 < pchCommand + COMMAND_SIZE; p1++)
|
|
|
|
{
|
|
|
|
if (*p1 == 0)
|
|
|
|
{
|
|
|
|
// Must be all zeros after the first zero
|
|
|
|
for (; p1 < pchCommand + COMMAND_SIZE; p1++)
|
|
|
|
if (*p1 != 0)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
else if (*p1 < ' ' || *p1 > 0x7E)
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Message size
|
|
|
|
if (nMessageSize > MAX_SIZE)
|
|
|
|
{
|
2015-01-08 11:44:25 +01:00
|
|
|
LogPrintf("CMessageHeader::IsValid(): (%s, %u bytes) nMessageSize > MAX_SIZE\n", GetCommand(), nMessageSize);
|
2011-08-11 18:14:53 +02:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
2011-08-11 18:40:12 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
2012-01-03 23:33:31 +01:00
|
|
|
CAddress::CAddress() : CService()
|
2011-08-11 18:40:12 +02:00
|
|
|
{
|
|
|
|
Init();
|
|
|
|
}
|
|
|
|
|
2016-06-08 19:12:22 +02:00
|
|
|
CAddress::CAddress(CService ipIn, ServiceFlags nServicesIn) : CService(ipIn)
|
2011-08-11 18:40:12 +02:00
|
|
|
{
|
|
|
|
Init();
|
2012-01-03 23:33:31 +01:00
|
|
|
nServices = nServicesIn;
|
2011-08-11 18:40:12 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
void CAddress::Init()
|
|
|
|
{
|
2016-06-08 19:12:22 +02:00
|
|
|
nServices = NODE_NONE;
|
2011-08-11 18:40:12 +02:00
|
|
|
nTime = 100000000;
|
|
|
|
}
|
|
|
|
|
2011-08-11 18:49:03 +02:00
|
|
|
CInv::CInv()
|
|
|
|
{
|
|
|
|
type = 0;
|
2014-12-15 09:11:16 +01:00
|
|
|
hash.SetNull();
|
2011-08-11 18:49:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
CInv::CInv(int typeIn, const uint256& hashIn)
|
|
|
|
{
|
|
|
|
type = typeIn;
|
|
|
|
hash = hashIn;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator<(const CInv& a, const CInv& b)
|
|
|
|
{
|
|
|
|
return (a.type < b.type || (a.type == b.type && a.hash < b.hash));
|
|
|
|
}
|
|
|
|
|
2015-11-06 01:32:04 +01:00
|
|
|
std::string CInv::GetCommand() const
|
2011-08-11 18:49:03 +02:00
|
|
|
{
|
2015-11-06 01:32:04 +01:00
|
|
|
std::string cmd;
|
|
|
|
if (type & MSG_WITNESS_FLAG)
|
|
|
|
cmd.append("witness-");
|
|
|
|
int masked = type & MSG_TYPE_MASK;
|
|
|
|
switch (masked)
|
|
|
|
{
|
|
|
|
case MSG_TX: return cmd.append(NetMsgType::TX);
|
|
|
|
case MSG_BLOCK: return cmd.append(NetMsgType::BLOCK);
|
|
|
|
case MSG_FILTERED_BLOCK: return cmd.append(NetMsgType::MERKLEBLOCK);
|
|
|
|
case MSG_CMPCT_BLOCK: return cmd.append(NetMsgType::CMPCTBLOCK);
|
|
|
|
default:
|
2015-01-08 11:44:25 +01:00
|
|
|
throw std::out_of_range(strprintf("CInv::GetCommand(): type=%d unknown type", type));
|
2015-11-06 01:32:04 +01:00
|
|
|
}
|
2011-08-11 18:49:03 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
std::string CInv::ToString() const
|
|
|
|
{
|
2016-11-09 11:09:16 +01:00
|
|
|
try {
|
|
|
|
return strprintf("%s %s", GetCommand(), hash.ToString());
|
|
|
|
} catch(const std::out_of_range &) {
|
|
|
|
return strprintf("0x%08x %s", type, hash.ToString());
|
|
|
|
}
|
2011-08-11 18:49:03 +02:00
|
|
|
}
|
2015-12-07 15:31:32 +01:00
|
|
|
|
|
|
|
const std::vector<std::string> &getAllNetMessageTypes()
|
|
|
|
{
|
|
|
|
return allNetMessageTypesVec;
|
|
|
|
}
|