54#include <boost/multi_index/hashed_index.hpp>
55#include <boost/multi_index/member.hpp>
56#include <boost/multi_index/ordered_index.hpp>
57#include <boost/multi_index_container.hpp>
115 "Max protocol message length must be greater than largest "
116 "possible INV message");
173 std::chrono::seconds(2),
174 std::chrono::seconds(2),
175 std::chrono::seconds(60),
182 std::chrono::seconds(2),
183 std::chrono::seconds(2),
184 std::chrono::seconds(60),
217 "MAX_BLOCKTXN_DEPTH too high");
276 std::chrono::seconds{1},
277 "INVENTORY_RELAY_MAX too low");
328 std::unique_ptr<PartiallyDownloadedBlock> partialBlock;
333 std::chrono::seconds timeAdded;
335 StalledTxId(
TxId txid_, std::chrono::seconds timeAdded_)
336 : txid(txid_), timeAdded(timeAdded_){};
342using StalledTxIdSet = boost::multi_index_container<
344 boost::multi_index::indexed_by<
346 boost::multi_index::hashed_unique<
347 boost::multi_index::tag<by_txid>,
348 boost::multi_index::member<StalledTxId, TxId, &StalledTxId::txid>,
351 boost::multi_index::ordered_non_unique<
352 boost::multi_index::tag<by_time>,
353 boost::multi_index::member<StalledTxId, std::chrono::seconds,
354 &StalledTxId::timeAdded>>>>;
391 std::atomic<ServiceFlags> m_their_services{
NODE_NONE};
394 Mutex m_misbehavior_mutex;
399 bool m_should_discourage
GUARDED_BY(m_misbehavior_mutex){
false};
402 Mutex m_block_inv_mutex;
408 std::vector<BlockHash> m_blocks_for_inv_relay
GUARDED_BY(m_block_inv_mutex);
414 std::vector<BlockHash>
415 m_blocks_for_headers_relay
GUARDED_BY(m_block_inv_mutex);
426 std::atomic<int> m_starting_height{-1};
429 std::atomic<uint64_t> m_ping_nonce_sent{0};
431 std::atomic<std::chrono::microseconds> m_ping_start{0us};
433 std::atomic<bool> m_ping_queued{
false};
443 std::chrono::microseconds m_next_send_feefilter
456 bool m_relay_txs
GUARDED_BY(m_bloom_filter_mutex){
false};
461 std::unique_ptr<CBloomFilter>
477 GUARDED_BY(m_tx_inventory_mutex){50000, 0.000001};
483 std::set<TxId> m_tx_inventory_to_send
GUARDED_BY(m_tx_inventory_mutex);
489 bool m_send_mempool
GUARDED_BY(m_tx_inventory_mutex){
false};
491 std::atomic<std::chrono::seconds> m_last_mempool_req{0s};
496 std::chrono::microseconds
497 m_next_inv_send_time
GUARDED_BY(m_tx_inventory_mutex){0};
503 std::atomic<Amount> m_fee_filter_received{
Amount::zero()};
509 m_avalanche_stalled_txids
GUARDED_BY(m_tx_inventory_mutex);
517 LOCK(m_tx_relay_mutex);
519 m_tx_relay = std::make_unique<Peer::TxRelay>();
520 return m_tx_relay.get();
524 return WITH_LOCK(m_tx_relay_mutex,
return m_tx_relay.get());
526 const TxRelay *GetTxRelay() const
528 return WITH_LOCK(m_tx_relay_mutex,
return m_tx_relay.get());
533 std::set<avalanche::ProofId>
534 m_proof_inventory_to_send
GUARDED_BY(m_proof_inventory_mutex);
537 GUARDED_BY(m_proof_inventory_mutex){10000, 0.000001};
544 std::chrono::microseconds m_next_inv_send_time{0};
548 std::atomic<std::chrono::seconds> lastSharedProofsUpdate{0s};
549 std::atomic<bool> compactproofs_requested{
false};
556 const std::unique_ptr<ProofRelay> m_proof_relay;
561 std::vector<CAddress>
573 std::unique_ptr<CRollingBloomFilter>
591 std::atomic_bool m_addr_relay_enabled{
false};
595 mutable Mutex m_addr_send_times_mutex;
597 std::chrono::microseconds
598 m_next_addr_send
GUARDED_BY(m_addr_send_times_mutex){0};
600 std::chrono::microseconds
601 m_next_local_addr_send
GUARDED_BY(m_addr_send_times_mutex){0};
606 std::atomic_bool m_wants_addrv2{
false};
610 mutable Mutex m_addr_token_bucket_mutex;
615 double m_addr_token_bucket
GUARDED_BY(m_addr_token_bucket_mutex){1.0};
617 std::chrono::microseconds
619 GetTime<std::chrono::microseconds>()};
621 std::atomic<uint64_t> m_addr_rate_limited{0};
626 std::atomic<uint64_t> m_addr_processed{0};
632 bool m_inv_triggered_getheaders_before_sync
636 Mutex m_getdata_requests_mutex;
638 std::deque<CInv> m_getdata_requests
GUARDED_BY(m_getdata_requests_mutex);
645 Mutex m_headers_sync_mutex;
650 std::unique_ptr<HeadersSyncState>
655 std::atomic<bool> m_sent_sendheaders{
false};
658 std::chrono::microseconds m_headers_sync_timeout
669 : m_id(id), m_our_services{our_services},
670 m_proof_relay(fRelayProofs ?
std::make_unique<ProofRelay>()
674 mutable Mutex m_tx_relay_mutex;
677 std::unique_ptr<TxRelay> m_tx_relay
GUARDED_BY(m_tx_relay_mutex);
680using PeerRef = std::shared_ptr<Peer>;
698 bool fSyncStarted{
false};
701 std::chrono::microseconds m_stalling_since{0us};
702 std::list<QueuedBlock> vBlocksInFlight;
705 std::chrono::microseconds m_downloading_since{0us};
707 bool fPreferredDownload{
false};
712 bool m_requested_hb_cmpctblocks{
false};
714 bool m_provides_cmpctblocks{
false};
742 struct ChainSyncTimeoutState {
745 std::chrono::seconds m_timeout{0s};
749 bool m_sent_getheaders{
false};
752 bool m_protect{
false};
755 ChainSyncTimeoutState m_chain_sync;
758 int64_t m_last_block_announcement{0};
761 const bool m_is_inbound;
763 CNodeState(
bool is_inbound) : m_is_inbound(is_inbound) {}
774 const std::shared_ptr<const CBlock> &pblock,
782 bool fInitialDownload)
override
788 const std::shared_ptr<const CBlock> &pblock)
override
797 !m_headers_presync_mutex);
799 std::atomic<bool> &interrupt)
override
801 !m_recent_confirmed_transactions_mutex,
802 !m_most_recent_block_mutex, !cs_proofrequest,
803 !m_headers_presync_mutex, g_msgproc_mutex);
806 !m_recent_confirmed_transactions_mutex,
807 !m_most_recent_block_mutex, !cs_proofrequest,
813 std::optional<std::string>
820 void RelayTransaction(const
TxId &txid) override
822 void RelayProof(const
avalanche::ProofId &proofid) override
824 void SetBestHeight(
int height)
override { m_best_height = height; };
827 Misbehaving(*
Assert(GetPeerRef(peer_id)),
"");
830 const std::string &msg_type,
DataStream &vRecv,
831 const std::chrono::microseconds time_received,
832 const std::atomic<bool> &interruptMsgProc)
override
834 !m_recent_confirmed_transactions_mutex,
835 !m_most_recent_block_mutex, !cs_proofrequest,
836 !m_headers_presync_mutex, g_msgproc_mutex);
838 int64_t time_in_seconds)
override;
845 void ConsiderEviction(
CNode &pto, Peer &peer,
846 std::chrono::seconds time_in_seconds)
853 void EvictExtraOutboundPeers(std::chrono::seconds now)
860 void ReattemptInitialBroadcast(
CScheduler &scheduler)
866 void UpdateAvalancheStatistics()
const;
871 void AvalanchePeriodicNetworking(
CScheduler &scheduler)
const;
889 void Misbehaving(Peer &peer,
const std::string &message);
901 void MaybePunishNodeForBlock(
NodeId nodeid,
903 bool via_compact_block,
904 const std::string &message =
"")
912 const
std::
string &message = "")
924 bool MaybeDiscourageAndDisconnect(
CNode &pnode, Peer &peer);
942 bool maybe_add_extra_compact_tx)
945 struct PackageToValidate {
947 const std::vector<NodeId> m_senders;
952 : m_txns{parent, child}, m_senders{parent_sender, child_sender} {}
955 Assume(m_txns.size() == 2);
957 "parent %s (sender=%d) + child %s (sender=%d)",
958 m_txns.front()->GetId().ToString(), m_senders.front(),
959 m_txns.back()->GetId().ToString(), m_senders.back());
968 void ProcessPackageResult(
const PackageToValidate &package_to_validate,
978 std::optional<PackageToValidate> Find1P1CPackage(
const CTransactionRef &ptx,
1005 bool ProcessOrphanTx(
const Config &config, Peer &peer)
1018 void ProcessHeadersMessage(
const Config &config,
CNode &pfrom, Peer &peer,
1019 std::vector<CBlockHeader> &&headers,
1020 bool via_compact_block)
1030 bool CheckHeadersPoW(
const std::vector<CBlockHeader> &headers,
1040 void HandleUnconnectingHeaders(
CNode &pfrom, Peer &peer,
1041 const std::vector<CBlockHeader> &headers)
1045 CheckHeadersAreContinuous(
const std::vector<CBlockHeader> &headers)
const;
1065 bool IsContinuationOfLowWorkHeadersSync(Peer &peer,
CNode &pfrom,
1066 std::vector<CBlockHeader> &headers)
1068 !m_headers_presync_mutex, g_msgproc_mutex);
1082 bool TryLowWorkHeadersSync(Peer &peer,
CNode &pfrom,
1084 std::vector<CBlockHeader> &headers)
1086 !m_headers_presync_mutex, g_msgproc_mutex);
1092 bool IsAncestorOfBestHeaderOrTip(
const CBlockIndex *header)
1106 void HeadersDirectFetchBlocks(
const Config &config,
CNode &pfrom,
1109 void UpdatePeerStateForReceivedHeaders(
CNode &pfrom, Peer &peer,
1111 bool received_new_header,
1112 bool may_have_more_headers)
1115 void SendBlockTransactions(
CNode &pfrom, Peer &peer,
const CBlock &block,
1124 std::chrono::microseconds current_time)
1134 std::chrono::microseconds current_time,
bool preferred)
1139 m_connman.PushMessage(&
node, std::move(
msg));
1141 template <
typename... Args>
1142 void MakeAndPushMessage(
CNode &
node, std::string msg_type,
1143 Args &&...args)
const {
1145 std::forward<Args>(args)...));
1149 void PushNodeVersion(
const Config &config,
CNode &pnode,
const Peer &peer);
1157 void MaybeSendPing(
CNode &node_to, Peer &peer,
1158 std::chrono::microseconds now);
1161 void MaybeSendAddr(
CNode &
node, Peer &peer,
1162 std::chrono::microseconds current_time)
1169 void MaybeSendSendHeaders(
CNode &
node, Peer &peer)
1173 void MaybeSendFeefilter(
CNode &
node, Peer &peer,
1174 std::chrono::microseconds current_time)
1186 void RelayAddress(
NodeId originator,
const CAddress &addr,
bool fReachable)
1207 Mutex cs_proofrequest;
1212 std::atomic<int> m_best_height{-1};
1217 const Options m_opts;
1219 bool RejectIncomingTxs(
const CNode &peer)
const;
1231 mutable Mutex m_peer_mutex;
1238 std::map<NodeId, PeerRef> m_peer_map
GUARDED_BY(m_peer_mutex);
1247 const CNodeState *State(
NodeId pnode)
const
1252 std::atomic<std::chrono::microseconds> m_next_inv_to_inbounds{0us};
1259 m_last_block_inv_triggering_headers_sync
GUARDED_BY(g_msgproc_mutex){};
1267 std::map<BlockHash, std::pair<NodeId, bool>>
1277 std::atomic<std::chrono::seconds> m_block_stalling_timeout{
1291 bool AlreadyHaveTx(
const TxId &txid,
bool include_reconsiderable)
1293 !m_recent_confirmed_transactions_mutex);
1356 mutable Mutex m_recent_confirmed_transactions_mutex;
1358 GUARDED_BY(m_recent_confirmed_transactions_mutex){24'000, 0.000'001};
1367 std::chrono::microseconds
1368 NextInvToInbounds(std::chrono::microseconds now,
1369 std::chrono::seconds average_interval)
1374 mutable Mutex m_most_recent_block_mutex;
1375 std::shared_ptr<const CBlock>
1376 m_most_recent_block
GUARDED_BY(m_most_recent_block_mutex);
1377 std::shared_ptr<const CBlockHeaderAndShortTxIDs>
1378 m_most_recent_compact_block
GUARDED_BY(m_most_recent_block_mutex);
1380 std::unique_ptr<const std::map<TxId, CTransactionRef>>
1381 m_most_recent_block_txs
GUARDED_BY(m_most_recent_block_mutex);
1386 Mutex m_headers_presync_mutex;
1397 using HeadersPresyncStats =
1398 std::pair<arith_uint256, std::optional<std::pair<int64_t, uint32_t>>>;
1400 std::map<NodeId, HeadersPresyncStats>
1401 m_headers_presync_stats
GUARDED_BY(m_headers_presync_mutex){};
1405 std::atomic_bool m_headers_presync_should_signal{
false};
1413 bool IsBlockRequested(
const BlockHash &hash)
1417 bool IsBlockRequestedFromOutbound(
const BlockHash &hash)
1428 void RemoveBlockRequest(
const BlockHash &hash,
1429 std::optional<NodeId> from_peer)
1438 bool BlockRequested(
const Config &config,
NodeId nodeid,
1440 std::list<QueuedBlock>::iterator **pit =
nullptr)
1449 void FindNextBlocksToDownload(const Peer &peer,
unsigned int count,
1455 void TryDownloadingHistoricalBlocks(
1456 const Peer &peer,
unsigned int count,
1490 const Peer &peer, CNodeState *state,
1492 int nWindowEnd, const
CChain *activeChain =
nullptr,
1493 NodeId *nodeStaller =
nullptr)
1503 std::atomic<
std::chrono::seconds> m_last_tip_update{0s};
1510 const std::chrono::seconds mempool_req,
1511 const std::chrono::seconds now)
1516 void ProcessGetData(
const Config &config,
CNode &pfrom, Peer &peer,
1517 const std::atomic<bool> &interruptMsgProc)
1519 peer.m_getdata_requests_mutex,
1525 const std::shared_ptr<const CBlock> &block,
1526 bool force_processing,
bool min_pow_checked);
1534 void MaybeSetPeerAsAnnouncingHeaderAndIDs(
NodeId nodeid)
1553 std::vector<CTransactionRef>
1554 vExtraTxnForCompact
GUARDED_BY(g_msgproc_mutex);
1556 size_t vExtraTxnForCompactIt
GUARDED_BY(g_msgproc_mutex) = 0;
1561 void ProcessBlockAvailability(
NodeId nodeid)
1576 bool BlockRequestAllowed(const
CBlockIndex *pindex)
1578 bool AlreadyHaveBlock(const
BlockHash &block_hash)
1580 bool AlreadyHaveProof(const
avalanche::ProofId &proofid);
1581 void ProcessGetBlockData(const
Config &config,
CNode &pfrom, Peer &peer,
1604 bool PrepareBlockFilterRequest(
CNode &
node, Peer &peer,
1606 uint32_t start_height,
1608 uint32_t max_height_diff,
1650 uint32_t GetAvalancheVoteForBlock(const
BlockHash &hash) const
1661 const
TxId &
id) const
1663 !m_recent_confirmed_transactions_mutex);
1672 bool SetupAddressRelay(const
CNode &
node, Peer &peer)
1675 void AddAddressKnown(Peer &peer, const
CAddress &addr)
1677 void PushAddress(Peer &peer, const
CAddress &addr)
1685 bool ReceivedAvalancheProof(
CNode &
node, Peer &peer,
1691 const
std::chrono::seconds now)
1694 bool isPreferredDownloadPeer(const
CNode &pfrom);
1697const CNodeState *PeerManagerImpl::State(
NodeId pnode) const
1699 std::map<NodeId, CNodeState>::const_iterator it = m_node_states.find(pnode);
1700 if (it == m_node_states.end()) {
1707CNodeState *PeerManagerImpl::State(
NodeId pnode)
1709 return const_cast<CNodeState *
>(std::as_const(*this).State(pnode));
1717static bool IsAddrCompatible(
const Peer &peer,
const CAddress &addr) {
1721void PeerManagerImpl::AddAddressKnown(Peer &peer,
const CAddress &addr) {
1722 assert(peer.m_addr_known);
1723 peer.m_addr_known->insert(addr.
GetKey());
1726void PeerManagerImpl::PushAddress(Peer &peer,
const CAddress &addr) {
1730 assert(peer.m_addr_known);
1731 if (addr.
IsValid() && !peer.m_addr_known->contains(addr.
GetKey()) &&
1732 IsAddrCompatible(peer, addr)) {
1733 if (peer.m_addrs_to_send.size() >= m_opts.max_addr_to_send) {
1734 peer.m_addrs_to_send[m_rng.randrange(peer.m_addrs_to_send.size())] =
1737 peer.m_addrs_to_send.push_back(addr);
1742static void AddKnownTx(Peer &peer,
const TxId &txid) {
1743 auto tx_relay = peer.GetTxRelay();
1748 LOCK(tx_relay->m_tx_inventory_mutex);
1749 tx_relay->m_tx_inventory_known_filter.insert(txid);
1753 if (peer.m_proof_relay !=
nullptr) {
1754 LOCK(peer.m_proof_relay->m_proof_inventory_mutex);
1755 peer.m_proof_relay->m_proof_inventory_known_filter.insert(proofid);
1759bool PeerManagerImpl::isPreferredDownloadPeer(
const CNode &pfrom) {
1761 const CNodeState *state = State(pfrom.
GetId());
1762 return state && state->fPreferredDownload;
1765static bool CanServeBlocks(
const Peer &peer) {
1773static bool IsLimitedPeer(
const Peer &peer) {
1778std::chrono::microseconds
1779PeerManagerImpl::NextInvToInbounds(std::chrono::microseconds now,
1780 std::chrono::seconds average_interval) {
1781 if (m_next_inv_to_inbounds.load() < now) {
1786 m_next_inv_to_inbounds =
1787 now + m_rng.rand_exp_duration(average_interval);
1789 return m_next_inv_to_inbounds;
1792bool PeerManagerImpl::IsBlockRequested(
const BlockHash &hash) {
1793 return mapBlocksInFlight.count(hash);
1796bool PeerManagerImpl::IsBlockRequestedFromOutbound(
const BlockHash &hash) {
1797 for (
auto range = mapBlocksInFlight.equal_range(hash);
1798 range.first != range.second; range.first++) {
1799 auto [nodeid, block_it] = range.first->second;
1800 CNodeState &nodestate = *
Assert(State(nodeid));
1801 if (!nodestate.m_is_inbound) {
1809void PeerManagerImpl::RemoveBlockRequest(
const BlockHash &hash,
1810 std::optional<NodeId> from_peer) {
1811 auto range = mapBlocksInFlight.equal_range(hash);
1812 if (range.first == range.second) {
1820 while (range.first != range.second) {
1821 auto [node_id, list_it] = range.first->second;
1823 if (from_peer && *from_peer != node_id) {
1828 CNodeState &state = *
Assert(State(node_id));
1830 if (state.vBlocksInFlight.begin() == list_it) {
1833 state.m_downloading_since =
1834 std::max(state.m_downloading_since,
1835 GetTime<std::chrono::microseconds>());
1837 state.vBlocksInFlight.erase(list_it);
1839 if (state.vBlocksInFlight.empty()) {
1841 m_peers_downloading_from--;
1843 state.m_stalling_since = 0us;
1845 range.first = mapBlocksInFlight.erase(range.first);
1849bool PeerManagerImpl::BlockRequested(
const Config &config,
NodeId nodeid,
1851 std::list<QueuedBlock>::iterator **pit) {
1854 CNodeState *state = State(nodeid);
1855 assert(state !=
nullptr);
1860 for (
auto range = mapBlocksInFlight.equal_range(hash);
1861 range.first != range.second; range.first++) {
1862 if (range.first->second.first == nodeid) {
1864 *pit = &range.first->second.second;
1871 RemoveBlockRequest(hash, nodeid);
1873 std::list<QueuedBlock>::iterator it = state->vBlocksInFlight.insert(
1874 state->vBlocksInFlight.end(),
1875 {&block, std::unique_ptr<PartiallyDownloadedBlock>(
1876 pit ? new PartiallyDownloadedBlock(config, &m_mempool)
1878 if (state->vBlocksInFlight.size() == 1) {
1880 state->m_downloading_since = GetTime<std::chrono::microseconds>();
1881 m_peers_downloading_from++;
1884 auto itInFlight = mapBlocksInFlight.insert(
1885 std::make_pair(hash, std::make_pair(nodeid, it)));
1888 *pit = &itInFlight->second.second;
1894void PeerManagerImpl::MaybeSetPeerAsAnnouncingHeaderAndIDs(
NodeId nodeid) {
1900 if (m_opts.ignore_incoming_txs) {
1904 CNodeState *nodestate = State(nodeid);
1909 if (!nodestate->m_provides_cmpctblocks) {
1912 int num_outbound_hb_peers = 0;
1913 for (std::list<NodeId>::iterator it = lNodesAnnouncingHeaderAndIDs.begin();
1914 it != lNodesAnnouncingHeaderAndIDs.end(); it++) {
1915 if (*it == nodeid) {
1916 lNodesAnnouncingHeaderAndIDs.erase(it);
1917 lNodesAnnouncingHeaderAndIDs.push_back(nodeid);
1920 CNodeState *state = State(*it);
1921 if (state !=
nullptr && !state->m_is_inbound) {
1922 ++num_outbound_hb_peers;
1925 if (nodestate->m_is_inbound) {
1928 if (lNodesAnnouncingHeaderAndIDs.size() >= 3 &&
1929 num_outbound_hb_peers == 1) {
1930 CNodeState *remove_node =
1931 State(lNodesAnnouncingHeaderAndIDs.front());
1932 if (remove_node !=
nullptr && !remove_node->m_is_inbound) {
1935 std::swap(lNodesAnnouncingHeaderAndIDs.front(),
1936 *std::next(lNodesAnnouncingHeaderAndIDs.begin()));
1943 if (lNodesAnnouncingHeaderAndIDs.size() >= 3) {
1947 lNodesAnnouncingHeaderAndIDs.front(), [
this](
CNode *pnodeStop) {
1948 MakeAndPushMessage(*pnodeStop, NetMsgType::SENDCMPCT,
1950 CMPCTBLOCKS_VERSION);
1953 pnodeStop->m_bip152_highbandwidth_to = false;
1956 lNodesAnnouncingHeaderAndIDs.pop_front();
1963 lNodesAnnouncingHeaderAndIDs.push_back(pfrom->
GetId());
1968bool PeerManagerImpl::TipMayBeStale() {
1971 if (m_last_tip_update.load() == 0s) {
1972 m_last_tip_update = GetTime<std::chrono::seconds>();
1974 return m_last_tip_update.load() <
1975 GetTime<std::chrono::seconds>() -
1978 mapBlocksInFlight.empty();
1981bool PeerManagerImpl::CanDirectFetch() {
1987static bool PeerHasHeader(CNodeState *state,
const CBlockIndex *pindex)
1989 if (state->pindexBestKnownBlock &&
1990 pindex == state->pindexBestKnownBlock->GetAncestor(pindex->nHeight)) {
1993 if (state->pindexBestHeaderSent &&
1994 pindex == state->pindexBestHeaderSent->GetAncestor(pindex->nHeight)) {
2000void PeerManagerImpl::ProcessBlockAvailability(
NodeId nodeid) {
2001 CNodeState *state = State(nodeid);
2002 assert(state !=
nullptr);
2004 if (!state->hashLastUnknownBlock.IsNull()) {
2008 if (state->pindexBestKnownBlock ==
nullptr ||
2009 pindex->
nChainWork >= state->pindexBestKnownBlock->nChainWork) {
2010 state->pindexBestKnownBlock = pindex;
2012 state->hashLastUnknownBlock.SetNull();
2017void PeerManagerImpl::UpdateBlockAvailability(
NodeId nodeid,
2019 CNodeState *state = State(nodeid);
2020 assert(state !=
nullptr);
2022 ProcessBlockAvailability(nodeid);
2027 if (state->pindexBestKnownBlock ==
nullptr ||
2028 pindex->
nChainWork >= state->pindexBestKnownBlock->nChainWork) {
2029 state->pindexBestKnownBlock = pindex;
2034 state->hashLastUnknownBlock = hash;
2040void PeerManagerImpl::FindNextBlocksToDownload(
2041 const Peer &peer,
unsigned int count,
2042 std::vector<const CBlockIndex *> &vBlocks,
NodeId &nodeStaller) {
2047 vBlocks.reserve(vBlocks.size() +
count);
2048 CNodeState *state = State(peer.m_id);
2049 assert(state !=
nullptr);
2052 ProcessBlockAvailability(peer.m_id);
2054 if (state->pindexBestKnownBlock ==
nullptr ||
2055 state->pindexBestKnownBlock->nChainWork <
2057 state->pindexBestKnownBlock->nChainWork <
2067 const CBlockIndex *snap_base{m_chainman.GetSnapshotBaseBlock()};
2068 if (snap_base && state->pindexBestKnownBlock->GetAncestor(
2069 snap_base->nHeight) != snap_base) {
2071 "Not downloading blocks from peer=%d, which doesn't have the "
2072 "snapshot block in its best chain.\n",
2081 if (state->pindexLastCommonBlock ==
nullptr ||
2083 state->pindexLastCommonBlock->nHeight < snap_base->nHeight)) {
2084 state->pindexLastCommonBlock =
2086 .
ActiveChain()[std::min(state->pindexBestKnownBlock->nHeight,
2093 state->pindexLastCommonBlock, state->pindexBestKnownBlock);
2094 if (state->pindexLastCommonBlock == state->pindexBestKnownBlock) {
2098 const CBlockIndex *pindexWalk = state->pindexLastCommonBlock;
2106 FindNextBlocks(vBlocks, peer, state, pindexWalk,
count, nWindowEnd,
2110void PeerManagerImpl::TryDownloadingHistoricalBlocks(
2111 const Peer &peer,
unsigned int count,
2112 std::vector<const CBlockIndex *> &vBlocks,
const CBlockIndex *from_tip,
2117 if (vBlocks.size() >=
count) {
2121 vBlocks.reserve(
count);
2122 CNodeState *state =
Assert(State(peer.m_id));
2124 if (state->pindexBestKnownBlock ==
nullptr ||
2125 state->pindexBestKnownBlock->GetAncestor(target_block->
nHeight) !=
2140 FindNextBlocks(vBlocks, peer, state, from_tip,
count,
2145void PeerManagerImpl::FindNextBlocks(std::vector<const CBlockIndex *> &vBlocks,
2146 const Peer &peer, CNodeState *state,
2148 unsigned int count,
int nWindowEnd,
2149 const CChain *activeChain,
2151 std::vector<const CBlockIndex *> vToFetch;
2153 std::min<int>(state->pindexBestKnownBlock->nHeight, nWindowEnd + 1);
2155 while (pindexWalk->
nHeight < nMaxHeight) {
2160 int nToFetch = std::min(nMaxHeight - pindexWalk->
nHeight,
2161 std::max<int>(
count - vBlocks.size(), 128));
2162 vToFetch.resize(nToFetch);
2163 pindexWalk = state->pindexBestKnownBlock->
GetAncestor(
2164 pindexWalk->
nHeight + nToFetch);
2165 vToFetch[nToFetch - 1] = pindexWalk;
2166 for (
unsigned int i = nToFetch - 1; i > 0; i--) {
2167 vToFetch[i - 1] = vToFetch[i]->
pprev;
2180 if (pindex->nStatus.hasData() ||
2181 (activeChain && activeChain->
Contains(pindex))) {
2183 state->pindexLastCommonBlock = pindex;
2185 }
else if (!IsBlockRequested(pindex->
GetBlockHash())) {
2187 if (pindex->
nHeight > nWindowEnd) {
2189 if (vBlocks.size() == 0 && waitingfor != peer.m_id) {
2193 *nodeStaller = waitingfor;
2198 vBlocks.push_back(pindex);
2199 if (vBlocks.size() ==
count) {
2202 }
else if (waitingfor == -1) {
2214template <
class InvId>
2218 return !
node.HasPermission(
2231template <
class InvId>
2232static std::chrono::microseconds
2236 std::chrono::microseconds current_time,
bool preferred) {
2237 auto delay = std::chrono::microseconds{0};
2249 return current_time + delay;
2252void PeerManagerImpl::PushNodeVersion(
const Config &config,
CNode &pnode,
2254 uint64_t my_services{peer.m_our_services};
2255 const int64_t nTime{
count_seconds(GetTime<std::chrono::seconds>())};
2257 const int nNodeStartingHeight{m_best_height};
2268 const bool tx_relay{!RejectIncomingTxs(pnode)};
2277 nNodeStartingHeight, tx_relay, extraEntropy);
2281 "send version message: version %d, blocks=%d, them=%s, "
2282 "txrelay=%d, peer=%d\n",
2287 "send version message: version %d, blocks=%d, "
2288 "txrelay=%d, peer=%d\n",
2293void PeerManagerImpl::AddTxAnnouncement(
2295 std::chrono::microseconds current_time) {
2303 const bool preferred = isPreferredDownloadPeer(
node);
2305 current_time, preferred);
2307 m_txrequest.ReceivedInv(
node.GetId(), txid, preferred, reqtime);
2310void PeerManagerImpl::AddProofAnnouncement(
2312 std::chrono::microseconds current_time,
bool preferred) {
2323 m_proofrequest.ReceivedInv(
node.GetId(), proofid, preferred, reqtime);
2326void PeerManagerImpl::UpdateLastBlockAnnounceTime(
NodeId node,
2327 int64_t time_in_seconds) {
2329 CNodeState *state = State(
node);
2331 state->m_last_block_announcement = time_in_seconds;
2335void PeerManagerImpl::InitializeNode(
const Config &config,
CNode &
node,
2340 m_node_states.emplace_hint(m_node_states.end(),
2341 std::piecewise_construct,
2342 std::forward_as_tuple(nodeid),
2343 std::forward_as_tuple(
node.IsInboundConn()));
2344 assert(m_txrequest.Count(nodeid) == 0);
2352 PeerRef peer = std::make_shared<Peer>(nodeid, our_services, !!m_avalanche);
2355 m_peer_map.emplace_hint(m_peer_map.end(), nodeid, peer);
2357 if (!
node.IsInboundConn()) {
2358 PushNodeVersion(config,
node, *peer);
2362void PeerManagerImpl::ReattemptInitialBroadcast(
CScheduler &scheduler) {
2365 for (
const TxId &txid : unbroadcast_txids) {
2367 if (m_mempool.
exists(txid)) {
2368 RelayTransaction(txid);
2379 auto unbroadcasted_proofids =
2383 auto it = unbroadcasted_proofids.begin();
2384 while (it != unbroadcasted_proofids.end()) {
2387 if (!pm.isBoundToPeer(*it)) {
2388 pm.removeUnbroadcastProof(*it);
2389 it = unbroadcasted_proofids.erase(it);
2396 return unbroadcasted_proofids;
2400 for (
const auto &proofid : unbroadcasted_proofids) {
2401 RelayProof(proofid);
2408 const auto reattemptBroadcastInterval =
2410 scheduler.
scheduleFromNow([&] { ReattemptInitialBroadcast(scheduler); },
2411 reattemptBroadcastInterval);
2414void PeerManagerImpl::UpdateAvalancheStatistics()
const {
2420void PeerManagerImpl::AvalanchePeriodicNetworking(
CScheduler &scheduler)
const {
2421 const auto now = GetTime<std::chrono::seconds>();
2422 std::vector<NodeId> avanode_ids;
2423 bool fQuorumEstablished;
2424 bool fShouldRequestMoreNodes;
2434 fShouldRequestMoreNodes =
2442 avanode_ids.push_back(pnode->GetId());
2445 PeerRef peer = GetPeerRef(pnode->
GetId());
2446 if (peer ==
nullptr) {
2450 if (peer->m_proof_relay &&
2451 now > (peer->m_proof_relay->lastSharedProofsUpdate.load() +
2453 peer->m_proof_relay->sharedProofs = {};
2457 if (avanode_ids.empty()) {
2465 for (
NodeId avanodeId : avanode_ids) {
2466 const bool sentGetavaaddr =
2469 MakeAndPushMessage(*pavanode, NetMsgType::GETAVAADDR);
2470 PeerRef peer = GetPeerRef(avanodeId);
2471 WITH_LOCK(peer->m_addr_token_bucket_mutex,
2472 peer->m_addr_token_bucket +=
2473 m_opts.max_addr_to_send);
2481 if (sentGetavaaddr && fQuorumEstablished && !fShouldRequestMoreNodes) {
2496 avanode_ids.resize(std::min<size_t>(avanode_ids.size(), 3));
2499 for (
NodeId nodeid : avanode_ids) {
2502 PeerRef peer = GetPeerRef(nodeid);
2503 if (peer->m_proof_relay) {
2505 peer->m_proof_relay->compactproofs_requested =
true;
2515 const auto avalanchePeriodicNetworkingInterval =
2517 scheduler.
scheduleFromNow([&] { AvalanchePeriodicNetworking(scheduler); },
2518 avalanchePeriodicNetworkingInterval);
2521void PeerManagerImpl::FinalizeNode(
const Config &config,
const CNode &
node) {
2531 PeerRef peer = RemovePeer(nodeid);
2534 m_peer_map.erase(nodeid);
2536 CNodeState *state = State(nodeid);
2537 assert(state !=
nullptr);
2539 if (state->fSyncStarted) {
2543 for (
const QueuedBlock &entry : state->vBlocksInFlight) {
2545 mapBlocksInFlight.equal_range(entry.pindex->GetBlockHash());
2546 while (range.first != range.second) {
2547 auto [node_id, list_it] = range.first->second;
2548 if (node_id != nodeid) {
2551 range.first = mapBlocksInFlight.erase(range.first);
2558 m_txrequest.DisconnectedPeer(nodeid);
2559 m_num_preferred_download_peers -= state->fPreferredDownload;
2560 m_peers_downloading_from -= (!state->vBlocksInFlight.empty());
2561 assert(m_peers_downloading_from >= 0);
2562 m_outbound_peers_with_protect_from_disconnect -=
2563 state->m_chain_sync.m_protect;
2564 assert(m_outbound_peers_with_protect_from_disconnect >= 0);
2566 m_node_states.erase(nodeid);
2568 if (m_node_states.empty()) {
2570 assert(mapBlocksInFlight.empty());
2571 assert(m_num_preferred_download_peers == 0);
2572 assert(m_peers_downloading_from == 0);
2573 assert(m_outbound_peers_with_protect_from_disconnect == 0);
2574 assert(m_txrequest.Size() == 0);
2576 return orphanage.Size();
2581 if (
node.fSuccessfullyConnected && !
node.IsBlockOnlyConn() &&
2582 !
node.IsInboundConn()) {
2589 LOCK(m_headers_presync_mutex);
2590 m_headers_presync_stats.erase(nodeid);
2593 WITH_LOCK(cs_proofrequest, m_proofrequest.DisconnectedPeer(nodeid));
2598PeerRef PeerManagerImpl::GetPeerRef(
NodeId id)
const {
2600 auto it = m_peer_map.find(
id);
2601 return it != m_peer_map.end() ? it->second :
nullptr;
2604PeerRef PeerManagerImpl::RemovePeer(
NodeId id) {
2607 auto it = m_peer_map.find(
id);
2608 if (it != m_peer_map.end()) {
2609 ret = std::move(it->second);
2610 m_peer_map.erase(it);
2615bool PeerManagerImpl::GetNodeStateStats(
NodeId nodeid,
2619 const CNodeState *state = State(nodeid);
2620 if (state ==
nullptr) {
2624 ? state->pindexBestKnownBlock->nHeight
2627 ? state->pindexLastCommonBlock->nHeight
2629 for (
const QueuedBlock &queue : state->vBlocksInFlight) {
2636 PeerRef peer = GetPeerRef(nodeid);
2637 if (peer ==
nullptr) {
2649 auto ping_wait{0us};
2650 if ((0 != peer->m_ping_nonce_sent) &&
2651 (0 != peer->m_ping_start.load().count())) {
2653 GetTime<std::chrono::microseconds>() - peer->m_ping_start.load();
2656 if (
auto tx_relay = peer->GetTxRelay()) {
2658 return tx_relay->m_relay_txs);
2670 LOCK(peer->m_headers_sync_mutex);
2671 if (peer->m_headers_sync) {
2679void PeerManagerImpl::AddToCompactExtraTransactions(
const CTransactionRef &tx) {
2680 if (m_opts.max_extra_txs <= 0) {
2684 if (!vExtraTxnForCompact.size()) {
2685 vExtraTxnForCompact.resize(m_opts.max_extra_txs);
2688 vExtraTxnForCompact[vExtraTxnForCompactIt] = tx;
2689 vExtraTxnForCompactIt = (vExtraTxnForCompactIt + 1) % m_opts.max_extra_txs;
2692void PeerManagerImpl::Misbehaving(Peer &peer,
const std::string &message) {
2693 LOCK(peer.m_misbehavior_mutex);
2695 const std::string message_prefixed =
2696 message.empty() ?
"" : (
": " + message);
2697 peer.m_should_discourage =
true;
2702void PeerManagerImpl::MaybePunishNodeForBlock(
NodeId nodeid,
2704 bool via_compact_block,
2705 const std::string &message) {
2706 PeerRef peer{GetPeerRef(nodeid)};
2717 if (!via_compact_block) {
2719 Misbehaving(*peer, message);
2726 CNodeState *node_state = State(nodeid);
2727 if (node_state ==
nullptr) {
2734 if (!via_compact_block && !node_state->m_is_inbound) {
2736 Misbehaving(*peer, message);
2746 Misbehaving(*peer, message);
2752 Misbehaving(*peer, message);
2758 if (message !=
"") {
2763void PeerManagerImpl::MaybePunishNodeForTx(
NodeId nodeid,
2765 const std::string &message) {
2766 PeerRef peer{GetPeerRef(nodeid)};
2773 Misbehaving(*peer, message);
2791 if (message !=
"") {
2796bool PeerManagerImpl::BlockRequestAllowed(
const CBlockIndex *pindex) {
2802 (m_chainman.m_best_header !=
nullptr) &&
2803 (m_chainman.m_best_header->GetBlockTime() - pindex->
GetBlockTime() <
2806 *m_chainman.m_best_header, *pindex, *m_chainman.m_best_header,
2810std::optional<std::string>
2811PeerManagerImpl::FetchBlock(
const Config &config,
NodeId peer_id,
2814 return "Loading blocks ...";
2820 CNodeState *state = State(peer_id);
2821 if (state ==
nullptr) {
2822 return "Peer does not exist";
2826 RemoveBlockRequest(block_index.
GetBlockHash(), std::nullopt);
2829 if (!BlockRequested(config, peer_id, block_index)) {
2830 return "Already requested from this peer";
2839 this->MakeAndPushMessage(*node, NetMsgType::GETDATA, invs);
2842 return "Node not fully connected";
2847 return std::nullopt;
2850std::unique_ptr<PeerManager>
2854 return std::make_unique<PeerManagerImpl>(connman, addrman, banman, chainman,
2863 : m_rng{opts.deterministic_rng},
2865 m_chainparams(chainman.GetParams()), m_connman(connman),
2866 m_addrman(addrman), m_banman(banman), m_chainman(chainman),
2867 m_mempool(pool), m_avalanche(
avalanche), m_opts{opts} {}
2869void PeerManagerImpl::StartScheduledTasks(
CScheduler &scheduler) {
2876 "peer eviction timer should be less than stale tip check timer");
2879 this->CheckForStaleTipAndEvictPeers();
2885 const auto reattemptBroadcastInterval =
2887 scheduler.
scheduleFromNow([&] { ReattemptInitialBroadcast(scheduler); },
2888 reattemptBroadcastInterval);
2893 UpdateAvalancheStatistics();
2899 const auto avalanchePeriodicNetworkingInterval =
2901 scheduler.
scheduleFromNow([&] { AvalanchePeriodicNetworking(scheduler); },
2902 avalanchePeriodicNetworkingInterval);
2911void PeerManagerImpl::BlockConnected(
2912 ChainstateRole role,
const std::shared_ptr<const CBlock> &pblock,
2916 m_last_tip_update = GetTime<std::chrono::seconds>();
2920 auto stalling_timeout = m_block_stalling_timeout.load();
2923 const auto new_timeout =
2924 std::max(std::chrono::duration_cast<std::chrono::seconds>(
2925 stalling_timeout * 0.85),
2927 if (m_block_stalling_timeout.compare_exchange_strong(stalling_timeout,
2947 LOCK(m_recent_confirmed_transactions_mutex);
2949 m_recent_confirmed_transactions.insert(ptx->GetId());
2954 for (
const auto &ptx : pblock->vtx) {
2955 m_txrequest.ForgetInvId(ptx->GetId());
2960void PeerManagerImpl::BlockDisconnected(
2961 const std::shared_ptr<const CBlock> &block,
const CBlockIndex *pindex) {
2970 LOCK(m_recent_confirmed_transactions_mutex);
2971 m_recent_confirmed_transactions.reset();
2978void PeerManagerImpl::NewPoWValidBlock(
2979 const CBlockIndex *pindex,
const std::shared_ptr<const CBlock> &pblock) {
2980 std::shared_ptr<const CBlockHeaderAndShortTxIDs> pcmpctblock =
2981 std::make_shared<const CBlockHeaderAndShortTxIDs>(
2986 if (pindex->
nHeight <= m_highest_fast_announce) {
2989 m_highest_fast_announce = pindex->
nHeight;
2992 const std::shared_future<CSerializedNetMsg> lazy_ser{
2993 std::async(std::launch::deferred, [&] {
2998 auto most_recent_block_txs =
2999 std::make_unique<std::map<TxId, CTransactionRef>>();
3000 for (
const auto &tx : pblock->vtx) {
3001 most_recent_block_txs->emplace(tx->GetId(), tx);
3004 LOCK(m_most_recent_block_mutex);
3005 m_most_recent_block_hash = hashBlock;
3006 m_most_recent_block = pblock;
3007 m_most_recent_compact_block = pcmpctblock;
3008 m_most_recent_block_txs = std::move(most_recent_block_txs);
3012 [
this, pindex, &lazy_ser, &hashBlock](
CNode *pnode)
3020 ProcessBlockAvailability(pnode->
GetId());
3021 CNodeState &state = *State(pnode->
GetId());
3025 if (state.m_requested_hb_cmpctblocks &&
3026 !PeerHasHeader(&state, pindex) &&
3027 PeerHasHeader(&state, pindex->
pprev)) {
3029 "%s sending header-and-ids %s to peer=%d\n",
3030 "PeerManager::NewPoWValidBlock",
3031 hashBlock.ToString(), pnode->
GetId());
3034 PushMessage(*pnode, ser_cmpctblock.Copy());
3035 state.pindexBestHeaderSent = pindex;
3044void PeerManagerImpl::UpdatedBlockTip(
const CBlockIndex *pindexNew,
3046 bool fInitialDownload) {
3047 SetBestHeight(pindexNew->
nHeight);
3051 if (fInitialDownload) {
3056 std::vector<BlockHash> vHashes;
3058 while (pindexToAnnounce != pindexFork) {
3060 pindexToAnnounce = pindexToAnnounce->
pprev;
3070 for (
auto &it : m_peer_map) {
3071 Peer &peer = *it.second;
3072 LOCK(peer.m_block_inv_mutex);
3074 peer.m_blocks_for_headers_relay.push_back(hash);
3086void PeerManagerImpl::BlockChecked(
const CBlock &block,
3091 std::map<BlockHash, std::pair<NodeId, bool>>::iterator it =
3092 mapBlockSource.find(hash);
3096 if (state.
IsInvalid() && it != mapBlockSource.end() &&
3097 State(it->second.first)) {
3098 MaybePunishNodeForBlock(it->second.first, state,
3099 !it->second.second);
3108 mapBlocksInFlight.count(hash) == mapBlocksInFlight.size()) {
3109 if (it != mapBlockSource.end()) {
3110 MaybeSetPeerAsAnnouncingHeaderAndIDs(it->second.first);
3114 if (it != mapBlockSource.end()) {
3115 mapBlockSource.erase(it);
3124bool PeerManagerImpl::AlreadyHaveTx(
const TxId &txid,
3125 bool include_reconsiderable) {
3127 hashRecentRejectsChainTip) {
3132 hashRecentRejectsChainTip =
3134 m_recent_rejects.reset();
3135 m_recent_rejects_package_reconsiderable.reset();
3139 return orphanage.HaveTx(txid);
3145 return conflicting.HaveTx(txid);
3150 if (include_reconsiderable &&
3151 m_recent_rejects_package_reconsiderable.contains(txid)) {
3156 LOCK(m_recent_confirmed_transactions_mutex);
3157 if (m_recent_confirmed_transactions.contains(txid)) {
3162 return m_recent_rejects.contains(txid) || m_mempool.
exists(txid);
3165bool PeerManagerImpl::AlreadyHaveBlock(
const BlockHash &block_hash) {
3170 if (!
Assume(m_avalanche)) {
3175 if (localProof && localProof->getId() == proofid) {
3184void PeerManagerImpl::SendPings() {
3186 for (
auto &it : m_peer_map) {
3187 it.second->m_ping_queued =
true;
3191void PeerManagerImpl::RelayTransaction(
const TxId &txid) {
3193 for (
auto &it : m_peer_map) {
3194 Peer &peer = *it.second;
3195 auto tx_relay = peer.GetTxRelay();
3199 LOCK(tx_relay->m_tx_inventory_mutex);
3205 if (tx_relay->m_next_inv_send_time == 0s) {
3209 if (!tx_relay->m_tx_inventory_known_filter.contains(txid) ||
3210 tx_relay->m_avalanche_stalled_txids.count(txid) > 0) {
3211 tx_relay->m_tx_inventory_to_send.insert(txid);
3218 for (
auto &it : m_peer_map) {
3219 Peer &peer = *it.second;
3221 if (!peer.m_proof_relay) {
3224 LOCK(peer.m_proof_relay->m_proof_inventory_mutex);
3225 if (!peer.m_proof_relay->m_proof_inventory_known_filter.contains(
3227 peer.m_proof_relay->m_proof_inventory_to_send.insert(proofid);
3232void PeerManagerImpl::RelayAddress(
NodeId originator,
const CAddress &addr,
3248 const auto current_time{GetTime<std::chrono::seconds>()};
3251 const uint64_t time_addr{
3252 (
static_cast<uint64_t
>(
count_seconds(current_time)) + hash_addr) /
3262 unsigned int nRelayNodes = (fReachable || (hasher.Finalize() & 1)) ? 2 : 1;
3263 std::array<std::pair<uint64_t, Peer *>, 2> best{
3264 {{0,
nullptr}, {0,
nullptr}}};
3265 assert(nRelayNodes <= best.size());
3269 for (
auto &[
id, peer] : m_peer_map) {
3270 if (peer->m_addr_relay_enabled &&
id != originator &&
3271 IsAddrCompatible(*peer, addr)) {
3273 for (
unsigned int i = 0; i < nRelayNodes; i++) {
3274 if (hashKey > best[i].first) {
3275 std::copy(best.begin() + i, best.begin() + nRelayNodes - 1,
3276 best.begin() + i + 1);
3277 best[i] = std::make_pair(hashKey, peer.get());
3284 for (
unsigned int i = 0; i < nRelayNodes && best[i].first != 0; i++) {
3285 PushAddress(*best[i].second, addr);
3289void PeerManagerImpl::ProcessGetBlockData(
const Config &config,
CNode &pfrom,
3290 Peer &peer,
const CInv &inv) {
3293 std::shared_ptr<const CBlock> a_recent_block;
3294 std::shared_ptr<const CBlockHeaderAndShortTxIDs> a_recent_compact_block;
3296 LOCK(m_most_recent_block_mutex);
3297 a_recent_block = m_most_recent_block;
3298 a_recent_compact_block = m_most_recent_compact_block;
3301 bool need_activate_chain =
false;
3315 need_activate_chain =
true;
3319 if (need_activate_chain) {
3322 state, a_recent_block, m_avalanche)) {
3330 bool can_direct_fetch{
false};
3338 if (!BlockRequestAllowed(pindex)) {
3340 "%s: ignoring request from peer=%i for old "
3341 "block that isn't in the main chain\n",
3342 __func__, pfrom.
GetId());
3348 (((m_chainman.m_best_header !=
nullptr) &&
3349 (m_chainman.m_best_header->GetBlockTime() -
3357 "historical block serving limit reached, disconnect peer=%d\n",
3370 (tip->nHeight - pindex->
nHeight >
3373 "Ignore block request below NODE_NETWORK_LIMITED "
3374 "threshold, disconnect peer=%d\n",
3384 if (!pindex->nStatus.hasData()) {
3387 can_direct_fetch = CanDirectFetch();
3391 std::shared_ptr<const CBlock> pblock;
3392 auto handle_block_read_error = [&]() {
3394 return m_chainman.
m_blockman.IsBlockPruned(*pindex))) {
3396 "Block was pruned before it could be read, disconnect "
3400 LogError(
"Cannot load block from disk, disconnect peer=%d\n",
3406 if (a_recent_block && a_recent_block->GetHash() == pindex->
GetBlockHash()) {
3407 pblock = a_recent_block;
3411 std::vector<uint8_t> block_data;
3413 handle_block_read_error();
3420 std::shared_ptr<CBlock> pblockRead = std::make_shared<CBlock>();
3422 handle_block_read_error();
3425 pblock = pblockRead;
3431 bool sendMerkleBlock =
false;
3433 if (
auto tx_relay = peer.GetTxRelay()) {
3434 LOCK(tx_relay->m_bloom_filter_mutex);
3435 if (tx_relay->m_bloom_filter) {
3436 sendMerkleBlock =
true;
3441 if (sendMerkleBlock) {
3452 typedef std::pair<size_t, uint256> PairType;
3455 *pblock->vtx[pair.first]);
3465 if (can_direct_fetch &&
3467 if (a_recent_compact_block &&
3468 a_recent_compact_block->header.GetHash() ==
3471 *a_recent_compact_block);
3485 LOCK(peer.m_block_inv_mutex);
3488 if (hash == peer.m_continuation_block) {
3492 std::vector<CInv> vInv;
3495 peer.m_continuation_block =
BlockHash();
3501PeerManagerImpl::FindTxForGetData(
const Peer &peer,
const TxId &txid,
3502 const std::chrono::seconds mempool_req,
3503 const std::chrono::seconds now) {
3504 auto txinfo = m_mempool.
info(txid);
3509 if ((mempool_req.count() && txinfo.m_time <= mempool_req) ||
3511 return std::move(txinfo.tx);
3520 Assume(peer.GetTxRelay())->m_recently_announced_invs.contains(txid);
3521 if (recent && txinfo.tx) {
3522 return std::move(txinfo.tx);
3527 LOCK(m_most_recent_block_mutex);
3528 if (m_most_recent_block_txs !=
nullptr) {
3529 auto it = m_most_recent_block_txs->find(txid);
3530 if (it != m_most_recent_block_txs->end()) {
3543PeerManagerImpl::FindProofForGetData(
const Peer &peer,
3545 const std::chrono::seconds now) {
3548 bool send_unconditionally =
3574 if (send_unconditionally) {
3579 if (peer.m_proof_relay->m_recently_announced_proofs.contains(proofid)) {
3586void PeerManagerImpl::ProcessGetData(
3588 const std::atomic<bool> &interruptMsgProc) {
3591 auto tx_relay = peer.GetTxRelay();
3593 std::deque<CInv>::iterator it = peer.m_getdata_requests.begin();
3594 std::vector<CInv> vNotFound;
3596 const auto now{GetTime<std::chrono::seconds>()};
3598 const auto mempool_req = tx_relay !=
nullptr
3599 ? tx_relay->m_last_mempool_req.load()
3600 : std::chrono::seconds::min();
3605 while (it != peer.m_getdata_requests.end() &&
3606 (it->IsMsgProof() || it->IsMsgTx())) {
3607 if (interruptMsgProc) {
3616 const CInv &inv = *it++;
3620 vNotFound.push_back(inv);
3624 auto proof = FindProofForGetData(peer, proofid, now);
3631 vNotFound.push_back(inv);
3638 if (tx_relay ==
nullptr) {
3652 std::vector<TxId> parent_ids_to_add;
3655 auto txiter = m_mempool.
GetIter(tx->GetId());
3657 auto &pentry = *txiter;
3659 (*pentry)->GetMemPoolParentsConst();
3660 parent_ids_to_add.reserve(parents.size());
3661 for (
const auto &parent : parents) {
3662 if (parent.get()->GetTime() >
3664 parent_ids_to_add.push_back(
3665 parent.get()->GetTx().GetId());
3670 for (
const TxId &parent_txid : parent_ids_to_add) {
3673 if (
WITH_LOCK(tx_relay->m_tx_inventory_mutex,
3674 return !tx_relay->m_tx_inventory_known_filter
3675 .contains(parent_txid))) {
3676 tx_relay->m_recently_announced_invs.insert(parent_txid);
3680 vNotFound.push_back(inv);
3692 if (it != peer.m_getdata_requests.end() && !pfrom.
fPauseSend) {
3693 const CInv &inv = *it++;
3695 ProcessGetBlockData(config, pfrom, peer, inv);
3701 peer.m_getdata_requests.erase(peer.m_getdata_requests.begin(), it);
3703 if (!vNotFound.empty()) {
3720void PeerManagerImpl::SendBlockTransactions(
3724 for (
size_t i = 0; i < req.
indices.size(); i++) {
3726 Misbehaving(peer,
"getblocktxn with out-of-bounds tx indices");
3735bool PeerManagerImpl::CheckHeadersPoW(
const std::vector<CBlockHeader> &headers,
3740 Misbehaving(peer,
"header with invalid proof of work");
3745 if (!CheckHeadersAreContinuous(headers)) {
3746 Misbehaving(peer,
"non-continuous headers sequence");
3759 near_chaintip_work =
3772void PeerManagerImpl::HandleUnconnectingHeaders(
3773 CNode &pfrom, Peer &peer,
const std::vector<CBlockHeader> &headers) {
3777 if (MaybeSendGetHeaders(pfrom,
GetLocator(best_header), peer)) {
3780 "received header %s: missing prev block %s, sending getheaders "
3781 "(%d) to end (peer=%d)\n",
3783 headers[0].hashPrevBlock.ToString(), best_header->nHeight,
3791 UpdateBlockAvailability(pfrom.
GetId(), headers.back().GetHash()));
3794bool PeerManagerImpl::CheckHeadersAreContinuous(
3795 const std::vector<CBlockHeader> &headers)
const {
3798 if (!hashLastBlock.
IsNull() && header.hashPrevBlock != hashLastBlock) {
3801 hashLastBlock = header.GetHash();
3806bool PeerManagerImpl::IsContinuationOfLowWorkHeadersSync(
3807 Peer &peer,
CNode &pfrom, std::vector<CBlockHeader> &headers) {
3808 if (peer.m_headers_sync) {
3809 auto result = peer.m_headers_sync->ProcessNextHeaders(
3813 if (result.success) {
3814 peer.m_last_getheaders_timestamp = {};
3816 if (result.request_more) {
3817 auto locator = peer.m_headers_sync->NextHeadersRequestLocator();
3820 Assume(!locator.vHave.empty());
3824 if (!locator.vHave.empty()) {
3827 bool sent_getheaders =
3828 MaybeSendGetHeaders(pfrom, locator, peer);
3831 locator.vHave.front().ToString(), pfrom.
GetId());
3836 peer.m_headers_sync.reset(
nullptr);
3841 LOCK(m_headers_presync_mutex);
3842 m_headers_presync_stats.erase(pfrom.
GetId());
3845 HeadersPresyncStats stats;
3846 stats.first = peer.m_headers_sync->GetPresyncWork();
3847 if (peer.m_headers_sync->GetState() ==
3849 stats.second = {peer.m_headers_sync->GetPresyncHeight(),
3850 peer.m_headers_sync->GetPresyncTime()};
3854 LOCK(m_headers_presync_mutex);
3855 m_headers_presync_stats[pfrom.
GetId()] = stats;
3857 m_headers_presync_stats.find(m_headers_presync_bestpeer);
3858 bool best_updated =
false;
3859 if (best_it == m_headers_presync_stats.end()) {
3864 const HeadersPresyncStats *stat_best{
nullptr};
3865 for (
const auto &[_peer, _stat] : m_headers_presync_stats) {
3866 if (!stat_best || _stat > *stat_best) {
3871 m_headers_presync_bestpeer = peer_best;
3872 best_updated = (peer_best == pfrom.
GetId());
3873 }
else if (best_it->first == pfrom.
GetId() ||
3874 stats > best_it->second) {
3877 m_headers_presync_bestpeer = pfrom.
GetId();
3878 best_updated =
true;
3880 if (best_updated && stats.second.has_value()) {
3883 m_headers_presync_should_signal =
true;
3887 if (result.success) {
3890 headers.swap(result.pow_validated_headers);
3893 return result.success;
3901bool PeerManagerImpl::TryLowWorkHeadersSync(
3903 std::vector<CBlockHeader> &headers) {
3910 arith_uint256 minimum_chain_work = GetAntiDoSWorkThreshold();
3914 if (total_work < minimum_chain_work) {
3928 LOCK(peer.m_headers_sync_mutex);
3929 peer.m_headers_sync.reset(
3931 chain_start_header, minimum_chain_work));
3936 (void)IsContinuationOfLowWorkHeadersSync(peer, pfrom, headers);
3939 "Ignoring low-work chain (height=%u) from peer=%d\n",
3940 chain_start_header->
nHeight + headers.size(),
3952bool PeerManagerImpl::IsAncestorOfBestHeaderOrTip(
const CBlockIndex *header) {
3953 return header !=
nullptr &&
3954 ((m_chainman.m_best_header !=
nullptr &&
3956 m_chainman.m_best_header->GetAncestor(header->
nHeight)) ||
3960bool PeerManagerImpl::MaybeSendGetHeaders(
CNode &pfrom,
3967 if (current_time - peer.m_last_getheaders_timestamp >
3970 peer.m_last_getheaders_timestamp = current_time;
3982void PeerManagerImpl::HeadersDirectFetchBlocks(
const Config &config,
3986 CNodeState *nodestate = State(pfrom.
GetId());
3990 std::vector<const CBlockIndex *> vToFetch;
3996 if (!pindexWalk->nStatus.hasData() &&
3999 vToFetch.push_back(pindexWalk);
4001 pindexWalk = pindexWalk->
pprev;
4012 std::vector<CInv> vGetData;
4015 if (nodestate->vBlocksInFlight.size() >=
4021 BlockRequested(config, pfrom.
GetId(), *pindex);
4025 if (vGetData.size() > 1) {
4027 "Downloading blocks toward %s (%d) via headers "
4032 if (vGetData.size() > 0) {
4033 if (!m_opts.ignore_incoming_txs &&
4034 nodestate->m_provides_cmpctblocks && vGetData.size() == 1 &&
4035 mapBlocksInFlight.size() == 1 &&
4052void PeerManagerImpl::UpdatePeerStateForReceivedHeaders(
4054 bool received_new_header,
bool may_have_more_headers) {
4057 CNodeState *nodestate = State(pfrom.
GetId());
4065 if (received_new_header &&
4067 nodestate->m_last_block_announcement =
GetTime();
4075 if (nodestate->pindexBestKnownBlock &&
4076 nodestate->pindexBestKnownBlock->nChainWork <
4087 LogPrintf(
"Disconnecting outbound peer %d -- headers "
4088 "chain has insufficient work\n",
4102 nodestate->pindexBestKnownBlock !=
nullptr) {
4103 if (m_outbound_peers_with_protect_from_disconnect <
4105 nodestate->pindexBestKnownBlock->nChainWork >=
4107 !nodestate->m_chain_sync.m_protect) {
4110 nodestate->m_chain_sync.m_protect =
true;
4111 ++m_outbound_peers_with_protect_from_disconnect;
4116void PeerManagerImpl::ProcessHeadersMessage(
const Config &config,
CNode &pfrom,
4118 std::vector<CBlockHeader> &&headers,
4119 bool via_compact_block) {
4120 size_t nCount = headers.size();
4128 LOCK(peer.m_headers_sync_mutex);
4129 if (peer.m_headers_sync) {
4130 peer.m_headers_sync.reset(
nullptr);
4131 LOCK(m_headers_presync_mutex);
4132 m_headers_presync_stats.erase(pfrom.
GetId());
4137 peer.m_last_getheaders_timestamp = {};
4145 if (!CheckHeadersPoW(headers, m_chainparams.
GetConsensus(), peer)) {
4160 bool already_validated_work =
false;
4163 bool have_headers_sync =
false;
4165 LOCK(peer.m_headers_sync_mutex);
4167 already_validated_work =
4168 IsContinuationOfLowWorkHeadersSync(peer, pfrom, headers);
4180 if (headers.empty()) {
4184 have_headers_sync = !!peer.m_headers_sync;
4190 headers[0].hashPrevBlock))};
4191 bool headers_connect_blockindex{chain_start_header !=
nullptr};
4193 if (!headers_connect_blockindex) {
4197 HandleUnconnectingHeaders(pfrom, peer, headers);
4205 peer.m_last_getheaders_timestamp = {};
4214 last_received_header =
4216 if (IsAncestorOfBestHeaderOrTip(last_received_header)) {
4217 already_validated_work =
true;
4225 already_validated_work =
true;
4231 if (!already_validated_work &&
4232 TryLowWorkHeadersSync(peer, pfrom, chain_start_header, headers)) {
4244 bool received_new_header{last_received_header ==
nullptr};
4249 state, &pindexLast)) {
4251 MaybePunishNodeForBlock(pfrom.
GetId(), state, via_compact_block,
4252 "invalid header received");
4262 if (MaybeSendGetHeaders(pfrom,
GetLocator(pindexLast), peer)) {
4265 "more getheaders (%d) to end to peer=%d (startheight:%d)\n",
4266 pindexLast->
nHeight, pfrom.
GetId(), peer.m_starting_height);
4270 UpdatePeerStateForReceivedHeaders(pfrom, peer, *pindexLast,
4271 received_new_header,
4275 HeadersDirectFetchBlocks(config, pfrom, *pindexLast);
4278void PeerManagerImpl::ProcessInvalidTx(
NodeId nodeid,
4281 bool maybe_add_extra_compact_tx) {
4286 const TxId &txid = ptx->GetId();
4306 m_recent_rejects_package_reconsiderable.insert(txid);
4308 m_recent_rejects.insert(txid);
4310 m_txrequest.ForgetInvId(txid);
4313 AddToCompactExtraTransactions(ptx);
4316 MaybePunishNodeForTx(nodeid, state);
4322 return orphanage.EraseTx(txid);
4336 m_txrequest.ForgetInvId(tx->GetId());
4342 orphanage.
EraseTx(tx->GetId());
4347 "AcceptToMemoryPool: peer=%d: accepted %s (poolsz %u txn, %u kB)\n",
4348 nodeid, tx->GetId().ToString(), m_mempool.
size(),
4351 RelayTransaction(tx->GetId());
4354void PeerManagerImpl::ProcessPackageResult(
4355 const PackageToValidate &package_to_validate,
4361 const auto &
package = package_to_validate.m_txns;
4362 const auto &senders = package_to_validate.m_senders;
4365 m_recent_rejects_package_reconsiderable.insert(
GetPackageHash(package));
4369 if (!
Assume(package.size() == 2)) {
4375 auto package_iter = package.rbegin();
4376 auto senders_iter = senders.rbegin();
4377 while (package_iter != package.rend()) {
4378 const auto &tx = *package_iter;
4379 const NodeId nodeid = *senders_iter;
4380 const auto it_result{package_result.
m_tx_results.find(tx->GetId())};
4384 const auto &tx_result = it_result->second;
4385 switch (tx_result.m_result_type) {
4387 ProcessValidTx(nodeid, tx);
4397 ProcessInvalidTx(nodeid, tx, tx_result.m_state,
4414std::optional<PeerManagerImpl::PackageToValidate>
4420 const auto &parent_txid{ptx->GetId()};
4422 Assume(m_recent_rejects_package_reconsiderable.contains(parent_txid));
4428 const auto cpfp_candidates_same_peer{
4434 for (
const auto &child : cpfp_candidates_same_peer) {
4435 Package maybe_cpfp_package{ptx, child};
4436 if (!m_recent_rejects_package_reconsiderable.contains(
4438 return PeerManagerImpl::PackageToValidate{ptx, child, nodeid,
4452 const auto cpfp_candidates_different_peer{
4462 std::vector<size_t> tx_indices(cpfp_candidates_different_peer.size());
4463 std::iota(tx_indices.begin(), tx_indices.end(), 0);
4464 Shuffle(tx_indices.begin(), tx_indices.end(), m_rng);
4466 for (
const auto index : tx_indices) {
4469 const auto [child_tx, child_sender] =
4470 cpfp_candidates_different_peer.at(index);
4471 Package maybe_cpfp_package{ptx, child_tx};
4472 if (!m_recent_rejects_package_reconsiderable.contains(
4474 return PeerManagerImpl::PackageToValidate{ptx, child_tx, nodeid,
4478 return std::nullopt;
4481bool PeerManagerImpl::ProcessOrphanTx(
const Config &config, Peer &peer) {
4487 return orphanage.GetTxToReconsider(peer.m_id);
4492 const TxId &orphanTxId = porphanTx->GetId();
4497 ProcessValidTx(peer.m_id, porphanTx);
4503 " invalid orphan tx %s from peer=%d. %s\n",
4510 ProcessInvalidTx(peer.m_id, porphanTx, state,
4521bool PeerManagerImpl::PrepareBlockFilterRequest(
4523 const BlockHash &stop_hash, uint32_t max_height_diff,
4525 const bool supported_filter_type =
4528 if (!supported_filter_type) {
4530 "peer %d requested unsupported block filter type: %d\n",
4531 node.GetId(),
static_cast<uint8_t
>(filter_type));
4532 node.fDisconnect =
true;
4542 if (!stop_index || !BlockRequestAllowed(stop_index)) {
4545 node.fDisconnect =
true;
4550 uint32_t stop_height = stop_index->
nHeight;
4551 if (start_height > stop_height) {
4554 "peer %d sent invalid getcfilters/getcfheaders with "
4556 "start height %d and stop height %d\n",
4557 node.GetId(), start_height, stop_height);
4558 node.fDisconnect =
true;
4561 if (stop_height - start_height >= max_height_diff) {
4563 "peer %d requested too many cfilters/cfheaders: %d / %d\n",
4564 node.GetId(), stop_height - start_height + 1, max_height_diff);
4565 node.fDisconnect =
true;
4570 if (!filter_index) {
4579void PeerManagerImpl::ProcessGetCFilters(
CNode &
node, Peer &peer,
4581 uint8_t filter_type_ser;
4582 uint32_t start_height;
4585 vRecv >> filter_type_ser >> start_height >> stop_hash;
4592 if (!PrepareBlockFilterRequest(
node, peer, filter_type, start_height,
4598 std::vector<BlockFilter> filters;
4601 "Failed to find block filter in index: filter_type=%s, "
4602 "start_height=%d, stop_hash=%s\n",
4608 for (
const auto &filter : filters) {
4613void PeerManagerImpl::ProcessGetCFHeaders(
CNode &
node, Peer &peer,
4615 uint8_t filter_type_ser;
4616 uint32_t start_height;
4619 vRecv >> filter_type_ser >> start_height >> stop_hash;
4626 if (!PrepareBlockFilterRequest(
node, peer, filter_type, start_height,
4633 if (start_height > 0) {
4635 stop_index->
GetAncestor(
static_cast<int>(start_height - 1));
4638 "Failed to find block filter header in index: "
4639 "filter_type=%s, block_hash=%s\n",
4646 std::vector<uint256> filter_hashes;
4650 "Failed to find block filter hashes in index: filter_type=%s, "
4651 "start_height=%d, stop_hash=%s\n",
4658 stop_index->
GetBlockHash(), prev_header, filter_hashes);
4661void PeerManagerImpl::ProcessGetCFCheckPt(
CNode &
node, Peer &peer,
4663 uint8_t filter_type_ser;
4666 vRecv >> filter_type_ser >> stop_hash;
4673 if (!PrepareBlockFilterRequest(
4674 node, peer, filter_type, 0, stop_hash,
4675 std::numeric_limits<uint32_t>::max(),
4676 stop_index, filter_index)) {
4684 for (
int i = headers.size() - 1; i >= 0; i--) {
4690 "Failed to find block filter header in index: "
4691 "filter_type=%s, block_hash=%s\n",
4714PeerManagerImpl::GetAvalancheVoteForBlock(
const BlockHash &hash)
const {
4725 if (pindex->nStatus.isInvalid()) {
4730 if (pindex->nStatus.isOnParkedChain()) {
4738 if (pindex == pindexFork) {
4743 if (pindexFork != pindexTip) {
4748 if (!pindex->nStatus.hasData()) {
4759 const TxId &
id)
const {
4761 if (
WITH_LOCK(m_recent_confirmed_transactions_mutex,
4762 return m_recent_confirmed_transactions.contains(
id))) {
4771 if (m_recent_rejects.contains(
id)) {
4783 if (
auto iter = m_mempool.
GetIter(
id)) {
4784 mempool_tx = (**iter)->GetSharedTx();
4789 return conflicting.HaveTx(id);
4796 return orphanage.HaveTx(id);
4862 const std::shared_ptr<const CBlock> &block,
4863 bool force_processing,
4864 bool min_pow_checked) {
4865 bool new_block{
false};
4867 &new_block, m_avalanche);
4869 node.m_last_block_time = GetTime<std::chrono::seconds>();
4874 RemoveBlockRequest(block->GetHash(), std::nullopt);
4877 mapBlockSource.erase(block->GetHash());
4881void PeerManagerImpl::ProcessMessage(
4882 const Config &config,
CNode &pfrom,
const std::string &msg_type,
4883 DataStream &vRecv,
const std::chrono::microseconds time_received,
4884 const std::atomic<bool> &interruptMsgProc) {
4890 PeerRef peer = GetPeerRef(pfrom.
GetId());
4891 if (peer ==
nullptr) {
4897 "Avalanche is not initialized, ignoring %s message\n",
4912 uint64_t nNonce = 1;
4915 std::string cleanSubVer;
4916 int starting_height = -1;
4918 uint64_t nExtraEntropy = 1;
4920 vRecv >> nVersion >> Using<CustomUintFormatter<8>>(nServices) >> nTime;
4933 "peer=%d does not offer the expected services "
4934 "(%08x offered, %08x expected); disconnecting\n",
4935 pfrom.
GetId(), nServices,
4945 "peer=%d does not offer the avalanche service; disconnecting\n",
4954 "peer=%d using obsolete version %i; disconnecting\n",
4955 pfrom.
GetId(), nVersion);
4960 if (!vRecv.
empty()) {
4969 if (!vRecv.
empty()) {
4970 std::string strSubVer;
4974 if (!vRecv.
empty()) {
4975 vRecv >> starting_height;
4977 if (!vRecv.
empty()) {
4980 if (!vRecv.
empty()) {
4981 vRecv >> nExtraEntropy;
4985 LogPrintf(
"connected to self at %s, disconnecting\n",
4998 PushNodeVersion(config, pfrom, *peer);
5002 const int greatest_common_version =
5014 peer->m_their_services = nServices;
5018 pfrom.cleanSubVer = cleanSubVer;
5020 peer->m_starting_height = starting_height;
5028 (fRelay || (peer->m_our_services &
NODE_BLOOM))) {
5029 auto *
const tx_relay = peer->SetTxRelay();
5031 LOCK(tx_relay->m_bloom_filter_mutex);
5033 tx_relay->m_relay_txs = fRelay;
5046 CNodeState *state = State(pfrom.
GetId());
5047 state->fPreferredDownload =
5051 m_num_preferred_download_peers += state->fPreferredDownload;
5057 bool send_getaddr{
false};
5059 send_getaddr = SetupAddressRelay(pfrom, *peer);
5070 peer->m_getaddr_sent =
true;
5074 WITH_LOCK(peer->m_addr_token_bucket_mutex,
5075 peer->m_addr_token_bucket += m_opts.max_addr_to_send);
5096 std::string remoteAddr;
5102 "receive version message: [%s] %s: version %d, blocks=%d, "
5103 "us=%s, txrelay=%d, peer=%d%s\n",
5106 pfrom.
GetId(), remoteAddr);
5108 int64_t currentTime =
GetTime();
5109 int64_t nTimeOffset = nTime - currentTime;
5114 Misbehaving(*peer,
"Ignoring invalid timestamp in version message");
5124 "feeler connection completed peer=%d; disconnecting\n",
5133 Misbehaving(*peer,
"non-version message before version handshake");
5140 "ignoring redundant verack message from peer=%d\n",
5146 LogPrintf(
"New outbound peer connected: version: %d, blocks=%d, "
5148 pfrom.
nVersion.load(), peer->m_starting_height,
5171 AddKnownProof(*peer, localProof->getId());
5175 peer->m_proof_relay->m_recently_announced_proofs.insert(
5176 localProof->getId());
5181 if (
auto tx_relay = peer->GetTxRelay()) {
5190 return tx_relay->m_tx_inventory_to_send.empty() &&
5191 tx_relay->m_next_inv_send_time == 0s));
5200 Misbehaving(*peer,
"non-verack message before version handshake");
5205 const auto ser_params{
5214 std::vector<CAddress> vAddr;
5218 if (!SetupAddressRelay(pfrom, *peer)) {
5224 if (vAddr.size() > m_opts.max_addr_to_send) {
5225 Misbehaving(*peer,
strprintf(
"%s message size = %u", msg_type,
5231 std::vector<CAddress> vAddrOk;
5232 const auto current_a_time{Now<NodeSeconds>()};
5235 const auto current_time = GetTime<std::chrono::microseconds>();
5237 LOCK(peer->m_addr_token_bucket_mutex);
5240 const auto time_diff =
5241 std::max(current_time - peer->m_addr_token_timestamp, 0us);
5242 const double increment =
5244 peer->m_addr_token_bucket =
5245 std::min<double>(peer->m_addr_token_bucket + increment,
5249 peer->m_addr_token_timestamp = current_time;
5251 const bool rate_limited =
5253 uint64_t num_proc = 0;
5254 uint64_t num_rate_limit = 0;
5255 Shuffle(vAddr.begin(), vAddr.end(), m_rng);
5257 if (interruptMsgProc) {
5262 LOCK(peer->m_addr_token_bucket_mutex);
5264 if (peer->m_addr_token_bucket < 1.0) {
5270 peer->m_addr_token_bucket -= 1.0;
5283 addr.
nTime > current_a_time + 10min) {
5284 addr.
nTime = current_a_time - 5 * 24h;
5286 AddAddressKnown(*peer, addr);
5295 if (addr.
nTime > current_a_time - 10min && !peer->m_getaddr_sent &&
5298 RelayAddress(pfrom.
GetId(), addr, fReachable);
5302 vAddrOk.push_back(addr);
5305 peer->m_addr_processed += num_proc;
5306 peer->m_addr_rate_limited += num_rate_limit;
5308 "Received addr: %u addresses (%u processed, %u rate-limited) "
5310 vAddr.size(), num_proc, num_rate_limit, pfrom.
GetId());
5312 m_addrman.
Add(vAddrOk, pfrom.
addr, 2h);
5313 if (vAddr.size() < 1000) {
5314 peer->m_getaddr_sent =
false;
5321 "addrfetch connection completed peer=%d; disconnecting\n",
5329 peer->m_wants_addrv2 =
true;
5334 peer->m_prefers_headers =
true;
5339 bool sendcmpct_hb{
false};
5340 uint64_t sendcmpct_version{0};
5341 vRecv >> sendcmpct_hb >> sendcmpct_version;
5348 CNodeState *nodestate = State(pfrom.
GetId());
5349 nodestate->m_provides_cmpctblocks =
true;
5350 nodestate->m_requested_hb_cmpctblocks = sendcmpct_hb;
5359 std::vector<CInv> vInv;
5362 Misbehaving(*peer,
strprintf(
"inv message size = %u", vInv.size()));
5366 const bool reject_tx_invs{RejectIncomingTxs(pfrom)};
5368 const auto current_time{GetTime<std::chrono::microseconds>()};
5369 std::optional<BlockHash> best_block;
5371 auto logInv = [&](
const CInv &inv,
bool fAlreadyHave) {
5373 fAlreadyHave ?
"have" :
"new", pfrom.
GetId());
5376 for (
CInv &inv : vInv) {
5377 if (interruptMsgProc) {
5389 const bool fAlreadyHave = AlreadyHaveBlock(
BlockHash(inv.
hash));
5390 logInv(inv, fAlreadyHave);
5393 UpdateBlockAvailability(pfrom.
GetId(), hash);
5395 !IsBlockRequested(hash)) {
5402 best_block = std::move(hash);
5413 const bool fAlreadyHave = AlreadyHaveProof(proofid);
5414 logInv(inv, fAlreadyHave);
5415 AddKnownProof(*peer, proofid);
5417 if (!fAlreadyHave && m_avalanche &&
5419 const bool preferred = isPreferredDownloadPeer(pfrom);
5421 LOCK(cs_proofrequest);
5422 AddProofAnnouncement(pfrom, proofid, current_time,
5431 const bool fAlreadyHave =
5432 AlreadyHaveTx(txid,
true);
5433 logInv(inv, fAlreadyHave);
5435 AddKnownTx(*peer, txid);
5436 if (reject_tx_invs) {
5438 "transaction (%s) inv sent in violation of "
5439 "protocol, disconnecting peer=%d\n",
5443 }
else if (!fAlreadyHave &&
5445 AddTxAnnouncement(pfrom, txid, current_time);
5452 "Unknown inv type \"%s\" received from peer=%d\n",
5469 if (state.fSyncStarted ||
5470 (!peer->m_inv_triggered_getheaders_before_sync &&
5471 *best_block != m_last_block_inv_triggering_headers_sync)) {
5472 if (MaybeSendGetHeaders(
5473 pfrom,
GetLocator(m_chainman.m_best_header), *peer)) {
5475 m_chainman.m_best_header->nHeight,
5476 best_block->ToString(), pfrom.
GetId());
5478 if (!state.fSyncStarted) {
5479 peer->m_inv_triggered_getheaders_before_sync =
true;
5483 m_last_block_inv_triggering_headers_sync = *best_block;
5492 std::vector<CInv> vInv;
5496 strprintf(
"getdata message size = %u", vInv.size()));
5501 vInv.size(), pfrom.
GetId());
5503 if (vInv.size() > 0) {
5509 LOCK(peer->m_getdata_requests_mutex);
5510 peer->m_getdata_requests.insert(peer->m_getdata_requests.end(),
5511 vInv.begin(), vInv.end());
5512 ProcessGetData(config, pfrom, *peer, interruptMsgProc);
5521 vRecv >> locator >> hashStop;
5525 "getblocks locator size %lld > %d, disconnect peer=%d\n",
5539 std::shared_ptr<const CBlock> a_recent_block;
5541 LOCK(m_most_recent_block_mutex);
5542 a_recent_block = m_most_recent_block;
5546 state, a_recent_block, m_avalanche)) {
5564 (pindex ? pindex->
nHeight : -1),
5567 for (; pindex; pindex = m_chainman.
ActiveChain().Next(pindex)) {
5576 const int nPrunedBlocksLikelyToHave =
5580 (!pindex->nStatus.hasData() ||
5582 nPrunedBlocksLikelyToHave)) {
5585 " getblocks stopping, pruned or too old block at %d %s\n",
5590 peer->m_block_inv_mutex,
5591 peer->m_blocks_for_inv_relay.push_back(pindex->
GetBlockHash()));
5592 if (--nLimit <= 0) {
5598 peer->m_continuation_block = pindex->GetBlockHash();
5610 std::shared_ptr<const CBlock> recent_block;
5612 LOCK(m_most_recent_block_mutex);
5613 if (m_most_recent_block_hash == req.
blockhash) {
5614 recent_block = m_most_recent_block;
5619 SendBlockTransactions(pfrom, *peer, *recent_block, req);
5629 if (!pindex || !pindex->nStatus.hasData()) {
5632 "Peer %d sent us a getblocktxn for a block we don't have\n",
5643 if (!block_pos.IsNull()) {
5651 SendBlockTransactions(pfrom, *peer, block, req);
5663 "Peer %d sent us a getblocktxn for a block > %i deep\n",
5668 WITH_LOCK(peer->m_getdata_requests_mutex,
5669 peer->m_getdata_requests.push_back(inv));
5678 vRecv >> locator >> hashStop;
5682 "getheaders locator size %lld > %d, disconnect peer=%d\n",
5691 "Ignoring getheaders from peer=%d while importing/reindexing\n",
5705 if (m_chainman.
ActiveTip() ==
nullptr ||
5710 "Ignoring getheaders from peer=%d because active chain "
5711 "has too little work; sending empty response\n",
5716 std::vector<CBlock>());
5720 CNodeState *nodestate = State(pfrom.
GetId());
5729 if (!BlockRequestAllowed(pindex)) {
5731 "%s: ignoring request from peer=%i for old block "
5732 "header that isn't in the main chain\n",
5733 __func__, pfrom.
GetId());
5747 std::vector<CBlock> vHeaders;
5750 (pindex ? pindex->
nHeight : -1),
5753 for (; pindex; pindex = m_chainman.
ActiveChain().Next(pindex)) {
5755 if (--nLimit <= 0 || pindex->GetBlockHash() == hashStop) {
5772 nodestate->pindexBestHeaderSent =
5779 if (RejectIncomingTxs(pfrom)) {
5781 "transaction sent in violation of protocol peer=%d\n",
5797 const CTransaction &tx = *ptx;
5798 const TxId &txid = tx.GetId();
5799 AddKnownTx(*peer, txid);
5804 m_txrequest.ReceivedResponse(pfrom.
GetId(), txid);
5806 if (AlreadyHaveTx(txid,
true)) {
5812 if (!m_mempool.
exists(tx.GetId())) {
5814 "Not relaying non-mempool transaction %s from "
5815 "forcerelay peer=%d\n",
5816 tx.GetId().ToString(), pfrom.
GetId());
5818 LogPrintf(
"Force relaying tx %s from peer=%d\n",
5819 tx.GetId().ToString(), pfrom.
GetId());
5820 RelayTransaction(tx.GetId());
5824 if (m_recent_rejects_package_reconsiderable.contains(txid)) {
5832 "found tx %s in reconsiderable rejects, looking for "
5833 "child in orphanage\n",
5835 if (
auto package_to_validate{
5836 Find1P1CPackage(ptx, pfrom.
GetId())}) {
5839 package_to_validate->m_txns,
5842 "package evaluation for %s: %s (%s)\n",
5843 package_to_validate->ToString(),
5845 ?
"package accepted"
5846 :
"package rejected",
5848 ProcessPackageResult(package_to_validate.value(),
5877 ProcessValidTx(pfrom.
GetId(), ptx);
5883 bool fRejectedParents =
false;
5887 std::vector<TxId> unique_parents;
5888 unique_parents.reserve(tx.vin.size());
5889 for (
const CTxIn &txin : tx.vin) {
5892 unique_parents.push_back(txin.prevout.GetTxId());
5894 std::sort(unique_parents.begin(), unique_parents.end());
5895 unique_parents.erase(
5896 std::unique(unique_parents.begin(), unique_parents.end()),
5897 unique_parents.end());
5905 std::optional<TxId> rejected_parent_reconsiderable;
5906 for (
const TxId &parent_txid : unique_parents) {
5907 if (m_recent_rejects.contains(parent_txid)) {
5908 fRejectedParents =
true;
5912 if (m_recent_rejects_package_reconsiderable.contains(
5914 !m_mempool.
exists(parent_txid)) {
5919 if (rejected_parent_reconsiderable.has_value()) {
5920 fRejectedParents =
true;
5923 rejected_parent_reconsiderable = parent_txid;
5926 if (!fRejectedParents) {
5927 const auto current_time{
5928 GetTime<std::chrono::microseconds>()};
5930 for (
const TxId &parent_txid : unique_parents) {
5932 AddKnownTx(*peer, parent_txid);
5936 if (!AlreadyHaveTx(parent_txid,
5938 AddTxAnnouncement(pfrom, parent_txid, current_time);
5944 if (
unsigned int nEvicted =
5948 if (orphanage.AddTx(ptx,
5950 AddToCompactExtraTransactions(ptx);
5953 m_opts.max_orphan_txs, m_rng);
5956 "orphanage overflow, removed %u tx\n",
5962 m_txrequest.ForgetInvId(tx.GetId());
5966 "not keeping orphan with rejected parents %s\n",
5967 tx.GetId().ToString());
5970 m_recent_rejects.insert(tx.GetId());
5971 m_txrequest.ForgetInvId(tx.GetId());
5975 ProcessInvalidTx(pfrom.
GetId(), ptx, state,
5985 "tx %s failed but reconsiderable, looking for child in "
5988 if (
auto package_to_validate{
5989 Find1P1CPackage(ptx, pfrom.
GetId())}) {
5992 package_to_validate->m_txns,
false)};
5994 "package evaluation for %s: %s (%s)\n",
5995 package_to_validate->ToString(),
5997 ?
"package accepted"
5998 :
"package rejected",
6000 ProcessPackageResult(package_to_validate.value(),
6009 m_txrequest.ForgetInvId(tx.GetId());
6011 unsigned int nEvicted{0};
6018 m_opts.max_conflicting_txs, m_rng);
6023 "conflicting pool overflow, removed %u tx\n",
6036 "Unexpected cmpctblock message received from peer %d\n",
6043 vRecv >> cmpctblock;
6044 }
catch (std::ios_base::failure &e) {
6046 Misbehaving(*peer,
"cmpctblock-bad-indexes");
6050 bool received_new_header =
false;
6063 MaybeSendGetHeaders(
6064 pfrom,
GetLocator(m_chainman.m_best_header), *peer);
6070 GetAntiDoSWorkThreshold()) {
6074 "Ignoring low-work compact block from peer %d\n",
6080 received_new_header =
true;
6090 MaybePunishNodeForBlock(pfrom.
GetId(), state,
6092 "invalid header via cmpctblock");
6097 if (received_new_header) {
6098 LogInfo(
"Saw new cmpctblock header hash=%s peer=%d\n",
6099 blockhash.ToString(), pfrom.
GetId());
6106 bool fProcessBLOCKTXN =
false;
6112 bool fRevertToHeaderProcessing =
false;
6116 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
6117 bool fBlockReconstructed =
false;
6125 CNodeState *nodestate = State(pfrom.
GetId());
6129 if (received_new_header &&
6132 nodestate->m_last_block_announcement =
GetTime();
6135 if (pindex->nStatus.hasData()) {
6142 size_t already_in_flight =
6143 std::distance(range_flight.first, range_flight.second);
6144 bool requested_block_from_this_peer{
false};
6148 bool first_in_flight =
6149 already_in_flight == 0 ||
6150 (range_flight.first->second.first == pfrom.
GetId());
6152 while (range_flight.first != range_flight.second) {
6153 if (range_flight.first->second.first == pfrom.
GetId()) {
6154 requested_block_from_this_peer =
true;
6157 range_flight.first++;
6166 if (requested_block_from_this_peer) {
6170 std::vector<CInv> vInv(1);
6179 if (!already_in_flight && !CanDirectFetch()) {
6187 nodestate->vBlocksInFlight.size() <
6189 requested_block_from_this_peer) {
6190 std::list<QueuedBlock>::iterator *queuedBlockIt =
nullptr;
6191 if (!BlockRequested(config, pfrom.
GetId(), *pindex,
6193 if (!(*queuedBlockIt)->partialBlock) {
6195 ->partialBlock.reset(
6202 "we were already syncing!\n");
6208 *(*queuedBlockIt)->partialBlock;
6210 partialBlock.
InitData(cmpctblock, vExtraTxnForCompact);
6216 Misbehaving(*peer,
"invalid compact block");
6219 if (first_in_flight) {
6222 std::vector<CInv> vInv(1);
6235 for (
size_t i = 0; i < cmpctblock.
BlockTxCount(); i++) {
6246 fProcessBLOCKTXN =
true;
6247 }
else if (first_in_flight) {
6254 IsBlockRequestedFromOutbound(blockhash) ||
6277 tempBlock.InitData(cmpctblock, vExtraTxnForCompact);
6282 std::vector<CTransactionRef> dummy;
6283 status = tempBlock.FillBlock(*pblock, dummy);
6285 fBlockReconstructed =
true;
6289 if (requested_block_from_this_peer) {
6293 std::vector<CInv> vInv(1);
6300 fRevertToHeaderProcessing =
true;
6305 if (fProcessBLOCKTXN) {
6307 blockTxnMsg, time_received, interruptMsgProc);
6310 if (fRevertToHeaderProcessing) {
6316 return ProcessHeadersMessage(config, pfrom, *peer,
6321 if (fBlockReconstructed) {
6326 mapBlockSource.emplace(pblock->GetHash(),
6327 std::make_pair(pfrom.
GetId(),
false));
6338 ProcessBlock(config, pfrom, pblock,
true,
6347 RemoveBlockRequest(pblock->GetHash(), std::nullopt);
6357 "Unexpected blocktxn message received from peer %d\n",
6365 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
6366 bool fBlockRead =
false;
6370 auto range_flight = mapBlocksInFlight.equal_range(resp.
blockhash);
6371 size_t already_in_flight =
6372 std::distance(range_flight.first, range_flight.second);
6373 bool requested_block_from_this_peer{
false};
6377 bool first_in_flight =
6378 already_in_flight == 0 ||
6379 (range_flight.first->second.first == pfrom.
GetId());
6381 while (range_flight.first != range_flight.second) {
6382 auto [node_id, block_it] = range_flight.first->second;
6383 if (node_id == pfrom.
GetId() && block_it->partialBlock) {
6384 requested_block_from_this_peer =
true;
6387 range_flight.first++;
6390 if (!requested_block_from_this_peer) {
6392 "Peer %d sent us block transactions for block "
6393 "we weren't expecting\n",
6399 *range_flight.first->second.second->partialBlock;
6407 "invalid compact block/non-matching block transactions");
6410 if (first_in_flight) {
6412 std::vector<CInv> invs;
6419 "Peer %d sent us a compact block but it failed to "
6420 "reconstruct, waiting on first download to complete\n",
6453 std::make_pair(pfrom.
GetId(),
false));
6464 ProcessBlock(config, pfrom, pblock,
true,
6474 "Unexpected headers message received from peer %d\n",
6479 std::vector<CBlockHeader> headers;
6486 strprintf(
"too-many-headers: headers message size = %u",
6490 headers.resize(nCount);
6491 for (
unsigned int n = 0; n < nCount; n++) {
6492 vRecv >> headers[n];
6497 ProcessHeadersMessage(config, pfrom, *peer, std::move(headers),
6503 if (m_headers_presync_should_signal.exchange(
false)) {
6504 HeadersPresyncStats stats;
6506 LOCK(m_headers_presync_mutex);
6508 m_headers_presync_stats.find(m_headers_presync_bestpeer);
6509 if (it != m_headers_presync_stats.end()) {
6515 stats.first, stats.second->first, stats.second->second);
6526 "Unexpected block message received from peer %d\n",
6531 std::shared_ptr<CBlock> pblock = std::make_shared<CBlock>();
6535 pblock->GetHash().ToString(), pfrom.
GetId());
6540 pblock->hashPrevBlock))};
6544 "Received mutated block from peer=%d\n", peer->m_id);
6545 Misbehaving(*peer,
"mutated block");
6547 RemoveBlockRequest(pblock->GetHash(), peer->m_id));
6557 const BlockHash hash = pblock->GetHash();
6558 bool min_pow_checked =
false;
6563 forceProcessing = IsBlockRequested(hash);
6564 RemoveBlockRequest(hash, pfrom.
GetId());
6568 mapBlockSource.emplace(hash, std::make_pair(pfrom.
GetId(),
true));
6574 GetAntiDoSWorkThreshold()) {
6575 min_pow_checked =
true;
6578 ProcessBlock(config, pfrom, pblock, forceProcessing, min_pow_checked);
6588 if (pfrom.m_avalanche_pubkey.has_value()) {
6591 "Ignoring avahello from peer %d: already in our node set\n",
6597 vRecv >> delegation;
6604 if (!delegation.
verify(state, pubkey)) {
6605 Misbehaving(*peer,
"invalid-delegation");
6608 pfrom.m_avalanche_pubkey = std::move(pubkey);
6611 sighasher << delegation.
getId();
6619 if (!(*pfrom.m_avalanche_pubkey)
6620 .VerifySchnorr(sighasher.GetHash(),
sig)) {
6621 Misbehaving(*peer,
"invalid-avahello-signature");
6628 if (!AlreadyHaveProof(proofid)) {
6629 const bool preferred = isPreferredDownloadPeer(pfrom);
6630 LOCK(cs_proofrequest);
6631 AddProofAnnouncement(pfrom, proofid,
6632 GetTime<std::chrono::microseconds>(),
6640 vRecv >> max_elements;
6644 Misbehaving(*peer,
"avahello-max-elements-too-low");
6652 return pm.
addNode(pfrom.
GetId(), proofid, max_elements);
6663 WITH_LOCK(peer->m_addr_token_bucket_mutex,
6664 peer->m_addr_token_bucket += m_opts.max_addr_to_send);
6668 peer->m_proof_relay->compactproofs_requested =
true;
6679 const auto now = Now<SteadyMilliseconds>();
6685 last_poll + std::chrono::milliseconds(m_opts.avalanche_cooldown)) {
6687 "Ignoring repeated avapoll from peer %d: cooldown not "
6702 strprintf(
"too-many-ava-poll: poll message size = %u", nCount));
6706 std::vector<avalanche::Vote> votes;
6707 votes.reserve(nCount);
6709 bool fPreconsensus{
false};
6710 bool fStakingPreconsensus{
false};
6715 fStakingPreconsensus =
6719 for (
unsigned int n = 0; n < nCount; n++) {
6727 if (!quorum_established) {
6728 votes.emplace_back(vote, inv.
hash);
6735 if (fPreconsensus) {
6737 GetAvalancheVoteForTx(*m_avalanche,
TxId(inv.
hash));
6749 if (fStakingPreconsensus) {
6756 "poll inv type %d unknown from peer=%d\n",
6761 votes.emplace_back(vote, inv.
hash);
6787 if (!pfrom.m_avalanche_pubkey.has_value() ||
6788 !(*pfrom.m_avalanche_pubkey)
6789 .VerifySchnorr(verifier.GetHash(),
sig)) {
6790 Misbehaving(*peer,
"invalid-ava-response-signature");
6795 auto now = GetTime<std::chrono::seconds>();
6797 std::vector<avalanche::VoteItemUpdate> updates;
6798 bool disconnect{
false};
6801 disconnect, error)) {
6803 Misbehaving(*peer, error);
6821 "Repeated failure to register votes from peer %d: %s\n",
6822 pfrom.
GetId(), error);
6825 Misbehaving(*peer, error);
6838 auto logVoteUpdate = [](
const auto &voteUpdate,
6839 const std::string &voteItemTypeStr,
6840 const auto &voteItemId) {
6841 std::string voteOutcome;
6842 bool alwaysPrint =
false;
6843 switch (voteUpdate.getStatus()) {
6845 voteOutcome =
"invalidated";
6849 voteOutcome =
"rejected";
6852 voteOutcome =
"accepted";
6855 voteOutcome =
"finalized";
6858 alwaysPrint = voteItemTypeStr !=
"tx";
6861 voteOutcome =
"stalled";
6870 alwaysPrint &= (voteItemTypeStr !=
"contender");
6873 LogPrintf(
"Avalanche %s %s %s\n", voteOutcome, voteItemTypeStr,
6874 voteItemId.ToString());
6878 voteItemTypeStr, voteItemId.ToString());
6882 bool shouldActivateBestChain =
false;
6884 bool fPreconsensus{
false};
6885 bool fStakingPreconsensus{
false};
6890 fStakingPreconsensus =
6894 for (
const auto &u : updates) {
6899 if (
auto pitem = std::get_if<const avalanche::ProofRef>(&item)) {
6903 logVoteUpdate(u,
"proof", proofid);
6905 auto rejectionMode =
6907 auto nextCooldownTimePoint = GetTime<std::chrono::seconds>();
6908 switch (u.getStatus()) {
6924 return pm.rejectProof(proofid,
6928 "ERROR: Failed to reject proof: %s\n",
6934 nextCooldownTimePoint += std::chrono::seconds(
6935 m_opts.avalanche_peer_replacement_cooldown);
6941 avalanche::PeerManager::
6942 RegistrationMode::FORCE_ACCEPT);
6945 [&](const avalanche::Peer &peer) {
6946 pm.updateNextPossibleConflictTime(
6948 nextCooldownTimePoint);
6949 if (u.getStatus() ==
6950 avalanche::VoteStatus::
6952 pm.setFinalized(peer.peerid);
6960 "ERROR: Failed to accept proof: %s\n",
6967 auto getBlockFromIndex = [
this](
const CBlockIndex *pindex) {
6970 std::shared_ptr<const CBlock> pblock =
WITH_LOCK(
6971 m_most_recent_block_mutex,
return m_most_recent_block);
6973 if (!pblock || pblock->GetHash() != pindex->
GetBlockHash()) {
6974 std::shared_ptr<CBlock> pblockRead =
6975 std::make_shared<CBlock>();
6978 assert(!
"cannot load block from disk");
6980 pblock = pblockRead;
6985 if (
auto pitem = std::get_if<const CBlockIndex *>(&item)) {
6988 shouldActivateBestChain =
true;
6992 switch (u.getStatus()) {
6997 LogPrintf(
"ERROR: Database error: %s\n",
7006 LogPrintf(
"ERROR: Database error: %s\n",
7011 auto pblock = getBlockFromIndex(pindex);
7027 std::unique_ptr<node::CBlockTemplate> blockTemplate;
7031 chainstate.UnparkBlock(pindex);
7033 const bool newlyFinalized =
7034 !chainstate.IsBlockAvalancheFinalized(pindex) &&
7035 chainstate.AvalancheFinalizeBlock(pindex,
7040 if (fPreconsensus && newlyFinalized) {
7041 auto pblock = getBlockFromIndex(pindex);
7055 std::unordered_set<TxId, SaltedTxIdHasher>
7056 confirmedTxIdsInNonFinalizedBlocks;
7058 block !=
nullptr && block != pindex;
7059 block = block->
pprev) {
7061 getBlockFromIndex(block);
7063 for (
const auto &tx :
7064 currentBlock->vtx) {
7065 confirmedTxIdsInNonFinalizedBlocks
7066 .insert(tx->GetId());
7074 confirmedTxIdsInNonFinalizedBlocks);
7086 config, chainstate, &m_mempool,
7088 blockAssembler.pblocktemplate.reset(
7091 if (blockAssembler.pblocktemplate) {
7092 blockAssembler.addTxs(m_mempool);
7093 blockTemplate = std::move(
7094 blockAssembler.pblocktemplate);
7100 if (blockTemplate) {
7106 for (
const auto &templateEntry :
7120 if (fStakingPreconsensus) {
7122 std::get_if<const avalanche::StakeContenderId>(&item)) {
7124 logVoteUpdate(u,
"contender", contenderId);
7126 switch (u.getStatus()) {
7147 if (!fPreconsensus) {
7151 if (
auto pitem = std::get_if<const CTransactionRef>(&item)) {
7155 const TxId &txid = tx->GetId();
7156 const auto status{u.getStatus()};
7161 logVoteUpdate(u,
"tx", txid);
7173 std::shared_ptr<const std::vector<Coin>> spentCoins;
7185 _spentCoins.has_value()
7186 ? std::make_shared<const std::vector<Coin>>(
7187 std::move(*_spentCoins))
7191 if (m_mempool.
exists(txid)) {
7195 std::vector<CTransactionRef> conflictingTxs =
7201 if (conflictingTxs.size() > 0) {
7209 "Attempting to pull a now invalid "
7210 "conflicting tx %s to mempool\n",
7211 conflictingTxs[0]->GetId().
ToString());
7218 for (
const auto &conflictingTx :
7221 conflictingTx->GetId());
7240 m_recent_rejects.insert(txid);
7242 AddToCompactExtraTransactions(tx);
7257 return conflicting.HaveTx(txid);
7260 std::vector<CTransactionRef>
7261 mempool_conflicting_txs;
7262 for (
const auto &txin : tx->vin) {
7267 mempool_conflicting_txs.push_back(
7268 std::move(conflict));
7277 [&txid, &mempool_conflicting_txs](
7282 if (mempool_conflicting_txs.size() >
7285 mempool_conflicting_txs[0],
7294 auto it = m_mempool.
GetIter(txid);
7295 if (!it.has_value()) {
7298 "Error: finalized tx (%s) is not in the "
7304 std::vector<TxId> finalizedTxIds;
7305 m_mempool.setAvalancheFinalized(
7310 for (
const auto &finalized_txid : finalizedTxIds) {
7315 logVoteUpdate(u,
"tx", finalized_txid);
7323 std::vector<CTransactionRef>
7326 for (
const auto &conflictingTx :
7328 m_recent_rejects.insert(
7329 conflictingTx->GetId());
7331 conflictingTx->GetId());
7357 m_txrequest.ForgetInvId(txid);
7363 for (
auto &it : m_peer_map) {
7364 auto tx_relay = (*it.second).GetTxRelay();
7369 LOCK(tx_relay->m_tx_inventory_mutex);
7376 auto &stalled_by_time =
7377 tx_relay->m_avalanche_stalled_txids
7379 if (stalled_by_time.size() >=
7381 stalled_by_time.erase(
7382 stalled_by_time.begin()->timeAdded);
7385 tx_relay->m_avalanche_stalled_txids.insert(
7390 AddToCompactExtraTransactions(tx);
7398 if (shouldActivateBestChain) {
7401 state,
nullptr, m_avalanche)) {
7416 ReceivedAvalancheProof(pfrom, *peer, proof);
7425 if (peer->m_proof_relay ==
nullptr) {
7429 peer->m_proof_relay->lastSharedProofsUpdate =
7430 GetTime<std::chrono::seconds>();
7432 peer->m_proof_relay->sharedProofs =
7438 peer->m_proof_relay->sharedProofs);
7448 if (peer->m_proof_relay ==
nullptr) {
7453 if (!peer->m_proof_relay->compactproofs_requested) {
7457 peer->m_proof_relay->compactproofs_requested =
false;
7461 vRecv >> compactProofs;
7462 }
catch (std::ios_base::failure &e) {
7464 Misbehaving(*peer,
"avaproofs-bad-indexes");
7470 if (!ReceivedAvalancheProof(pfrom, *peer, prefilledProof.proof)) {
7500 auto shortIdProcessor =
7504 if (shortIdProcessor.hasOutOfBoundIndex()) {
7507 Misbehaving(*peer,
"avaproofs-bad-indexes");
7510 if (!shortIdProcessor.isEvenlyDistributed()) {
7515 std::vector<std::pair<avalanche::ProofId, bool>> remoteProofsStatus;
7522 shortIdProcessor.matchKnownItem(shortid, peer.
proof);
7529 remoteProofsStatus.emplace_back(peer.
getProofId(),
7540 for (
size_t i = 0; i < compactProofs.
size(); i++) {
7541 if (shortIdProcessor.getItem(i) ==
nullptr) {
7557 return pfrom.m_avalanche_pubkey.has_value())) {
7560 for (
const auto &[proofid, present] : remoteProofsStatus) {
7570 if (peer->m_proof_relay ==
nullptr) {
7577 auto requestedIndiceIt = proofreq.
indices.begin();
7578 uint32_t treeIndice = 0;
7579 peer->m_proof_relay->sharedProofs.forEachLeaf([&](
const auto &proof) {
7580 if (requestedIndiceIt == proofreq.
indices.end()) {
7585 if (treeIndice++ == *requestedIndiceIt) {
7587 requestedIndiceIt++;
7593 peer->m_proof_relay->sharedProofs = {};
7606 "Ignoring \"getaddr\" from %s connection. peer=%d\n",
7613 Assume(SetupAddressRelay(pfrom, *peer));
7617 if (peer->m_getaddr_recvd) {
7622 peer->m_getaddr_recvd =
true;
7624 peer->m_addrs_to_send.clear();
7625 std::vector<CAddress> vAddr;
7626 const size_t maxAddrToSend = m_opts.max_addr_to_send;
7634 for (
const CAddress &addr : vAddr) {
7635 PushAddress(*peer, addr);
7641 auto now = GetTime<std::chrono::seconds>();
7651 if (!SetupAddressRelay(pfrom, *peer)) {
7653 "Ignoring getavaaddr message from %s peer=%d\n",
7658 auto availabilityScoreComparator = [](
const CNode *lhs,
7661 double scoreRhs = rhs->getAvailabilityScore();
7663 if (scoreLhs != scoreRhs) {
7664 return scoreLhs > scoreRhs;
7673 std::set<
const CNode *,
decltype(availabilityScoreComparator)> avaNodes(
7674 availabilityScoreComparator);
7681 avaNodes.insert(pnode);
7682 if (avaNodes.size() > m_opts.max_addr_to_send) {
7683 avaNodes.erase(std::prev(avaNodes.end()));
7687 peer->m_addrs_to_send.clear();
7688 for (
const CNode *pnode : avaNodes) {
7689 PushAddress(*peer, pnode->
addr);
7700 "mempool request with bloom filters disabled, "
7701 "disconnect peer=%d\n",
7712 "mempool request with bandwidth limit reached, "
7713 "disconnect peer=%d\n",
7720 if (
auto tx_relay = peer->GetTxRelay()) {
7721 LOCK(tx_relay->m_tx_inventory_mutex);
7722 tx_relay->m_send_mempool =
true;
7751 const auto ping_end = time_received;
7754 bool bPingFinished =
false;
7755 std::string sProblem;
7757 if (nAvail >=
sizeof(nonce)) {
7762 if (peer->m_ping_nonce_sent != 0) {
7763 if (nonce == peer->m_ping_nonce_sent) {
7766 bPingFinished =
true;
7767 const auto ping_time = ping_end - peer->m_ping_start.load();
7768 if (ping_time.count() >= 0) {
7773 sProblem =
"Timing mishap";
7777 sProblem =
"Nonce mismatch";
7781 bPingFinished =
true;
7782 sProblem =
"Nonce zero";
7786 sProblem =
"Unsolicited pong without ping";
7791 bPingFinished =
true;
7792 sProblem =
"Short payload";
7795 if (!(sProblem.empty())) {
7797 "pong peer=%d: %s, %x expected, %x received, %u bytes\n",
7798 pfrom.
GetId(), sProblem, peer->m_ping_nonce_sent, nonce,
7801 if (bPingFinished) {
7802 peer->m_ping_nonce_sent = 0;
7810 "filterload received despite not offering bloom services "
7811 "from peer=%d; disconnecting\n",
7821 Misbehaving(*peer,
"too-large bloom filter");
7822 }
else if (
auto tx_relay = peer->GetTxRelay()) {
7824 LOCK(tx_relay->m_bloom_filter_mutex);
7825 tx_relay->m_bloom_filter.reset(
new CBloomFilter(filter));
7826 tx_relay->m_relay_txs =
true;
7836 "filteradd received despite not offering bloom services "
7837 "from peer=%d; disconnecting\n",
7842 std::vector<uint8_t> vData;
7851 }
else if (
auto tx_relay = peer->GetTxRelay()) {
7852 LOCK(tx_relay->m_bloom_filter_mutex);
7853 if (tx_relay->m_bloom_filter) {
7854 tx_relay->m_bloom_filter->insert(vData);
7862 Misbehaving(*peer,
"bad filteradd message");
7870 "filterclear received despite not offering bloom services "
7871 "from peer=%d; disconnecting\n",
7876 auto tx_relay = peer->GetTxRelay();
7882 LOCK(tx_relay->m_bloom_filter_mutex);
7883 tx_relay->m_bloom_filter =
nullptr;
7884 tx_relay->m_relay_txs =
true;
7893 vRecv >> newFeeFilter;
7895 if (
auto tx_relay = peer->GetTxRelay()) {
7896 tx_relay->m_fee_filter_received = newFeeFilter;
7905 ProcessGetCFilters(pfrom, *peer, vRecv);
7910 ProcessGetCFHeaders(pfrom, *peer, vRecv);
7915 ProcessGetCFCheckPt(pfrom, *peer, vRecv);
7920 std::vector<CInv> vInv;
7926 for (
CInv &inv : vInv) {
7932 m_txrequest.ReceivedResponse(pfrom.
GetId(),
TxId(inv.
hash));
7939 LOCK(cs_proofrequest);
7940 m_proofrequest.ReceivedResponse(
7954bool PeerManagerImpl::MaybeDiscourageAndDisconnect(
CNode &pnode, Peer &peer) {
7956 LOCK(peer.m_misbehavior_mutex);
7959 if (!peer.m_should_discourage) {
7963 peer.m_should_discourage =
false;
7969 LogPrintf(
"Warning: not punishing noban peer %d!\n", peer.m_id);
7975 LogPrintf(
"Warning: not punishing manually connected peer %d!\n",
7984 "Warning: disconnecting but not discouraging %s peer %d!\n",
8001bool PeerManagerImpl::ProcessMessages(
const Config &config,
CNode *pfrom,
8002 std::atomic<bool> &interruptMsgProc) {
8005 PeerRef peer = GetPeerRef(pfrom->
GetId());
8006 if (peer ==
nullptr) {
8011 LOCK(peer->m_getdata_requests_mutex);
8012 if (!peer->m_getdata_requests.empty()) {
8013 ProcessGetData(config, *pfrom, *peer, interruptMsgProc);
8017 const bool processed_orphan = ProcessOrphanTx(config, *peer);
8023 if (processed_orphan) {
8030 LOCK(peer->m_getdata_requests_mutex);
8031 if (!peer->m_getdata_requests.empty()) {
8048 bool fMoreWork = poll_result->second;
8052 msg.m_recv.size(),
msg.m_recv.data());
8054 if (m_opts.capture_messages) {
8060 ProcessMessage(config, *pfrom,
msg.m_type,
msg.m_recv,
msg.m_time,
8062 if (interruptMsgProc) {
8067 LOCK(peer->m_getdata_requests_mutex);
8068 if (!peer->m_getdata_requests.empty()) {
8077 return orphanage.HaveTxToReconsider(peer->m_id);
8081 }
catch (
const std::exception &e) {
8084 e.what(),
typeid(e).name());
8093void PeerManagerImpl::ConsiderEviction(
CNode &pto, Peer &peer,
8094 std::chrono::seconds time_in_seconds) {
8097 CNodeState &state = *State(pto.
GetId());
8100 state.fSyncStarted) {
8107 if (state.pindexBestKnownBlock !=
nullptr &&
8108 state.pindexBestKnownBlock->nChainWork >=
8110 if (state.m_chain_sync.m_timeout != 0s) {
8111 state.m_chain_sync.m_timeout = 0s;
8112 state.m_chain_sync.m_work_header =
nullptr;
8113 state.m_chain_sync.m_sent_getheaders =
false;
8115 }
else if (state.m_chain_sync.m_timeout == 0s ||
8116 (state.m_chain_sync.m_work_header !=
nullptr &&
8117 state.pindexBestKnownBlock !=
nullptr &&
8118 state.pindexBestKnownBlock->nChainWork >=
8119 state.m_chain_sync.m_work_header->nChainWork)) {
8125 state.m_chain_sync.m_work_header = m_chainman.
ActiveChain().
Tip();
8126 state.m_chain_sync.m_sent_getheaders =
false;
8127 }
else if (state.m_chain_sync.m_timeout > 0s &&
8128 time_in_seconds > state.m_chain_sync.m_timeout) {
8133 if (state.m_chain_sync.m_sent_getheaders) {
8136 "Disconnecting outbound peer %d for old chain, best known "
8139 state.pindexBestKnownBlock !=
nullptr
8140 ? state.pindexBestKnownBlock->GetBlockHash().ToString()
8144 assert(state.m_chain_sync.m_work_header);
8149 MaybeSendGetHeaders(
8150 pto,
GetLocator(state.m_chain_sync.m_work_header->pprev),
8154 "sending getheaders to outbound peer=%d to verify chain "
8155 "work (current best known block:%s, benchmark blockhash: "
8158 state.pindexBestKnownBlock !=
nullptr
8159 ? state.pindexBestKnownBlock->GetBlockHash().ToString()
8161 state.m_chain_sync.m_work_header->GetBlockHash()
8163 state.m_chain_sync.m_sent_getheaders =
true;
8170 state.m_chain_sync.m_timeout =
8177void PeerManagerImpl::EvictExtraOutboundPeers(std::chrono::seconds now) {
8186 std::pair<NodeId, std::chrono::seconds> youngest_peer{-1, 0},
8187 next_youngest_peer{-1, 0};
8193 if (pnode->
GetId() > youngest_peer.first) {
8194 next_youngest_peer = youngest_peer;
8195 youngest_peer.first = pnode->GetId();
8196 youngest_peer.second = pnode->m_last_block_time;
8200 NodeId to_disconnect = youngest_peer.first;
8201 if (youngest_peer.second > next_youngest_peer.second) {
8204 to_disconnect = next_youngest_peer.first;
8216 CNodeState *node_state = State(pnode->
GetId());
8217 if (node_state ==
nullptr ||
8219 node_state->vBlocksInFlight.empty())) {
8222 "disconnecting extra block-relay-only peer=%d "
8223 "(last block received at time %d)\n",
8230 "keeping block-relay-only peer=%d chosen for eviction "
8231 "(connect time: %d, blocks_in_flight: %d)\n",
8233 node_state->vBlocksInFlight.size());
8249 int64_t oldest_block_announcement = std::numeric_limits<int64_t>::max();
8260 CNodeState *state = State(pnode->
GetId());
8261 if (state ==
nullptr) {
8266 if (state->m_chain_sync.m_protect) {
8269 if (state->m_last_block_announcement < oldest_block_announcement ||
8270 (state->m_last_block_announcement == oldest_block_announcement &&
8271 pnode->
GetId() > worst_peer)) {
8272 worst_peer = pnode->
GetId();
8273 oldest_block_announcement = state->m_last_block_announcement;
8277 if (worst_peer == -1) {
8281 bool disconnected = m_connman.
ForNode(
8289 CNodeState &state = *State(pnode->
GetId());
8291 state.vBlocksInFlight.empty()) {
8293 "disconnecting extra outbound peer=%d (last block "
8294 "announcement received at time %d)\n",
8295 pnode->
GetId(), oldest_block_announcement);
8300 "keeping outbound peer=%d chosen for eviction "
8301 "(connect time: %d, blocks_in_flight: %d)\n",
8303 state.vBlocksInFlight.size());
8318void PeerManagerImpl::CheckForStaleTipAndEvictPeers() {
8321 auto now{GetTime<std::chrono::seconds>()};
8323 EvictExtraOutboundPeers(now);
8325 if (now > m_stale_tip_check_time) {
8331 LogPrintf(
"Potential stale tip detected, will try using extra "
8332 "outbound peer (last tip update: %d seconds ago)\n",
8341 if (!m_initial_sync_finished && CanDirectFetch()) {
8343 m_initial_sync_finished =
true;
8347void PeerManagerImpl::MaybeSendPing(
CNode &node_to, Peer &peer,
8348 std::chrono::microseconds now) {
8350 node_to, std::chrono::duration_cast<std::chrono::seconds>(now)) &&
8351 peer.m_ping_nonce_sent &&
8362 bool pingSend =
false;
8364 if (peer.m_ping_queued) {
8369 if (peer.m_ping_nonce_sent == 0 &&
8379 }
while (nonce == 0);
8380 peer.m_ping_queued =
false;
8381 peer.m_ping_start = now;
8383 peer.m_ping_nonce_sent = nonce;
8388 peer.m_ping_nonce_sent = 0;
8394void PeerManagerImpl::MaybeSendAddr(
CNode &
node, Peer &peer,
8395 std::chrono::microseconds current_time) {
8397 if (!peer.m_addr_relay_enabled) {
8401 LOCK(peer.m_addr_send_times_mutex);
8403 peer.m_next_local_addr_send < current_time) {
8410 if (peer.m_next_local_addr_send != 0us) {
8411 peer.m_addr_known->reset();
8414 CAddress local_addr{*local_service, peer.m_our_services,
8415 Now<NodeSeconds>()};
8416 PushAddress(peer, local_addr);
8418 peer.m_next_local_addr_send =
8424 if (current_time <= peer.m_next_addr_send) {
8428 peer.m_next_addr_send =
8431 const size_t max_addr_to_send = m_opts.max_addr_to_send;
8432 if (!
Assume(peer.m_addrs_to_send.size() <= max_addr_to_send)) {
8435 peer.m_addrs_to_send.resize(max_addr_to_send);
8440 auto addr_already_known =
8443 bool ret = peer.m_addr_known->contains(addr.
GetKey());
8445 peer.m_addr_known->insert(addr.
GetKey());
8449 peer.m_addrs_to_send.erase(std::remove_if(peer.m_addrs_to_send.begin(),
8450 peer.m_addrs_to_send.end(),
8451 addr_already_known),
8452 peer.m_addrs_to_send.end());
8455 if (peer.m_addrs_to_send.empty()) {
8459 const char *msg_type;
8461 if (peer.m_wants_addrv2) {
8471 peer.m_addrs_to_send));
8472 peer.m_addrs_to_send.clear();
8475 if (peer.m_addrs_to_send.capacity() > 40) {
8476 peer.m_addrs_to_send.shrink_to_fit();
8480void PeerManagerImpl::MaybeSendSendHeaders(
CNode &
node, Peer &peer) {
8485 if (!peer.m_sent_sendheaders &&
8488 CNodeState &state = *State(
node.GetId());
8489 if (state.pindexBestKnownBlock !=
nullptr &&
8490 state.pindexBestKnownBlock->nChainWork >
8497 peer.m_sent_sendheaders =
true;
8502void PeerManagerImpl::MaybeSendFeefilter(
8503 CNode &pto, Peer &peer, std::chrono::microseconds current_time) {
8504 if (m_opts.ignore_incoming_txs) {
8528 static const Amount MAX_FILTER{m_fee_filter_rounder.round(
MAX_MONEY)};
8529 if (peer.m_fee_filter_sent == MAX_FILTER) {
8532 peer.m_next_send_feefilter = 0us;
8535 if (current_time > peer.m_next_send_feefilter) {
8536 Amount filterToSend = m_fee_filter_rounder.round(currentFilter);
8540 if (filterToSend != peer.m_fee_filter_sent) {
8542 peer.m_fee_filter_sent = filterToSend;
8544 peer.m_next_send_feefilter =
8552 peer.m_next_send_feefilter &&
8553 (currentFilter < 3 * peer.m_fee_filter_sent / 4 ||
8554 currentFilter > 4 * peer.m_fee_filter_sent / 3)) {
8555 peer.m_next_send_feefilter =
8563class CompareInvMempoolOrder {
8567 explicit CompareInvMempoolOrder(
CTxMemPool *_mempool) : mp(_mempool) {}
8569 bool operator()(std::set<TxId>::iterator a, std::set<TxId>::iterator b) {
8579bool PeerManagerImpl::RejectIncomingTxs(
const CNode &peer)
const {
8588 if (m_opts.ignore_incoming_txs &&
8595bool PeerManagerImpl::SetupAddressRelay(
const CNode &
node, Peer &peer) {
8599 if (
node.IsBlockOnlyConn()) {
8603 if (!peer.m_addr_relay_enabled.exchange(
true)) {
8607 peer.m_addr_known = std::make_unique<CRollingBloomFilter>(5000, 0.001);
8613bool PeerManagerImpl::SendMessages(
const Config &config,
CNode *pto) {
8616 PeerRef peer = GetPeerRef(pto->
GetId());
8625 if (MaybeDiscourageAndDisconnect(*pto, *peer)) {
8634 const auto current_time{GetTime<std::chrono::microseconds>()};
8639 "addrfetch connection timeout; disconnecting peer=%d\n",
8645 MaybeSendPing(*pto, *peer, current_time);
8652 bool sync_blocks_and_headers_from_peer =
false;
8654 MaybeSendAddr(*pto, *peer, current_time);
8656 MaybeSendSendHeaders(*pto, *peer);
8661 CNodeState &state = *State(pto->
GetId());
8664 if (m_chainman.m_best_header ==
nullptr) {
8671 if (state.fPreferredDownload) {
8672 sync_blocks_and_headers_from_peer =
true;
8683 if (m_num_preferred_download_peers == 0 ||
8684 mapBlocksInFlight.empty()) {
8685 sync_blocks_and_headers_from_peer =
true;
8689 if (!state.fSyncStarted && CanServeBlocks(*peer) &&
8693 if ((nSyncStarted == 0 && sync_blocks_and_headers_from_peer) ||
8695 const CBlockIndex *pindexStart = m_chainman.m_best_header;
8704 if (pindexStart->
pprev) {
8705 pindexStart = pindexStart->
pprev;
8707 if (MaybeSendGetHeaders(*pto,
GetLocator(pindexStart), *peer)) {
8710 "initial getheaders (%d) to peer=%d (startheight:%d)\n",
8712 peer->m_starting_height);
8714 state.fSyncStarted =
true;
8715 peer->m_headers_sync_timeout =
8720 std::chrono::microseconds{
8722 Ticks<std::chrono::seconds>(
8724 m_chainman.m_best_header->Time()) /
8741 LOCK(peer->m_block_inv_mutex);
8742 std::vector<CBlock> vHeaders;
8744 ((!peer->m_prefers_headers &&
8745 (!state.m_requested_hb_cmpctblocks ||
8746 peer->m_blocks_for_headers_relay.size() > 1)) ||
8747 peer->m_blocks_for_headers_relay.size() >
8752 ProcessBlockAvailability(pto->
GetId());
8754 if (!fRevertToInv) {
8755 bool fFoundStartingHeader =
false;
8759 for (
const BlockHash &hash : peer->m_blocks_for_headers_relay) {
8765 fRevertToInv =
true;
8768 if (pBestIndex !=
nullptr && pindex->
pprev != pBestIndex) {
8779 fRevertToInv =
true;
8782 pBestIndex = pindex;
8783 if (fFoundStartingHeader) {
8786 }
else if (PeerHasHeader(&state, pindex)) {
8789 }
else if (pindex->
pprev ==
nullptr ||
8790 PeerHasHeader(&state, pindex->
pprev)) {
8793 fFoundStartingHeader =
true;
8798 fRevertToInv =
true;
8803 if (!fRevertToInv && !vHeaders.empty()) {
8804 if (vHeaders.size() == 1 && state.m_requested_hb_cmpctblocks) {
8809 "%s sending header-and-ids %s to peer=%d\n",
8810 __func__, vHeaders.front().GetHash().ToString(),
8813 std::optional<CSerializedNetMsg> cached_cmpctblock_msg;
8815 LOCK(m_most_recent_block_mutex);
8816 if (m_most_recent_block_hash ==
8818 cached_cmpctblock_msg =
8820 *m_most_recent_compact_block);
8823 if (cached_cmpctblock_msg.has_value()) {
8825 std::move(cached_cmpctblock_msg.value()));
8829 block, *pBestIndex)};
8836 state.pindexBestHeaderSent = pBestIndex;
8837 }
else if (peer->m_prefers_headers) {
8838 if (vHeaders.size() > 1) {
8840 "%s: %u headers, range (%s, %s), to peer=%d\n",
8841 __func__, vHeaders.size(),
8842 vHeaders.front().GetHash().ToString(),
8843 vHeaders.back().GetHash().ToString(),
8847 "%s: sending header %s to peer=%d\n", __func__,
8848 vHeaders.front().GetHash().ToString(),
8852 state.pindexBestHeaderSent = pBestIndex;
8854 fRevertToInv =
true;
8861 if (!peer->m_blocks_for_headers_relay.empty()) {
8863 peer->m_blocks_for_headers_relay.back();
8874 "Announcing block %s not on main chain (tip=%s)\n",
8883 if (!PeerHasHeader(&state, pindex)) {
8884 peer->m_blocks_for_inv_relay.push_back(hashToAnnounce);
8886 "%s: sending inv peer=%d hash=%s\n", __func__,
8891 peer->m_blocks_for_headers_relay.clear();
8898 std::vector<CInv> vInv;
8899 auto addInvAndMaybeFlush = [&](uint32_t type,
const uint256 &hash) {
8900 vInv.emplace_back(type, hash);
8911 LOCK(peer->m_block_inv_mutex);
8913 vInv.reserve(std::max<size_t>(peer->m_blocks_for_inv_relay.size(),
8919 for (
const BlockHash &hash : peer->m_blocks_for_inv_relay) {
8922 peer->m_blocks_for_inv_relay.clear();
8925 auto computeNextInvSendTime =
8926 [&](std::chrono::microseconds &next)
8930 if (next < current_time) {
8931 fSendTrickle =
true;
8933 next = NextInvToInbounds(
8938 next = current_time;
8942 return fSendTrickle;
8946 if (peer->m_proof_relay !=
nullptr) {
8947 LOCK(peer->m_proof_relay->m_proof_inventory_mutex);
8949 if (computeNextInvSendTime(
8950 peer->m_proof_relay->m_next_inv_send_time)) {
8952 peer->m_proof_relay->m_proof_inventory_to_send.begin();
8954 peer->m_proof_relay->m_proof_inventory_to_send.end()) {
8957 it = peer->m_proof_relay->m_proof_inventory_to_send.erase(
8960 if (peer->m_proof_relay->m_proof_inventory_known_filter
8961 .contains(proofid)) {
8965 peer->m_proof_relay->m_proof_inventory_known_filter.insert(
8968 peer->m_proof_relay->m_recently_announced_proofs.insert(
8974 if (
auto tx_relay = peer->GetTxRelay()) {
8975 LOCK(tx_relay->m_tx_inventory_mutex);
8977 const bool fSendTrickle =
8978 computeNextInvSendTime(tx_relay->m_next_inv_send_time);
8983 LOCK(tx_relay->m_bloom_filter_mutex);
8984 if (!tx_relay->m_relay_txs) {
8985 tx_relay->m_tx_inventory_to_send.clear();
8990 if (fSendTrickle && tx_relay->m_send_mempool) {
8991 auto vtxinfo = m_mempool.
infoAll();
8992 tx_relay->m_send_mempool =
false;
8994 tx_relay->m_fee_filter_received.load()};
8996 LOCK(tx_relay->m_bloom_filter_mutex);
8998 for (
const auto &txinfo : vtxinfo) {
8999 const TxId &txid = txinfo.tx->GetId();
9000 tx_relay->m_tx_inventory_to_send.erase(txid);
9003 if (txinfo.fee < filterrate.GetFee(txinfo.vsize)) {
9006 if (tx_relay->m_bloom_filter &&
9007 !tx_relay->m_bloom_filter->IsRelevantAndUpdate(
9011 tx_relay->m_tx_inventory_known_filter.insert(txid);
9014 addInvAndMaybeFlush(
MSG_TX, txid);
9016 tx_relay->m_last_mempool_req =
9017 std::chrono::duration_cast<std::chrono::seconds>(
9024 std::vector<std::set<TxId>::iterator> vInvTx;
9025 vInvTx.reserve(tx_relay->m_tx_inventory_to_send.size());
9026 for (std::set<TxId>::iterator it =
9027 tx_relay->m_tx_inventory_to_send.begin();
9028 it != tx_relay->m_tx_inventory_to_send.end(); it++) {
9029 vInvTx.push_back(it);
9032 tx_relay->m_fee_filter_received.load()};
9037 CompareInvMempoolOrder compareInvMempoolOrder(&m_mempool);
9038 std::make_heap(vInvTx.begin(), vInvTx.end(),
9039 compareInvMempoolOrder);
9043 unsigned int nRelayedTransactions = 0;
9044 LOCK(tx_relay->m_bloom_filter_mutex);
9045 while (!vInvTx.empty() &&
9050 std::pop_heap(vInvTx.begin(), vInvTx.end(),
9051 compareInvMempoolOrder);
9052 std::set<TxId>::iterator it = vInvTx.back();
9054 const TxId txid = *it;
9056 tx_relay->m_tx_inventory_to_send.erase(it);
9058 if (tx_relay->m_tx_inventory_known_filter.contains(txid) &&
9059 tx_relay->m_avalanche_stalled_txids.count(txid) == 0) {
9063 auto txinfo = m_mempool.
info(txid);
9069 if (txinfo.fee < filterrate.GetFee(txinfo.vsize)) {
9072 if (tx_relay->m_bloom_filter &&
9073 !tx_relay->m_bloom_filter->IsRelevantAndUpdate(
9078 tx_relay->m_recently_announced_invs.insert(txid);
9079 addInvAndMaybeFlush(
MSG_TX, txid);
9080 nRelayedTransactions++;
9081 tx_relay->m_tx_inventory_known_filter.insert(txid);
9082 tx_relay->m_avalanche_stalled_txids.erase(txid);
9088 if (!vInv.empty()) {
9095 CNodeState &state = *State(pto->
GetId());
9098 auto stalling_timeout = m_block_stalling_timeout.load();
9099 if (state.m_stalling_since.count() &&
9100 state.m_stalling_since < current_time - stalling_timeout) {
9105 LogPrintf(
"Peer=%d is stalling block download, disconnecting\n",
9110 const auto new_timeout =
9112 if (stalling_timeout != new_timeout &&
9113 m_block_stalling_timeout.compare_exchange_strong(
9114 stalling_timeout, new_timeout)) {
9117 "Increased stalling timeout temporarily to %d seconds\n",
9129 if (state.vBlocksInFlight.size() > 0) {
9130 QueuedBlock &queuedBlock = state.vBlocksInFlight.front();
9131 int nOtherPeersWithValidatedDownloads =
9132 m_peers_downloading_from - 1;
9134 state.m_downloading_since +
9135 std::chrono::seconds{consensusParams.nPowTargetSpacing} *
9138 nOtherPeersWithValidatedDownloads)) {
9139 LogPrintf(
"Timeout downloading block %s from peer=%d, "
9141 queuedBlock.pindex->GetBlockHash().ToString(),
9149 if (state.fSyncStarted &&
9150 peer->m_headers_sync_timeout < std::chrono::microseconds::max()) {
9153 if (current_time > peer->m_headers_sync_timeout &&
9154 nSyncStarted == 1 &&
9155 (m_num_preferred_download_peers -
9156 state.fPreferredDownload >=
9165 LogPrintf(
"Timeout downloading headers from peer=%d, "
9171 LogPrintf(
"Timeout downloading headers from noban "
9172 "peer=%d, not disconnecting\n",
9178 state.fSyncStarted =
false;
9180 peer->m_headers_sync_timeout = 0us;
9186 peer->m_headers_sync_timeout = std::chrono::microseconds::max();
9192 ConsiderEviction(*pto, *peer, GetTime<std::chrono::seconds>());
9195 std::vector<CInv> vGetData;
9203 CNodeState &state = *State(pto->
GetId());
9205 if (CanServeBlocks(*peer) &&
9206 ((sync_blocks_and_headers_from_peer && !IsLimitedPeer(*peer)) ||
9209 std::vector<const CBlockIndex *> vToDownload;
9211 auto get_inflight_budget = [&state]() {
9214 static_cast<int>(state.vBlocksInFlight.size()));
9220 FindNextBlocksToDownload(*peer, get_inflight_budget(), vToDownload,
9223 !IsLimitedPeer(*peer)) {
9229 m_chainman.GetSnapshotBaseBlock());
9231 TryDownloadingHistoricalBlocks(
9232 *peer, get_inflight_budget(), vToDownload, from_tip,
9233 Assert(m_chainman.GetSnapshotBaseBlock()));
9237 BlockRequested(config, pto->
GetId(), *pindex);
9242 if (state.vBlocksInFlight.empty() && staller != -1) {
9243 if (State(staller)->m_stalling_since == 0us) {
9244 State(staller)->m_stalling_since = current_time;
9251 auto addGetDataAndMaybeFlush = [&](uint32_t type,
const uint256 &hash) {
9252 CInv inv(type, hash);
9255 vGetData.push_back(std::move(inv));
9266 LOCK(cs_proofrequest);
9267 std::vector<std::pair<NodeId, avalanche::ProofId>> expired;
9269 m_proofrequest.GetRequestable(pto->
GetId(), current_time, &expired);
9270 for (
const auto &entry : expired) {
9272 "timeout of inflight proof %s from peer=%d\n",
9273 entry.second.ToString(), entry.first);
9275 for (
const auto &proofid : requestable) {
9276 if (!AlreadyHaveProof(proofid)) {
9278 m_proofrequest.RequestedData(
9279 pto->
GetId(), proofid,
9286 m_proofrequest.ForgetInvId(proofid);
9296 std::vector<std::pair<NodeId, TxId>> expired;
9298 m_txrequest.GetRequestable(pto->
GetId(), current_time, &expired);
9299 for (
const auto &entry : expired) {
9301 entry.second.ToString(), entry.first);
9303 for (
const TxId &txid : requestable) {
9307 if (!AlreadyHaveTx(txid,
false)) {
9308 addGetDataAndMaybeFlush(
MSG_TX, txid);
9309 m_txrequest.RequestedData(
9316 m_txrequest.ForgetInvId(txid);
9320 if (!vGetData.empty()) {
9325 MaybeSendFeefilter(*pto, *peer, current_time);
9329bool PeerManagerImpl::ReceivedAvalancheProof(
CNode &
node, Peer &peer,
9331 assert(proof !=
nullptr);
9335 AddKnownProof(peer, proofid);
9347 return node.m_avalanche_pubkey.has_value());
9348 auto saveProofIfStaker = [
this, isStaker](
const CNode &
node,
9350 const NodeId nodeid) ->
bool {
9362 LOCK(cs_proofrequest);
9363 m_proofrequest.ReceivedResponse(nodeid, proofid);
9365 if (AlreadyHaveProof(proofid)) {
9366 m_proofrequest.ForgetInvId(proofid);
9367 saveProofIfStaker(
node, proofid, nodeid);
9377 return pm.registerProof(proof, state);
9379 WITH_LOCK(cs_proofrequest, m_proofrequest.ForgetInvId(proofid));
9380 RelayProof(proofid);
9382 node.m_last_proof_time = GetTime<std::chrono::seconds>();
9385 nodeid, proofid.ToString());
9407 "Not polling the avalanche proof (%s): peer=%d, proofid %s\n",
9408 state.
IsValid() ?
"not-worth-polling"
9410 nodeid, proofid.ToString());
9413 saveProofIfStaker(
node, proofid, nodeid);
bool MoneyRange(const Amount nValue)
static constexpr Amount MAX_MONEY
No amount larger than this (in satoshi) is valid.
enum ReadStatus_t ReadStatus
const std::string & BlockFilterTypeName(BlockFilterType filter_type)
Get the human-readable name for a filter type.
BlockFilterIndex * GetBlockFilterIndex(BlockFilterType filter_type)
Get a block filter index by type.
static constexpr int CFCHECKPT_INTERVAL
Interval between compact filter checkpoints.
@ CHAIN
Outputs do not overspend inputs, no double spends, coinbase output ok, no immature coinbase spends,...
@ TRANSACTIONS
Only first tx is coinbase, 2 <= coinbase input script length <= 100, transactions valid,...
@ SCRIPTS
Scripts & signatures ok.
@ TREE
All parent headers found, difficulty matches, timestamp >= median previous, checkpoint.
arith_uint256 GetBlockProof(const CBlockIndex &block)
CBlockLocator GetLocator(const CBlockIndex *index)
Get a locator for a block index entry.
int64_t GetBlockProofEquivalentTime(const CBlockIndex &to, const CBlockIndex &from, const CBlockIndex &tip, const Consensus::Params ¶ms)
Return the time it would take to redo the work difference between from and to, assuming the current h...
const CBlockIndex * LastCommonAncestor(const CBlockIndex *pa, const CBlockIndex *pb)
Find the last common ancestor two blocks have.
#define Assert(val)
Identity function.
#define Assume(val)
Assume is the identity function.
Stochastic address manager.
void Connected(const CService &addr, NodeSeconds time=Now< NodeSeconds >())
We have successfully connected to this peer.
void Good(const CService &addr, bool test_before_evict=true, NodeSeconds time=Now< NodeSeconds >())
Mark an entry as accessible, possibly moving it from "new" to "tried".
bool Add(const std::vector< CAddress > &vAddr, const CNetAddr &source, std::chrono::seconds time_penalty=0s)
Attempt to add one or more addresses to addrman's new table.
void SetServices(const CService &addr, ServiceFlags nServices)
Update an entry's service bits.
void Discourage(const CNetAddr &net_addr)
bool IsBanned(const CNetAddr &net_addr)
Return whether net_addr is banned.
bool IsDiscouraged(const CNetAddr &net_addr)
Return whether net_addr is discouraged.
BlockFilterIndex is used to store and retrieve block filters, hashes, and headers for a range of bloc...
bool LookupFilterRange(int start_height, const CBlockIndex *stop_index, std::vector< BlockFilter > &filters_out) const
Get a range of filters between two heights on a chain.
bool LookupFilterHashRange(int start_height, const CBlockIndex *stop_index, std::vector< uint256 > &hashes_out) const
Get a range of filter hashes between two heights on a chain.
bool LookupFilterHeader(const CBlockIndex *block_index, uint256 &header_out) EXCLUSIVE_LOCKS_REQUIRED(!m_cs_headers_cache)
Get a single filter header by block.
std::vector< CTransactionRef > txn
std::vector< uint32_t > indices
A CService with information about it as peer.
ServiceFlags nServices
Serialized as uint64_t in V1, and as CompactSize in V2.
static constexpr SerParams V1_NETWORK
NodeSeconds nTime
Always included in serialization, except in the network format on INIT_PROTO_VERSION.
static constexpr SerParams V2_NETWORK
size_t BlockTxCount() const
std::vector< CTransactionRef > vtx
The block chain is a tree shaped structure starting with the genesis block at the root,...
bool IsValid(enum BlockValidity nUpTo=BlockValidity::TRANSACTIONS) const EXCLUSIVE_LOCKS_REQUIRED(
Check whether this block index entry is valid up to the passed validity level.
CBlockIndex * pprev
pointer to the index of the predecessor of this block
CBlockHeader GetBlockHeader() const
arith_uint256 nChainWork
(memory only) Total amount of work (expected number of hashes) in the chain up to and including this ...
bool HaveNumChainTxs() const
Check whether this block and all previous blocks back to the genesis block or an assumeutxo snapshot ...
int64_t GetBlockTime() const
unsigned int nTx
Number of transactions in this block.
CBlockIndex * GetAncestor(int height)
Efficiently find an ancestor of this block.
BlockHash GetBlockHash() const
int nHeight
height of the entry in the chain. The genesis block has height 0
FlatFilePos GetBlockPos() const EXCLUSIVE_LOCKS_REQUIRED(
BloomFilter is a probabilistic filter which SPV clients provide so that we can filter the transaction...
bool IsWithinSizeConstraints() const
True if the size is <= MAX_BLOOM_FILTER_SIZE and the number of hash functions is <= MAX_HASH_FUNCS (c...
An in-memory indexed chain of blocks.
CBlockIndex * Tip() const
Returns the index entry for the tip of this chain, or nullptr if none.
CBlockIndex * Next(const CBlockIndex *pindex) const
Find the successor of a block in this chain, or nullptr if the given index is not found or is the tip...
int Height() const
Return the maximal height in the chain.
bool Contains(const CBlockIndex *pindex) const
Efficiently check whether a block is present in this chain.
CChainParams defines various tweakable parameters of a given instance of the Bitcoin system.
const CBlock & GenesisBlock() const
const Consensus::Params & GetConsensus() const
CCoinsView that adds a memory cache for transactions to another CCoinsView.
CCoinsView that brings transactions from a mempool into view.
void ForEachNode(const NodeFn &func)
bool OutboundTargetReached(bool historicalBlockServingLimit) const
check if the outbound target is reached.
bool ForNode(NodeId id, std::function< bool(CNode *pnode)> func)
bool GetNetworkActive() const
bool GetTryNewOutboundPeer() const
void SetTryNewOutboundPeer(bool flag)
int GetExtraBlockRelayCount() const
void WakeMessageHandler() EXCLUSIVE_LOCKS_REQUIRED(!mutexMsgProc)
void StartExtraBlockRelayPeers()
bool DisconnectNode(const std::string &node)
CSipHasher GetDeterministicRandomizer(uint64_t id) const
Get a unique deterministic randomizer.
int GetExtraFullOutboundCount() const
std::vector< CAddress > GetAddresses(size_t max_addresses, size_t max_pct, std::optional< Network > network) const
Return all or many randomly selected addresses, optionally by network.
bool CheckIncomingNonce(uint64_t nonce)
bool ShouldRunInactivityChecks(const CNode &node, std::chrono::seconds now) const
Return true if we should disconnect the peer for failing an inactivity check.
bool GetUseAddrmanOutgoing() const
Fee rate in satoshis per kilobyte: Amount / kB.
Amount GetFeePerK() const
Return the fee in satoshis for a size of 1000 bytes.
Inv(ventory) message data.
bool IsMsgCmpctBlk() const
std::string ToString() const
bool IsMsgStakeContender() const
bool IsMsgFilteredBlk() const
void TransactionInvalidated(const CTransactionRef &tx, std::shared_ptr< const std::vector< Coin > > spent_coins)
Used to create a Merkle proof (usually from a subset of transactions), which consists of a block head...
std::vector< std::pair< size_t, uint256 > > vMatchedTxn
Public only for unit testing and relay testing (not relayed).
bool IsRelayable() const
Whether this address should be relayed to other peers even if we can't reach it ourselves.
static constexpr SerParams V1
bool IsAddrV1Compatible() const
Check if the current object can be serialized in pre-ADDRv2/BIP155 format.
Transport protocol agnostic message container.
Information about a peer.
Mutex cs_avalanche_pubkey
bool IsFeelerConn() const
const std::chrono::seconds m_connected
Unix epoch time at peer connection.
bool ExpectServicesFromConn() const
std::atomic< int > nVersion
std::atomic_bool m_has_all_wanted_services
Whether this peer provides all services that we want.
bool IsInboundConn() const
bool HasPermission(NetPermissionFlags permission) const
bool IsOutboundOrBlockRelayConn() const
bool IsManualConn() const
std::atomic< int64_t > nTimeOffset
const std::string m_addr_name
std::string ConnectionTypeAsString() const
void SetCommonVersion(int greatest_common_version)
std::atomic< bool > m_bip152_highbandwidth_to
std::atomic_bool m_relays_txs
Whether we should relay transactions to this peer.
std::atomic< bool > m_bip152_highbandwidth_from
void PongReceived(std::chrono::microseconds ping_time)
A ping-pong round trip has completed successfully.
std::atomic_bool fSuccessfullyConnected
bool IsAddrFetchConn() const
uint64_t GetLocalNonce() const
void SetAddrLocal(const CService &addrLocalIn) EXCLUSIVE_LOCKS_REQUIRED(!m_addr_local_mutex)
May not be called more than once.
bool IsBlockOnlyConn() const
int GetCommonVersion() const
bool IsFullOutboundConn() const
uint64_t nRemoteHostNonce
Mutex m_subver_mutex
cleanSubVer is a sanitized string of the user agent byte array we read from the wire.
std::atomic_bool fPauseSend
std::chrono::seconds m_nextGetAvaAddr
uint64_t nRemoteExtraEntropy
std::optional< std::pair< CNetMessage, bool > > PollMessage() EXCLUSIVE_LOCKS_REQUIRED(!m_msg_process_queue_mutex)
Poll the next message from the processing queue of this connection.
uint64_t GetLocalExtraEntropy() const
SteadyMilliseconds m_last_poll
double getAvailabilityScore() const
std::atomic_bool m_bloom_filter_loaded
Whether this peer has loaded a bloom filter.
void updateAvailabilityScore(double decayFactor)
The availability score is calculated using an exponentially weighted average.
std::atomic< std::chrono::seconds > m_avalanche_last_message_fault
const bool m_inbound_onion
Whether this peer is an inbound onion, i.e.
std::atomic< int > m_avalanche_message_fault_counter
How much faulty messages did this node accumulate.
std::atomic< bool > m_avalanche_enabled
std::atomic< std::chrono::seconds > m_last_block_time
UNIX epoch time of the last block received from this peer that we had not yet seen (e....
std::atomic_bool fDisconnect
std::atomic< int > m_avalanche_message_fault_score
This score is incremented for every new faulty message received when m_avalanche_message_fault_counte...
std::atomic< std::chrono::seconds > m_last_tx_time
UNIX epoch time of the last transaction received from this peer that we had not yet seen (e....
void invsVoted(uint32_t count)
The node voted for count invs.
bool IsAvalancheOutboundConnection() const
An encapsulated public key.
RollingBloomFilter is a probabilistic "keep track of most recently inserted" set.
Simple class for background tasks that should be run periodically or once "after a while".
void scheduleEvery(Predicate p, std::chrono::milliseconds delta) EXCLUSIVE_LOCKS_REQUIRED(!newTaskMutex)
Repeat p until it return false.
void scheduleFromNow(Function f, std::chrono::milliseconds delta) EXCLUSIVE_LOCKS_REQUIRED(!newTaskMutex)
Call f once after the delta has passed.
A combination of a network address (CNetAddr) and a (TCP) port.
std::vector< uint8_t > GetKey() const
std::string ToStringAddrPort() const
uint64_t Finalize() const
Compute the 64-bit SipHash-2-4 of the data written so far.
CSipHasher & Write(uint64_t data)
Hash a 64-bit integer worth of data.
std::set< std::reference_wrapper< const CTxMemPoolEntryRef >, CompareIteratorById > Parents
CTxMemPool stores valid-according-to-the-current-best-chain transactions that may be included in the ...
void removeConflicts(const CTransaction &tx) EXCLUSIVE_LOCKS_REQUIRED(cs)
void RemoveUnbroadcastTx(const TxId &txid, const bool unchecked=false)
Removes a transaction from the unbroadcast set.
CFeeRate GetMinFee() const
The minimum fee to get into the mempool, which may itself not be enough for larger-sized transactions...
RecursiveMutex cs
This mutex needs to be locked when accessing mapTx or other members that are guarded by it.
void removeRecursive(const CTransaction &tx, MemPoolRemovalReason reason) EXCLUSIVE_LOCKS_REQUIRED(cs)
bool CompareTopologically(const TxId &txida, const TxId &txidb) const
TxMempoolInfo info(const TxId &txid) const
size_t DynamicMemoryUsage() const
bool setAvalancheFinalized(const CTxMemPoolEntryRef &tx, const Consensus::Params ¶ms, const CBlockIndex &active_chain_tip, std::vector< TxId > &finalizedTxIds) EXCLUSIVE_LOCKS_REQUIRED(bool isAvalancheFinalizedPreConsensus(const TxId &txid) const EXCLUSIVE_LOCKS_REQUIRED(cs)
std::vector< TxMempoolInfo > infoAll() const
CTransactionRef GetConflictTx(const COutPoint &prevout) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Get the transaction in the pool that spends the same prevout.
bool exists(const TxId &txid) const
std::set< TxId > GetUnbroadcastTxs() const
Returns transactions in unbroadcast set.
auto withOrphanage(Callable &&func) const EXCLUSIVE_LOCKS_REQUIRED(!cs_orphanage)
const CFeeRate m_min_relay_feerate
auto withConflicting(Callable &&func) const EXCLUSIVE_LOCKS_REQUIRED(!cs_conflicting)
void removeForFinalizedBlock(const std::unordered_set< TxId, SaltedTxIdHasher > &confirmedTxIdsInNonFinalizedBlocks) EXCLUSIVE_LOCKS_REQUIRED(cs)
unsigned long size() const
std::optional< txiter > GetIter(const TxId &txid) const EXCLUSIVE_LOCKS_REQUIRED(cs)
Returns an iterator to the given txid, if found.
virtual void NewPoWValidBlock(const CBlockIndex *pindex, const std::shared_ptr< const CBlock > &block)
Notifies listeners that a block which builds directly on our current tip has been received and connec...
virtual void BlockConnected(ChainstateRole role, const std::shared_ptr< const CBlock > &block, const CBlockIndex *pindex)
Notifies listeners of a block being connected.
virtual void BlockChecked(const CBlock &, const BlockValidationState &)
Notifies listeners of a block validation result.
virtual void UpdatedBlockTip(const CBlockIndex *pindexNew, const CBlockIndex *pindexFork, bool fInitialDownload)
Notifies listeners when the block chain tip advances.
virtual void BlockDisconnected(const std::shared_ptr< const CBlock > &block, const CBlockIndex *pindex)
Notifies listeners of a block being disconnected.
Provides an interface for creating and interacting with one or two chainstates: an IBD chainstate gen...
SnapshotCompletionResult MaybeCompleteSnapshotValidation() EXCLUSIVE_LOCKS_REQUIRED(const CBlockIndex *GetSnapshotBaseBlock() const EXCLUSIVE_LOCKS_REQUIRED(Chainstate ActiveChainstate)() const
Once the background validation chainstate has reached the height which is the base of the UTXO snapsh...
const CBlockIndex * GetBackgroundSyncTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
The tip of the background sync chain.
MempoolAcceptResult ProcessTransaction(const CTransactionRef &tx, bool test_accept=false) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
Try to add a transaction to the memory pool.
bool IsInitialBlockDownload() const
Check whether we are doing an initial block download (synchronizing from disk or network)
bool ProcessNewBlock(const std::shared_ptr< const CBlock > &block, bool force_processing, bool min_pow_checked, bool *new_block, avalanche::Processor *const avalanche=nullptr) LOCKS_EXCLUDED(cs_main)
Process an incoming block.
RecursiveMutex & GetMutex() const LOCK_RETURNED(
Alias for cs_main.
CBlockIndex * ActiveTip() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
bool BackgroundSyncInProgress() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
The state of a background sync (for net processing)
bool ProcessNewBlockHeaders(const std::vector< CBlockHeader > &block, bool min_pow_checked, BlockValidationState &state, const CBlockIndex **ppindex=nullptr, const std::optional< CCheckpointData > &test_checkpoints=std::nullopt) LOCKS_EXCLUDED(cs_main)
Process incoming block headers.
const arith_uint256 & MinimumChainWork() const
CChain & ActiveChain() const EXCLUSIVE_LOCKS_REQUIRED(GetMutex())
void MaybeRebalanceCaches() EXCLUSIVE_LOCKS_REQUIRED(void ReportHeadersPresync(const arith_uint256 &work, int64_t height, int64_t timestamp)
Check to see if caches are out of balance and if so, call ResizeCoinsCaches() as needed.
node::BlockManager m_blockman
A single BlockManager instance is shared across each constructed chainstate to avoid duplicating bloc...
virtual uint64_t GetMaxBlockSize() const =0
Double ended buffer combining vector and stream-like interfaces.
void ignore(size_t num_ignore)
uint64_t rand64() noexcept
Generate a random 64-bit integer.
Reads data from an underlying stream, while hashing the read data.
A writer stream (for serialization) that computes a 256-bit hash.
size_t Count(NodeId peer) const
Count how many announcements a peer has (REQUESTED, CANDIDATE, and COMPLETED combined).
size_t CountInFlight(NodeId peer) const
Count how many REQUESTED announcements a peer has.
Interface for message handling.
static Mutex g_msgproc_mutex
Mutex for anything that is only accessed via the msg processing thread.
virtual bool ProcessMessages(const Config &config, CNode *pnode, std::atomic< bool > &interrupt) EXCLUSIVE_LOCKS_REQUIRED(g_msgproc_mutex)=0
Process protocol messages received from a given node.
virtual bool SendMessages(const Config &config, CNode *pnode) EXCLUSIVE_LOCKS_REQUIRED(g_msgproc_mutex)=0
Send queued protocol messages to a given node.
virtual void InitializeNode(const Config &config, CNode &node, ServiceFlags our_services)=0
Initialize a peer (setup state, queue any initial messages)
virtual void FinalizeNode(const Config &config, const CNode &node)=0
Handle removal of a peer (clear state)
static bool HasFlag(NetPermissionFlags flags, NetPermissionFlags f)
ReadStatus InitData(const CBlockHeaderAndShortTxIDs &cmpctblock, const std::vector< CTransactionRef > &extra_txn)
bool IsTxAvailable(size_t index) const
ReadStatus FillBlock(CBlock &block, const std::vector< CTransactionRef > &vtx_missing)
virtual std::optional< std::string > FetchBlock(const Config &config, NodeId peer_id, const CBlockIndex &block_index)=0
Attempt to manually fetch block from a given peer.
virtual void SendPings()=0
Send ping message to all peers.
static std::unique_ptr< PeerManager > make(CConnman &connman, AddrMan &addrman, BanMan *banman, ChainstateManager &chainman, CTxMemPool &pool, avalanche::Processor *const avalanche, Options opts)
virtual void StartScheduledTasks(CScheduler &scheduler)=0
Begin running background tasks, should only be called once.
virtual bool IgnoresIncomingTxs()=0
Whether this node ignores txs received over p2p.
virtual void ProcessMessage(const Config &config, CNode &pfrom, const std::string &msg_type, DataStream &vRecv, const std::chrono::microseconds time_received, const std::atomic< bool > &interruptMsgProc) EXCLUSIVE_LOCKS_REQUIRED(g_msgproc_mutex)=0
Process a single message from a peer.
virtual bool GetNodeStateStats(NodeId nodeid, CNodeStateStats &stats) const =0
Get statistics from node state.
virtual void UnitTestMisbehaving(const NodeId peer_id)=0
Public for unit testing.
virtual void UpdateLastBlockAnnounceTime(NodeId node, int64_t time_in_seconds)=0
This function is used for testing the stale tip eviction logic, see denialofservice_tests....
virtual void CheckForStaleTipAndEvictPeers()=0
Evict extra outbound peers.
static RCUPtr make(Args &&...args)
Construct a new object that is owned by the pointer.
I randrange(I range) noexcept
Generate a random integer in the range [0..range), with range > 0.
A Span is an object that can refer to a contiguous sequence of objects.
int EraseTx(const TxId &txid) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Erase a tx by txid.
void EraseForPeer(NodeId peer) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Erase all txs announced by a peer (eg, after that peer disconnects)
std::vector< CTransactionRef > GetChildrenFromSamePeer(const CTransactionRef &parent, NodeId nodeid) const EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Get all children that spend from this tx and were received from nodeid.
bool AddTx(const CTransactionRef &tx, NodeId peer) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Add a new transaction to the pool.
unsigned int LimitTxs(unsigned int max_txs, FastRandomContext &rng) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Limit the txs to the given maximum.
void EraseForBlock(const CBlock &block) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Erase all txs included in or invalidated by a new block.
std::vector< CTransactionRef > GetConflictTxs(const CTransactionRef &tx) const EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
void AddChildrenToWorkSet(const CTransaction &tx) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Add any tx that list a particular tx as a parent into the from peer's work set.
std::vector< std::pair< CTransactionRef, NodeId > > GetChildrenFromDifferentPeer(const CTransactionRef &parent, NodeId nodeid) const EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
Get all children that spend from this tx but were not received from nodeid.
std::string GetRejectReason() const
std::string ToString() const
256-bit unsigned big integer.
const std::vector< PrefilledProof > & getPrefilledProofs() const
uint64_t getShortID(const ProofId &proofid) const
const std::vector< uint64_t > & getShortIDs() const
ProofId getProofId() const
bool verify(DelegationState &state, CPubKey &auth) const
const DelegationId & getId() const
const LimitedProofId & getLimitedProofId() const
bool addNode(NodeId nodeid, const ProofId &proofid, size_t max_elements)
Node API.
bool shouldRequestMoreNodes()
Returns true if we encountered a lack of node since the last call.
bool exists(const ProofId &proofid) const
Return true if the (valid) proof exists, but only for non-dangling proofs.
bool forPeer(const ProofId &proofid, Callable &&func) const
void removeUnbroadcastProof(const ProofId &proofid)
const ProofRadixTree & getShareableProofsSnapshot() const
bool isBoundToPeer(const ProofId &proofid) const
bool saveRemoteProof(const ProofId &proofid, const NodeId nodeid, const bool present)
void forEachPeer(Callable &&func) const
void setInvalid(const ProofId &proofid)
bool isInvalid(const ProofId &proofid) const
bool isImmature(const ProofId &proofid) const
auto getUnbroadcastProofs() const
bool isInConflictingPool(const ProofId &proofid) const
void sendResponse(CNode *pfrom, Response response) const
bool addToReconcile(const AnyVoteItem &item) EXCLUSIVE_LOCKS_REQUIRED(!cs_finalizedItems)
bool isStakingPreconsensusActivated(const CBlockIndex *pprev) const
int64_t getAvaproofsNodeCounter() const
bool sendHello(CNode *pfrom) EXCLUSIVE_LOCKS_REQUIRED(!cs_delayedAvahelloNodeIds)
Send a avahello message.
void setRecentlyFinalized(const uint256 &itemId) EXCLUSIVE_LOCKS_REQUIRED(!cs_finalizedItems)
size_t getMaxElementPoll() const
bool isQuorumEstablished() LOCKS_EXCLUDED(cs_main) EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager
void cleanupStakingRewards(const int minHeight) EXCLUSIVE_LOCKS_REQUIRED(!cs_stakingRewards
ProofRef getLocalProof() const
void acceptStakeContender(const StakeContenderId &contenderId) EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager)
bool reconcileOrFinalize(const ProofRef &proof) EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager
Wrapper around the addToReconcile for proofs that adds back the finalization flag to the peer if it i...
int getStakeContenderStatus(const StakeContenderId &contenderId) const EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager
Track votes on stake contenders.
void sendDelayedAvahello() EXCLUSIVE_LOCKS_REQUIRED(!cs_delayedAvahelloNodeIds)
void finalizeStakeContender(const StakeContenderId &contenderId) EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager
bool isPreconsensusActivated(const CBlockIndex *pprev) const
auto withPeerManager(Callable &&func) const EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager)
bool registerVotes(NodeId nodeid, const Response &response, std::vector< VoteItemUpdate > &updates, bool &disconnect, std::string &error) EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager
void rejectStakeContender(const StakeContenderId &contenderId) EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager)
void avaproofsSent(NodeId nodeid) LOCKS_EXCLUDED(cs_main) EXCLUSIVE_LOCKS_REQUIRED(!cs_peerManager)
std::vector< uint32_t > indices
std::string ToString() const
std::string GetHex() const
Generate a new block, without valid proof-of-work.
bool ReadRawBlock(std::vector< uint8_t > &block, const FlatFilePos &pos) const
CBlockIndex * LookupBlockIndex(const BlockHash &hash) EXCLUSIVE_LOCKS_REQUIRED(cs_main)
bool LoadingBlocks() const
bool IsPruneMode() const
Whether running in -prune mode.
bool ReadBlock(CBlock &block, const FlatFilePos &pos) const
Functions for disk access for blocks.
static const uint256 ZERO
@ BLOCK_CHECKPOINT
the block failed to meet one of our checkpoints
@ BLOCK_HEADER_LOW_WORK
the block header may be on a too-little-work chain
@ BLOCK_INVALID_HEADER
invalid proof of work or time too old
@ BLOCK_CACHED_INVALID
this block was cached as being invalid and we didn't store the reason why
@ BLOCK_CONSENSUS
invalid by consensus rules (excluding any below reasons)
@ BLOCK_MISSING_PREV
We don't have the previous block the checked one is built on.
@ BLOCK_INVALID_PREV
A block this one builds on is invalid.
@ BLOCK_MUTATED
the block's data didn't match the data committed to by the PoW
@ BLOCK_TIME_FUTURE
block timestamp was > 2 hours in the future (or our clock is bad)
@ BLOCK_RESULT_UNSET
initial value. Block has not yet been rejected
@ TX_MISSING_INPUTS
transaction was missing some of its inputs
@ TX_CHILD_BEFORE_PARENT
This tx outputs are already spent in the mempool.
@ TX_MEMPOOL_POLICY
violated mempool's fee/size/descendant/etc limits
@ TX_PACKAGE_RECONSIDERABLE
fails some policy, but might be acceptable if submitted in a (different) package
@ TX_UNKNOWN
transaction was not validated because package failed
@ TX_PREMATURE_SPEND
transaction spends a coinbase too early, or violates locktime/sequence locks
@ TX_DUPLICATE
Tx already in mempool or in the chain.
@ TX_INPUTS_NOT_STANDARD
inputs failed policy rules
@ TX_CONFLICT
Tx conflicts with a finalized tx, i.e.
@ TX_NOT_STANDARD
otherwise didn't meet our local policy rules
@ TX_AVALANCHE_RECONSIDERABLE
fails some policy, but might be reconsidered by avalanche voting
@ TX_NO_MEMPOOL
this node does not have a mempool so can't validate the transaction
@ TX_RESULT_UNSET
initial value. Tx has not yet been rejected
@ TX_CONSENSUS
invalid by consensus rules
static size_t RecursiveDynamicUsage(const CScript &script)
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
ChainstateRole
This enum describes the various roles a specific Chainstate instance can take.
std::array< uint8_t, CPubKey::SCHNORR_SIZE > SchnorrSig
a Schnorr signature
#define LogPrintLevel(category, level,...)
#define LogPrint(category,...)
#define LogDebug(category,...)
CSerializedNetMsg Make(std::string msg_type, Args &&...args)
const char * FILTERLOAD
The filterload message tells the receiving peer to filter all relayed transactions and requested merk...
const char * CFHEADERS
cfheaders is a response to a getcfheaders request containing a filter header and a vector of filter h...
const char * AVAPROOFSREQ
Request for missing avalanche proofs after an avaproofs message has been processed.
const char * CFILTER
cfilter is a response to a getcfilters request containing a single compact filter.
const char * BLOCK
The block message transmits a single serialized block.
const char * FILTERCLEAR
The filterclear message tells the receiving peer to remove a previously-set bloom filter.
const char * HEADERS
The headers message sends one or more block headers to a node which previously requested certain head...
const char * ADDRV2
The addrv2 message relays connection information for peers on the network just like the addr message,...
const char * SENDHEADERS
Indicates that a node prefers to receive new block announcements via a "headers" message rather than ...
const char * AVAPROOFS
The avaproofs message the proof short ids of all the valid proofs that we know.
const char * PONG
The pong message replies to a ping message, proving to the pinging node that the ponging node is stil...
const char * GETAVAPROOFS
The getavaproofs message requests an avaproofs message that provides the proof short ids of all the v...
const char * SENDCMPCT
Contains a 1-byte bool and 8-byte LE version number.
const char * GETADDR
The getaddr message requests an addr message from the receiving node, preferably one with lots of IP ...
const char * GETCFCHECKPT
getcfcheckpt requests evenly spaced compact filter headers, enabling parallelized download and valida...
const char * NOTFOUND
The notfound message is a reply to a getdata message which requested an object the receiving node doe...
const char * GETAVAADDR
The getavaaddr message requests an addr message from the receiving node, containing IP addresses of t...
const char * CMPCTBLOCK
Contains a CBlockHeaderAndShortTxIDs object - providing a header and list of "short txids".
const char * MEMPOOL
The mempool message requests the TXIDs of transactions that the receiving node has verified as valid ...
const char * GETCFILTERS
getcfilters requests compact filters for a range of blocks.
const char * TX
The tx message transmits a single transaction.
const char * AVAHELLO
Contains a delegation and a signature.
const char * FILTERADD
The filteradd message tells the receiving peer to add a single element to a previously-set bloom filt...
const char * ADDR
The addr (IP address) message relays connection information for peers on the network.
const char * VERSION
The version message provides information about the transmitting node to the receiving node at the beg...
const char * GETBLOCKS
The getblocks message requests an inv message that provides block header hashes starting from a parti...
const char * FEEFILTER
The feefilter message tells the receiving peer not to inv us any txs which do not meet the specified ...
const char * GETHEADERS
The getheaders message requests a headers message that provides block headers starting from a particu...
const char * AVARESPONSE
Contains an avalanche::Response.
const char * GETDATA
The getdata message requests one or more data objects from another node.
const char * VERACK
The verack message acknowledges a previously-received version message, informing the connecting node ...
const char * BLOCKTXN
Contains a BlockTransactions.
const char * GETCFHEADERS
getcfheaders requests a compact filter header and the filter hashes for a range of blocks,...
const char * SENDADDRV2
The sendaddrv2 message signals support for receiving ADDRV2 messages (BIP155).
const char * PING
The ping message is sent periodically to help confirm that the receiving peer is still connected.
const char * AVAPOLL
Contains an avalanche::Poll.
const char * MERKLEBLOCK
The merkleblock message is a reply to a getdata message which requested a block using the inventory t...
const char * AVAPROOF
Contains an avalanche::Proof.
const char * CFCHECKPT
cfcheckpt is a response to a getcfcheckpt request containing a vector of evenly spaced filter headers...
const char * GETBLOCKTXN
Contains a BlockTransactionsRequest Peer should respond with "blocktxn" message.
const char * INV
The inv message (inventory message) transmits one or more inventories of objects known to the transmi...
ShortIdProcessor< PrefilledProof, ShortIdProcessorPrefilledProofAdapter, ProofRefCompare > ProofShortIdProcessor
std::variant< const ProofRef, const CBlockIndex *, const StakeContenderId, const CTransactionRef > AnyVoteItem
RCUPtr< const Proof > ProofRef
Implement std::hash so RCUPtr can be used as a key for maps or sets.
std::optional< CService > GetLocalAddrForPeer(CNode &node)
Returns a local address that we should advertise to this peer.
std::function< void(const CAddress &addr, const std::string &msg_type, Span< const uint8_t > data, bool is_incoming)> CaptureMessage
Defaults to CaptureMessageToFile(), but can be overridden by unit tests.
std::string userAgent(const Config &config)
bool IsReachable(enum Network net)
bool SeenLocal(const CService &addr)
vote for a local address
static const unsigned int MAX_SUBVERSION_LENGTH
Maximum length of the user agent string in version message.
static constexpr std::chrono::minutes TIMEOUT_INTERVAL
Time after which to disconnect, after waiting for a ping response (or inactivity).
@ BypassProofRequestLimits
static constexpr auto HEADERS_RESPONSE_TIME
How long to wait for a peer to respond to a getheaders request.
static constexpr size_t MAX_ADDR_PROCESSING_TOKEN_BUCKET
The soft limit of the address processing token bucket (the regular MAX_ADDR_RATE_PER_SECOND based inc...
static constexpr size_t MAX_AVALANCHE_STALLED_TXIDS_PER_PEER
Maximum number of stalled avalanche txids to store per peer.
static const int MAX_BLOCKS_IN_TRANSIT_PER_PEER
Number of blocks that can be requested at any given time from a single peer.
static constexpr auto BLOCK_STALLING_TIMEOUT_DEFAULT
Default time during which a peer must stall block download progress before being disconnected.
static constexpr auto GETAVAADDR_INTERVAL
Minimum time between 2 successives getavaaddr messages from the same peer.
static constexpr auto AVG_FEEFILTER_BROADCAST_INTERVAL
Verify that INVENTORY_MAX_RECENT_RELAY is enough to cache everything typically relayed before uncondi...
static constexpr unsigned int INVENTORY_BROADCAST_MAX_PER_MB
Maximum number of inventory items to send per transmission.
static constexpr auto EXTRA_PEER_CHECK_INTERVAL
How frequently to check for extra outbound peers and disconnect.
static const unsigned int BLOCK_DOWNLOAD_WINDOW
Size of the "block download window": how far ahead of our current height do we fetch?...
static uint32_t getAvalancheVoteForProof(const avalanche::Processor &avalanche, const avalanche::ProofId &id)
Decide a response for an Avalanche poll about the given proof.
static constexpr int STALE_RELAY_AGE_LIMIT
Age after which a stale block will no longer be served if requested as protection against fingerprint...
static constexpr int HISTORICAL_BLOCK_AGE
Age after which a block is considered historical for purposes of rate limiting block relay.
static constexpr auto ROTATE_ADDR_RELAY_DEST_INTERVAL
Delay between rotating the peers we relay a particular address to.
static constexpr auto MINIMUM_CONNECT_TIME
Minimum time an outbound-peer-eviction candidate must be connected for, in order to evict.
static constexpr auto CHAIN_SYNC_TIMEOUT
Timeout for (unprotected) outbound peers to sync to our chainwork.
static const unsigned int NODE_NETWORK_LIMITED_MIN_BLOCKS
Minimum blocks required to signal NODE_NETWORK_LIMITED.
static constexpr auto AVG_LOCAL_ADDRESS_BROADCAST_INTERVAL
Average delay between local address broadcasts.
static const int MAX_BLOCKTXN_DEPTH
Maximum depth of blocks we're willing to respond to GETBLOCKTXN requests for.
static constexpr uint64_t CMPCTBLOCKS_VERSION
The compactblocks version we support.
bool IsAvalancheMessageType(const std::string &msg_type)
static constexpr int32_t MAX_OUTBOUND_PEERS_TO_PROTECT_FROM_DISCONNECT
Protect at least this many outbound peers from disconnection due to slow/behind headers chain.
static std::chrono::microseconds ComputeRequestTime(const CNode &node, const InvRequestTracker< InvId > &requestTracker, const DataRequestParameters &requestParams, std::chrono::microseconds current_time, bool preferred)
Compute the request time for this announcement, current time plus delays for:
static constexpr auto INBOUND_INVENTORY_BROADCAST_INTERVAL
Average delay between trickled inventory transmissions for inbound peers.
static constexpr DataRequestParameters TX_REQUEST_PARAMS
static constexpr auto MAX_FEEFILTER_CHANGE_DELAY
Maximum feefilter broadcast delay after significant change.
static constexpr uint32_t MAX_GETCFILTERS_SIZE
Maximum number of compact filters that may be requested with one getcfilters.
static constexpr auto HEADERS_DOWNLOAD_TIMEOUT_BASE
Headers download timeout.
static const unsigned int MAX_GETDATA_SZ
Limit to avoid sending big packets.
static constexpr double BLOCK_DOWNLOAD_TIMEOUT_BASE
Block download timeout base, expressed in multiples of the block interval (i.e.
static constexpr auto AVALANCHE_AVAPROOFS_TIMEOUT
If no proof was requested from a compact proof message after this timeout expired,...
static constexpr auto STALE_CHECK_INTERVAL
How frequently to check for stale tips.
static constexpr unsigned int INVENTORY_MAX_RECENT_RELAY
The number of most recently announced transactions a peer can request.
static constexpr auto UNCONDITIONAL_RELAY_DELAY
How long a transaction has to be in the mempool before it can unconditionally be relayed.
static constexpr auto AVG_ADDRESS_BROADCAST_INTERVAL
Average delay between peer address broadcasts.
static const unsigned int MAX_LOCATOR_SZ
The maximum number of entries in a locator.
static constexpr double BLOCK_DOWNLOAD_TIMEOUT_PER_PEER
Additional block download timeout per parallel downloading peer (i.e.
static constexpr double MAX_ADDR_RATE_PER_SECOND
The maximum rate of address records we're willing to process on average.
static constexpr auto PING_INTERVAL
Time between pings automatically sent out for latency probing and keepalive.
static const int MAX_CMPCTBLOCK_DEPTH
Maximum depth of blocks we're willing to serve as compact blocks to peers when requested.
static constexpr DataRequestParameters PROOF_REQUEST_PARAMS
static const unsigned int MAX_BLOCKS_TO_ANNOUNCE
Maximum number of headers to announce when relaying blocks with headers message.
static bool TooManyAnnouncements(const CNode &node, const InvRequestTracker< InvId > &requestTracker, const DataRequestParameters &requestParams)
static constexpr uint32_t MAX_GETCFHEADERS_SIZE
Maximum number of cf hashes that may be requested with one getcfheaders.
static constexpr auto BLOCK_STALLING_TIMEOUT_MAX
Maximum timeout for stalling block download.
static constexpr auto HEADERS_DOWNLOAD_TIMEOUT_PER_HEADER
static constexpr uint64_t RANDOMIZER_ID_ADDRESS_RELAY
SHA256("main address relay")[0:8].
static constexpr size_t MAX_PCT_ADDR_TO_SEND
the maximum percentage of addresses from our addrman to return in response to a getaddr message.
static const unsigned int MAX_INV_SZ
The maximum number of entries in an 'inv' protocol message.
static constexpr unsigned int INVENTORY_BROADCAST_PER_SECOND
Maximum rate of inventory items to send per second.
static constexpr size_t MAX_ADDR_TO_SEND
The maximum number of address records permitted in an ADDR message.
static const unsigned int MAX_CMPCTBLOCKS_INFLIGHT_PER_BLOCK
Maximum number of outstanding CMPCTBLOCK requests for the same block.
static const unsigned int MAX_HEADERS_RESULTS
Number of headers sent in one getheaders result.
bool IsProxy(const CNetAddr &addr)
static constexpr NodeId NO_NODE
Special NodeId that represent no node.
uint256 GetPackageHash(const Package &package)
std::vector< CTransactionRef > Package
A package is an ordered list of transactions.
static constexpr Amount DEFAULT_MIN_RELAY_TX_FEE_PER_KB(1000 *SATOSHI)
Default for -minrelaytxfee, minimum relay fee for transactions.
std::shared_ptr< const CTransaction > CTransactionRef
static constexpr size_t AVALANCHE_MAX_ELEMENT_POLL_LEGACY
Legacy maximum element poll.
void SetServiceFlagsIBDCache(bool state)
Set the current IBD status in order to figure out the desirable service flags.
ServiceFlags GetDesirableServiceFlags(ServiceFlags services)
Gets the set of service flags which are "desirable" for a given peer.
static const unsigned int MAX_PROTOCOL_MESSAGE_LENGTH
Maximum length of incoming protocol messages (Currently 2MB).
static bool HasAllDesirableServiceFlags(ServiceFlags services)
A shortcut for (services & GetDesirableServiceFlags(services)) == GetDesirableServiceFlags(services),...
@ MSG_AVA_STAKE_CONTENDER
@ MSG_CMPCT_BLOCK
Defined in BIP152.
ServiceFlags
nServices flags.
static bool MayHaveUsefulAddressDB(ServiceFlags services)
Checks if a peer with the given service flags may be capable of having a robust address-storage DB.
static const int SHORT_IDS_BLOCKS_VERSION
short-id-based block download starts with this version
static const int SENDHEADERS_VERSION
"sendheaders" command and announcing blocks with headers starts with this version
static const int PROTOCOL_VERSION
network protocol versioning
static const int FEEFILTER_VERSION
"feefilter" tells peers to filter invs to you by fee starts with this version
static const int MIN_PEER_PROTO_VERSION
disconnect from peers older than this proto version
static const int INVALID_CB_NO_BAN_VERSION
not banning for invalid compact blocks starts with this version
static const int BIP0031_VERSION
BIP 0031, pong message, is enabled for all versions AFTER this one.
static const int AVALANCHE_MAX_ELEMENT_BUMP_VERSION
Avalanche can poll up to 1024 items per message starting with this version.
void Shuffle(I first, I last, R &&rng)
More efficient than using std::shuffle on a FastRandomContext.
reverse_range< T > reverse_iterate(T &x)
static const unsigned int MAX_SCRIPT_ELEMENT_SIZE
static std::string ToString(const CService &ip)
void Unserialize(Stream &, V)=delete
#define LIMITED_STRING(obj, n)
static auto WithParams(const Params ¶ms, T &&t)
Return a wrapper around t that (de)serializes it with specified parameter params.
uint64_t ReadCompactSize(Stream &is, bool range_check=true)
Decode a CompactSize-encoded variable-length integer.
constexpr auto MakeUCharSpan(V &&v) -> decltype(UCharSpanCast(Span{std::forward< V >(v)}))
Like the Span constructor, but for (const) uint8_t member types only.
static const double AVALANCHE_STATISTICS_DECAY_FACTOR
Pre-computed decay factor for the avalanche statistics computation.
static constexpr std::chrono::minutes AVALANCHE_STATISTICS_REFRESH_PERIOD
Refresh period for the avalanche statistics computation.
static constexpr Amount zero() noexcept
A BlockHash is a unqiue identifier for a block.
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
std::vector< BlockHash > vHave
std::chrono::microseconds m_ping_wait
Amount m_fee_filter_received
std::vector< int > vHeightInFlight
bool m_addr_relay_enabled
uint64_t m_addr_rate_limited
uint64_t m_addr_processed
ServiceFlags their_services
Parameters that influence chain consensus.
int64_t nPowTargetSpacing
std::chrono::seconds PowTargetSpacing() const
const std::chrono::seconds overloaded_peer_delay
How long to delay requesting data from overloaded peers (see max_peer_request_in_flight).
const size_t max_peer_announcements
Maximum number of inventories to consider for requesting, per peer.
const std::chrono::seconds nonpref_peer_delay
How long to delay requesting data from non-preferred peers.
const NetPermissionFlags bypass_request_limits_permissions
Permission flags a peer requires to bypass the request limits tracking limits and delay penalty.
const std::chrono::microseconds getdata_interval
How long to wait (in microseconds) before a data request from an additional peer.
const size_t max_peer_request_in_flight
Maximum number of in-flight data requests from a peer.
Validation result for a transaction evaluated by MemPoolAccept (single or package).
const ResultType m_result_type
Result type.
const TxValidationState m_state
Contains information about why the transaction failed.
@ MEMPOOL_ENTRY
Valid, transaction was already in the mempool.
@ VALID
Fully validated, valid.
static time_point now() noexcept
Return current system time or mocked time, if set.
std::chrono::time_point< NodeClock > time_point
Validation result for package mempool acceptance.
PackageValidationState m_state
std::map< TxId, MempoolAcceptResult > m_tx_results
Map from txid to finished MempoolAcceptResults.
This is a radix tree storing values identified by a unique key.
A TxId is the identifier of a transaction.
std::chrono::seconds registration_time
const ProofId & getProofId() const
StakeContenderIds are unique for each block to ensure that the peer polling for their acceptance has ...
#define AssertLockNotHeld(cs)
#define WITH_LOCK(cs, code)
Run code while locking a mutex.
#define EXCLUSIVE_LOCKS_REQUIRED(...)
#define LOCKS_EXCLUDED(...)
#define NO_THREAD_SAFETY_ANALYSIS
int64_t GetTime()
DEPRECATED Use either ClockType::now() or Now<TimePointType>() if a cast is needed.
constexpr int64_t count_microseconds(std::chrono::microseconds t)
constexpr int64_t count_seconds(std::chrono::seconds t)
std::chrono::time_point< NodeClock, std::chrono::seconds > NodeSeconds
double CountSecondsDouble(SecondsDouble t)
Helper to count the seconds in any std::chrono::duration type.
NodeClock::time_point GetAdjustedTime()
void AddTimeData(const CNetAddr &ip, int64_t nOffsetSample)
#define TRACE6(context, event, a, b, c, d, e, f)
@ AVALANCHE
Removed by avalanche vote.
std::string SanitizeString(std::string_view str, int rule)
Remove unsafe chars.
arith_uint256 CalculateHeadersWork(const std::vector< CBlockHeader > &headers)
Return the sum of the work on a given set of headers.
bool HasValidProofOfWork(const std::vector< CBlockHeader > &headers, const Consensus::Params &consensusParams)
Check with the proof of work on each blockheader matches the value in nBits.
PackageMempoolAcceptResult ProcessNewPackage(Chainstate &active_chainstate, CTxMemPool &pool, const Package &package, bool test_accept)
Validate (and maybe submit) a package to the mempool.
bool IsBlockMutated(const CBlock &block)
Check if a block has been mutated (with respect to its merkle root).
std::optional< std::vector< Coin > > GetSpentCoins(const CTransactionRef &ptx, const CCoinsViewCache &coins_view)
Get the coins spent by ptx from the coins_view.
static const unsigned int MIN_BLOCKS_TO_KEEP
Block files containing a block-height within MIN_BLOCKS_TO_KEEP of ActiveChain().Tip() will not be pr...
CMainSignals & GetMainSignals()