2010-08-29 12:58:15 -04:00
// Copyright (c) 2009-2010 Satoshi Nakamoto
2019-01-02 09:13:50 -03:00
// Copyright (c) 2009-2019 The Bitcoin Core developers
2014-12-13 01:09:33 -03:00
// Distributed under the MIT software license, see the accompanying
2012-05-18 10:02:28 -04:00
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
2010-08-29 12:58:15 -04:00
2013-05-27 19:55:01 -04:00
# if defined(HAVE_CONFIG_H)
2017-11-09 21:57:53 -03:00
# include <config/bitcoin-config.h>
2013-05-27 19:55:01 -04:00
# endif
2017-11-09 21:57:53 -03:00
# include <net.h>
# include <chainparams.h>
# include <clientversion.h>
# include <consensus/consensus.h>
# include <crypto/common.h>
# include <crypto/sha256.h>
# include <primitives/transaction.h>
# include <netbase.h>
# include <scheduler.h>
# include <ui_interface.h>
2018-10-22 19:51:11 -03:00
# include <util/strencodings.h>
2013-04-13 02:13:08 -03:00
2011-10-07 12:02:21 -03:00
# ifdef WIN32
2011-07-01 21:59:37 -04:00
# include <string.h>
2013-04-13 02:13:08 -03:00
# else
2013-07-17 04:51:40 -04:00
# include <fcntl.h>
# endif
2018-09-26 22:54:52 -03:00
# ifdef USE_POLL
# include <poll.h>
# endif
2011-03-26 09:01:27 -03:00
# ifdef USE_UPNP
# include <miniupnpc/miniupnpc.h>
2013-04-13 02:13:08 -03:00
# include <miniupnpc/miniwget.h>
2011-03-26 09:01:27 -03:00
# include <miniupnpc/upnpcommands.h>
# include <miniupnpc/upnperrors.h>
# endif
2018-09-26 22:54:52 -03:00
# include <unordered_map>
2014-01-30 06:55:55 -03:00
2015-04-08 15:20:00 -03:00
# include <math.h>
2017-10-05 13:46:54 -03:00
// Dump addresses to peers.dat every 15 minutes (900s)
static constexpr int DUMP_PEERS_INTERVAL = 15 * 60 ;
2016-06-17 00:10:07 -04:00
// We add a random period time (0 to 1 seconds) to feeler connections to prevent synchronization.
# define FEELER_SLEEP_WINDOW 1
2018-03-11 16:19:26 -03:00
// MSG_NOSIGNAL is not available on some platforms, if it doesn't exist define it as 0
# if !defined(MSG_NOSIGNAL)
2013-05-27 19:55:01 -04:00
# define MSG_NOSIGNAL 0
# endif
2013-06-23 18:23:28 -04:00
2017-03-05 06:32:00 -03:00
// MSG_DONTWAIT is not available on some platforms, if it doesn't exist define it as 0
2018-03-11 16:19:26 -03:00
# if !defined(MSG_DONTWAIT)
2017-03-05 06:32:00 -03:00
# define MSG_DONTWAIT 0
# endif
2014-06-24 03:03:18 -04:00
// Fix for ancient MinGW versions, that don't have defined these in ws2tcpip.h.
// Todo: Can be removed when our pull-tester is upgraded to a modern MinGW version.
# ifdef WIN32
# ifndef PROTECTION_LEVEL_UNRESTRICTED
# define PROTECTION_LEVEL_UNRESTRICTED 10
# endif
# ifndef IPV6_PROTECTION_LEVEL
# define IPV6_PROTECTION_LEVEL 23
# endif
# endif
2017-06-01 06:34:02 -04:00
/** Used to pass flags to the Bind() function */
enum BindFlags {
BF_NONE = 0 ,
BF_EXPLICIT = ( 1U < < 0 ) ,
BF_REPORT_ERROR = ( 1U < < 1 ) ,
BF_WHITELIST = ( 1U < < 2 ) ,
} ;
2018-10-29 17:30:30 -03:00
// The set of sockets cannot be modified while waiting
// The sleep time needs to be small to avoid new sockets stalling
static const uint64_t SELECT_TIMEOUT_MILLISECONDS = 50 ;
2019-01-02 09:13:50 -03:00
const std : : string NET_MESSAGE_COMMAND_OTHER = " *other* " ;
2015-08-25 11:30:31 -03:00
2016-09-09 07:48:10 -03:00
static const uint64_t RANDOMIZER_ID_NETGROUP = 0x6c0edd8036ef4036ULL ; // SHA256("netgroup")[0:8]
2016-10-26 16:10:15 -03:00
static const uint64_t RANDOMIZER_ID_LOCALHOSTNONCE = 0xd93e69e2bbfa5735ULL ; // SHA256("localhostnonce")[0:8]
2010-08-29 12:58:15 -04:00
//
// Global state variables
//
2012-05-24 13:02:21 -04:00
bool fDiscover = true ;
2014-05-29 06:33:17 -04:00
bool fListen = true ;
2016-05-12 09:00:22 -03:00
bool fRelayTxes = true ;
2014-05-05 07:22:28 -04:00
CCriticalSection cs_mapLocalHost ;
2018-10-08 10:34:39 -03:00
std : : map < CNetAddr , LocalServiceInfo > mapLocalHost GUARDED_BY ( cs_mapLocalHost ) ;
static bool vfLimited [ NET_MAX ] GUARDED_BY ( cs_mapLocalHost ) = { } ;
2015-07-31 13:05:42 -03:00
std : : string strSubVersion ;
2010-08-29 12:58:15 -04:00
2016-04-11 13:52:29 -03:00
limitedmap < uint256 , int64_t > mapAlreadyAskedFor ( MAX_INV_SZ ) ;
2010-08-29 12:58:15 -04:00
2016-04-16 18:51:01 -03:00
void CConnman : : AddOneShot ( const std : : string & strDest )
2012-04-23 21:15:00 -03:00
{
LOCK ( cs_vOneShots ) ;
vOneShots . push_back ( strDest ) ;
}
2011-04-21 11:45:08 -03:00
unsigned short GetListenPort ( )
{
2017-08-01 15:17:40 -04:00
return ( unsigned short ) ( gArgs . GetArg ( " -port " , Params ( ) . GetDefaultPort ( ) ) ) ;
2011-04-21 11:45:08 -03:00
}
2010-08-29 12:58:15 -04:00
2012-02-12 09:45:24 -03:00
// find 'best' local address for a particular peer
2012-05-10 14:35:13 -04:00
bool GetLocal ( CService & addr , const CNetAddr * paddrPeer )
2012-02-12 09:45:24 -03:00
{
2014-05-29 06:33:17 -04:00
if ( ! fListen )
2012-02-12 09:45:24 -03:00
return false ;
2010-08-29 12:58:15 -04:00
2012-05-12 18:41:24 -04:00
int nBestScore = - 1 ;
2012-02-12 09:45:24 -03:00
int nBestReachability = - 1 ;
{
LOCK ( cs_mapLocalHost ) ;
2017-06-04 16:02:43 -04:00
for ( const auto & entry : mapLocalHost )
2012-02-12 09:45:24 -03:00
{
2017-06-04 16:02:43 -04:00
int nScore = entry . second . nScore ;
int nReachability = entry . first . GetReachabilityFrom ( paddrPeer ) ;
2012-05-12 18:41:24 -04:00
if ( nReachability > nBestReachability | | ( nReachability = = nBestReachability & & nScore > nBestScore ) )
2012-02-12 09:45:24 -03:00
{
2017-06-04 16:02:43 -04:00
addr = CService ( entry . first , entry . second . nPort ) ;
2012-02-12 09:45:24 -03:00
nBestReachability = nReachability ;
2012-05-12 18:41:24 -04:00
nBestScore = nScore ;
2012-02-12 09:45:24 -03:00
}
}
}
2012-05-12 18:41:24 -04:00
return nBestScore > = 0 ;
2012-02-12 09:45:24 -03:00
}
2010-08-29 12:58:15 -04:00
2018-04-14 20:19:40 -03:00
//! Convert the pnSeed6 array into usable address objects.
2015-01-24 01:40:50 -03:00
static std : : vector < CAddress > convertSeed6 ( const std : : vector < SeedSpec6 > & vSeedsIn )
{
// It'll only connect to one or two seed nodes because once it connects,
// it'll get a pile of addresses with newer timestamps.
// Seed nodes are given a random 'last seen time' of between one and two
// weeks ago.
const int64_t nOneWeek = 7 * 24 * 60 * 60 ;
std : : vector < CAddress > vSeedsOut ;
vSeedsOut . reserve ( vSeedsIn . size ( ) ) ;
2018-10-31 19:02:24 -03:00
FastRandomContext rng ;
2017-07-20 05:32:47 -04:00
for ( const auto & seed_in : vSeedsIn ) {
2015-01-24 01:40:50 -03:00
struct in6_addr ip ;
2017-07-20 05:32:47 -04:00
memcpy ( & ip , seed_in . addr , sizeof ( ip ) ) ;
2017-10-19 18:32:33 -03:00
CAddress addr ( CService ( ip , seed_in . port ) , GetDesirableServiceFlags ( NODE_NONE ) ) ;
2018-10-31 19:02:24 -03:00
addr . nTime = GetTime ( ) - rng . randrange ( nOneWeek ) - nOneWeek ;
2015-01-24 01:40:50 -03:00
vSeedsOut . push_back ( addr ) ;
}
return vSeedsOut ;
}
2012-02-12 09:45:24 -03:00
// get best local address for a particular peer as a CAddress
2014-07-21 02:32:25 -04:00
// Otherwise, return the unroutable 0.0.0.0 but filled in with
// the normal parameters, since the IP may be changed to a useful
// one by discovery.
2016-04-19 01:04:58 -03:00
CAddress GetLocalAddress ( const CNetAddr * paddrPeer , ServiceFlags nLocalServices )
2012-02-12 09:45:24 -03:00
{
2017-05-18 16:57:53 -04:00
CAddress ret ( CService ( CNetAddr ( ) , GetListenPort ( ) ) , nLocalServices ) ;
2012-05-10 14:35:13 -04:00
CService addr ;
2012-02-12 09:45:24 -03:00
if ( GetLocal ( addr , paddrPeer ) )
{
2016-05-25 11:18:37 -04:00
ret = CAddress ( addr , nLocalServices ) ;
2012-02-12 09:45:24 -03:00
}
2014-07-21 02:32:25 -04:00
ret . nTime = GetAdjustedTime ( ) ;
2012-02-12 09:45:24 -03:00
return ret ;
}
2010-08-29 12:58:15 -04:00
2018-05-02 12:14:48 -03:00
static int GetnScore ( const CService & addr )
2012-02-12 09:45:24 -03:00
{
2014-07-21 02:32:25 -04:00
LOCK ( cs_mapLocalHost ) ;
if ( mapLocalHost . count ( addr ) = = LOCAL_NONE )
return 0 ;
return mapLocalHost [ addr ] . nScore ;
}
// Is our peer's addrLocal potentially useful as an external IP source?
bool IsPeerAddrLocalGood ( CNode * pnode )
{
2017-02-06 14:18:51 -03:00
CService addrLocal = pnode - > GetAddrLocal ( ) ;
return fDiscover & & pnode - > addr . IsRoutable ( ) & & addrLocal . IsRoutable ( ) & &
2019-01-09 21:41:37 -03:00
IsReachable ( addrLocal . GetNetwork ( ) ) ;
2014-07-21 02:32:25 -04:00
}
// pushes our own address to a peer
2016-02-12 15:35:32 -03:00
void AdvertiseLocal ( CNode * pnode )
2014-07-21 02:32:25 -04:00
{
if ( fListen & & pnode - > fSuccessfullyConnected )
2012-02-12 09:45:24 -03:00
{
2016-04-19 01:04:58 -03:00
CAddress addrLocal = GetLocalAddress ( & pnode - > addr , pnode - > GetLocalServices ( ) ) ;
2017-05-11 09:59:57 -03:00
if ( gArgs . GetBoolArg ( " -addrmantest " , false ) ) {
// use IPv4 loopback during addrmantest
addrLocal = CAddress ( CService ( LookupNumeric ( " 127.0.0.1 " , GetListenPort ( ) ) ) , pnode - > GetLocalServices ( ) ) ;
}
2014-07-21 02:32:25 -04:00
// If discovery is enabled, sometimes give our peer the address it
// tells us that it sees us as in case it has a better idea of our
// address than we do.
2018-10-31 19:02:24 -03:00
FastRandomContext rng ;
2014-07-21 02:32:25 -04:00
if ( IsPeerAddrLocalGood ( pnode ) & & ( ! addrLocal . IsRoutable ( ) | |
2018-10-31 19:02:24 -03:00
rng . randbits ( ( GetnScore ( addrLocal ) > LOCAL_MANUAL ) ? 3 : 1 ) = = 0 ) )
2012-02-12 09:45:24 -03:00
{
2017-02-06 14:18:51 -03:00
addrLocal . SetIP ( pnode - > GetAddrLocal ( ) ) ;
2014-07-21 02:32:25 -04:00
}
2017-05-11 09:59:57 -03:00
if ( addrLocal . IsRoutable ( ) | | gArgs . GetBoolArg ( " -addrmantest " , false ) )
2014-07-21 02:32:25 -04:00
{
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " AdvertiseLocal: advertising address %s \n " , addrLocal . ToString ( ) ) ;
2018-10-31 19:02:24 -03:00
pnode - > PushAddress ( addrLocal , rng ) ;
2012-02-12 09:45:24 -03:00
}
}
}
// learn a new local address
2012-05-10 14:35:13 -04:00
bool AddLocal ( const CService & addr , int nScore )
2012-02-12 09:45:24 -03:00
{
if ( ! addr . IsRoutable ( ) )
return false ;
2012-05-24 13:02:21 -04:00
if ( ! fDiscover & & nScore < LOCAL_MANUAL )
2012-05-13 08:11:53 -04:00
return false ;
2019-01-09 21:41:37 -03:00
if ( ! IsReachable ( addr ) )
2012-05-13 09:11:51 -04:00
return false ;
2014-01-16 12:15:27 -03:00
LogPrintf ( " AddLocal(%s,%i) \n " , addr . ToString ( ) , nScore ) ;
2012-02-12 09:45:24 -03:00
{
LOCK ( cs_mapLocalHost ) ;
2012-05-12 18:41:24 -04:00
bool fAlready = mapLocalHost . count ( addr ) > 0 ;
LocalServiceInfo & info = mapLocalHost [ addr ] ;
if ( ! fAlready | | nScore > = info . nScore ) {
2012-08-28 20:33:25 -04:00
info . nScore = nScore + ( fAlready ? 1 : 0 ) ;
info . nPort = addr . GetPort ( ) ;
2012-05-12 18:41:24 -04:00
}
2012-02-12 09:45:24 -03:00
}
return true ;
}
2012-05-12 19:26:14 -04:00
bool AddLocal ( const CNetAddr & addr , int nScore )
2012-05-10 14:35:13 -04:00
{
2012-05-12 19:26:14 -04:00
return AddLocal ( CService ( addr , GetListenPort ( ) ) , nScore ) ;
2012-05-10 14:35:13 -04:00
}
2018-07-27 02:22:42 -04:00
void RemoveLocal ( const CService & addr )
2015-09-08 12:48:45 -03:00
{
LOCK ( cs_mapLocalHost ) ;
LogPrintf ( " RemoveLocal(%s) \n " , addr . ToString ( ) ) ;
mapLocalHost . erase ( addr ) ;
}
2019-01-09 21:41:37 -03:00
void SetReachable ( enum Network net , bool reachable )
2012-05-04 10:46:22 -04:00
{
2017-05-23 20:04:38 -04:00
if ( net = = NET_UNROUTABLE | | net = = NET_INTERNAL )
2012-05-14 11:15:58 -04:00
return ;
2012-05-04 10:46:22 -04:00
LOCK ( cs_mapLocalHost ) ;
2019-01-09 21:41:37 -03:00
vfLimited [ net ] = ! reachable ;
2012-05-04 10:46:22 -04:00
}
2019-01-09 21:41:37 -03:00
bool IsReachable ( enum Network net )
2012-05-04 10:46:22 -04:00
{
LOCK ( cs_mapLocalHost ) ;
2019-01-09 21:41:37 -03:00
return ! vfLimited [ net ] ;
2012-05-14 11:15:58 -04:00
}
2019-01-09 21:41:37 -03:00
bool IsReachable ( const CNetAddr & addr )
2012-05-14 11:15:58 -04:00
{
2019-01-09 21:41:37 -03:00
return IsReachable ( addr . GetNetwork ( ) ) ;
2012-05-04 10:46:22 -04:00
}
/** vote for a local address */
2012-05-10 14:35:13 -04:00
bool SeenLocal ( const CService & addr )
2012-02-12 09:45:24 -03:00
{
{
LOCK ( cs_mapLocalHost ) ;
if ( mapLocalHost . count ( addr ) = = 0 )
return false ;
2012-05-12 18:41:24 -04:00
mapLocalHost [ addr ] . nScore + + ;
2012-02-12 09:45:24 -03:00
}
return true ;
}
2014-07-21 02:32:25 -04:00
2012-05-04 10:46:22 -04:00
/** check whether a given address is potentially local */
2012-05-10 14:35:13 -04:00
bool IsLocal ( const CService & addr )
2012-02-12 09:45:24 -03:00
{
LOCK ( cs_mapLocalHost ) ;
return mapLocalHost . count ( addr ) > 0 ;
}
2010-08-29 12:58:15 -04:00
2016-04-16 20:13:12 -03:00
CNode * CConnman : : FindNode ( const CNetAddr & ip )
2010-08-29 12:58:15 -04:00
{
2013-04-04 06:30:55 -03:00
LOCK ( cs_vNodes ) ;
2017-07-20 05:32:47 -04:00
for ( CNode * pnode : vNodes ) {
2017-06-01 10:13:35 -04:00
if ( static_cast < CNetAddr > ( pnode - > addr ) = = ip ) {
2017-07-20 05:32:47 -04:00
return pnode ;
}
}
2017-08-07 01:36:37 -04:00
return nullptr ;
2010-08-29 12:58:15 -04:00
}
2016-04-16 20:13:12 -03:00
CNode * CConnman : : FindNode ( const CSubNet & subNet )
2015-05-25 15:03:51 -03:00
{
LOCK ( cs_vNodes ) ;
2017-07-20 05:32:47 -04:00
for ( CNode * pnode : vNodes ) {
2017-06-01 10:13:35 -04:00
if ( subNet . Match ( static_cast < CNetAddr > ( pnode - > addr ) ) ) {
2017-07-20 05:32:47 -04:00
return pnode ;
}
}
2017-08-07 01:36:37 -04:00
return nullptr ;
2015-05-25 15:03:51 -03:00
}
2016-04-16 20:13:12 -03:00
CNode * CConnman : : FindNode ( const std : : string & addrName )
2012-04-19 12:38:03 -03:00
{
LOCK ( cs_vNodes ) ;
2017-06-01 21:18:57 -04:00
for ( CNode * pnode : vNodes ) {
2017-02-06 14:04:34 -03:00
if ( pnode - > GetAddrName ( ) = = addrName ) {
2017-07-20 05:32:47 -04:00
return pnode ;
2017-02-06 14:04:34 -03:00
}
}
2017-08-07 01:36:37 -04:00
return nullptr ;
2012-04-19 12:38:03 -03:00
}
2016-04-16 20:13:12 -03:00
CNode * CConnman : : FindNode ( const CService & addr )
2010-08-29 12:58:15 -04:00
{
2013-04-04 06:30:55 -03:00
LOCK ( cs_vNodes ) ;
2017-07-20 05:32:47 -04:00
for ( CNode * pnode : vNodes ) {
2017-06-01 10:13:35 -04:00
if ( static_cast < CService > ( pnode - > addr ) = = addr ) {
2017-07-20 05:32:47 -04:00
return pnode ;
}
}
2017-08-07 01:36:37 -04:00
return nullptr ;
2010-08-29 12:58:15 -04:00
}
2016-04-17 21:21:58 -03:00
bool CConnman : : CheckIncomingNonce ( uint64_t nonce )
{
LOCK ( cs_vNodes ) ;
2018-06-18 01:58:28 -04:00
for ( const CNode * pnode : vNodes ) {
2016-04-17 21:21:58 -03:00
if ( ! pnode - > fSuccessfullyConnected & & ! pnode - > fInbound & & pnode - > GetLocalNonce ( ) = = nonce )
return false ;
}
return true ;
}
2017-05-30 05:59:42 -04:00
/** Get the bind address for a socket as CAddress */
static CAddress GetBindAddress ( SOCKET sock )
{
CAddress addr_bind ;
struct sockaddr_storage sockaddr_bind ;
socklen_t sockaddr_bind_len = sizeof ( sockaddr_bind ) ;
if ( sock ! = INVALID_SOCKET ) {
if ( ! getsockname ( sock , ( struct sockaddr * ) & sockaddr_bind , & sockaddr_bind_len ) ) {
addr_bind . SetSockAddr ( ( const struct sockaddr * ) & sockaddr_bind ) ;
} else {
LogPrint ( BCLog : : NET , " Warning: getsockname failed \n " ) ;
}
}
return addr_bind ;
}
2018-02-28 17:15:01 -03:00
CNode * CConnman : : ConnectNode ( CAddress addrConnect , const char * pszDest , bool fCountFailure , bool manual_connection )
2010-08-29 12:58:15 -04:00
{
2017-08-07 01:36:37 -04:00
if ( pszDest = = nullptr ) {
2012-02-12 09:45:24 -03:00
if ( IsLocal ( addrConnect ) )
2017-08-07 01:36:37 -04:00
return nullptr ;
2010-08-29 12:58:15 -04:00
2012-04-19 12:38:03 -03:00
// Look for an existing connection
2017-06-01 10:13:35 -04:00
CNode * pnode = FindNode ( static_cast < CService > ( addrConnect ) ) ;
2012-04-19 12:38:03 -03:00
if ( pnode )
{
2017-01-24 18:51:22 -03:00
LogPrintf ( " Failed to open new connection, already connected \n " ) ;
2017-08-07 01:36:37 -04:00
return nullptr ;
2012-04-19 12:38:03 -03:00
}
2010-08-29 12:58:15 -04:00
}
/// debug print
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " trying connection %s lastseen=%.1fhrs \n " ,
2014-01-16 12:15:27 -03:00
pszDest ? pszDest : addrConnect . ToString ( ) ,
2014-05-24 05:14:52 -04:00
pszDest ? 0.0 : ( double ) ( GetAdjustedTime ( ) - addrConnect . nTime ) / 3600.0 ) ;
2010-08-29 12:58:15 -04:00
2017-06-22 14:01:04 -04:00
// Resolve
const int default_port = Params ( ) . GetDefaultPort ( ) ;
if ( pszDest ) {
std : : vector < CService > resolved ;
if ( Lookup ( pszDest , resolved , default_port , fNameLookup & & ! HaveNameProxy ( ) , 256 ) & & ! resolved . empty ( ) ) {
addrConnect = CAddress ( resolved [ GetRand ( resolved . size ( ) ) ] , NODE_NONE ) ;
if ( ! addrConnect . IsValid ( ) ) {
2018-05-02 09:50:38 -03:00
LogPrint ( BCLog : : NET , " Resolver returned invalid address %s for %s \n " , addrConnect . ToString ( ) , pszDest ) ;
2017-06-22 14:01:04 -04:00
return nullptr ;
}
2016-05-28 10:22:02 -04:00
// It is possible that we already have a connection to the IP/port pszDest resolved to.
// In that case, drop the connection that was just created, and return the existing CNode instead.
// Also store the name we used to connect in that CNode, so that future FindNode() calls to that
// name catch this early.
2017-01-24 18:50:27 -03:00
LOCK ( cs_vNodes ) ;
2017-06-01 10:13:35 -04:00
CNode * pnode = FindNode ( static_cast < CService > ( addrConnect ) ) ;
2016-05-28 10:22:02 -04:00
if ( pnode )
{
2017-02-06 14:04:34 -03:00
pnode - > MaybeSetAddrName ( std : : string ( pszDest ) ) ;
2017-01-24 18:51:22 -03:00
LogPrintf ( " Failed to open new connection, already connected \n " ) ;
2017-08-07 01:36:37 -04:00
return nullptr ;
2016-05-28 10:22:02 -04:00
}
}
2017-06-22 14:01:04 -04:00
}
2016-05-28 10:22:02 -04:00
2017-06-22 14:01:04 -04:00
// Connect
bool connected = false ;
2018-02-01 12:29:15 -03:00
SOCKET hSocket = INVALID_SOCKET ;
2017-06-22 14:01:04 -04:00
proxyType proxy ;
if ( addrConnect . IsValid ( ) ) {
bool proxyConnectionFailed = false ;
2017-09-18 19:45:51 -03:00
if ( GetProxy ( addrConnect . GetNetwork ( ) , proxy ) ) {
hSocket = CreateSocket ( proxy . proxy ) ;
if ( hSocket = = INVALID_SOCKET ) {
return nullptr ;
}
2017-06-22 14:01:04 -04:00
connected = ConnectThroughProxy ( proxy , addrConnect . ToStringIP ( ) , addrConnect . GetPort ( ) , hSocket , nConnectTimeout , & proxyConnectionFailed ) ;
2017-09-18 19:45:51 -03:00
} else {
// no proxy needed (none set for target network)
hSocket = CreateSocket ( addrConnect ) ;
if ( hSocket = = INVALID_SOCKET ) {
return nullptr ;
}
2018-02-28 17:15:01 -03:00
connected = ConnectSocketDirectly ( addrConnect , hSocket , nConnectTimeout , manual_connection ) ;
2017-09-18 19:45:51 -03:00
}
2017-06-22 14:01:04 -04:00
if ( ! proxyConnectionFailed ) {
// If a connection to the node was attempted, and failure (if any) is not caused by a problem connecting to
// the proxy, mark this as an attempt.
addrman . Attempt ( addrConnect , fCountFailure ) ;
}
} else if ( pszDest & & GetNameProxy ( proxy ) ) {
2017-09-18 19:45:51 -03:00
hSocket = CreateSocket ( proxy . proxy ) ;
if ( hSocket = = INVALID_SOCKET ) {
return nullptr ;
}
2017-06-22 14:01:04 -04:00
std : : string host ;
int port = default_port ;
SplitHostPort ( std : : string ( pszDest ) , port , host ) ;
connected = ConnectThroughProxy ( proxy , host , port , hSocket , nConnectTimeout , nullptr ) ;
}
2017-10-02 15:18:32 -03:00
if ( ! connected ) {
CloseSocket ( hSocket ) ;
return nullptr ;
2010-08-29 12:58:15 -04:00
}
2014-05-24 05:14:52 -04:00
2017-10-02 15:18:32 -03:00
// Add node
NodeId id = GetNewNodeId ( ) ;
uint64_t nonce = GetDeterministicRandomizer ( RANDOMIZER_ID_LOCALHOSTNONCE ) . Write ( id ) . Finalize ( ) ;
CAddress addr_bind = GetBindAddress ( hSocket ) ;
CNode * pnode = new CNode ( id , nLocalServices , GetBestHeight ( ) , hSocket , addrConnect , CalculateKeyedNetGroup ( addrConnect ) , nonce , addr_bind , pszDest ? pszDest : " " , false ) ;
pnode - > AddRef ( ) ;
return pnode ;
2010-08-29 12:58:15 -04:00
}
2017-10-05 14:10:58 -03:00
void BanMan : : DumpBanlist ( )
2016-04-18 18:59:31 -03:00
{
2016-04-16 18:43:11 -03:00
SweepBanned ( ) ; // clean unused entries (if bantime has expired)
2016-04-18 18:59:31 -03:00
2016-04-16 18:43:11 -03:00
if ( ! BannedSetIsDirty ( ) )
2016-04-18 18:59:31 -03:00
return ;
int64_t nStart = GetTimeMillis ( ) ;
banmap_t banmap ;
2016-04-16 18:43:11 -03:00
GetBanned ( banmap ) ;
2017-10-05 14:35:20 -03:00
if ( m_ban_db . Write ( banmap ) ) {
2017-04-19 13:49:11 -03:00
SetBannedSetDirty ( false ) ;
}
2016-04-18 18:59:31 -03:00
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " Flushed %d banned node ips/subnets to banlist.dat %dms \n " ,
2016-04-18 18:59:31 -03:00
banmap . size ( ) , GetTimeMillis ( ) - nStart ) ;
}
2010-08-29 12:58:15 -04:00
void CNode : : CloseSocketDisconnect ( )
{
fDisconnect = true ;
2017-02-06 16:05:45 -03:00
LOCK ( cs_hSocket ) ;
2010-08-29 12:58:15 -04:00
if ( hSocket ! = INVALID_SOCKET )
{
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " disconnecting peer=%d \n " , id ) ;
2014-07-10 06:13:03 -04:00
CloseSocket ( hSocket ) ;
2010-08-29 12:58:15 -04:00
}
}
2017-10-05 14:10:58 -03:00
void BanMan : : ClearBanned ( )
2011-09-06 17:09:04 -03:00
{
2016-04-18 18:02:43 -03:00
{
LOCK ( cs_setBanned ) ;
setBanned . clear ( ) ;
setBannedIsDirty = true ;
}
DumpBanlist ( ) ; //store banlist to disk
2016-05-25 21:26:46 -04:00
if ( clientInterface )
clientInterface - > BannedListChanged ( ) ;
2011-09-06 17:09:04 -03:00
}
2017-10-05 14:10:58 -03:00
bool BanMan : : IsBanned ( CNetAddr ip )
2011-09-06 17:09:04 -03:00
{
2017-06-08 22:05:53 -04:00
LOCK ( cs_setBanned ) ;
2017-07-20 05:32:47 -04:00
for ( const auto & it : setBanned ) {
CSubNet subNet = it . first ;
CBanEntry banEntry = it . second ;
2015-05-25 15:03:51 -03:00
2017-06-08 22:05:53 -04:00
if ( subNet . Match ( ip ) & & GetTime ( ) < banEntry . nBanUntil ) {
return true ;
2015-05-25 15:03:51 -03:00
}
}
2017-06-08 22:05:53 -04:00
return false ;
2015-05-25 15:03:51 -03:00
}
2017-10-05 14:10:58 -03:00
bool BanMan : : IsBanned ( CSubNet subnet )
2015-05-25 15:03:51 -03:00
{
2017-06-08 22:05:53 -04:00
LOCK ( cs_setBanned ) ;
banmap_t : : iterator i = setBanned . find ( subnet ) ;
if ( i ! = setBanned . end ( ) )
2015-05-25 15:03:51 -03:00
{
2017-06-08 22:05:53 -04:00
CBanEntry banEntry = ( * i ) . second ;
if ( GetTime ( ) < banEntry . nBanUntil ) {
return true ;
2011-09-06 17:09:04 -03:00
}
}
2017-06-08 22:05:53 -04:00
return false ;
2011-09-06 17:09:04 -03:00
}
2017-10-05 14:10:58 -03:00
void BanMan : : Ban ( const CNetAddr & addr , const BanReason & banReason , int64_t bantimeoffset , bool sinceUnixEpoch ) {
2015-06-29 15:37:22 -03:00
CSubNet subNet ( addr ) ;
2015-06-26 16:38:33 -03:00
Ban ( subNet , banReason , bantimeoffset , sinceUnixEpoch ) ;
2015-05-25 15:03:51 -03:00
}
2017-10-05 14:10:58 -03:00
void BanMan : : Ban ( const CSubNet & subNet , const BanReason & banReason , int64_t bantimeoffset , bool sinceUnixEpoch ) {
2015-06-26 16:38:33 -03:00
CBanEntry banEntry ( GetTime ( ) ) ;
banEntry . banReason = banReason ;
if ( bantimeoffset < = 0 )
{
2017-10-05 14:41:45 -03:00
bantimeoffset = m_default_ban_time ;
2015-06-26 16:38:33 -03:00
sinceUnixEpoch = false ;
}
banEntry . nBanUntil = ( sinceUnixEpoch ? 0 : GetTime ( ) ) + bantimeoffset ;
2016-04-18 18:02:43 -03:00
{
LOCK ( cs_setBanned ) ;
if ( setBanned [ subNet ] . nBanUntil < banEntry . nBanUntil ) {
setBanned [ subNet ] = banEntry ;
setBannedIsDirty = true ;
}
else
return ;
}
2016-05-25 21:26:46 -04:00
if ( clientInterface )
clientInterface - > BannedListChanged ( ) ;
2016-04-18 18:02:43 -03:00
if ( banReason = = BanReasonManuallyAdded )
DumpBanlist ( ) ; //store banlist to disk immediately if user requested ban
2011-09-06 17:09:04 -03:00
}
2017-10-05 14:10:58 -03:00
bool BanMan : : Unban ( const CNetAddr & addr ) {
2015-06-29 15:37:22 -03:00
CSubNet subNet ( addr ) ;
2015-05-25 15:03:51 -03:00
return Unban ( subNet ) ;
}
2017-10-05 14:10:58 -03:00
bool BanMan : : Unban ( const CSubNet & subNet ) {
2015-06-19 10:27:37 -03:00
{
2016-04-18 18:02:43 -03:00
LOCK ( cs_setBanned ) ;
if ( ! setBanned . erase ( subNet ) )
return false ;
2015-06-19 10:27:37 -03:00
setBannedIsDirty = true ;
}
2016-05-25 21:26:46 -04:00
if ( clientInterface )
clientInterface - > BannedListChanged ( ) ;
2016-04-18 18:02:43 -03:00
DumpBanlist ( ) ; //store banlist to disk immediately
return true ;
2015-05-19 05:07:23 -03:00
}
2017-10-05 14:10:58 -03:00
void BanMan : : GetBanned ( banmap_t & banMap )
2015-05-19 05:07:23 -03:00
{
LOCK ( cs_setBanned ) ;
2017-04-19 13:49:11 -03:00
// Sweep the banlist so expired bans are not returned
SweepBanned ( ) ;
2015-05-19 05:07:23 -03:00
banMap = setBanned ; //create a thread safe copy
}
2017-10-05 14:10:58 -03:00
void BanMan : : SetBanned ( const banmap_t & banMap )
2015-06-19 10:27:37 -03:00
{
LOCK ( cs_setBanned ) ;
setBanned = banMap ;
setBannedIsDirty = true ;
}
2017-10-05 14:10:58 -03:00
void BanMan : : SweepBanned ( )
2015-06-19 10:27:37 -03:00
{
int64_t now = GetTime ( ) ;
2017-11-19 04:38:27 -03:00
bool notifyUI = false ;
2015-06-19 10:27:37 -03:00
{
2017-11-19 04:38:27 -03:00
LOCK ( cs_setBanned ) ;
banmap_t : : iterator it = setBanned . begin ( ) ;
while ( it ! = setBanned . end ( ) )
2015-06-19 10:27:37 -03:00
{
2017-11-19 04:38:27 -03:00
CSubNet subNet = ( * it ) . first ;
CBanEntry banEntry = ( * it ) . second ;
if ( now > banEntry . nBanUntil )
{
setBanned . erase ( it + + ) ;
setBannedIsDirty = true ;
notifyUI = true ;
LogPrint ( BCLog : : NET , " %s: Removed banned node ip/subnet from banlist.dat: %s \n " , __func__ , subNet . ToString ( ) ) ;
}
else
+ + it ;
2015-06-19 10:27:37 -03:00
}
}
2017-11-05 23:34:40 -03:00
// update UI
2017-11-19 04:38:27 -03:00
if ( notifyUI & & clientInterface ) {
2017-11-05 23:34:40 -03:00
clientInterface - > BannedListChanged ( ) ;
2015-06-19 10:27:37 -03:00
}
}
2017-10-05 14:10:58 -03:00
bool BanMan : : BannedSetIsDirty ( )
2015-06-19 10:27:37 -03:00
{
LOCK ( cs_setBanned ) ;
return setBannedIsDirty ;
}
2017-10-05 14:10:58 -03:00
void BanMan : : SetBannedSetDirty ( bool dirty )
2015-06-19 10:27:37 -03:00
{
LOCK ( cs_setBanned ) ; //reuse setBanned lock for the isDirty flag
setBannedIsDirty = dirty ;
}
2014-06-21 07:34:36 -04:00
2016-04-17 19:34:32 -03:00
bool CConnman : : IsWhitelistedRange ( const CNetAddr & addr ) {
2017-06-01 21:18:57 -04:00
for ( const CSubNet & subnet : vWhitelistedRange ) {
2014-06-21 07:34:36 -04:00
if ( subnet . Match ( addr ) )
return true ;
}
return false ;
}
2017-02-06 14:04:34 -03:00
std : : string CNode : : GetAddrName ( ) const {
LOCK ( cs_addrName ) ;
return addrName ;
}
void CNode : : MaybeSetAddrName ( const std : : string & addrNameIn ) {
LOCK ( cs_addrName ) ;
if ( addrName . empty ( ) ) {
addrName = addrNameIn ;
}
}
2017-02-06 14:18:51 -03:00
CService CNode : : GetAddrLocal ( ) const {
LOCK ( cs_addrLocal ) ;
return addrLocal ;
}
void CNode : : SetAddrLocal ( const CService & addrLocalIn ) {
LOCK ( cs_addrLocal ) ;
if ( addrLocal . IsValid ( ) ) {
error ( " Addr local already set for node: %i. Refusing to change from %s to %s " , id , addrLocal . ToString ( ) , addrLocalIn . ToString ( ) ) ;
} else {
addrLocal = addrLocalIn ;
}
}
2012-06-29 17:24:53 -04:00
# undef X
# define X(name) stats.name = name
void CNode : : copyStats ( CNodeStats & stats )
{
2013-11-17 21:25:17 -03:00
stats . nodeid = this - > GetId ( ) ;
2012-06-29 17:24:53 -04:00
X ( nServices ) ;
2016-10-04 20:27:11 -03:00
X ( addr ) ;
2017-05-30 05:59:42 -04:00
X ( addrBind ) ;
2017-02-06 13:53:34 -03:00
{
LOCK ( cs_filter ) ;
X ( fRelayTxes ) ;
}
2012-06-29 17:24:53 -04:00
X ( nLastSend ) ;
X ( nLastRecv ) ;
X ( nTimeConnected ) ;
2014-12-15 07:06:15 -03:00
X ( nTimeOffset ) ;
2017-02-06 14:04:34 -03:00
stats . addrName = GetAddrName ( ) ;
2012-06-29 17:24:53 -04:00
X ( nVersion ) ;
2017-02-06 14:08:31 -03:00
{
LOCK ( cs_SubVer ) ;
X ( cleanSubVer ) ;
}
2012-06-29 17:24:53 -04:00
X ( fInbound ) ;
2017-10-05 12:49:16 -03:00
X ( m_manual_connection ) ;
2012-06-29 17:24:53 -04:00
X ( nStartingHeight ) ;
2017-02-06 04:34:57 -03:00
{
LOCK ( cs_vSend ) ;
X ( mapSendBytesPerMsgCmd ) ;
X ( nSendBytes ) ;
}
{
LOCK ( cs_vRecv ) ;
X ( mapRecvBytesPerMsgCmd ) ;
X ( nRecvBytes ) ;
}
2014-06-21 07:34:36 -04:00
X ( fWhitelisted ) ;
2018-10-08 10:33:39 -03:00
{
LOCK ( cs_feeFilter ) ;
X ( minFeeFilter ) ;
}
2013-11-15 08:24:34 -03:00
2013-08-22 07:34:33 -04:00
// It is common for nodes with good ping times to suddenly become lagged,
// due to a new block arriving or other large transfer.
// Merely reporting pingtime might fool the caller into thinking the node was still responsive,
// since pingtime does not update until the ping is complete, which might take a while.
// So, if a ping is taking an unusually long time in flight,
// the caller can immediately detect that this is happening.
2013-04-13 02:13:08 -03:00
int64_t nPingUsecWait = 0 ;
2013-08-22 07:34:33 -04:00
if ( ( 0 ! = nPingNonceSent ) & & ( 0 ! = nPingUsecStart ) ) {
nPingUsecWait = GetTimeMicros ( ) - nPingUsecStart ;
}
2013-11-15 08:24:34 -03:00
2013-08-22 07:34:33 -04:00
// Raw ping time is in microseconds, but show it to user as whole seconds (Bitcoin users should be well used to small numbers with many decimal places by now :)
stats . dPingTime = ( ( ( double ) nPingUsecTime ) / 1e6 ) ;
2016-10-14 11:11:38 -03:00
stats . dMinPing = ( ( ( double ) nMinPingUsecTime ) / 1e6 ) ;
2013-08-22 07:34:33 -04:00
stats . dPingWait = ( ( ( double ) nPingUsecWait ) / 1e6 ) ;
2013-11-15 08:24:34 -03:00
2013-08-22 01:50:19 -04:00
// Leave string empty if addrLocal invalid (not filled in yet)
2017-02-06 14:18:51 -03:00
CService addrLocalUnlocked = GetAddrLocal ( ) ;
stats . addrLocal = addrLocalUnlocked . IsValid ( ) ? addrLocalUnlocked . ToString ( ) : " " ;
2012-06-29 17:24:53 -04:00
}
# undef X
2010-08-29 12:58:15 -04:00
2016-04-18 22:33:54 -03:00
bool CNode : : ReceiveMsgBytes ( const char * pch , unsigned int nBytes , bool & complete )
2012-11-15 21:41:12 -03:00
{
2016-04-18 22:33:54 -03:00
complete = false ;
2016-12-31 04:05:17 -03:00
int64_t nTimeMicros = GetTimeMicros ( ) ;
2017-02-06 04:34:57 -03:00
LOCK ( cs_vRecv ) ;
2016-12-31 04:05:17 -03:00
nLastRecv = nTimeMicros / 1000000 ;
nRecvBytes + = nBytes ;
2012-11-15 21:41:12 -03:00
while ( nBytes > 0 ) {
// get current incomplete message, or create a new one
2013-02-28 21:41:28 -03:00
if ( vRecvMsg . empty ( ) | |
2012-11-15 21:41:12 -03:00
vRecvMsg . back ( ) . complete ( ) )
2016-12-31 04:05:15 -03:00
vRecvMsg . push_back ( CNetMessage ( Params ( ) . MessageStart ( ) , SER_NETWORK , INIT_PROTO_VERSION ) ) ;
2012-11-15 21:41:12 -03:00
CNetMessage & msg = vRecvMsg . back ( ) ;
// absorb network data
int handled ;
if ( ! msg . in_data )
handled = msg . readHeader ( pch , nBytes ) ;
else
handled = msg . readData ( pch , nBytes ) ;
if ( handled < 0 )
2017-03-06 13:54:08 -03:00
return false ;
2012-11-15 21:41:12 -03:00
2015-03-05 09:01:22 -03:00
if ( msg . in_data & & msg . hdr . nMessageSize > MAX_PROTOCOL_MESSAGE_LENGTH ) {
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " Oversized message from peer=%i, disconnecting \n " , GetId ( ) ) ;
2015-03-05 09:01:22 -03:00
return false ;
}
2012-11-15 21:41:12 -03:00
pch + = handled ;
nBytes - = handled ;
2014-07-06 10:06:46 -04:00
2015-04-05 06:35:37 -03:00
if ( msg . complete ( ) ) {
2015-08-25 11:30:31 -03:00
//store received bytes per message command
//to prevent a memory DOS, only allow valid commands
mapMsgCmdSize : : iterator i = mapRecvBytesPerMsgCmd . find ( msg . hdr . pchCommand ) ;
if ( i = = mapRecvBytesPerMsgCmd . end ( ) )
i = mapRecvBytesPerMsgCmd . find ( NET_MESSAGE_COMMAND_OTHER ) ;
assert ( i ! = mapRecvBytesPerMsgCmd . end ( ) ) ;
i - > second + = msg . hdr . nMessageSize + CMessageHeader : : HEADER_SIZE ;
2016-12-31 04:05:17 -03:00
msg . nTime = nTimeMicros ;
2016-04-18 22:33:54 -03:00
complete = true ;
2015-04-05 06:35:37 -03:00
}
2012-11-15 21:41:12 -03:00
}
return true ;
}
2017-02-02 16:33:41 -03:00
void CNode : : SetSendVersion ( int nVersionIn )
{
// Send version may only be changed in the version message, and
// only one version message is allowed per session. We can therefore
// treat this value as const and even atomic as long as it's only used
// once a version message has been successfully processed. Any attempt to
// set this twice is an error.
if ( nSendVersion ! = 0 ) {
error ( " Send version already set for node: %i. Refusing to change from %i to %i " , id , nSendVersion , nVersionIn ) ;
} else {
nSendVersion = nVersionIn ;
}
}
int CNode : : GetSendVersion ( ) const
{
// The send version should always be explicitly set to
// INIT_PROTO_VERSION rather than using this value until SetSendVersion
// has been called.
if ( nSendVersion = = 0 ) {
error ( " Requesting unset send version for node: %i. Using %i " , id , INIT_PROTO_VERSION ) ;
return INIT_PROTO_VERSION ;
}
return nSendVersion ;
}
2012-11-15 21:41:12 -03:00
int CNetMessage : : readHeader ( const char * pch , unsigned int nBytes )
{
// copy data to temporary parsing buffer
unsigned int nRemaining = 24 - nHdrPos ;
unsigned int nCopy = std : : min ( nRemaining , nBytes ) ;
memcpy ( & hdrbuf [ nHdrPos ] , pch , nCopy ) ;
nHdrPos + = nCopy ;
// if header incomplete, exit
if ( nHdrPos < 24 )
return nCopy ;
// deserialize to CMessageHeader
try {
hdrbuf > > hdr ;
}
2014-12-07 09:29:06 -03:00
catch ( const std : : exception & ) {
2012-11-15 21:41:12 -03:00
return - 1 ;
}
// reject messages larger than MAX_SIZE
if ( hdr . nMessageSize > MAX_SIZE )
2017-03-06 13:54:08 -03:00
return - 1 ;
2012-11-15 21:41:12 -03:00
// switch state to reading message data
in_data = true ;
return nCopy ;
}
int CNetMessage : : readData ( const char * pch , unsigned int nBytes )
{
unsigned int nRemaining = hdr . nMessageSize - nDataPos ;
unsigned int nCopy = std : : min ( nRemaining , nBytes ) ;
2014-06-21 11:00:38 -04:00
if ( vRecv . size ( ) < nDataPos + nCopy ) {
// Allocate up to 256 KiB ahead, but never more than the total message size.
vRecv . resize ( std : : min ( hdr . nMessageSize , nDataPos + nCopy + 256 * 1024 ) ) ;
}
2016-10-30 19:02:16 -03:00
hasher . Write ( ( const unsigned char * ) pch , nCopy ) ;
2012-11-15 21:41:12 -03:00
memcpy ( & vRecv [ nDataPos ] , pch , nCopy ) ;
nDataPos + = nCopy ;
return nCopy ;
}
2016-10-30 19:02:16 -03:00
const uint256 & CNetMessage : : GetMessageHash ( ) const
{
assert ( complete ( ) ) ;
if ( data_hash . IsNull ( ) )
hasher . Finalize ( data_hash . begin ( ) ) ;
return data_hash ;
}
2018-10-08 10:34:39 -03:00
size_t CConnman : : SocketSendData ( CNode * pnode ) const EXCLUSIVE_LOCKS_REQUIRED ( pnode - > cs_vSend )
2012-11-15 20:04:52 -03:00
{
2016-11-10 22:17:30 -03:00
auto it = pnode - > vSendMsg . begin ( ) ;
2016-05-21 06:04:02 -04:00
size_t nSentSize = 0 ;
2013-03-24 12:52:24 -03:00
while ( it ! = pnode - > vSendMsg . end ( ) ) {
2016-11-10 22:17:30 -03:00
const auto & data = * it ;
2013-03-24 12:52:24 -03:00
assert ( data . size ( ) > pnode - > nSendOffset ) ;
2017-02-06 16:05:45 -03:00
int nBytes = 0 ;
{
LOCK ( pnode - > cs_hSocket ) ;
if ( pnode - > hSocket = = INVALID_SOCKET )
break ;
nBytes = send ( pnode - > hSocket , reinterpret_cast < const char * > ( data . data ( ) ) + pnode - > nSendOffset , data . size ( ) - pnode - > nSendOffset , MSG_NOSIGNAL | MSG_DONTWAIT ) ;
}
2013-03-24 12:52:24 -03:00
if ( nBytes > 0 ) {
2017-01-19 15:01:18 -03:00
pnode - > nLastSend = GetSystemTimeInSeconds ( ) ;
2013-04-07 14:31:13 -03:00
pnode - > nSendBytes + = nBytes ;
2013-03-24 12:52:24 -03:00
pnode - > nSendOffset + = nBytes ;
2016-05-21 06:04:02 -04:00
nSentSize + = nBytes ;
2013-03-24 12:52:24 -03:00
if ( pnode - > nSendOffset = = data . size ( ) ) {
pnode - > nSendOffset = 0 ;
pnode - > nSendSize - = data . size ( ) ;
2016-12-31 04:05:32 -03:00
pnode - > fPauseSend = pnode - > nSendSize > nSendBufferMaxSize ;
2013-03-24 12:52:24 -03:00
it + + ;
} else {
// could not send full message; stop sending more
break ;
}
} else {
if ( nBytes < 0 ) {
// error
int nErr = WSAGetLastError ( ) ;
if ( nErr ! = WSAEWOULDBLOCK & & nErr ! = WSAEMSGSIZE & & nErr ! = WSAEINTR & & nErr ! = WSAEINPROGRESS )
{
2014-05-08 08:15:19 -04:00
LogPrintf ( " socket send error %s \n " , NetworkErrorString ( nErr ) ) ;
2013-03-24 12:52:24 -03:00
pnode - > CloseSocketDisconnect ( ) ;
}
}
// couldn't send anything at all
break ;
2012-11-15 20:04:52 -03:00
}
}
2013-03-24 12:52:24 -03:00
if ( it = = pnode - > vSendMsg . end ( ) ) {
assert ( pnode - > nSendOffset = = 0 ) ;
assert ( pnode - > nSendSize = = 0 ) ;
}
pnode - > vSendMsg . erase ( pnode - > vSendMsg . begin ( ) , it ) ;
2016-05-21 06:04:02 -04:00
return nSentSize ;
2012-11-15 20:04:52 -03:00
}
2010-08-29 12:58:15 -04:00
2016-04-18 16:58:19 -03:00
struct NodeEvictionCandidate
{
NodeId id ;
int64_t nTimeConnected ;
int64_t nMinPingUsecTime ;
2016-05-22 01:55:15 -04:00
int64_t nLastBlockTime ;
int64_t nLastTXTime ;
2016-10-31 21:08:47 -03:00
bool fRelevantServices ;
2016-05-22 01:55:15 -04:00
bool fRelayTxes ;
bool fBloomFilter ;
2016-04-18 16:58:19 -03:00
CAddress addr ;
2016-05-25 09:38:32 -04:00
uint64_t nKeyedNetGroup ;
2015-08-20 21:29:04 -03:00
} ;
2016-04-18 16:58:19 -03:00
static bool ReverseCompareNodeMinPingTime ( const NodeEvictionCandidate & a , const NodeEvictionCandidate & b )
2015-08-13 06:58:58 -03:00
{
2016-04-18 16:58:19 -03:00
return a . nMinPingUsecTime > b . nMinPingUsecTime ;
2015-08-13 06:58:58 -03:00
}
2016-04-18 16:58:19 -03:00
static bool ReverseCompareNodeTimeConnected ( const NodeEvictionCandidate & a , const NodeEvictionCandidate & b )
2015-08-13 06:58:58 -03:00
{
2016-04-18 16:58:19 -03:00
return a . nTimeConnected > b . nTimeConnected ;
2015-08-13 06:58:58 -03:00
}
2016-05-23 03:21:05 -04:00
static bool CompareNetGroupKeyed ( const NodeEvictionCandidate & a , const NodeEvictionCandidate & b ) {
2016-05-25 09:38:32 -04:00
return a . nKeyedNetGroup < b . nKeyedNetGroup ;
2016-05-22 01:55:15 -04:00
}
static bool CompareNodeBlockTime ( const NodeEvictionCandidate & a , const NodeEvictionCandidate & b )
{
// There is a fall-through here because it is common for a node to have many peers which have not yet relayed a block.
if ( a . nLastBlockTime ! = b . nLastBlockTime ) return a . nLastBlockTime < b . nLastBlockTime ;
2016-10-31 21:08:47 -03:00
if ( a . fRelevantServices ! = b . fRelevantServices ) return b . fRelevantServices ;
2016-05-22 01:55:15 -04:00
return a . nTimeConnected > b . nTimeConnected ;
}
static bool CompareNodeTXTime ( const NodeEvictionCandidate & a , const NodeEvictionCandidate & b )
{
// There is a fall-through here because it is common for a node to have more than a few peers that have not yet relayed txn.
if ( a . nLastTXTime ! = b . nLastTXTime ) return a . nLastTXTime < b . nLastTXTime ;
if ( a . fRelayTxes ! = b . fRelayTxes ) return b . fRelayTxes ;
if ( a . fBloomFilter ! = b . fBloomFilter ) return a . fBloomFilter ;
return a . nTimeConnected > b . nTimeConnected ;
}
2015-08-13 06:58:58 -03:00
2017-10-19 04:53:01 -03:00
//! Sort an array by the specified comparator, then erase the last K elements.
template < typename T , typename Comparator >
static void EraseLastKElements ( std : : vector < T > & elements , Comparator comparator , size_t k )
{
std : : sort ( elements . begin ( ) , elements . end ( ) , comparator ) ;
size_t eraseSize = std : : min ( k , elements . size ( ) ) ;
elements . erase ( elements . end ( ) - eraseSize , elements . end ( ) ) ;
}
2016-04-29 11:23:51 -03:00
/** Try to find a connection to evict when the node is full.
* Extreme care must be taken to avoid opening the node to attacker
* triggered network partitioning .
* The strategy used here is to protect a small number of peers
* for each of several distinct characteristics which are difficult
* to forge . In order to partition a node the attacker must be
* simultaneously better at all of them than honest peers .
*/
2016-04-16 20:13:12 -03:00
bool CConnman : : AttemptToEvictConnection ( )
{
2016-04-18 16:58:19 -03:00
std : : vector < NodeEvictionCandidate > vEvictionCandidates ;
2015-08-13 06:58:58 -03:00
{
LOCK ( cs_vNodes ) ;
2017-07-20 05:32:47 -04:00
for ( const CNode * node : vNodes ) {
2015-08-13 06:58:58 -03:00
if ( node - > fWhitelisted )
continue ;
if ( ! node - > fInbound )
continue ;
if ( node - > fDisconnect )
continue ;
2018-02-02 20:11:01 -03:00
LOCK ( node - > cs_filter ) ;
2017-04-11 13:13:55 -03:00
NodeEvictionCandidate candidate = { node - > GetId ( ) , node - > nTimeConnected , node - > nMinPingUsecTime ,
2016-10-31 21:08:47 -03:00
node - > nLastBlockTime , node - > nLastTXTime ,
2017-10-04 18:59:30 -03:00
HasAllDesirableServiceFlags ( node - > nServices ) ,
2017-08-07 01:36:37 -04:00
node - > fRelayTxes , node - > pfilter ! = nullptr , node - > addr , node - > nKeyedNetGroup } ;
2016-04-18 16:58:19 -03:00
vEvictionCandidates . push_back ( candidate ) ;
2015-08-13 06:58:58 -03:00
}
}
// Protect connections with certain characteristics
2015-08-20 20:47:49 -03:00
// Deterministically select 4 peers to protect by netgroup.
2016-05-23 03:21:05 -04:00
// An attacker cannot predict which netgroups will be protected
2017-10-19 04:53:01 -03:00
EraseLastKElements ( vEvictionCandidates , CompareNetGroupKeyed , 4 ) ;
2016-04-29 11:23:51 -03:00
// Protect the 8 nodes with the lowest minimum ping time.
2015-08-20 20:47:49 -03:00
// An attacker cannot manipulate this metric without physically moving nodes closer to the target.
2017-10-19 04:53:01 -03:00
EraseLastKElements ( vEvictionCandidates , ReverseCompareNodeMinPingTime , 8 ) ;
2016-05-22 01:55:15 -04:00
// Protect 4 nodes that most recently sent us transactions.
// An attacker cannot manipulate this metric without performing useful work.
2017-10-19 04:53:01 -03:00
EraseLastKElements ( vEvictionCandidates , CompareNodeTXTime , 4 ) ;
2016-05-22 01:55:15 -04:00
// Protect 4 nodes that most recently sent us blocks.
// An attacker cannot manipulate this metric without performing useful work.
2017-10-19 04:53:01 -03:00
EraseLastKElements ( vEvictionCandidates , CompareNodeBlockTime , 4 ) ;
2015-08-25 20:31:13 -03:00
// Protect the half of the remaining nodes which have been connected the longest.
2016-04-29 11:23:51 -03:00
// This replicates the non-eviction implicit behavior, and precludes attacks that start later.
2017-10-19 04:53:01 -03:00
EraseLastKElements ( vEvictionCandidates , ReverseCompareNodeTimeConnected , vEvictionCandidates . size ( ) / 2 ) ;
2015-08-13 06:58:58 -03:00
2015-08-22 19:15:39 -03:00
if ( vEvictionCandidates . empty ( ) ) return false ;
2015-08-13 06:58:58 -03:00
2015-11-23 00:48:54 -03:00
// Identify the network group with the most connections and youngest member.
// (vEvictionCandidates is already sorted by reverse connect time)
2016-05-25 09:38:32 -04:00
uint64_t naMostConnections ;
2015-08-13 06:58:58 -03:00
unsigned int nMostConnections = 0 ;
2015-11-23 00:48:54 -03:00
int64_t nMostConnectionsTime = 0 ;
2016-06-10 10:09:06 -04:00
std : : map < uint64_t , std : : vector < NodeEvictionCandidate > > mapNetGroupNodes ;
2017-06-01 21:18:57 -04:00
for ( const NodeEvictionCandidate & node : vEvictionCandidates ) {
2017-10-19 04:53:01 -03:00
std : : vector < NodeEvictionCandidate > & group = mapNetGroupNodes [ node . nKeyedNetGroup ] ;
group . push_back ( node ) ;
int64_t grouptime = group [ 0 ] . nTimeConnected ;
2015-08-13 06:58:58 -03:00
2017-10-19 04:53:01 -03:00
if ( group . size ( ) > nMostConnections | | ( group . size ( ) = = nMostConnections & & grouptime > nMostConnectionsTime ) ) {
nMostConnections = group . size ( ) ;
2015-11-23 00:48:54 -03:00
nMostConnectionsTime = grouptime ;
2016-05-25 09:38:32 -04:00
naMostConnections = node . nKeyedNetGroup ;
2015-08-13 06:58:58 -03:00
}
}
2015-08-25 21:06:15 -03:00
// Reduce to the network group with the most connections
2016-06-10 10:09:06 -04:00
vEvictionCandidates = std : : move ( mapNetGroupNodes [ naMostConnections ] ) ;
2015-08-13 06:58:58 -03:00
2015-11-23 00:48:54 -03:00
// Disconnect from the network group with the most connections
2016-04-18 16:58:19 -03:00
NodeId evicted = vEvictionCandidates . front ( ) . id ;
LOCK ( cs_vNodes ) ;
2017-07-20 05:32:47 -04:00
for ( CNode * pnode : vNodes ) {
if ( pnode - > GetId ( ) = = evicted ) {
pnode - > fDisconnect = true ;
2016-04-18 16:58:19 -03:00
return true ;
}
}
return false ;
2015-08-13 06:58:58 -03:00
}
2016-04-16 15:47:18 -03:00
void CConnman : : AcceptConnection ( const ListenSocket & hListenSocket ) {
2015-08-13 06:00:10 -03:00
struct sockaddr_storage sockaddr ;
socklen_t len = sizeof ( sockaddr ) ;
SOCKET hSocket = accept ( hListenSocket . socket , ( struct sockaddr * ) & sockaddr , & len ) ;
CAddress addr ;
int nInbound = 0 ;
2016-08-31 14:17:28 -03:00
int nMaxInbound = nMaxConnections - ( nMaxOutbound + nMaxFeeler ) ;
2015-08-13 06:00:10 -03:00
2017-05-30 05:59:42 -04:00
if ( hSocket ! = INVALID_SOCKET ) {
if ( ! addr . SetSockAddr ( ( const struct sockaddr * ) & sockaddr ) ) {
2015-08-13 06:00:10 -03:00
LogPrintf ( " Warning: Unknown socket family \n " ) ;
2017-05-30 05:59:42 -04:00
}
}
2015-08-13 06:00:10 -03:00
2016-04-17 19:34:32 -03:00
bool whitelisted = hListenSocket . whitelisted | | IsWhitelistedRange ( addr ) ;
2015-08-13 06:00:10 -03:00
{
LOCK ( cs_vNodes ) ;
2017-07-20 05:32:47 -04:00
for ( const CNode * pnode : vNodes ) {
if ( pnode - > fInbound ) nInbound + + ;
}
2015-08-13 06:00:10 -03:00
}
if ( hSocket = = INVALID_SOCKET )
{
int nErr = WSAGetLastError ( ) ;
if ( nErr ! = WSAEWOULDBLOCK )
LogPrintf ( " socket error accept failed: %s \n " , NetworkErrorString ( nErr ) ) ;
2015-08-13 06:16:46 -03:00
return ;
2015-08-13 06:00:10 -03:00
}
2015-08-13 06:16:46 -03:00
2013-03-25 22:33:25 -03:00
if ( ! fNetworkActive ) {
LogPrintf ( " connection from %s dropped: not accepting new connections \n " , addr . ToString ( ) ) ;
CloseSocket ( hSocket ) ;
return ;
}
2015-08-13 06:16:46 -03:00
if ( ! IsSelectableSocket ( hSocket ) )
2015-08-13 06:00:10 -03:00
{
LogPrintf ( " connection from %s dropped: non-selectable socket \n " , addr . ToString ( ) ) ;
CloseSocket ( hSocket ) ;
2015-08-13 06:16:46 -03:00
return ;
2015-08-13 06:00:10 -03:00
}
2015-08-13 06:16:46 -03:00
2015-10-21 20:52:29 -03:00
// According to the internet TCP_NODELAY is not carried into accepted sockets
// on all platforms. Set it again here just to be sure.
2017-03-23 02:02:02 -03:00
SetSocketNoDelay ( hSocket ) ;
2015-10-21 20:52:29 -03:00
2017-10-05 14:10:58 -03:00
if ( m_banman & & m_banman - > IsBanned ( addr ) & & ! whitelisted )
2015-08-13 06:00:10 -03:00
{
2017-10-30 19:56:37 -03:00
LogPrint ( BCLog : : NET , " connection from %s dropped (banned) \n " , addr . ToString ( ) ) ;
2015-08-13 06:00:10 -03:00
CloseSocket ( hSocket ) ;
2015-08-13 06:16:46 -03:00
return ;
2015-08-13 06:00:10 -03:00
}
2015-08-13 06:16:46 -03:00
2015-08-13 06:19:17 -03:00
if ( nInbound > = nMaxInbound )
2015-08-13 06:00:10 -03:00
{
2016-05-22 01:55:15 -04:00
if ( ! AttemptToEvictConnection ( ) ) {
2015-08-13 06:58:58 -03:00
// No connection to evict, disconnect the new connection
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " failed to find an eviction candidate - connection dropped (full) \n " ) ;
2015-08-13 06:58:58 -03:00
CloseSocket ( hSocket ) ;
return ;
}
2015-08-13 06:00:10 -03:00
}
2016-10-26 16:10:15 -03:00
NodeId id = GetNewNodeId ( ) ;
uint64_t nonce = GetDeterministicRandomizer ( RANDOMIZER_ID_LOCALHOSTNONCE ) . Write ( id ) . Finalize ( ) ;
2017-05-30 05:59:42 -04:00
CAddress addr_bind = GetBindAddress ( hSocket ) ;
2016-10-26 16:10:15 -03:00
2017-05-30 05:59:42 -04:00
CNode * pnode = new CNode ( id , nLocalServices , GetBestHeight ( ) , hSocket , addr , CalculateKeyedNetGroup ( addr ) , nonce , addr_bind , " " , true ) ;
2015-08-13 06:16:46 -03:00
pnode - > AddRef ( ) ;
pnode - > fWhitelisted = whitelisted ;
2017-07-06 14:08:23 -04:00
m_msgproc - > InitializeNode ( pnode ) ;
2015-08-13 06:00:10 -03:00
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " connection from %s accepted \n " , addr . ToString ( ) ) ;
2015-08-13 06:16:46 -03:00
{
LOCK ( cs_vNodes ) ;
vNodes . push_back ( pnode ) ;
2015-08-13 06:00:10 -03:00
}
}
2018-09-24 17:36:58 -03:00
void CConnman : : DisconnectNodes ( )
2010-08-29 12:58:15 -04:00
{
{
2018-09-24 17:30:53 -03:00
LOCK ( cs_vNodes ) ;
2018-05-10 13:23:22 -03:00
2018-09-24 17:36:58 -03:00
if ( ! fNetworkActive ) {
// Disconnect any connected nodes
for ( CNode * pnode : vNodes ) {
if ( ! pnode - > fDisconnect ) {
LogPrint ( BCLog : : NET , " Network not active, dropping peer=%d \n " , pnode - > GetId ( ) ) ;
pnode - > fDisconnect = true ;
2018-05-10 13:23:22 -03:00
}
}
2018-09-24 17:36:58 -03:00
}
2018-05-10 13:23:22 -03:00
2018-09-24 17:36:58 -03:00
// Disconnect unused nodes
std : : vector < CNode * > vNodesCopy = vNodes ;
for ( CNode * pnode : vNodesCopy )
{
if ( pnode - > fDisconnect )
2010-08-29 12:58:15 -04:00
{
2018-09-24 17:36:58 -03:00
// remove from vNodes
vNodes . erase ( remove ( vNodes . begin ( ) , vNodes . end ( ) , pnode ) , vNodes . end ( ) ) ;
2010-08-29 12:58:15 -04:00
2018-09-24 17:36:58 -03:00
// release outbound grant (if any)
pnode - > grantOutbound . Release ( ) ;
2012-04-04 11:01:57 -03:00
2018-09-24 17:36:58 -03:00
// close socket and cleanup
pnode - > CloseSocketDisconnect ( ) ;
2010-08-29 12:58:15 -04:00
2018-09-24 17:36:58 -03:00
// hold in disconnected pool until all refs are released
pnode - > Release ( ) ;
vNodesDisconnected . push_back ( pnode ) ;
2010-08-29 12:58:15 -04:00
}
2013-07-24 20:25:25 -04:00
}
2018-09-24 17:36:58 -03:00
}
{
// Delete disconnected nodes
std : : list < CNode * > vNodesDisconnectedCopy = vNodesDisconnected ;
for ( CNode * pnode : vNodesDisconnectedCopy )
2013-07-24 20:25:25 -04:00
{
2018-09-24 17:36:58 -03:00
// wait until threads are done using it
if ( pnode - > GetRefCount ( ) < = 0 ) {
bool fDelete = false ;
{
TRY_LOCK ( pnode - > cs_inventory , lockInv ) ;
if ( lockInv ) {
TRY_LOCK ( pnode - > cs_vSend , lockSend ) ;
if ( lockSend ) {
fDelete = true ;
2012-04-06 13:39:12 -03:00
}
}
2018-09-24 17:36:58 -03:00
}
if ( fDelete ) {
vNodesDisconnected . remove ( pnode ) ;
DeleteNode ( pnode ) ;
2010-08-29 12:58:15 -04:00
}
}
}
2018-09-24 17:36:58 -03:00
}
}
void CConnman : : NotifyNumConnectionsChanged ( )
{
size_t vNodesSize ;
{
LOCK ( cs_vNodes ) ;
vNodesSize = vNodes . size ( ) ;
}
if ( vNodesSize ! = nPrevNodeCount ) {
nPrevNodeCount = vNodesSize ;
if ( clientInterface )
clientInterface - > NotifyNumConnectionsChanged ( vNodesSize ) ;
}
}
2018-09-24 17:43:00 -03:00
void CConnman : : InactivityCheck ( CNode * pnode )
{
int64_t nTime = GetSystemTimeInSeconds ( ) ;
2018-11-15 20:30:26 -03:00
if ( nTime - pnode - > nTimeConnected > m_peer_connect_timeout )
2018-09-24 17:43:00 -03:00
{
if ( pnode - > nLastRecv = = 0 | | pnode - > nLastSend = = 0 )
{
2018-11-15 20:30:26 -03:00
LogPrint ( BCLog : : NET , " socket no message in first %i seconds, %d %d from %d \n " , m_peer_connect_timeout , pnode - > nLastRecv ! = 0 , pnode - > nLastSend ! = 0 , pnode - > GetId ( ) ) ;
2018-09-24 17:43:00 -03:00
pnode - > fDisconnect = true ;
}
else if ( nTime - pnode - > nLastSend > TIMEOUT_INTERVAL )
2016-11-26 00:29:55 -03:00
{
2018-09-24 17:43:00 -03:00
LogPrintf ( " socket sending timeout: %is \n " , nTime - pnode - > nLastSend ) ;
pnode - > fDisconnect = true ;
}
else if ( nTime - pnode - > nLastRecv > ( pnode - > nVersion > BIP0031_VERSION ? TIMEOUT_INTERVAL : 90 * 60 ) )
{
LogPrintf ( " socket receive timeout: %is \n " , nTime - pnode - > nLastRecv ) ;
pnode - > fDisconnect = true ;
}
else if ( pnode - > nPingNonceSent & & pnode - > nPingUsecStart + TIMEOUT_INTERVAL * 1000000 < GetTimeMicros ( ) )
{
LogPrintf ( " ping timeout: %fs \n " , 0.000001 * ( GetTimeMicros ( ) - pnode - > nPingUsecStart ) ) ;
pnode - > fDisconnect = true ;
2016-11-26 00:29:55 -03:00
}
2018-09-24 17:43:00 -03:00
else if ( ! pnode - > fSuccessfullyConnected )
{
LogPrint ( BCLog : : NET , " version handshake timeout from %d \n " , pnode - > GetId ( ) ) ;
pnode - > fDisconnect = true ;
2010-08-29 12:58:15 -04:00
}
2018-09-24 17:43:00 -03:00
}
}
2010-08-29 12:58:15 -04:00
2018-09-25 16:32:07 -03:00
bool CConnman : : GenerateSelectSet ( std : : set < SOCKET > & recv_set , std : : set < SOCKET > & send_set , std : : set < SOCKET > & error_set )
2018-09-24 17:36:58 -03:00
{
2018-09-24 18:03:17 -03:00
for ( const ListenSocket & hListenSocket : vhListenSocket ) {
2018-09-25 16:32:07 -03:00
recv_set . insert ( hListenSocket . socket ) ;
2018-09-24 18:03:17 -03:00
}
2014-06-24 03:09:45 -04:00
2018-09-24 18:03:17 -03:00
{
LOCK ( cs_vNodes ) ;
for ( CNode * pnode : vNodes )
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
// Implement the following logic:
// * If there is data to send, select() for sending data. As this only
// happens when optimistic write failed, we choose to first drain the
// write buffer in this case before receiving more. This avoids
// needlessly queueing received data, if the remote peer is not themselves
// receiving data. This means properly utilizing TCP flow control signalling.
// * Otherwise, if there is space left in the receive buffer, select() for
// receiving data.
// * Hand off all complete messages to the processor, to be handled without
// blocking here.
bool select_recv = ! pnode - > fPauseRecv ;
bool select_send ;
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
LOCK ( pnode - > cs_vSend ) ;
select_send = ! pnode - > vSendMsg . empty ( ) ;
}
2017-02-06 15:47:24 -03:00
2018-09-24 18:03:17 -03:00
LOCK ( pnode - > cs_hSocket ) ;
if ( pnode - > hSocket = = INVALID_SOCKET )
continue ;
2017-02-06 16:05:45 -03:00
2018-09-25 16:32:07 -03:00
error_set . insert ( pnode - > hSocket ) ;
2018-09-24 18:03:17 -03:00
if ( select_send ) {
2018-09-25 16:32:07 -03:00
send_set . insert ( pnode - > hSocket ) ;
2018-09-24 18:03:17 -03:00
continue ;
}
if ( select_recv ) {
2018-09-25 16:32:07 -03:00
recv_set . insert ( pnode - > hSocket ) ;
2010-08-29 12:58:15 -04:00
}
}
2018-09-24 18:03:17 -03:00
}
2010-08-29 12:58:15 -04:00
2018-09-25 16:32:07 -03:00
return ! recv_set . empty ( ) | | ! send_set . empty ( ) | | ! error_set . empty ( ) ;
}
2018-09-26 22:54:52 -03:00
# ifdef USE_POLL
void CConnman : : SocketEvents ( std : : set < SOCKET > & recv_set , std : : set < SOCKET > & send_set , std : : set < SOCKET > & error_set )
{
std : : set < SOCKET > recv_select_set , send_select_set , error_select_set ;
if ( ! GenerateSelectSet ( recv_select_set , send_select_set , error_select_set ) ) {
interruptNet . sleep_for ( std : : chrono : : milliseconds ( SELECT_TIMEOUT_MILLISECONDS ) ) ;
return ;
}
std : : unordered_map < SOCKET , struct pollfd > pollfds ;
for ( SOCKET socket_id : recv_select_set ) {
pollfds [ socket_id ] . fd = socket_id ;
pollfds [ socket_id ] . events | = POLLIN ;
}
for ( SOCKET socket_id : send_select_set ) {
pollfds [ socket_id ] . fd = socket_id ;
pollfds [ socket_id ] . events | = POLLOUT ;
}
for ( SOCKET socket_id : error_select_set ) {
pollfds [ socket_id ] . fd = socket_id ;
// These flags are ignored, but we set them for clarity
pollfds [ socket_id ] . events | = POLLERR | POLLHUP ;
}
std : : vector < struct pollfd > vpollfds ;
vpollfds . reserve ( pollfds . size ( ) ) ;
for ( auto it : pollfds ) {
vpollfds . push_back ( std : : move ( it . second ) ) ;
}
if ( poll ( vpollfds . data ( ) , vpollfds . size ( ) , SELECT_TIMEOUT_MILLISECONDS ) < 0 ) return ;
if ( interruptNet ) return ;
for ( struct pollfd pollfd_entry : vpollfds ) {
if ( pollfd_entry . revents & POLLIN ) recv_set . insert ( pollfd_entry . fd ) ;
if ( pollfd_entry . revents & POLLOUT ) send_set . insert ( pollfd_entry . fd ) ;
if ( pollfd_entry . revents & ( POLLERR | POLLHUP ) ) error_set . insert ( pollfd_entry . fd ) ;
}
}
# else
2018-09-26 22:51:46 -03:00
void CConnman : : SocketEvents ( std : : set < SOCKET > & recv_set , std : : set < SOCKET > & send_set , std : : set < SOCKET > & error_set )
2018-09-25 16:32:07 -03:00
{
std : : set < SOCKET > recv_select_set , send_select_set , error_select_set ;
if ( ! GenerateSelectSet ( recv_select_set , send_select_set , error_select_set ) ) {
interruptNet . sleep_for ( std : : chrono : : milliseconds ( SELECT_TIMEOUT_MILLISECONDS ) ) ;
return ;
}
//
// Find which sockets have data to receive
//
struct timeval timeout ;
timeout . tv_sec = 0 ;
timeout . tv_usec = SELECT_TIMEOUT_MILLISECONDS * 1000 ; // frequency to poll pnode->vSend
fd_set fdsetRecv ;
fd_set fdsetSend ;
fd_set fdsetError ;
FD_ZERO ( & fdsetRecv ) ;
FD_ZERO ( & fdsetSend ) ;
FD_ZERO ( & fdsetError ) ;
SOCKET hSocketMax = 0 ;
for ( SOCKET hSocket : recv_select_set ) {
FD_SET ( hSocket , & fdsetRecv ) ;
hSocketMax = std : : max ( hSocketMax , hSocket ) ;
}
for ( SOCKET hSocket : send_select_set ) {
FD_SET ( hSocket , & fdsetSend ) ;
hSocketMax = std : : max ( hSocketMax , hSocket ) ;
}
for ( SOCKET hSocket : error_select_set ) {
FD_SET ( hSocket , & fdsetError ) ;
hSocketMax = std : : max ( hSocketMax , hSocket ) ;
}
int nSelect = select ( hSocketMax + 1 , & fdsetRecv , & fdsetSend , & fdsetError , & timeout ) ;
2018-09-24 18:03:17 -03:00
if ( interruptNet )
return ;
2013-03-07 00:31:26 -03:00
2018-09-24 18:03:17 -03:00
if ( nSelect = = SOCKET_ERROR )
{
2018-09-25 16:32:07 -03:00
int nErr = WSAGetLastError ( ) ;
LogPrintf ( " socket select error %s \n " , NetworkErrorString ( nErr ) ) ;
for ( unsigned int i = 0 ; i < = hSocketMax ; i + + )
FD_SET ( i , & fdsetRecv ) ;
2018-09-24 18:03:17 -03:00
FD_ZERO ( & fdsetSend ) ;
FD_ZERO ( & fdsetError ) ;
2018-10-29 17:30:30 -03:00
if ( ! interruptNet . sleep_for ( std : : chrono : : milliseconds ( SELECT_TIMEOUT_MILLISECONDS ) ) )
2018-09-24 18:03:17 -03:00
return ;
}
2010-08-29 12:58:15 -04:00
2018-09-26 22:51:46 -03:00
for ( SOCKET hSocket : recv_select_set ) {
if ( FD_ISSET ( hSocket , & fdsetRecv ) ) {
recv_set . insert ( hSocket ) ;
}
}
for ( SOCKET hSocket : send_select_set ) {
if ( FD_ISSET ( hSocket , & fdsetSend ) ) {
send_set . insert ( hSocket ) ;
}
}
for ( SOCKET hSocket : error_select_set ) {
if ( FD_ISSET ( hSocket , & fdsetError ) ) {
error_set . insert ( hSocket ) ;
}
}
}
2018-09-26 22:54:52 -03:00
# endif
2018-09-26 22:51:46 -03:00
void CConnman : : SocketHandler ( )
{
std : : set < SOCKET > recv_set , send_set , error_set ;
SocketEvents ( recv_set , send_set , error_set ) ;
if ( interruptNet ) return ;
2018-09-24 18:03:17 -03:00
//
// Accept new connections
//
for ( const ListenSocket & hListenSocket : vhListenSocket )
{
2018-09-26 22:51:46 -03:00
if ( hListenSocket . socket ! = INVALID_SOCKET & & recv_set . count ( hListenSocket . socket ) > 0 )
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
AcceptConnection ( hListenSocket ) ;
2010-08-29 12:58:15 -04:00
}
2018-09-24 18:03:17 -03:00
}
//
// Service each socket
//
std : : vector < CNode * > vNodesCopy ;
{
LOCK ( cs_vNodes ) ;
vNodesCopy = vNodes ;
for ( CNode * pnode : vNodesCopy )
pnode - > AddRef ( ) ;
}
for ( CNode * pnode : vNodesCopy )
{
if ( interruptNet )
return ;
2010-08-29 12:58:15 -04:00
//
2018-09-24 18:03:17 -03:00
// Receive
2010-08-29 12:58:15 -04:00
//
2018-09-24 18:03:17 -03:00
bool recvSet = false ;
bool sendSet = false ;
bool errorSet = false ;
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
LOCK ( pnode - > cs_hSocket ) ;
if ( pnode - > hSocket = = INVALID_SOCKET )
continue ;
2018-09-26 22:51:46 -03:00
recvSet = recv_set . count ( pnode - > hSocket ) > 0 ;
sendSet = send_set . count ( pnode - > hSocket ) > 0 ;
errorSet = error_set . count ( pnode - > hSocket ) > 0 ;
2010-08-29 12:58:15 -04:00
}
2018-09-24 18:03:17 -03:00
if ( recvSet | | errorSet )
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
// typical socket buffer is 8K-64K
char pchBuf [ 0x10000 ] ;
int nBytes = 0 ;
2017-02-06 16:05:45 -03:00
{
LOCK ( pnode - > cs_hSocket ) ;
if ( pnode - > hSocket = = INVALID_SOCKET )
continue ;
2018-09-24 18:03:17 -03:00
nBytes = recv ( pnode - > hSocket , pchBuf , sizeof ( pchBuf ) , MSG_DONTWAIT ) ;
2017-02-06 16:05:45 -03:00
}
2018-09-24 18:03:17 -03:00
if ( nBytes > 0 )
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
bool notify = false ;
if ( ! pnode - > ReceiveMsgBytes ( pchBuf , nBytes , notify ) )
2017-03-06 13:54:08 -03:00
pnode - > CloseSocketDisconnect ( ) ;
2018-09-24 18:03:17 -03:00
RecordBytesRecv ( nBytes ) ;
if ( notify ) {
size_t nSizeAdded = 0 ;
auto it ( pnode - > vRecvMsg . begin ( ) ) ;
for ( ; it ! = pnode - > vRecvMsg . end ( ) ; + + it ) {
if ( ! it - > complete ( ) )
break ;
nSizeAdded + = it - > vRecv . size ( ) + CMessageHeader : : HEADER_SIZE ;
}
2017-03-06 13:54:08 -03:00
{
2018-09-24 18:03:17 -03:00
LOCK ( pnode - > cs_vProcessMsg ) ;
pnode - > vProcessMsg . splice ( pnode - > vProcessMsg . end ( ) , pnode - > vRecvMsg , pnode - > vRecvMsg . begin ( ) , it ) ;
pnode - > nProcessQueueSize + = nSizeAdded ;
pnode - > fPauseRecv = pnode - > nProcessQueueSize > nReceiveFloodSize ;
2010-08-29 12:58:15 -04:00
}
2018-09-24 18:03:17 -03:00
WakeMessageHandler ( ) ;
2010-08-29 12:58:15 -04:00
}
}
2018-09-24 18:03:17 -03:00
else if ( nBytes = = 0 )
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
// socket closed gracefully
if ( ! pnode - > fDisconnect ) {
LogPrint ( BCLog : : NET , " socket closed \n " ) ;
2016-04-18 22:44:42 -03:00
}
2018-09-24 18:03:17 -03:00
pnode - > CloseSocketDisconnect ( ) ;
2010-08-29 12:58:15 -04:00
}
2018-09-24 18:03:17 -03:00
else if ( nBytes < 0 )
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
// error
int nErr = WSAGetLastError ( ) ;
if ( nErr ! = WSAEWOULDBLOCK & & nErr ! = WSAEMSGSIZE & & nErr ! = WSAEINTR & & nErr ! = WSAEINPROGRESS )
2017-02-07 17:23:17 -03:00
{
2018-09-24 18:03:17 -03:00
if ( ! pnode - > fDisconnect )
LogPrintf ( " socket recv error %s \n " , NetworkErrorString ( nErr ) ) ;
pnode - > CloseSocketDisconnect ( ) ;
2017-02-07 17:23:17 -03:00
}
2010-08-29 12:58:15 -04:00
}
}
2018-09-24 18:03:17 -03:00
//
// Send
//
if ( sendSet )
2010-08-29 12:58:15 -04:00
{
2018-09-24 18:03:17 -03:00
LOCK ( pnode - > cs_vSend ) ;
size_t nBytes = SocketSendData ( pnode ) ;
if ( nBytes ) {
RecordBytesSent ( nBytes ) ;
}
2010-08-29 12:58:15 -04:00
}
2018-09-24 18:03:17 -03:00
InactivityCheck ( pnode ) ;
}
{
LOCK ( cs_vNodes ) ;
for ( CNode * pnode : vNodesCopy )
pnode - > Release ( ) ;
}
}
void CConnman : : ThreadSocketHandler ( )
{
while ( ! interruptNet )
{
DisconnectNodes ( ) ;
NotifyNumConnectionsChanged ( ) ;
SocketHandler ( ) ;
2010-08-29 12:58:15 -04:00
}
}
2016-12-31 04:05:21 -03:00
void CConnman : : WakeMessageHandler ( )
{
2016-12-31 04:05:26 -03:00
{
std : : lock_guard < std : : mutex > lock ( mutexMsgProc ) ;
fMsgProcWake = true ;
}
2016-12-31 04:05:21 -03:00
condMsgProc . notify_one ( ) ;
}
2010-08-29 12:58:15 -04:00
2011-03-26 09:01:27 -03:00
# ifdef USE_UPNP
2018-02-07 19:20:16 -03:00
static CThreadInterrupt g_upnp_interrupt ;
static std : : thread g_upnp_thread ;
2018-05-02 12:14:48 -03:00
static void ThreadMapPort ( )
2011-03-26 09:01:27 -03:00
{
2012-09-05 18:36:19 -03:00
std : : string port = strprintf ( " %u " , GetListenPort ( ) ) ;
2017-06-21 15:10:00 -04:00
const char * multicastif = nullptr ;
const char * minissdpdpath = nullptr ;
struct UPNPDev * devlist = nullptr ;
2011-03-26 09:01:27 -03:00
char lanaddr [ 64 ] ;
2011-12-10 13:52:50 -03:00
# ifndef UPNPDISCOVER_SUCCESS
/* miniupnpc 1.5 */
devlist = upnpDiscover ( 2000 , multicastif , minissdpdpath , 0 ) ;
2015-08-23 17:53:49 -03:00
# elif MINIUPNPC_API_VERSION < 14
2011-12-10 13:52:50 -03:00
/* miniupnpc 1.6 */
int error = 0 ;
2011-08-11 18:20:07 -04:00
devlist = upnpDiscover ( 2000 , multicastif , minissdpdpath , 0 , 0 , & error ) ;
2015-08-23 17:53:49 -03:00
# else
/* miniupnpc 1.9.20150730 */
int error = 0 ;
devlist = upnpDiscover ( 2000 , multicastif , minissdpdpath , 0 , 0 , 2 , & error ) ;
2011-12-10 13:52:50 -03:00
# endif
2011-03-26 09:01:27 -03:00
struct UPNPUrls urls ;
struct IGDdatas data ;
int r ;
2011-04-16 15:35:45 -03:00
r = UPNP_GetValidIGD ( devlist , & urls , & data , lanaddr , sizeof ( lanaddr ) ) ;
if ( r = = 1 )
2011-03-26 09:01:27 -03:00
{
2012-05-24 13:02:21 -04:00
if ( fDiscover ) {
2012-02-10 00:41:42 -03:00
char externalIPAddress [ 40 ] ;
r = UPNP_GetExternalIPAddress ( urls . controlURL , data . first . servicetype , externalIPAddress ) ;
if ( r ! = UPNPCOMMAND_SUCCESS )
2013-09-18 07:38:08 -03:00
LogPrintf ( " UPnP: GetExternalIPAddress() returned %d \n " , r ) ;
2012-02-10 00:41:42 -03:00
else
{
if ( externalIPAddress [ 0 ] )
{
2016-05-31 13:05:52 -04:00
CNetAddr resolved ;
if ( LookupHost ( externalIPAddress , resolved , false ) ) {
LogPrintf ( " UPnP: ExternalIPAddress = %s \n " , resolved . ToString ( ) . c_str ( ) ) ;
AddLocal ( resolved , LOCAL_UPNP ) ;
}
2012-02-10 00:41:42 -03:00
}
else
2013-09-18 07:38:08 -03:00
LogPrintf ( " UPnP: GetExternalIPAddress failed. \n " ) ;
2012-02-10 00:41:42 -03:00
}
}
2016-04-15 20:53:45 -03:00
std : : string strDesc = " Bitcoin " + FormatFullVersion ( ) ;
2011-08-11 18:20:07 -04:00
2018-02-07 19:20:16 -03:00
do {
2012-01-31 19:36:25 -03:00
# ifndef UPNPDISCOVER_SUCCESS
2018-02-07 19:20:16 -03:00
/* miniupnpc 1.5 */
r = UPNP_AddPortMapping ( urls . controlURL , data . first . servicetype ,
port . c_str ( ) , port . c_str ( ) , lanaddr , strDesc . c_str ( ) , " TCP " , 0 ) ;
2012-01-31 19:36:25 -03:00
# else
2018-02-07 19:20:16 -03:00
/* miniupnpc 1.6 */
r = UPNP_AddPortMapping ( urls . controlURL , data . first . servicetype ,
port . c_str ( ) , port . c_str ( ) , lanaddr , strDesc . c_str ( ) , " TCP " , 0 , " 0 " ) ;
2012-01-31 19:36:25 -03:00
# endif
2018-02-07 19:20:16 -03:00
if ( r ! = UPNPCOMMAND_SUCCESS )
LogPrintf ( " AddPortMapping(%s, %s, %s) failed with code %d (%s) \n " ,
port , port , lanaddr , r , strupnperror ( r ) ) ;
else
LogPrintf ( " UPnP Port Mapping successful. \n " ) ;
2011-03-26 09:01:27 -03:00
}
2018-02-07 19:20:16 -03:00
while ( g_upnp_interrupt . sleep_for ( std : : chrono : : minutes ( 20 ) ) ) ;
r = UPNP_DeletePortMapping ( urls . controlURL , data . first . servicetype , port . c_str ( ) , " TCP " , 0 ) ;
LogPrintf ( " UPNP_DeletePortMapping() returned: %d \n " , r ) ;
freeUPNPDevlist ( devlist ) ; devlist = nullptr ;
FreeUPNPUrls ( & urls ) ;
2011-03-26 09:01:27 -03:00
} else {
2013-09-18 07:38:08 -03:00
LogPrintf ( " No valid UPnP IGDs found \n " ) ;
2017-06-21 15:10:00 -04:00
freeUPNPDevlist ( devlist ) ; devlist = nullptr ;
2011-04-16 15:35:45 -03:00
if ( r ! = 0 )
FreeUPNPUrls ( & urls ) ;
2011-03-26 09:01:27 -03:00
}
}
2018-02-07 19:20:16 -03:00
void StartMapPort ( )
2011-03-26 09:01:27 -03:00
{
2018-02-07 19:20:16 -03:00
if ( ! g_upnp_thread . joinable ( ) ) {
assert ( ! g_upnp_interrupt ) ;
g_upnp_thread = std : : thread ( ( std : : bind ( & TraceThread < void ( * ) ( ) > , " upnp " , & ThreadMapPort ) ) ) ;
}
}
2013-03-07 00:31:26 -03:00
2018-02-07 19:20:16 -03:00
void InterruptMapPort ( )
{
if ( g_upnp_thread . joinable ( ) ) {
g_upnp_interrupt ( ) ;
2013-03-07 00:31:26 -03:00
}
2018-02-07 19:20:16 -03:00
}
void StopMapPort ( )
{
if ( g_upnp_thread . joinable ( ) ) {
g_upnp_thread . join ( ) ;
g_upnp_interrupt . reset ( ) ;
2011-03-26 09:01:27 -03:00
}
}
2013-03-07 00:31:26 -03:00
2011-08-09 12:38:17 -04:00
# else
2018-02-07 19:20:16 -03:00
void StartMapPort ( )
{
// Intentionally left blank.
}
void InterruptMapPort ( )
{
// Intentionally left blank.
}
void StopMapPort ( )
2011-08-09 12:38:17 -04:00
{
// Intentionally left blank.
}
2011-03-26 09:01:27 -03:00
# endif
2016-04-16 15:47:18 -03:00
void CConnman : : ThreadDNSAddressSeed ( )
2011-11-21 14:25:00 -03:00
{
2014-07-29 11:04:46 -04:00
// goal: only query DNS seeds if address need is acute
2016-10-17 20:11:35 -03:00
// Avoiding DNS seeds when we don't need them improves user privacy by
// creating fewer identifying DNS requests, reduces trust by giving seeds
// less influence on the network topology, and reduces traffic to the seeds.
2014-07-29 11:04:46 -04:00
if ( ( addrman . size ( ) > 0 ) & &
2017-08-01 15:17:40 -04:00
( ! gArgs . GetBoolArg ( " -forcednsseed " , DEFAULT_FORCEDNSSEED ) ) ) {
2016-12-27 19:12:44 -03:00
if ( ! interruptNet . sleep_for ( std : : chrono : : seconds ( 11 ) ) )
return ;
2014-07-29 11:04:46 -04:00
LOCK ( cs_vNodes ) ;
2016-10-17 20:11:35 -03:00
int nRelevant = 0 ;
2018-06-18 01:58:28 -04:00
for ( const CNode * pnode : vNodes ) {
2017-10-06 18:27:56 -03:00
nRelevant + = pnode - > fSuccessfullyConnected & & ! pnode - > fFeeler & & ! pnode - > fOneShot & & ! pnode - > m_manual_connection & & ! pnode - > fInbound ;
2016-10-17 20:11:35 -03:00
}
if ( nRelevant > = 2 ) {
2014-07-29 11:04:46 -04:00
LogPrintf ( " P2P peers available. Skipped DNS seeding. \n " ) ;
return ;
}
}
2017-10-19 18:32:45 -03:00
const std : : vector < std : : string > & vSeeds = Params ( ) . DNSSeeds ( ) ;
2011-03-09 00:40:50 -03:00
int found = 0 ;
2013-09-18 07:38:08 -03:00
LogPrintf ( " Loading addresses from DNS seeds (could take a while) \n " ) ;
2013-01-30 01:13:17 -03:00
2017-10-19 18:32:45 -03:00
for ( const std : : string & seed : vSeeds ) {
2017-04-14 17:29:57 -03:00
if ( interruptNet ) {
return ;
}
2013-01-30 01:13:17 -03:00
if ( HaveNameProxy ( ) ) {
2017-10-19 18:32:45 -03:00
AddOneShot ( seed ) ;
2013-01-30 01:13:17 -03:00
} else {
2016-04-15 20:53:45 -03:00
std : : vector < CNetAddr > vIPs ;
std : : vector < CAddress > vAdd ;
2017-10-04 18:59:30 -03:00
ServiceFlags requiredServiceBits = GetDesirableServiceFlags ( NODE_NONE ) ;
2017-10-19 18:32:45 -03:00
std : : string host = strprintf ( " x%x.%s " , requiredServiceBits , seed ) ;
2017-05-23 20:48:08 -04:00
CNetAddr resolveSource ;
if ( ! resolveSource . SetInternal ( host ) ) {
continue ;
}
2018-03-06 20:26:29 -03:00
unsigned int nMaxIPs = 256 ; // Limits number of IPs learned from a DNS seed
if ( LookupHost ( host . c_str ( ) , vIPs , nMaxIPs , true ) )
2013-01-30 01:13:17 -03:00
{
2017-06-01 21:18:57 -04:00
for ( const CNetAddr & ip : vIPs )
2011-05-02 10:34:42 -03:00
{
2013-01-30 01:13:17 -03:00
int nOneDay = 24 * 3600 ;
2016-05-21 17:55:22 -04:00
CAddress addr = CAddress ( CService ( ip , Params ( ) . GetDefaultPort ( ) ) , requiredServiceBits ) ;
2013-01-30 01:13:17 -03:00
addr . nTime = GetTime ( ) - 3 * nOneDay - GetRand ( 4 * nOneDay ) ; // use a random age between 3 and 7 days old
vAdd . push_back ( addr ) ;
found + + ;
2011-05-02 10:34:42 -03:00
}
2017-05-23 20:48:08 -04:00
addrman . Add ( vAdd , resolveSource ) ;
2017-10-19 18:32:45 -03:00
} else {
// We now avoid directly using results from DNS Seeds which do not support service bit filtering,
// instead using them as a oneshot to get nodes with our desired service bits.
AddOneShot ( seed ) ;
2016-04-12 21:38:06 -03:00
}
2011-03-09 00:40:50 -03:00
}
}
2013-09-18 07:38:08 -03:00
LogPrintf ( " %d addresses found from DNS seeds \n " , found ) ;
2011-03-09 00:40:50 -03:00
}
2010-08-29 12:58:15 -04:00
2011-11-21 14:25:00 -03:00
2016-04-16 18:43:11 -03:00
void CConnman : : DumpAddresses ( )
2012-01-04 19:39:45 -03:00
{
2013-04-13 02:13:08 -03:00
int64_t nStart = GetTimeMillis ( ) ;
2012-05-16 22:11:19 -04:00
2012-01-04 19:39:45 -03:00
CAddrDB adb ;
2012-05-16 22:11:19 -04:00
adb . Write ( addrman ) ;
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " Flushed %d addresses to peers.dat %dms \n " ,
2012-05-16 22:11:19 -04:00
addrman . size ( ) , GetTimeMillis ( ) - nStart ) ;
2012-01-04 19:39:45 -03:00
}
2010-08-29 12:58:15 -04:00
2016-04-16 15:47:18 -03:00
void CConnman : : ProcessOneShot ( )
2012-04-23 21:15:00 -03:00
{
2016-04-15 20:53:45 -03:00
std : : string strDest ;
2012-04-23 21:15:00 -03:00
{
LOCK ( cs_vOneShots ) ;
if ( vOneShots . empty ( ) )
return ;
strDest = vOneShots . front ( ) ;
vOneShots . pop_front ( ) ;
}
CAddress addr ;
2012-05-10 12:44:07 -04:00
CSemaphoreGrant grant ( * semOutbound , true ) ;
if ( grant ) {
2018-02-01 16:04:49 -03:00
OpenNetworkConnection ( addr , false , & grant , strDest . c_str ( ) , true ) ;
2012-05-10 12:44:07 -04:00
}
2012-04-23 21:15:00 -03:00
}
2017-10-23 14:36:15 -03:00
bool CConnman : : GetTryNewOutboundPeer ( )
{
return m_try_another_outbound_peer ;
}
void CConnman : : SetTryNewOutboundPeer ( bool flag )
{
m_try_another_outbound_peer = flag ;
2017-10-24 17:56:07 -03:00
LogPrint ( BCLog : : NET , " net: setting try another outbound peer=%s \n " , flag ? " true " : " false " ) ;
2017-10-23 14:36:15 -03:00
}
// Return the number of peers we have over our outbound connection limit
// Exclude peers that are marked for disconnect, or are going to be
// disconnected soon (eg one-shots and feelers)
// Also exclude peers that haven't finished initial connection handshake yet
// (so that we don't decide we're over our desired connection limit, and then
// evict some peer that has finished the handshake)
int CConnman : : GetExtraOutboundCount ( )
{
int nOutbound = 0 ;
{
LOCK ( cs_vNodes ) ;
2018-06-18 01:58:28 -04:00
for ( const CNode * pnode : vNodes ) {
2017-10-23 14:36:15 -03:00
if ( ! pnode - > fInbound & & ! pnode - > m_manual_connection & & ! pnode - > fFeeler & & ! pnode - > fDisconnect & & ! pnode - > fOneShot & & pnode - > fSuccessfullyConnected ) {
+ + nOutbound ;
}
}
}
return std : : max ( nOutbound - nMaxOutbound , 0 ) ;
}
2017-06-15 03:39:07 -04:00
void CConnman : : ThreadOpenConnections ( const std : : vector < std : : string > connect )
2010-08-29 12:58:15 -04:00
{
// Connect to specific addresses
2017-06-15 03:39:07 -04:00
if ( ! connect . empty ( ) )
2010-08-29 12:58:15 -04:00
{
2013-04-13 02:13:08 -03:00
for ( int64_t nLoop = 0 ; ; nLoop + + )
2010-08-29 12:58:15 -04:00
{
2012-04-23 21:15:00 -03:00
ProcessOneShot ( ) ;
2017-06-15 03:39:07 -04:00
for ( const std : : string & strAddr : connect )
2010-08-29 12:58:15 -04:00
{
2016-06-08 13:12:22 -04:00
CAddress addr ( CService ( ) , NODE_NONE ) ;
2017-10-04 18:59:30 -03:00
OpenNetworkConnection ( addr , false , nullptr , strAddr . c_str ( ) , false , false , true ) ;
2010-08-29 12:58:15 -04:00
for ( int i = 0 ; i < 10 & & i < nLoop ; i + + )
{
2016-12-27 19:12:44 -03:00
if ( ! interruptNet . sleep_for ( std : : chrono : : milliseconds ( 500 ) ) )
return ;
2010-08-29 12:58:15 -04:00
}
}
2016-12-27 19:12:44 -03:00
if ( ! interruptNet . sleep_for ( std : : chrono : : milliseconds ( 500 ) ) )
return ;
2010-08-29 12:58:15 -04:00
}
}
// Initiate network connections
2013-04-13 02:13:08 -03:00
int64_t nStart = GetTime ( ) ;
2016-06-17 00:10:07 -04:00
// Minimum time before next feeler connection (in microseconds).
int64_t nNextFeeler = PoissonNextSend ( nStart * 1000 * 1000 , FEELER_INTERVAL ) ;
2016-12-27 19:12:44 -03:00
while ( ! interruptNet )
2010-08-29 12:58:15 -04:00
{
2012-04-23 21:15:00 -03:00
ProcessOneShot ( ) ;
2016-12-27 19:12:44 -03:00
if ( ! interruptNet . sleep_for ( std : : chrono : : milliseconds ( 500 ) ) )
return ;
2012-02-15 17:17:15 -03:00
2012-05-10 12:44:07 -04:00
CSemaphoreGrant grant ( * semOutbound ) ;
2016-12-27 19:12:44 -03:00
if ( interruptNet )
return ;
2010-08-29 12:58:15 -04:00
2013-05-07 09:16:25 -04:00
// Add seed nodes if DNS seeds are all down (an infrastructure attack?).
if ( addrman . size ( ) = = 0 & & ( GetTime ( ) - nStart > 60 ) ) {
static bool done = false ;
if ( ! done ) {
2013-09-18 07:38:08 -03:00
LogPrintf ( " Adding fixed seed nodes as DNS doesn't seem to be available. \n " ) ;
2016-05-31 13:05:52 -04:00
CNetAddr local ;
2017-06-21 15:45:20 -04:00
local . SetInternal ( " fixedseeds " ) ;
2016-05-31 13:05:52 -04:00
addrman . Add ( convertSeed6 ( Params ( ) . FixedSeeds ( ) ) , local ) ;
2013-05-07 09:16:25 -04:00
done = true ;
2010-08-29 12:58:15 -04:00
}
}
//
// Choose an address to connect to based on most recently seen
//
CAddress addrConnect ;
2012-07-01 20:23:26 -04:00
// Only connect out to one peer per network group (/16 for IPv4).
2012-05-10 12:44:07 -04:00
int nOutbound = 0 ;
2016-04-15 20:53:45 -03:00
std : : set < std : : vector < unsigned char > > setConnected ;
2012-04-06 13:39:12 -03:00
{
LOCK ( cs_vNodes ) ;
2018-06-18 01:58:28 -04:00
for ( const CNode * pnode : vNodes ) {
2017-10-05 12:49:16 -03:00
if ( ! pnode - > fInbound & & ! pnode - > m_manual_connection ) {
2016-12-11 17:26:06 -03:00
// Netgroups for inbound and addnode peers are not excluded because our goal here
// is to not use multiple of our limited outbound slots on a single netgroup
// but inbound and addnode peers do not use our outbound slots. Inbound peers
// also have the added issue that they're attacker controlled and could be used
// to prevent us from connecting to particular hosts if we used them here.
2012-07-01 20:23:26 -04:00
setConnected . insert ( pnode - > addr . GetGroup ( ) ) ;
2012-05-10 12:44:07 -04:00
nOutbound + + ;
2012-07-01 20:23:26 -04:00
}
2012-05-10 12:44:07 -04:00
}
2012-04-06 13:39:12 -03:00
}
2010-08-29 12:58:15 -04:00
2016-06-17 00:10:07 -04:00
// Feeler Connections
//
// Design goals:
// * Increase the number of connectable addresses in the tried table.
//
// Method:
2017-05-27 06:00:37 -04:00
// * Choose a random address from new and attempt to connect to it if we can connect
2016-06-17 00:10:07 -04:00
// successfully it is added to tried.
2017-05-27 06:00:37 -04:00
// * Start attempting feeler connections only after node finishes making outbound
2016-06-17 00:10:07 -04:00
// connections.
// * Only make a feeler connection once every few minutes.
//
bool fFeeler = false ;
2017-10-23 14:36:15 -03:00
if ( nOutbound > = nMaxOutbound & & ! GetTryNewOutboundPeer ( ) ) {
2016-06-17 00:10:07 -04:00
int64_t nTime = GetTimeMicros ( ) ; // The current time right now (in microseconds).
if ( nTime > nNextFeeler ) {
nNextFeeler = PoissonNextSend ( nTime , FEELER_INTERVAL ) ;
fFeeler = true ;
} else {
continue ;
}
}
2011-10-04 00:41:47 -03:00
2016-10-27 14:55:39 -03:00
addrman . ResolveCollisions ( ) ;
2016-06-17 00:10:07 -04:00
int64_t nANow = GetAdjustedTime ( ) ;
2012-01-04 19:39:45 -03:00
int nTries = 0 ;
2016-12-27 19:12:44 -03:00
while ( ! interruptNet )
2010-08-29 12:58:15 -04:00
{
2016-10-27 14:55:39 -03:00
CAddrInfo addr = addrman . SelectTriedCollision ( ) ;
// SelectTriedCollision returns an invalid address if it is empty.
if ( ! fFeeler | | ! addr . IsValid ( ) ) {
addr = addrman . Select ( fFeeler ) ;
}
2010-08-29 12:58:15 -04:00
2012-01-04 19:39:45 -03:00
// if we selected an invalid address, restart
2012-03-31 12:58:25 -03:00
if ( ! addr . IsValid ( ) | | setConnected . count ( addr . GetGroup ( ) ) | | IsLocal ( addr ) )
2012-01-04 19:39:45 -03:00
break ;
2010-08-29 12:58:15 -04:00
2012-08-21 11:32:04 -04:00
// If we didn't find an appropriate destination after trying 100 addresses fetched from addrman,
// stop this loop, and let the outer loop run again (which sleeps, adds seed nodes, recalculates
// already-connected network ranges, ...) before trying new addrman addresses.
2012-01-04 19:39:45 -03:00
nTries + + ;
2012-08-21 11:32:04 -04:00
if ( nTries > 100 )
break ;
2010-08-29 12:58:15 -04:00
2019-01-09 21:41:37 -03:00
if ( ! IsReachable ( addr ) )
2012-05-04 10:46:22 -04:00
continue ;
2012-01-04 19:39:45 -03:00
// only consider very recently tried nodes after 30 failed attempts
if ( nANow - addr . nLastTry < 600 & & nTries < 30 )
continue ;
2017-10-04 18:59:30 -03:00
// for non-feelers, require all the services we'll want,
// for feelers, only require they be a full node (only because most
// SPV clients don't have a good address DB available)
if ( ! fFeeler & & ! HasAllDesirableServiceFlags ( addr . nServices ) ) {
continue ;
} else if ( fFeeler & & ! MayHaveUsefulAddressDB ( addr . nServices ) ) {
2015-11-16 20:20:49 -03:00
continue ;
2017-05-24 17:00:27 -04:00
}
2015-11-16 20:20:49 -03:00
2012-01-04 19:39:45 -03:00
// do not allow non-default ports, unless after 50 invalid addresses selected already
2013-05-07 09:16:25 -04:00
if ( addr . GetPort ( ) ! = Params ( ) . GetDefaultPort ( ) & & nTries < 50 )
2012-01-04 19:39:45 -03:00
continue ;
addrConnect = addr ;
break ;
2010-08-29 12:58:15 -04:00
}
2016-06-17 00:10:07 -04:00
if ( addrConnect . IsValid ( ) ) {
if ( fFeeler ) {
// Add small amount of random noise before connection to avoid synchronization.
int randsleep = GetRandInt ( FEELER_SLEEP_WINDOW * 1000 ) ;
2016-12-27 19:12:44 -03:00
if ( ! interruptNet . sleep_for ( std : : chrono : : milliseconds ( randsleep ) ) )
return ;
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " Making feeler connection to %s \n " , addrConnect . ToString ( ) ) ;
2016-06-17 00:10:07 -04:00
}
2017-08-07 01:36:37 -04:00
OpenNetworkConnection ( addrConnect , ( int ) setConnected . size ( ) > = std : : min ( nMaxConnections - 1 , 2 ) , & grant , nullptr , false , fFeeler ) ;
2016-06-17 00:10:07 -04:00
}
2010-08-29 12:58:15 -04:00
}
}
2016-04-16 19:12:58 -03:00
std : : vector < AddedNodeInfo > CConnman : : GetAddedNodeInfo ( )
2011-12-16 21:48:03 -03:00
{
2016-05-28 09:32:30 -04:00
std : : vector < AddedNodeInfo > ret ;
std : : list < std : : string > lAddresses ( 0 ) ;
2012-07-02 13:55:16 -04:00
{
LOCK ( cs_vAddedNodes ) ;
2016-05-28 09:32:30 -04:00
ret . reserve ( vAddedNodes . size ( ) ) ;
2017-07-20 05:32:47 -04:00
std : : copy ( vAddedNodes . cbegin ( ) , vAddedNodes . cend ( ) , std : : back_inserter ( lAddresses ) ) ;
2012-07-02 13:55:16 -04:00
}
2011-12-16 21:48:03 -03:00
2016-05-28 09:32:30 -04:00
// Build a map of all already connected addresses (by IP:port and by name) to inbound/outbound and resolved CService
std : : map < CService , bool > mapConnected ;
std : : map < std : : string , std : : pair < bool , CService > > mapConnectedByName ;
{
LOCK ( cs_vNodes ) ;
for ( const CNode * pnode : vNodes ) {
if ( pnode - > addr . IsValid ( ) ) {
mapConnected [ pnode - > addr ] = pnode - > fInbound ;
2012-07-02 13:55:16 -04:00
}
2017-02-06 14:04:34 -03:00
std : : string addrName = pnode - > GetAddrName ( ) ;
if ( ! addrName . empty ( ) ) {
mapConnectedByName [ std : : move ( addrName ) ] = std : : make_pair ( pnode - > fInbound , static_cast < const CService & > ( pnode - > addr ) ) ;
2012-04-19 12:38:03 -03:00
}
}
}
2017-06-01 21:18:57 -04:00
for ( const std : : string & strAddNode : lAddresses ) {
2016-08-04 16:37:49 -04:00
CService service ( LookupNumeric ( strAddNode . c_str ( ) , Params ( ) . GetDefaultPort ( ) ) ) ;
2018-01-26 07:48:56 -03:00
AddedNodeInfo addedNode { strAddNode , CService ( ) , false , false } ;
2016-05-28 09:32:30 -04:00
if ( service . IsValid ( ) ) {
// strAddNode is an IP:port
auto it = mapConnected . find ( service ) ;
if ( it ! = mapConnected . end ( ) ) {
2018-01-26 07:48:56 -03:00
addedNode . resolvedAddress = service ;
addedNode . fConnected = true ;
addedNode . fInbound = it - > second ;
2016-05-28 09:32:30 -04:00
}
} else {
// strAddNode is a name
auto it = mapConnectedByName . find ( strAddNode ) ;
if ( it ! = mapConnectedByName . end ( ) ) {
2018-01-26 07:48:56 -03:00
addedNode . resolvedAddress = it - > second . second ;
addedNode . fConnected = true ;
addedNode . fInbound = it - > second . first ;
2012-04-19 12:38:03 -03:00
}
}
2018-01-26 07:48:56 -03:00
ret . emplace_back ( std : : move ( addedNode ) ) ;
2012-04-19 12:38:03 -03:00
}
2016-05-28 09:32:30 -04:00
return ret ;
}
2016-04-16 15:47:18 -03:00
void CConnman : : ThreadOpenAddedConnections ( )
2016-05-28 09:32:30 -04:00
{
2016-12-11 01:39:26 -03:00
while ( true )
2011-12-16 21:48:03 -03:00
{
2016-12-11 01:39:26 -03:00
CSemaphoreGrant grant ( * semAddnode ) ;
2016-05-28 09:32:30 -04:00
std : : vector < AddedNodeInfo > vInfo = GetAddedNodeInfo ( ) ;
2016-12-11 01:39:26 -03:00
bool tried = false ;
2016-05-28 09:32:30 -04:00
for ( const AddedNodeInfo & info : vInfo ) {
if ( ! info . fConnected ) {
2016-12-11 01:39:26 -03:00
if ( ! grant . TryAcquire ( ) ) {
2018-03-18 11:26:45 -03:00
// If we've used up our semaphore and need a new one, let's not wait here since while we are waiting
2016-12-11 01:39:26 -03:00
// the addednodeinfo state might change.
break ;
}
tried = true ;
2017-06-23 12:29:50 -04:00
CAddress addr ( CService ( ) , NODE_NONE ) ;
OpenNetworkConnection ( addr , false , & grant , info . strAddedNode . c_str ( ) , false , false , true ) ;
2016-12-27 19:12:44 -03:00
if ( ! interruptNet . sleep_for ( std : : chrono : : milliseconds ( 500 ) ) )
return ;
2016-05-28 09:32:30 -04:00
}
2012-07-02 13:55:16 -04:00
}
2016-12-11 01:39:26 -03:00
// Retry every 60 seconds if a connection was attempted, otherwise two seconds
2017-01-07 14:49:14 -03:00
if ( ! interruptNet . sleep_for ( std : : chrono : : seconds ( tried ? 60 : 2 ) ) )
2016-12-27 19:12:44 -03:00
return ;
2011-12-16 21:48:03 -03:00
}
}
2012-07-25 20:48:39 -04:00
// if successful, this moves the passed grant to the constructed node
2018-02-01 16:04:49 -03:00
void CConnman : : OpenNetworkConnection ( const CAddress & addrConnect , bool fCountFailure , CSemaphoreGrant * grantOutbound , const char * pszDest , bool fOneShot , bool fFeeler , bool manual_connection )
2010-08-29 12:58:15 -04:00
{
//
// Initiate outbound network connection
//
2016-12-27 19:12:44 -03:00
if ( interruptNet ) {
2018-02-01 16:04:49 -03:00
return ;
2016-12-27 19:12:44 -03:00
}
2013-03-25 22:33:25 -03:00
if ( ! fNetworkActive ) {
2018-02-01 16:04:49 -03:00
return ;
2013-03-25 22:33:25 -03:00
}
2014-05-24 05:14:52 -04:00
if ( ! pszDest ) {
2012-02-12 09:45:24 -03:00
if ( IsLocal ( addrConnect ) | |
2017-10-05 14:10:58 -03:00
FindNode ( static_cast < CNetAddr > ( addrConnect ) ) | | ( m_banman & & m_banman - > IsBanned ( addrConnect ) ) | |
2014-07-21 09:00:42 -04:00
FindNode ( addrConnect . ToStringIPPort ( ) ) )
2018-02-01 16:04:49 -03:00
return ;
2015-05-31 10:44:22 -03:00
} else if ( FindNode ( std : : string ( pszDest ) ) )
2018-02-01 16:04:49 -03:00
return ;
2010-08-29 12:58:15 -04:00
2018-02-28 17:15:01 -03:00
CNode * pnode = ConnectNode ( addrConnect , pszDest , fCountFailure , manual_connection ) ;
2013-03-07 00:31:26 -03:00
2010-08-29 12:58:15 -04:00
if ( ! pnode )
2018-02-01 16:04:49 -03:00
return ;
2012-05-10 12:44:07 -04:00
if ( grantOutbound )
grantOutbound - > MoveTo ( pnode - > grantOutbound ) ;
2012-04-23 21:15:00 -03:00
if ( fOneShot )
pnode - > fOneShot = true ;
2016-06-17 00:10:07 -04:00
if ( fFeeler )
pnode - > fFeeler = true ;
2017-10-05 12:49:16 -03:00
if ( manual_connection )
pnode - > m_manual_connection = true ;
2010-08-29 12:58:15 -04:00
2017-07-06 14:08:23 -04:00
m_msgproc - > InitializeNode ( pnode ) ;
2017-01-24 18:51:22 -03:00
{
LOCK ( cs_vNodes ) ;
vNodes . push_back ( pnode ) ;
}
2010-08-29 12:58:15 -04:00
}
2016-04-16 15:47:18 -03:00
void CConnman : : ThreadMessageHandler ( )
2010-08-29 12:58:15 -04:00
{
2016-12-27 19:12:44 -03:00
while ( ! flagInterruptMsgProc )
2010-08-29 12:58:15 -04:00
{
2016-04-15 20:53:45 -03:00
std : : vector < CNode * > vNodesCopy ;
2010-08-29 12:58:15 -04:00
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_vNodes ) ;
2010-08-29 12:58:15 -04:00
vNodesCopy = vNodes ;
2017-06-01 21:18:57 -04:00
for ( CNode * pnode : vNodesCopy ) {
2010-08-29 12:58:15 -04:00
pnode - > AddRef ( ) ;
2013-04-04 19:43:04 -03:00
}
2010-08-29 12:58:15 -04:00
}
2016-12-31 04:05:26 -03:00
bool fMoreWork = false ;
2013-11-15 08:24:34 -03:00
2017-06-01 21:18:57 -04:00
for ( CNode * pnode : vNodesCopy )
2010-08-29 12:58:15 -04:00
{
2013-02-28 21:41:28 -03:00
if ( pnode - > fDisconnect )
continue ;
2010-08-29 12:58:15 -04:00
// Receive messages
2017-07-06 14:08:23 -04:00
bool fMoreNodeWork = m_msgproc - > ProcessMessages ( pnode , flagInterruptMsgProc ) ;
2016-12-31 04:05:36 -03:00
fMoreWork | = ( fMoreNodeWork & & ! pnode - > fPauseSend ) ;
2016-12-27 19:12:44 -03:00
if ( flagInterruptMsgProc )
return ;
2010-08-29 12:58:15 -04:00
// Send messages
2012-04-06 13:39:12 -03:00
{
2017-01-13 01:08:52 -03:00
LOCK ( pnode - > cs_sendProcessing ) ;
2018-07-01 22:26:47 -04:00
m_msgproc - > SendMessages ( pnode ) ;
2012-04-06 13:39:12 -03:00
}
2017-07-06 13:40:09 -04:00
2016-12-27 19:12:44 -03:00
if ( flagInterruptMsgProc )
return ;
2010-08-29 12:58:15 -04:00
}
{
2012-04-06 13:39:12 -03:00
LOCK ( cs_vNodes ) ;
2017-06-01 21:18:57 -04:00
for ( CNode * pnode : vNodesCopy )
2010-08-29 12:58:15 -04:00
pnode - > Release ( ) ;
}
2013-11-15 08:24:34 -03:00
2017-11-08 19:07:40 -03:00
WAIT_LOCK ( mutexMsgProc , lock ) ;
2016-12-31 04:05:26 -03:00
if ( ! fMoreWork ) {
condMsgProc . wait_until ( lock , std : : chrono : : steady_clock : : now ( ) + std : : chrono : : milliseconds ( 100 ) , [ this ] { return fMsgProcWake ; } ) ;
2016-12-27 19:12:44 -03:00
}
2016-12-31 04:05:26 -03:00
fMsgProcWake = false ;
2010-08-29 12:58:15 -04:00
}
}
2016-04-16 16:46:00 -03:00
bool CConnman : : BindListenPort ( const CService & addrBind , std : : string & strError , bool fWhitelisted )
2010-08-29 12:58:15 -04:00
{
strError = " " ;
int nOne = 1 ;
// Create socket for listening for incoming connections
2012-05-11 09:28:59 -04:00
struct sockaddr_storage sockaddr ;
socklen_t len = sizeof ( sockaddr ) ;
if ( ! addrBind . GetSockAddr ( ( struct sockaddr * ) & sockaddr , & len ) )
{
2014-05-24 05:14:52 -04:00
strError = strprintf ( " Error: Bind address family for %s not supported " , addrBind . ToString ( ) ) ;
2014-01-16 12:15:27 -03:00
LogPrintf ( " %s \n " , strError ) ;
2012-05-11 09:28:59 -04:00
return false ;
}
2017-10-02 17:31:37 -03:00
SOCKET hListenSocket = CreateSocket ( addrBind ) ;
2010-08-29 12:58:15 -04:00
if ( hListenSocket = = INVALID_SOCKET )
{
2014-05-08 08:15:19 -04:00
strError = strprintf ( " Error: Couldn't open socket for incoming connections (socket returned error %s) " , NetworkErrorString ( WSAGetLastError ( ) ) ) ;
2014-01-16 12:15:27 -03:00
LogPrintf ( " %s \n " , strError ) ;
2010-08-29 12:58:15 -04:00
return false ;
}
2018-01-26 07:48:56 -03:00
2010-08-29 12:58:15 -04:00
// Allow binding if the port is still in TIME_WAIT state after
2015-08-20 16:50:13 -03:00
// the program was closed and restarted.
2018-01-26 07:48:56 -03:00
setsockopt ( hListenSocket , SOL_SOCKET , SO_REUSEADDR , ( sockopt_arg_type ) & nOne , sizeof ( int ) ) ;
2010-08-29 12:58:15 -04:00
2012-05-11 09:28:59 -04:00
// some systems don't have IPV6_V6ONLY but are always v6only; others do have the option
// and enable it by default or not. Try to enable it, if possible.
if ( addrBind . IsIPv6 ( ) ) {
# ifdef IPV6_V6ONLY
2018-01-26 07:48:56 -03:00
setsockopt ( hListenSocket , IPPROTO_IPV6 , IPV6_V6ONLY , ( sockopt_arg_type ) & nOne , sizeof ( int ) ) ;
2013-07-13 07:05:04 -04:00
# endif
2012-05-11 09:28:59 -04:00
# ifdef WIN32
2014-06-24 03:03:18 -04:00
int nProtLevel = PROTECTION_LEVEL_UNRESTRICTED ;
setsockopt ( hListenSocket , IPPROTO_IPV6 , IPV6_PROTECTION_LEVEL , ( const char * ) & nProtLevel , sizeof ( int ) ) ;
2012-05-11 09:28:59 -04:00
# endif
}
if ( : : bind ( hListenSocket , ( struct sockaddr * ) & sockaddr , len ) = = SOCKET_ERROR )
2010-08-29 12:58:15 -04:00
{
int nErr = WSAGetLastError ( ) ;
if ( nErr = = WSAEADDRINUSE )
2015-12-09 07:53:12 -03:00
strError = strprintf ( _ ( " Unable to bind to %s on this computer. %s is probably already running. " ) , addrBind . ToString ( ) , _ ( PACKAGE_NAME ) ) ;
2010-08-29 12:58:15 -04:00
else
2014-05-08 08:15:19 -04:00
strError = strprintf ( _ ( " Unable to bind to %s on this computer (bind returned error %s) " ) , addrBind . ToString ( ) , NetworkErrorString ( nErr ) ) ;
2014-01-16 12:15:27 -03:00
LogPrintf ( " %s \n " , strError ) ;
2014-07-17 16:33:58 -04:00
CloseSocket ( hListenSocket ) ;
2010-08-29 12:58:15 -04:00
return false ;
}
2014-01-16 12:15:27 -03:00
LogPrintf ( " Bound to %s \n " , addrBind . ToString ( ) ) ;
2010-08-29 12:58:15 -04:00
// Listen for incoming connections
if ( listen ( hListenSocket , SOMAXCONN ) = = SOCKET_ERROR )
{
2014-05-08 08:15:19 -04:00
strError = strprintf ( _ ( " Error: Listening for incoming connections failed (listen returned error %s) " ) , NetworkErrorString ( WSAGetLastError ( ) ) ) ;
2014-01-16 12:15:27 -03:00
LogPrintf ( " %s \n " , strError ) ;
2014-07-17 16:33:58 -04:00
CloseSocket ( hListenSocket ) ;
2010-08-29 12:58:15 -04:00
return false ;
}
2014-06-21 07:34:36 -04:00
vhListenSocket . push_back ( ListenSocket ( hListenSocket , fWhitelisted ) ) ;
2012-05-11 09:28:59 -04:00
2014-06-21 07:34:36 -04:00
if ( addrBind . IsRoutable ( ) & & fDiscover & & ! fWhitelisted )
2012-05-11 09:28:59 -04:00
AddLocal ( addrBind , LOCAL_BIND ) ;
2010-08-29 12:58:15 -04:00
return true ;
}
2018-02-07 19:42:39 -03:00
void Discover ( )
2010-08-29 12:58:15 -04:00
{
2012-05-24 13:02:21 -04:00
if ( ! fDiscover )
2012-02-19 16:44:35 -03:00
return ;
2010-08-29 12:58:15 -04:00
2011-10-07 12:02:21 -03:00
# ifdef WIN32
2012-07-25 20:48:39 -04:00
// Get local host IP
2014-11-13 11:23:15 -03:00
char pszHostName [ 256 ] = " " ;
2010-08-29 12:58:15 -04:00
if ( gethostname ( pszHostName , sizeof ( pszHostName ) ) ! = SOCKET_ERROR )
{
2016-04-15 20:53:45 -03:00
std : : vector < CNetAddr > vaddr ;
2016-04-12 21:23:16 -03:00
if ( LookupHost ( pszHostName , vaddr , 0 , true ) )
2012-04-30 19:44:59 -04:00
{
2017-06-01 21:18:57 -04:00
for ( const CNetAddr & addr : vaddr )
2012-04-30 19:44:59 -04:00
{
2014-11-13 11:20:57 -03:00
if ( AddLocal ( addr , LOCAL_IF ) )
LogPrintf ( " %s: %s - %s \n " , __func__ , pszHostName , addr . ToString ( ) ) ;
2012-04-30 19:44:59 -04:00
}
}
2010-08-29 12:58:15 -04:00
}
2018-07-19 14:21:05 -04:00
# elif (HAVE_DECL_GETIFADDRS && HAVE_DECL_FREEIFADDRS)
2010-08-29 12:58:15 -04:00
// Get local host ip
struct ifaddrs * myaddrs ;
if ( getifaddrs ( & myaddrs ) = = 0 )
{
2017-08-07 01:36:37 -04:00
for ( struct ifaddrs * ifa = myaddrs ; ifa ! = nullptr ; ifa = ifa - > ifa_next )
2010-08-29 12:58:15 -04:00
{
2017-08-07 01:36:37 -04:00
if ( ifa - > ifa_addr = = nullptr ) continue ;
2010-08-29 12:58:15 -04:00
if ( ( ifa - > ifa_flags & IFF_UP ) = = 0 ) continue ;
if ( strcmp ( ifa - > ifa_name , " lo " ) = = 0 ) continue ;
if ( strcmp ( ifa - > ifa_name , " lo0 " ) = = 0 ) continue ;
if ( ifa - > ifa_addr - > sa_family = = AF_INET )
{
struct sockaddr_in * s4 = ( struct sockaddr_in * ) ( ifa - > ifa_addr ) ;
2012-02-12 09:45:24 -03:00
CNetAddr addr ( s4 - > sin_addr ) ;
2012-03-31 12:58:25 -03:00
if ( AddLocal ( addr , LOCAL_IF ) )
2014-11-13 11:20:57 -03:00
LogPrintf ( " %s: IPv4 %s: %s \n " , __func__ , ifa - > ifa_name , addr . ToString ( ) ) ;
2010-08-29 12:58:15 -04:00
}
else if ( ifa - > ifa_addr - > sa_family = = AF_INET6 )
{
struct sockaddr_in6 * s6 = ( struct sockaddr_in6 * ) ( ifa - > ifa_addr ) ;
2012-02-12 09:45:24 -03:00
CNetAddr addr ( s6 - > sin6_addr ) ;
2012-03-31 12:58:25 -03:00
if ( AddLocal ( addr , LOCAL_IF ) )
2014-11-13 11:20:57 -03:00
LogPrintf ( " %s: IPv6 %s: %s \n " , __func__ , ifa - > ifa_name , addr . ToString ( ) ) ;
2010-08-29 12:58:15 -04:00
}
}
freeifaddrs ( myaddrs ) ;
}
# endif
2012-02-19 16:44:35 -03:00
}
2013-03-25 22:33:25 -03:00
void CConnman : : SetNetworkActive ( bool active )
{
2016-12-25 17:19:40 -03:00
LogPrint ( BCLog : : NET , " SetNetworkActive: %s \n " , active ) ;
2013-03-25 22:33:25 -03:00
2017-07-14 10:00:19 -04:00
if ( fNetworkActive = = active ) {
return ;
}
fNetworkActive = active ;
2013-03-25 22:33:25 -03:00
2013-03-25 23:07:06 -03:00
uiInterface . NotifyNetworkActiveChanged ( fNetworkActive ) ;
2013-03-25 22:33:25 -03:00
}
2016-09-09 07:48:10 -03:00
CConnman : : CConnman ( uint64_t nSeed0In , uint64_t nSeed1In ) : nSeed0 ( nSeed0In ) , nSeed1 ( nSeed1In )
2016-04-16 15:47:18 -03:00
{
2017-10-23 14:36:15 -03:00
SetTryNewOutboundPeer ( false ) ;
[net] Fix use of uninitialized value in getnetworkinfo(const JSONRPCRequest& request)
When running test_bitcoin under Valgrind I found the following issue:
```
$ valgrind src/test/test_bitcoin
...
==10465== Use of uninitialised value of size 8
==10465== at 0x6D09B61: ??? (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D0B1BB: std::ostreambuf_iterator<char, std::char_traits<char> > std::num_put<char, std::ostreambuf_iterator<char, std::char_traits<char> > >::_M_insert_int<unsigned long>(std::ostreambuf_iterator<char, std::char_traits<char> >, std::ios_base&, char, unsigned long) const (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D0B36C: std::num_put<char, std::ostreambuf_iterator<char, std::char_traits<char> > >::do_put(std::ostreambuf_iterator<char, std::char_traits<char> >, std::ios_base&, char, unsigned long) const (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D17699: std::ostream& std::ostream::_M_insert<unsigned long>(unsigned long) (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x4CAAD7: operator<< (ostream:171)
==10465== by 0x4CAAD7: formatValue<ServiceFlags> (tinyformat.h:345)
==10465== by 0x4CAAD7: void tinyformat::detail::FormatArg::formatImpl<ServiceFlags>(std::ostream&, char const*, char const*, int, void const*) (tinyformat.h:523)
==10465== by 0x1924D4: format (tinyformat.h:510)
==10465== by 0x1924D4: tinyformat::detail::formatImpl(std::ostream&, char const*, tinyformat::detail::FormatArg const*, int) (tinyformat.h:803)
==10465== by 0x553A55: vformat (tinyformat.h:947)
==10465== by 0x553A55: format<ServiceFlags> (tinyformat.h:957)
==10465== by 0x553A55: std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > tinyformat::format<ServiceFlags>(char const*, ServiceFlags const&) (tinyformat.h:966)
==10465== by 0x54C952: getnetworkinfo(JSONRPCRequest const&) (net.cpp:462)
==10465== by 0x28EDB5: CallRPC(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >) (rpc_tests.cpp:31)
==10465== by 0x293947: rpc_tests::rpc_togglenetwork::test_method() (rpc_tests.cpp:88)
==10465== by 0x2950E5: rpc_tests::rpc_togglenetwork_invoker() (rpc_tests.cpp:84)
==10465== by 0x182496: invoke<void (*)()> (callback.hpp:56)
==10465== by 0x182496: boost::unit_test::ut_detail::callback0_impl_t<boost::unit_test::ut_detail::unused, void (*)()>::invoke() (callback.hpp:89)
...
```
The read of the uninitialized variable nLocalServices is triggered by g_connman->GetLocalServices()
in getnetworkinfo(const JSONRPCRequest& request) (net.cpp:462):
```c++
UniValue getnetworkinfo(const JSONRPCRequest& request)
{
...
if(g_connman)
obj.push_back(Pair("localservices", strprintf("%016x", g_connman->GetLocalServices())));
...
}
```
The reason for the uninitialized nLocalServices is that CConnman::Start(...) is not called
by the tests, and hence the initialization normally performed by CConnman::Start(...) is
not done.
This commit adds a method Init(const Options& connOptions) which is called by both the
constructor and CConnman::Start(...). This method initializes nLocalServices and the other
relevant values from the supplied Options object.
2017-08-02 08:02:36 -04:00
Options connOptions ;
Init ( connOptions ) ;
2016-04-16 15:47:18 -03:00
}
2016-04-17 21:20:34 -03:00
NodeId CConnman : : GetNewNodeId ( )
2016-04-16 18:43:11 -03:00
{
2016-04-17 21:20:34 -03:00
return nLastNodeId . fetch_add ( 1 , std : : memory_order_relaxed ) ;
}
2016-04-16 18:43:11 -03:00
2017-06-01 06:34:02 -04:00
bool CConnman : : Bind ( const CService & addr , unsigned int flags ) {
2019-01-09 21:41:37 -03:00
if ( ! ( flags & BF_EXPLICIT ) & & ! IsReachable ( addr ) )
2017-06-01 06:34:02 -04:00
return false ;
std : : string strError ;
if ( ! BindListenPort ( addr , strError , ( flags & BF_WHITELIST ) ! = 0 ) ) {
if ( ( flags & BF_REPORT_ERROR ) & & clientInterface ) {
clientInterface - > ThreadSafeMessageBox ( strError , " " , CClientUIInterface : : MSG_ERROR ) ;
}
return false ;
}
return true ;
}
bool CConnman : : InitBinds ( const std : : vector < CService > & binds , const std : : vector < CService > & whiteBinds ) {
bool fBound = false ;
for ( const auto & addrBind : binds ) {
fBound | = Bind ( addrBind , ( BF_EXPLICIT | BF_REPORT_ERROR ) ) ;
}
for ( const auto & addrBind : whiteBinds ) {
fBound | = Bind ( addrBind , ( BF_EXPLICIT | BF_REPORT_ERROR | BF_WHITELIST ) ) ;
}
if ( binds . empty ( ) & & whiteBinds . empty ( ) ) {
struct in_addr inaddr_any ;
inaddr_any . s_addr = INADDR_ANY ;
2018-07-16 02:29:27 -04:00
struct in6_addr inaddr6_any = IN6ADDR_ANY_INIT ;
fBound | = Bind ( CService ( inaddr6_any , GetListenPort ( ) ) , BF_NONE ) ;
2017-06-01 06:34:02 -04:00
fBound | = Bind ( CService ( inaddr_any , GetListenPort ( ) ) , ! fBound ? BF_REPORT_ERROR : BF_NONE ) ;
}
return fBound ;
}
[net] Fix use of uninitialized value in getnetworkinfo(const JSONRPCRequest& request)
When running test_bitcoin under Valgrind I found the following issue:
```
$ valgrind src/test/test_bitcoin
...
==10465== Use of uninitialised value of size 8
==10465== at 0x6D09B61: ??? (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D0B1BB: std::ostreambuf_iterator<char, std::char_traits<char> > std::num_put<char, std::ostreambuf_iterator<char, std::char_traits<char> > >::_M_insert_int<unsigned long>(std::ostreambuf_iterator<char, std::char_traits<char> >, std::ios_base&, char, unsigned long) const (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D0B36C: std::num_put<char, std::ostreambuf_iterator<char, std::char_traits<char> > >::do_put(std::ostreambuf_iterator<char, std::char_traits<char> >, std::ios_base&, char, unsigned long) const (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D17699: std::ostream& std::ostream::_M_insert<unsigned long>(unsigned long) (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x4CAAD7: operator<< (ostream:171)
==10465== by 0x4CAAD7: formatValue<ServiceFlags> (tinyformat.h:345)
==10465== by 0x4CAAD7: void tinyformat::detail::FormatArg::formatImpl<ServiceFlags>(std::ostream&, char const*, char const*, int, void const*) (tinyformat.h:523)
==10465== by 0x1924D4: format (tinyformat.h:510)
==10465== by 0x1924D4: tinyformat::detail::formatImpl(std::ostream&, char const*, tinyformat::detail::FormatArg const*, int) (tinyformat.h:803)
==10465== by 0x553A55: vformat (tinyformat.h:947)
==10465== by 0x553A55: format<ServiceFlags> (tinyformat.h:957)
==10465== by 0x553A55: std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > tinyformat::format<ServiceFlags>(char const*, ServiceFlags const&) (tinyformat.h:966)
==10465== by 0x54C952: getnetworkinfo(JSONRPCRequest const&) (net.cpp:462)
==10465== by 0x28EDB5: CallRPC(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >) (rpc_tests.cpp:31)
==10465== by 0x293947: rpc_tests::rpc_togglenetwork::test_method() (rpc_tests.cpp:88)
==10465== by 0x2950E5: rpc_tests::rpc_togglenetwork_invoker() (rpc_tests.cpp:84)
==10465== by 0x182496: invoke<void (*)()> (callback.hpp:56)
==10465== by 0x182496: boost::unit_test::ut_detail::callback0_impl_t<boost::unit_test::ut_detail::unused, void (*)()>::invoke() (callback.hpp:89)
...
```
The read of the uninitialized variable nLocalServices is triggered by g_connman->GetLocalServices()
in getnetworkinfo(const JSONRPCRequest& request) (net.cpp:462):
```c++
UniValue getnetworkinfo(const JSONRPCRequest& request)
{
...
if(g_connman)
obj.push_back(Pair("localservices", strprintf("%016x", g_connman->GetLocalServices())));
...
}
```
The reason for the uninitialized nLocalServices is that CConnman::Start(...) is not called
by the tests, and hence the initialization normally performed by CConnman::Start(...) is
not done.
This commit adds a method Init(const Options& connOptions) which is called by both the
constructor and CConnman::Start(...). This method initializes nLocalServices and the other
relevant values from the supplied Options object.
2017-08-02 08:02:36 -04:00
bool CConnman : : Start ( CScheduler & scheduler , const Options & connOptions )
2016-04-17 21:20:34 -03:00
{
[net] Fix use of uninitialized value in getnetworkinfo(const JSONRPCRequest& request)
When running test_bitcoin under Valgrind I found the following issue:
```
$ valgrind src/test/test_bitcoin
...
==10465== Use of uninitialised value of size 8
==10465== at 0x6D09B61: ??? (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D0B1BB: std::ostreambuf_iterator<char, std::char_traits<char> > std::num_put<char, std::ostreambuf_iterator<char, std::char_traits<char> > >::_M_insert_int<unsigned long>(std::ostreambuf_iterator<char, std::char_traits<char> >, std::ios_base&, char, unsigned long) const (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D0B36C: std::num_put<char, std::ostreambuf_iterator<char, std::char_traits<char> > >::do_put(std::ostreambuf_iterator<char, std::char_traits<char> >, std::ios_base&, char, unsigned long) const (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x6D17699: std::ostream& std::ostream::_M_insert<unsigned long>(unsigned long) (in /usr/lib/x86_64-linux-gnu/libstdc++.so.6.0.21)
==10465== by 0x4CAAD7: operator<< (ostream:171)
==10465== by 0x4CAAD7: formatValue<ServiceFlags> (tinyformat.h:345)
==10465== by 0x4CAAD7: void tinyformat::detail::FormatArg::formatImpl<ServiceFlags>(std::ostream&, char const*, char const*, int, void const*) (tinyformat.h:523)
==10465== by 0x1924D4: format (tinyformat.h:510)
==10465== by 0x1924D4: tinyformat::detail::formatImpl(std::ostream&, char const*, tinyformat::detail::FormatArg const*, int) (tinyformat.h:803)
==10465== by 0x553A55: vformat (tinyformat.h:947)
==10465== by 0x553A55: format<ServiceFlags> (tinyformat.h:957)
==10465== by 0x553A55: std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> > tinyformat::format<ServiceFlags>(char const*, ServiceFlags const&) (tinyformat.h:966)
==10465== by 0x54C952: getnetworkinfo(JSONRPCRequest const&) (net.cpp:462)
==10465== by 0x28EDB5: CallRPC(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char> >) (rpc_tests.cpp:31)
==10465== by 0x293947: rpc_tests::rpc_togglenetwork::test_method() (rpc_tests.cpp:88)
==10465== by 0x2950E5: rpc_tests::rpc_togglenetwork_invoker() (rpc_tests.cpp:84)
==10465== by 0x182496: invoke<void (*)()> (callback.hpp:56)
==10465== by 0x182496: boost::unit_test::ut_detail::callback0_impl_t<boost::unit_test::ut_detail::unused, void (*)()>::invoke() (callback.hpp:89)
...
```
The read of the uninitialized variable nLocalServices is triggered by g_connman->GetLocalServices()
in getnetworkinfo(const JSONRPCRequest& request) (net.cpp:462):
```c++
UniValue getnetworkinfo(const JSONRPCRequest& request)
{
...
if(g_connman)
obj.push_back(Pair("localservices", strprintf("%016x", g_connman->GetLocalServices())));
...
}
```
The reason for the uninitialized nLocalServices is that CConnman::Start(...) is not called
by the tests, and hence the initialization normally performed by CConnman::Start(...) is
not done.
This commit adds a method Init(const Options& connOptions) which is called by both the
constructor and CConnman::Start(...). This method initializes nLocalServices and the other
relevant values from the supplied Options object.
2017-08-02 08:02:36 -04:00
Init ( connOptions ) ;
2017-11-21 14:02:57 -03:00
{
LOCK ( cs_totalBytesRecv ) ;
nTotalBytesRecv = 0 ;
}
{
LOCK ( cs_totalBytesSent ) ;
nTotalBytesSent = 0 ;
nMaxOutboundTotalBytesSentInCycle = 0 ;
nMaxOutboundCycleStartTime = 0 ;
}
2016-04-18 22:44:42 -03:00
2017-06-01 06:34:02 -04:00
if ( fListen & & ! InitBinds ( connOptions . vBinds , connOptions . vWhiteBinds ) ) {
if ( clientInterface ) {
clientInterface - > ThreadSafeMessageBox (
_ ( " Failed to listen on any port. Use -listen=0 if you want this. " ) ,
" " , CClientUIInterface : : MSG_ERROR ) ;
}
return false ;
}
2017-05-27 06:00:37 -04:00
for ( const auto & strDest : connOptions . vSeedNodes ) {
AddOneShot ( strDest ) ;
}
2017-02-25 21:41:51 -03:00
if ( clientInterface ) {
clientInterface - > InitMessage ( _ ( " Loading P2P addresses... " ) ) ;
}
2015-07-03 04:46:17 -03:00
// Load addresses from peers.dat
2014-09-18 09:08:43 -03:00
int64_t nStart = GetTimeMillis ( ) ;
{
CAddrDB adb ;
2015-07-03 04:46:17 -03:00
if ( adb . Read ( addrman ) )
LogPrintf ( " Loaded %i addresses from peers.dat %dms \n " , addrman . size ( ) , GetTimeMillis ( ) - nStart ) ;
2016-02-02 06:45:11 -03:00
else {
2016-03-16 13:54:30 -03:00
addrman . Clear ( ) ; // Addrman can be in an inconsistent state after failure, reset it
2014-09-18 09:08:43 -03:00
LogPrintf ( " Invalid or missing peers.dat; recreating \n " ) ;
2016-02-02 06:45:11 -03:00
DumpAddresses ( ) ;
}
2014-09-18 09:08:43 -03:00
}
2015-06-19 10:27:37 -03:00
2016-07-22 10:01:12 -04:00
uiInterface . InitMessage ( _ ( " Starting network threads... " ) ) ;
2014-09-18 09:08:43 -03:00
fAddressesInitialized = true ;
2017-08-07 01:36:37 -04:00
if ( semOutbound = = nullptr ) {
2012-05-10 12:44:07 -04:00
// initialize semaphore
2017-08-15 02:46:56 -03:00
semOutbound = MakeUnique < CSemaphore > ( std : : min ( ( nMaxOutbound + nMaxFeeler ) , nMaxConnections ) ) ;
2012-05-10 12:44:07 -04:00
}
2017-08-07 01:36:37 -04:00
if ( semAddnode = = nullptr ) {
2016-12-11 01:39:26 -03:00
// initialize semaphore
2017-08-15 02:46:56 -03:00
semAddnode = MakeUnique < CSemaphore > ( nMaxAddnode ) ;
2016-12-11 01:39:26 -03:00
}
2012-05-10 12:44:07 -04:00
2010-08-29 12:58:15 -04:00
//
// Start threads
//
2017-07-06 13:40:09 -04:00
assert ( m_msgproc ) ;
2016-12-27 19:13:31 -03:00
InterruptSocks5 ( false ) ;
2016-12-27 19:12:44 -03:00
interruptNet . reset ( ) ;
flagInterruptMsgProc = false ;
2010-08-29 12:58:15 -04:00
2016-12-31 04:05:26 -03:00
{
2017-11-08 19:07:40 -03:00
LOCK ( mutexMsgProc ) ;
2016-12-31 04:05:26 -03:00
fMsgProcWake = false ;
}
2016-12-27 19:11:57 -03:00
// Send and receive from sockets, accept connections
2016-12-27 19:12:44 -03:00
threadSocketHandler = std : : thread ( & TraceThread < std : : function < void ( ) > > , " net " , std : : function < void ( ) > ( std : : bind ( & CConnman : : ThreadSocketHandler , this ) ) ) ;
2010-08-29 12:58:15 -04:00
2017-08-01 15:17:40 -04:00
if ( ! gArgs . GetBoolArg ( " -dnsseed " , true ) )
2013-09-18 07:38:08 -03:00
LogPrintf ( " DNS seeding disabled \n " ) ;
2011-11-21 14:25:00 -03:00
else
2016-12-27 19:12:44 -03:00
threadDNSAddressSeed = std : : thread ( & TraceThread < std : : function < void ( ) > > , " dnsseed " , std : : function < void ( ) > ( std : : bind ( & CConnman : : ThreadDNSAddressSeed , this ) ) ) ;
2010-08-29 12:58:15 -04:00
2011-12-16 21:48:03 -03:00
// Initiate outbound connections from -addnode
2016-12-27 19:12:44 -03:00
threadOpenAddedConnections = std : : thread ( & TraceThread < std : : function < void ( ) > > , " addcon " , std : : function < void ( ) > ( std : : bind ( & CConnman : : ThreadOpenAddedConnections , this ) ) ) ;
2011-12-16 21:48:03 -03:00
2017-06-15 03:39:07 -04:00
if ( connOptions . m_use_addrman_outgoing & & ! connOptions . m_specified_outgoing . empty ( ) ) {
if ( clientInterface ) {
clientInterface - > ThreadSafeMessageBox (
_ ( " Cannot provide specific connections and have addrman find outgoing connections at the same. " ) ,
" " , CClientUIInterface : : MSG_ERROR ) ;
}
return false ;
}
if ( connOptions . m_use_addrman_outgoing | | ! connOptions . m_specified_outgoing . empty ( ) )
threadOpenConnections = std : : thread ( & TraceThread < std : : function < void ( ) > > , " opencon " , std : : function < void ( ) > ( std : : bind ( & CConnman : : ThreadOpenConnections , this , connOptions . m_specified_outgoing ) ) ) ;
2010-08-29 12:58:15 -04:00
// Process messages
2016-12-27 19:12:44 -03:00
threadMessageHandler = std : : thread ( & TraceThread < std : : function < void ( ) > > , " msghand " , std : : function < void ( ) > ( std : : bind ( & CConnman : : ThreadMessageHandler , this ) ) ) ;
2010-08-29 12:58:15 -04:00
2016-04-16 18:43:11 -03:00
// Dump network addresses
2017-10-05 13:46:54 -03:00
scheduler . scheduleEvery ( std : : bind ( & CConnman : : DumpAddresses , this ) , DUMP_PEERS_INTERVAL * 1000 ) ;
2016-04-16 18:43:11 -03:00
2016-04-16 15:47:18 -03:00
return true ;
2010-08-29 12:58:15 -04:00
}
2017-10-05 14:41:45 -03:00
BanMan : : BanMan ( fs : : path ban_file , CClientUIInterface * client_interface , int64_t default_ban_time ) : clientInterface ( client_interface ) , m_ban_db ( std : : move ( ban_file ) ) , m_default_ban_time ( default_ban_time )
2017-10-05 14:10:58 -03:00
{
if ( clientInterface ) clientInterface - > InitMessage ( _ ( " Loading banlist... " ) ) ;
int64_t nStart = GetTimeMillis ( ) ;
setBannedIsDirty = false ;
banmap_t banmap ;
2017-10-05 14:35:20 -03:00
if ( m_ban_db . Read ( banmap ) ) {
2017-10-05 14:10:58 -03:00
SetBanned ( banmap ) ; // thread save setter
SetBannedSetDirty ( false ) ; // no need to write down, just read data
SweepBanned ( ) ; // sweep out unused entries
LogPrint ( BCLog : : NET , " Loaded %d banned node ips/subnets from banlist.dat %dms \n " ,
banmap . size ( ) , GetTimeMillis ( ) - nStart ) ;
} else {
LogPrintf ( " Invalid or missing banlist.dat; recreating \n " ) ;
SetBannedSetDirty ( true ) ; // force write
DumpBanlist ( ) ;
}
}
BanMan : : ~ BanMan ( )
{
DumpBanlist ( ) ;
}
2010-08-29 12:58:15 -04:00
class CNetCleanup
{
public :
2014-05-24 05:14:52 -04:00
CNetCleanup ( ) { }
2010-08-29 12:58:15 -04:00
~ CNetCleanup ( )
{
2011-10-07 12:02:21 -03:00
# ifdef WIN32
2010-08-29 12:58:15 -04:00
// Shutdown Windows Sockets
WSACleanup ( ) ;
# endif
}
}
instance_of_cnetcleanup ;
2012-08-12 23:26:30 -04:00
2016-12-27 19:12:44 -03:00
void CConnman : : Interrupt ( )
2016-04-16 15:47:18 -03:00
{
2016-12-27 19:12:44 -03:00
{
std : : lock_guard < std : : mutex > lock ( mutexMsgProc ) ;
flagInterruptMsgProc = true ;
}
condMsgProc . notify_all ( ) ;
interruptNet ( ) ;
2016-12-27 19:13:31 -03:00
InterruptSocks5 ( true ) ;
2016-12-27 19:12:44 -03:00
2017-03-08 16:55:28 -03:00
if ( semOutbound ) {
for ( int i = 0 ; i < ( nMaxOutbound + nMaxFeeler ) ; i + + ) {
2016-04-16 15:47:18 -03:00
semOutbound - > post ( ) ;
2017-03-08 16:55:28 -03:00
}
}
2017-03-08 16:41:57 -03:00
2017-03-08 16:55:28 -03:00
if ( semAddnode ) {
for ( int i = 0 ; i < nMaxAddnode ; i + + ) {
2017-03-08 16:41:57 -03:00
semAddnode - > post ( ) ;
2017-03-08 16:55:28 -03:00
}
}
2016-12-27 19:12:44 -03:00
}
void CConnman : : Stop ( )
{
if ( threadMessageHandler . joinable ( ) )
threadMessageHandler . join ( ) ;
if ( threadOpenConnections . joinable ( ) )
threadOpenConnections . join ( ) ;
if ( threadOpenAddedConnections . joinable ( ) )
threadOpenAddedConnections . join ( ) ;
if ( threadDNSAddressSeed . joinable ( ) )
threadDNSAddressSeed . join ( ) ;
if ( threadSocketHandler . joinable ( ) )
threadSocketHandler . join ( ) ;
2016-04-16 15:47:18 -03:00
2016-04-16 18:43:11 -03:00
if ( fAddressesInitialized )
{
2017-10-05 13:46:54 -03:00
DumpAddresses ( ) ;
2016-04-16 18:43:11 -03:00
fAddressesInitialized = false ;
}
2016-04-16 15:47:18 -03:00
// Close sockets
2017-06-01 21:18:57 -04:00
for ( CNode * pnode : vNodes )
2017-02-06 16:05:45 -03:00
pnode - > CloseSocketDisconnect ( ) ;
2017-06-01 21:18:57 -04:00
for ( ListenSocket & hListenSocket : vhListenSocket )
2016-04-16 15:47:18 -03:00
if ( hListenSocket . socket ! = INVALID_SOCKET )
if ( ! CloseSocket ( hListenSocket . socket ) )
LogPrintf ( " CloseSocket(hListenSocket) failed with error %s \n " , NetworkErrorString ( WSAGetLastError ( ) ) ) ;
// clean up some globals (to help leak detection)
2017-06-01 21:18:57 -04:00
for ( CNode * pnode : vNodes ) {
2016-05-24 18:59:16 -04:00
DeleteNode ( pnode ) ;
}
2017-06-01 21:18:57 -04:00
for ( CNode * pnode : vNodesDisconnected ) {
2016-05-24 18:59:16 -04:00
DeleteNode ( pnode ) ;
}
2016-04-16 15:47:18 -03:00
vNodes . clear ( ) ;
vNodesDisconnected . clear ( ) ;
vhListenSocket . clear ( ) ;
2017-08-09 10:07:22 -04:00
semOutbound . reset ( ) ;
semAddnode . reset ( ) ;
2016-04-16 15:47:18 -03:00
}
2016-05-24 18:59:16 -04:00
void CConnman : : DeleteNode ( CNode * pnode )
{
assert ( pnode ) ;
bool fUpdateConnectionTime = false ;
2017-07-06 13:40:09 -04:00
m_msgproc - > FinalizeNode ( pnode - > GetId ( ) , fUpdateConnectionTime ) ;
if ( fUpdateConnectionTime ) {
2016-05-24 18:59:16 -04:00
addrman . Connected ( pnode - > addr ) ;
2017-07-06 13:40:09 -04:00
}
2016-05-24 18:59:16 -04:00
delete pnode ;
}
2016-04-16 15:47:18 -03:00
CConnman : : ~ CConnman ( )
{
2016-12-27 19:12:44 -03:00
Interrupt ( ) ;
2016-09-13 15:42:55 -03:00
Stop ( ) ;
2016-04-16 15:47:18 -03:00
}
2012-08-12 23:26:30 -04:00
2016-04-16 18:43:11 -03:00
size_t CConnman : : GetAddressCount ( ) const
{
return addrman . size ( ) ;
}
void CConnman : : SetServices ( const CService & addr , ServiceFlags nServices )
{
addrman . SetServices ( addr , nServices ) ;
}
void CConnman : : MarkAddressGood ( const CAddress & addr )
{
addrman . Good ( addr ) ;
}
void CConnman : : AddNewAddresses ( const std : : vector < CAddress > & vAddr , const CAddress & addrFrom , int64_t nTimePenalty )
{
addrman . Add ( vAddr , addrFrom , nTimePenalty ) ;
}
std : : vector < CAddress > CConnman : : GetAddresses ( )
{
return addrman . GetAddr ( ) ;
}
2016-04-16 19:12:58 -03:00
bool CConnman : : AddNode ( const std : : string & strNode )
{
LOCK ( cs_vAddedNodes ) ;
2017-07-20 05:32:47 -04:00
for ( const std : : string & it : vAddedNodes ) {
if ( strNode = = it ) return false ;
2016-04-16 19:12:58 -03:00
}
vAddedNodes . push_back ( strNode ) ;
return true ;
}
bool CConnman : : RemoveAddedNode ( const std : : string & strNode )
{
LOCK ( cs_vAddedNodes ) ;
for ( std : : vector < std : : string > : : iterator it = vAddedNodes . begin ( ) ; it ! = vAddedNodes . end ( ) ; + + it ) {
if ( strNode = = * it ) {
vAddedNodes . erase ( it ) ;
return true ;
}
}
return false ;
}
2016-04-16 19:30:03 -03:00
size_t CConnman : : GetNodeCount ( NumConnections flags )
{
LOCK ( cs_vNodes ) ;
if ( flags = = CConnman : : CONNECTIONS_ALL ) // Shortcut if we want total
return vNodes . size ( ) ;
int nNum = 0 ;
2017-07-20 05:32:47 -04:00
for ( const auto & pnode : vNodes ) {
if ( flags & ( pnode - > fInbound ? CONNECTIONS_IN : CONNECTIONS_OUT ) ) {
2016-04-16 19:30:03 -03:00
nNum + + ;
2017-07-20 05:32:47 -04:00
}
}
2016-04-16 19:30:03 -03:00
return nNum ;
}
void CConnman : : GetNodeStats ( std : : vector < CNodeStats > & vstats )
{
vstats . clear ( ) ;
LOCK ( cs_vNodes ) ;
vstats . reserve ( vNodes . size ( ) ) ;
2017-07-20 05:32:47 -04:00
for ( CNode * pnode : vNodes ) {
2017-02-06 13:44:38 -03:00
vstats . emplace_back ( ) ;
pnode - > copyStats ( vstats . back ( ) ) ;
2016-04-16 19:30:03 -03:00
}
}
bool CConnman : : DisconnectNode ( const std : : string & strNode )
{
2017-01-24 18:50:27 -03:00
LOCK ( cs_vNodes ) ;
2016-04-16 19:30:03 -03:00
if ( CNode * pnode = FindNode ( strNode ) ) {
pnode - > fDisconnect = true ;
return true ;
}
return false ;
}
2017-10-04 19:25:34 -03:00
bool CConnman : : DisconnectNode ( const CSubNet & subnet )
{
bool disconnected = false ;
LOCK ( cs_vNodes ) ;
for ( CNode * pnode : vNodes ) {
if ( subnet . Match ( pnode - > addr ) ) {
pnode - > fDisconnect = true ;
disconnected = true ;
}
}
return disconnected ;
}
bool CConnman : : DisconnectNode ( const CNetAddr & addr )
{
return DisconnectNode ( CSubNet ( addr ) ) ;
}
2016-04-16 19:30:03 -03:00
bool CConnman : : DisconnectNode ( NodeId id )
{
LOCK ( cs_vNodes ) ;
for ( CNode * pnode : vNodes ) {
2017-04-11 13:13:55 -03:00
if ( id = = pnode - > GetId ( ) ) {
2016-04-16 19:30:03 -03:00
pnode - > fDisconnect = true ;
return true ;
}
}
return false ;
}
2016-04-18 22:44:42 -03:00
void CConnman : : RecordBytesRecv ( uint64_t bytes )
2013-08-22 12:09:32 -04:00
{
LOCK ( cs_totalBytesRecv ) ;
nTotalBytesRecv + = bytes ;
}
2016-04-18 22:44:42 -03:00
void CConnman : : RecordBytesSent ( uint64_t bytes )
2013-08-22 12:09:32 -04:00
{
LOCK ( cs_totalBytesSent ) ;
nTotalBytesSent + = bytes ;
2015-09-02 12:03:27 -03:00
uint64_t now = GetTime ( ) ;
if ( nMaxOutboundCycleStartTime + nMaxOutboundTimeframe < now )
{
// timeframe expired, reset cycle
nMaxOutboundCycleStartTime = now ;
nMaxOutboundTotalBytesSentInCycle = 0 ;
}
// TODO, exclude whitebind peers
nMaxOutboundTotalBytesSentInCycle + = bytes ;
}
2016-04-18 22:44:42 -03:00
void CConnman : : SetMaxOutboundTarget ( uint64_t limit )
2015-09-02 12:03:27 -03:00
{
LOCK ( cs_totalBytesSent ) ;
nMaxOutboundLimit = limit ;
}
2016-04-18 22:44:42 -03:00
uint64_t CConnman : : GetMaxOutboundTarget ( )
2015-09-02 12:03:27 -03:00
{
LOCK ( cs_totalBytesSent ) ;
return nMaxOutboundLimit ;
}
2016-04-18 22:44:42 -03:00
uint64_t CConnman : : GetMaxOutboundTimeframe ( )
2015-09-02 12:03:27 -03:00
{
LOCK ( cs_totalBytesSent ) ;
return nMaxOutboundTimeframe ;
}
2016-04-18 22:44:42 -03:00
uint64_t CConnman : : GetMaxOutboundTimeLeftInCycle ( )
2015-09-02 12:03:27 -03:00
{
LOCK ( cs_totalBytesSent ) ;
if ( nMaxOutboundLimit = = 0 )
return 0 ;
if ( nMaxOutboundCycleStartTime = = 0 )
return nMaxOutboundTimeframe ;
uint64_t cycleEndTime = nMaxOutboundCycleStartTime + nMaxOutboundTimeframe ;
uint64_t now = GetTime ( ) ;
return ( cycleEndTime < now ) ? 0 : cycleEndTime - GetTime ( ) ;
}
2016-04-18 22:44:42 -03:00
void CConnman : : SetMaxOutboundTimeframe ( uint64_t timeframe )
2015-09-02 12:03:27 -03:00
{
LOCK ( cs_totalBytesSent ) ;
if ( nMaxOutboundTimeframe ! = timeframe )
{
// reset measure-cycle in case of changing
// the timeframe
nMaxOutboundCycleStartTime = GetTime ( ) ;
}
nMaxOutboundTimeframe = timeframe ;
}
2016-04-18 22:44:42 -03:00
bool CConnman : : OutboundTargetReached ( bool historicalBlockServingLimit )
2015-09-02 12:03:27 -03:00
{
LOCK ( cs_totalBytesSent ) ;
if ( nMaxOutboundLimit = = 0 )
return false ;
if ( historicalBlockServingLimit )
{
2016-01-17 08:03:56 -03:00
// keep a large enough buffer to at least relay each block once
2015-09-02 12:03:27 -03:00
uint64_t timeLeftInCycle = GetMaxOutboundTimeLeftInCycle ( ) ;
2016-01-03 14:54:50 -03:00
uint64_t buffer = timeLeftInCycle / 600 * MAX_BLOCK_SERIALIZED_SIZE ;
2015-09-02 12:03:27 -03:00
if ( buffer > = nMaxOutboundLimit | | nMaxOutboundTotalBytesSentInCycle > = nMaxOutboundLimit - buffer )
return true ;
}
else if ( nMaxOutboundTotalBytesSentInCycle > = nMaxOutboundLimit )
return true ;
return false ;
}
2016-04-18 22:44:42 -03:00
uint64_t CConnman : : GetOutboundTargetBytesLeft ( )
2015-09-02 12:03:27 -03:00
{
LOCK ( cs_totalBytesSent ) ;
if ( nMaxOutboundLimit = = 0 )
return 0 ;
return ( nMaxOutboundTotalBytesSentInCycle > = nMaxOutboundLimit ) ? 0 : nMaxOutboundLimit - nMaxOutboundTotalBytesSentInCycle ;
2013-08-22 12:09:32 -04:00
}
2016-04-18 22:44:42 -03:00
uint64_t CConnman : : GetTotalBytesRecv ( )
2013-08-22 12:09:32 -04:00
{
LOCK ( cs_totalBytesRecv ) ;
return nTotalBytesRecv ;
}
2016-04-18 22:44:42 -03:00
uint64_t CConnman : : GetTotalBytesSent ( )
2013-08-22 12:09:32 -04:00
{
LOCK ( cs_totalBytesSent ) ;
return nTotalBytesSent ;
}
2013-10-28 03:28:00 -03:00
2016-04-19 01:04:58 -03:00
ServiceFlags CConnman : : GetLocalServices ( ) const
{
return nLocalServices ;
}
2016-05-24 16:42:17 -04:00
void CConnman : : SetBestHeight ( int height )
{
nBestHeight . store ( height , std : : memory_order_release ) ;
}
int CConnman : : GetBestHeight ( ) const
{
return nBestHeight . load ( std : : memory_order_acquire ) ;
}
2016-04-19 01:01:19 -03:00
unsigned int CConnman : : GetReceiveFloodSize ( ) const { return nReceiveFloodSize ; }
2014-08-20 23:17:21 -04:00
2019-01-05 08:11:04 -03:00
CNode : : CNode ( NodeId idIn , ServiceFlags nLocalServicesIn , int nMyStartingHeightIn , SOCKET hSocketIn , const CAddress & addrIn , uint64_t nKeyedNetGroupIn , uint64_t nLocalHostNonceIn , const CAddress & addrBindIn , const std : : string & addrNameIn , bool fInboundIn )
: nTimeConnected ( GetSystemTimeInSeconds ( ) ) ,
2016-05-25 09:38:32 -04:00
addr ( addrIn ) ,
2017-05-30 05:59:42 -04:00
addrBind ( addrBindIn ) ,
2016-10-31 18:06:15 -03:00
fInbound ( fInboundIn ) ,
2016-09-09 07:48:10 -03:00
nKeyedNetGroup ( nKeyedNetGroupIn ) ,
2015-07-19 16:43:34 -03:00
addrKnown ( 5000 , 0.001 ) ,
2016-10-31 18:06:15 -03:00
filterInventoryKnown ( 50000 , 0.000001 ) ,
2017-04-11 13:11:27 -03:00
id ( idIn ) ,
2016-10-26 16:10:15 -03:00
nLocalHostNonce ( nLocalHostNonceIn ) ,
2016-10-31 18:06:15 -03:00
nLocalServices ( nLocalServicesIn ) ,
2019-01-05 08:11:04 -03:00
nMyStartingHeight ( nMyStartingHeightIn )
2014-08-20 23:17:21 -04:00
{
hSocket = hSocketIn ;
addrName = addrNameIn = = " " ? addr . ToStringIPPort ( ) : addrNameIn ;
strSubVer = " " ;
2014-12-15 05:11:16 -03:00
hashContinue = uint256 ( ) ;
2015-11-29 06:52:51 -03:00
filterInventoryKnown . reset ( ) ;
2017-08-15 02:46:56 -03:00
pfilter = MakeUnique < CBloomFilter > ( ) ;
2016-04-17 21:21:58 -03:00
2017-06-01 21:18:57 -04:00
for ( const std : : string & msg : getAllNetMessageTypes ( ) )
2015-12-07 11:31:32 -03:00
mapRecvBytesPerMsgCmd [ msg ] = 0 ;
2015-08-25 11:30:31 -03:00
mapRecvBytesPerMsgCmd [ NET_MESSAGE_COMMAND_OTHER ] = 0 ;
2014-08-20 23:17:21 -04:00
2016-12-25 17:19:40 -03:00
if ( fLogIPs ) {
LogPrint ( BCLog : : NET , " Added connection to %s peer=%d \n " , addrName , id ) ;
} else {
LogPrint ( BCLog : : NET , " Added connection peer=%d \n " , id ) ;
}
2014-08-20 23:17:21 -04:00
}
CNode : : ~ CNode ( )
{
CloseSocket ( hSocket ) ;
}
void CNode : : AskFor ( const CInv & inv )
{
2015-11-22 22:54:23 -03:00
if ( mapAskFor . size ( ) > MAPASKFOR_MAX_SZ | | setAskFor . size ( ) > SETASKFOR_MAX_SZ )
2014-09-09 04:18:05 -03:00
return ;
2015-11-22 22:54:23 -03:00
// a peer may not have multiple non-responded queue positions for a single inv item
2014-07-16 17:31:41 -04:00
if ( ! setAskFor . insert ( inv . hash ) . second )
return ;
2014-08-20 23:17:21 -04:00
// We're using mapAskFor as a priority queue,
// the key is the earliest time the request can be sent
int64_t nRequestTime ;
2016-04-11 13:52:29 -03:00
limitedmap < uint256 , int64_t > : : const_iterator it = mapAlreadyAskedFor . find ( inv . hash ) ;
2014-08-20 23:17:21 -04:00
if ( it ! = mapAlreadyAskedFor . end ( ) )
nRequestTime = it - > second ;
else
nRequestTime = 0 ;
2018-02-28 12:46:31 -03:00
LogPrint ( BCLog : : NET , " askfor %s %d (%s) peer=%d \n " , inv . ToString ( ) , nRequestTime , FormatISO8601Time ( nRequestTime / 1000000 ) , id ) ;
2014-08-20 23:17:21 -04:00
// Make sure not to reuse time indexes to keep things in the same order
int64_t nNow = GetTimeMicros ( ) - 1000000 ;
static int64_t nLastTime ;
+ + nLastTime ;
nNow = std : : max ( nNow , nLastTime ) ;
nLastTime = nNow ;
// Each retry is 2 minutes after the last
nRequestTime = std : : max ( nRequestTime + 2 * 60 * 1000000 , nNow ) ;
if ( it ! = mapAlreadyAskedFor . end ( ) )
mapAlreadyAskedFor . update ( it , nRequestTime ) ;
else
2016-04-11 13:52:29 -03:00
mapAlreadyAskedFor . insert ( std : : make_pair ( inv . hash , nRequestTime ) ) ;
2014-08-20 23:17:21 -04:00
mapAskFor . insert ( std : : make_pair ( nRequestTime , inv ) ) ;
}
2017-01-20 22:34:57 -03:00
bool CConnman : : NodeFullyConnected ( const CNode * pnode )
{
return pnode & & pnode - > fSuccessfullyConnected & & ! pnode - > fDisconnect ;
}
2016-11-10 22:17:30 -03:00
void CConnman : : PushMessage ( CNode * pnode , CSerializedNetMsg & & msg )
2016-09-12 21:00:33 -03:00
{
2016-11-10 22:17:30 -03:00
size_t nMessageSize = msg . data . size ( ) ;
size_t nTotalSize = nMessageSize + CMessageHeader : : HEADER_SIZE ;
2017-04-11 13:13:55 -03:00
LogPrint ( BCLog : : NET , " sending %s (%d bytes) peer=%d \n " , SanitizeString ( msg . command . c_str ( ) ) , nMessageSize , pnode - > GetId ( ) ) ;
2016-09-12 21:00:33 -03:00
2016-11-10 22:17:30 -03:00
std : : vector < unsigned char > serializedHeader ;
serializedHeader . reserve ( CMessageHeader : : HEADER_SIZE ) ;
uint256 hash = Hash ( msg . data . data ( ) , msg . data . data ( ) + nMessageSize ) ;
CMessageHeader hdr ( Params ( ) . MessageStart ( ) , msg . command . c_str ( ) , nMessageSize ) ;
memcpy ( hdr . pchChecksum , hash . begin ( ) , CMessageHeader : : CHECKSUM_SIZE ) ;
2016-09-12 21:00:33 -03:00
2016-11-10 22:17:30 -03:00
CVectorWriter { SER_NETWORK , INIT_PROTO_VERSION , serializedHeader , 0 , hdr } ;
2016-09-12 21:00:33 -03:00
size_t nBytesSent = 0 ;
{
LOCK ( pnode - > cs_vSend ) ;
bool optimisticSend ( pnode - > vSendMsg . empty ( ) ) ;
//log total amount of bytes per command
2016-11-10 22:17:30 -03:00
pnode - > mapSendBytesPerMsgCmd [ msg . command ] + = nTotalSize ;
pnode - > nSendSize + = nTotalSize ;
2016-12-31 04:05:32 -03:00
if ( pnode - > nSendSize > nSendBufferMaxSize )
pnode - > fPauseSend = true ;
2016-11-10 22:17:30 -03:00
pnode - > vSendMsg . push_back ( std : : move ( serializedHeader ) ) ;
if ( nMessageSize )
pnode - > vSendMsg . push_back ( std : : move ( msg . data ) ) ;
2016-09-12 21:00:33 -03:00
// If write queue empty, attempt "optimistic write"
if ( optimisticSend = = true )
nBytesSent = SocketSendData ( pnode ) ;
}
if ( nBytesSent )
RecordBytesSent ( nBytesSent ) ;
}
2016-04-16 20:13:12 -03:00
bool CConnman : : ForNode ( NodeId id , std : : function < bool ( CNode * pnode ) > func )
{
CNode * found = nullptr ;
LOCK ( cs_vNodes ) ;
for ( auto & & pnode : vNodes ) {
2017-04-11 13:13:55 -03:00
if ( pnode - > GetId ( ) = = id ) {
2016-04-16 20:13:12 -03:00
found = pnode ;
break ;
}
}
2017-01-20 22:34:57 -03:00
return found ! = nullptr & & NodeFullyConnected ( found ) & & func ( found ) ;
2016-04-16 20:13:12 -03:00
}
2018-05-21 15:02:40 -04:00
int64_t CConnman : : PoissonNextSendInbound ( int64_t now , int average_interval_seconds )
{
if ( m_next_send_inv_to_incoming < now ) {
// If this function were called from multiple threads simultaneously
// it would possible that both update the next send variable, and return a different result to their caller.
// This is not possible in practice as only the net processing thread invokes this function.
m_next_send_inv_to_incoming = PoissonNextSend ( now , average_interval_seconds ) ;
}
return m_next_send_inv_to_incoming ;
}
int64_t PoissonNextSend ( int64_t now , int average_interval_seconds )
{
return now + ( int64_t ) ( log1p ( GetRand ( 1ULL < < 48 ) * - 0.0000000000000035527136788 /* -1/2^48 */ ) * average_interval_seconds * - 1000000.0 + 0.5 ) ;
2015-04-08 15:20:00 -03:00
}
2016-05-25 09:38:32 -04:00
2017-01-23 22:32:52 -03:00
CSipHasher CConnman : : GetDeterministicRandomizer ( uint64_t id ) const
2016-05-25 09:38:32 -04:00
{
2016-09-09 07:48:10 -03:00
return CSipHasher ( nSeed0 , nSeed1 ) . Write ( id ) ;
}
2016-05-25 09:38:32 -04:00
2017-01-23 22:32:52 -03:00
uint64_t CConnman : : CalculateKeyedNetGroup ( const CAddress & ad ) const
2016-09-09 07:48:10 -03:00
{
2016-05-25 09:38:32 -04:00
std : : vector < unsigned char > vchNetGroup ( ad . GetGroup ( ) ) ;
2017-02-19 15:18:04 -03:00
return GetDeterministicRandomizer ( RANDOMIZER_ID_NETGROUP ) . Write ( vchNetGroup . data ( ) , vchNetGroup . size ( ) ) . Finalize ( ) ;
2016-05-25 09:38:32 -04:00
}