Coverage Report

Created: 2026-04-29 19:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/tmp/bitcoin/src/txrequest.cpp
Line
Count
Source
1
// Copyright (c) 2020-present The Bitcoin Core developers
2
// Distributed under the MIT software license, see the accompanying
3
// file COPYING or http://www.opensource.org/licenses/mit-license.php.
4
5
#include <txrequest.h>
6
7
#include <crypto/siphash.h>
8
#include <net.h>
9
#include <primitives/transaction.h>
10
#include <random.h>
11
#include <uint256.h>
12
13
#include <boost/multi_index/indexed_by.hpp>
14
#include <boost/multi_index/ordered_index.hpp>
15
#include <boost/multi_index/sequenced_index.hpp>
16
#include <boost/multi_index/tag.hpp>
17
#include <boost/multi_index_container.hpp>
18
#include <boost/tuple/tuple.hpp>
19
20
#include <chrono>
21
#include <unordered_map>
22
#include <utility>
23
24
#include <cassert>
25
26
namespace {
27
28
/** The various states a (txhash,peer) pair can be in.
29
 *
30
 * Note that CANDIDATE is split up into 3 substates (DELAYED, BEST, READY), allowing more efficient implementation.
31
 * Also note that the sorting order of ByTxHashView relies on the specific order of values in this enum.
32
 *
33
 * Expected behaviour is:
34
 *   - When first announced by a peer, the state is CANDIDATE_DELAYED until reqtime is reached.
35
 *   - Announcements that have reached their reqtime but not been requested will be either CANDIDATE_READY or
36
 *     CANDIDATE_BEST. Neither of those has an expiration time; they remain in that state until they're requested or
37
 *     no longer needed. CANDIDATE_READY announcements are promoted to CANDIDATE_BEST when they're the best one left.
38
 *   - When requested, an announcement will be in state REQUESTED until expiry is reached.
39
 *   - If expiry is reached, or the peer replies to the request (either with NOTFOUND or the tx), the state becomes
40
 *     COMPLETED.
41
 */
42
enum class State : uint8_t {
43
    /** A CANDIDATE announcement whose reqtime is in the future. */
44
    CANDIDATE_DELAYED,
45
    /** A CANDIDATE announcement that's not CANDIDATE_DELAYED or CANDIDATE_BEST. */
46
    CANDIDATE_READY,
47
    /** The best CANDIDATE for a given txhash; only if there is no REQUESTED announcement already for that txhash.
48
     *  The CANDIDATE_BEST is the highest-priority announcement among all CANDIDATE_READY (and _BEST) ones for that
49
     *  txhash. */
50
    CANDIDATE_BEST,
51
    /** A REQUESTED announcement. */
52
    REQUESTED,
53
    /** A COMPLETED announcement. */
54
    COMPLETED,
55
};
56
57
//! Type alias for sequence numbers.
58
using SequenceNumber = uint64_t;
59
60
/** An announcement. This is the data we track for each txid or wtxid that is announced to us by each peer. */
61
struct Announcement {
62
    /** Txid or wtxid that was announced. */
63
    const GenTxid m_gtxid;
64
    /** For CANDIDATE_{DELAYED,BEST,READY} the reqtime; for REQUESTED the expiry. */
65
    std::chrono::microseconds m_time;
66
    /** What peer the request was from. */
67
    const NodeId m_peer;
68
    /** What sequence number this announcement has. */
69
    const SequenceNumber m_sequence : 59;
70
    /** Whether the request is preferred. */
71
    const bool m_preferred : 1;
72
    /** What state this announcement is in. */
73
    State m_state : 3 {State::CANDIDATE_DELAYED};
74
51.6M
    State GetState() const { return m_state; }
75
93.3k
    void SetState(State state) { m_state = state; }
76
77
    /** Whether this announcement is selected. There can be at most 1 selected peer per txhash. */
78
    bool IsSelected() const
79
4.74k
    {
80
4.74k
        return GetState() == State::CANDIDATE_BEST || GetState() == State::REQUESTED;
81
4.74k
    }
82
83
    /** Whether this announcement is waiting for a certain time to pass. */
84
    bool IsWaiting() const
85
4.23M
    {
86
4.23M
        return GetState() == State::REQUESTED || GetState() == State::CANDIDATE_DELAYED;
87
4.23M
    }
88
89
    /** Whether this announcement can feasibly be selected if the current IsSelected() one disappears. */
90
    bool IsSelectable() const
91
2.51M
    {
92
2.51M
        return GetState() == State::CANDIDATE_READY || GetState() == State::CANDIDATE_BEST;
93
2.51M
    }
94
95
    /** Construct a new announcement from scratch, initially in CANDIDATE_DELAYED state. */
96
    Announcement(const GenTxid& gtxid, NodeId peer, bool preferred, std::chrono::microseconds reqtime,
97
                 SequenceNumber sequence)
98
30.9k
        : m_gtxid(gtxid), m_time(reqtime), m_peer(peer), m_sequence(sequence), m_preferred(preferred) {}
99
};
100
101
//! Type alias for priorities.
102
using Priority = uint64_t;
103
104
/** A functor with embedded salt that computes priority of an announcement.
105
 *
106
 * Higher priorities are selected first.
107
 */
108
class PriorityComputer {
109
    const uint64_t m_k0, m_k1;
110
public:
111
    explicit PriorityComputer(bool deterministic) :
112
1.23k
        m_k0{deterministic ? 0 : FastRandomContext().rand64()},
113
1.23k
        m_k1{deterministic ? 0 : FastRandomContext().rand64()} {}
114
115
    Priority operator()(const uint256& txhash, NodeId peer, bool preferred) const
116
4.66M
    {
117
4.66M
        uint64_t low_bits = CSipHasher(m_k0, m_k1).Write(txhash).Write(peer).Finalize() >> 1;
118
4.66M
        return low_bits | uint64_t{preferred} << 63;
119
4.66M
    }
120
121
    Priority operator()(const Announcement& ann) const
122
2.43M
    {
123
2.43M
        return operator()(ann.m_gtxid.ToUint256(), ann.m_peer, ann.m_preferred);
124
2.43M
    }
125
};
126
127
// Definitions for the 3 indexes used in the main data structure.
128
//
129
// Each index has a By* type to identify it, a By*View data type to represent the view of announcement it is sorted
130
// by, and an By*ViewExtractor type to convert an announcement into the By*View type.
131
// See https://www.boost.org/doc/libs/1_58_0/libs/multi_index/doc/reference/key_extraction.html#key_extractors
132
// for more information about the key extraction concept.
133
134
// The ByPeer index is sorted by (peer, state == CANDIDATE_BEST, txhash)
135
//
136
// Uses:
137
// * Looking up existing announcements by peer/txhash, by checking both (peer, false, txhash) and
138
//   (peer, true, txhash).
139
// * Finding all CANDIDATE_BEST announcements for a given peer in GetRequestable.
140
struct ByPeer {};
141
using ByPeerView = std::tuple<NodeId, bool, const uint256&>;
142
struct ByPeerViewExtractor
143
{
144
    using result_type = ByPeerView;
145
    result_type operator()(const Announcement& ann) const
146
2.06M
    {
147
2.06M
        return ByPeerView{ann.m_peer, ann.GetState() == State::CANDIDATE_BEST, ann.m_gtxid.ToUint256()};
148
2.06M
    }
149
};
150
151
// The ByTxHash index is sorted by (txhash, state, priority).
152
//
153
// Note: priority == 0 whenever state != CANDIDATE_READY.
154
//
155
// Uses:
156
// * Deleting all announcements with a given txhash in ForgetTxHash.
157
// * Finding the best CANDIDATE_READY to convert to CANDIDATE_BEST, when no other CANDIDATE_READY or REQUESTED
158
//   announcement exists for that txhash.
159
// * Determining when no more non-COMPLETED announcements for a given txhash exist, so the COMPLETED ones can be
160
//   deleted.
161
struct ByTxHash {};
162
using ByTxHashView = std::tuple<const uint256&, State, Priority>;
163
class ByTxHashViewExtractor {
164
    const PriorityComputer& m_computer;
165
public:
166
1.23k
    explicit ByTxHashViewExtractor(const PriorityComputer& computer) : m_computer(computer) {}
167
    using result_type = ByTxHashView;
168
    result_type operator()(const Announcement& ann) const
169
650k
    {
170
650k
        const Priority prio = (ann.GetState() == State::CANDIDATE_READY) ? m_computer(ann) : 0;
171
650k
        return ByTxHashView{ann.m_gtxid.ToUint256(), ann.GetState(), prio};
172
650k
    }
173
};
174
175
enum class WaitState {
176
    //! Used for announcements that need efficient testing of "is their timestamp in the future?".
177
    FUTURE_EVENT,
178
    //! Used for announcements whose timestamp is not relevant.
179
    NO_EVENT,
180
    //! Used for announcements that need efficient testing of "is their timestamp in the past?".
181
    PAST_EVENT,
182
};
183
184
WaitState GetWaitState(const Announcement& ann)
185
1.10M
{
186
1.10M
    if (ann.IsWaiting()) return WaitState::FUTURE_EVENT;
187
475k
    if (ann.IsSelectable()) return WaitState::PAST_EVENT;
188
24.2k
    return WaitState::NO_EVENT;
189
475k
}
190
191
// The ByTime index is sorted by (wait_state, time).
192
//
193
// All announcements with a timestamp in the future can be found by iterating the index forward from the beginning.
194
// All announcements with a timestamp in the past can be found by iterating the index backwards from the end.
195
//
196
// Uses:
197
// * Finding CANDIDATE_DELAYED announcements whose reqtime has passed, and REQUESTED announcements whose expiry has
198
//   passed.
199
// * Finding CANDIDATE_READY/BEST announcements whose reqtime is in the future (when the clock time went backwards).
200
struct ByTime {};
201
using ByTimeView = std::pair<WaitState, std::chrono::microseconds>;
202
struct ByTimeViewExtractor
203
{
204
    using result_type = ByTimeView;
205
    result_type operator()(const Announcement& ann) const
206
1.10M
    {
207
1.10M
        return ByTimeView{GetWaitState(ann), ann.m_time};
208
1.10M
    }
209
};
210
211
212
/** Data type for the main data structure (Announcement objects with ByPeer/ByTxHash/ByTime indexes). */
213
using Index = boost::multi_index_container<
214
    Announcement,
215
    boost::multi_index::indexed_by<
216
        boost::multi_index::ordered_unique<boost::multi_index::tag<ByPeer>, ByPeerViewExtractor>,
217
        boost::multi_index::ordered_non_unique<boost::multi_index::tag<ByTxHash>, ByTxHashViewExtractor>,
218
        boost::multi_index::ordered_non_unique<boost::multi_index::tag<ByTime>, ByTimeViewExtractor>
219
    >
220
>;
221
222
/** Helper type to simplify syntax of iterator types. */
223
template<typename Tag>
224
using Iter = typename Index::index<Tag>::type::iterator;
225
226
/** Per-peer statistics object. */
227
struct PeerInfo {
228
    size_t m_total = 0; //!< Total number of announcements for this peer.
229
    size_t m_completed = 0; //!< Number of COMPLETED announcements for this peer.
230
    size_t m_requested = 0; //!< Number of REQUESTED announcements for this peer.
231
};
232
233
/** Per-txhash statistics object. Only used for sanity checking. */
234
struct TxHashInfo
235
{
236
    //! Number of CANDIDATE_DELAYED announcements for this txhash.
237
    size_t m_candidate_delayed = 0;
238
    //! Number of CANDIDATE_READY announcements for this txhash.
239
    size_t m_candidate_ready = 0;
240
    //! Number of CANDIDATE_BEST announcements for this txhash (at most one).
241
    size_t m_candidate_best = 0;
242
    //! Number of REQUESTED announcements for this txhash (at most one; mutually exclusive with CANDIDATE_BEST).
243
    size_t m_requested = 0;
244
    //! The priority of the CANDIDATE_BEST announcement if one exists, or max() otherwise.
245
    Priority m_priority_candidate_best = std::numeric_limits<Priority>::max();
246
    //! The highest priority of all CANDIDATE_READY announcements (or min() if none exist).
247
    Priority m_priority_best_candidate_ready = std::numeric_limits<Priority>::min();
248
    //! All peers we have an announcement for this txhash for.
249
    std::vector<NodeId> m_peers;
250
};
251
252
/** Compare two PeerInfo objects. Only used for sanity checking. */
253
bool operator==(const PeerInfo& a, const PeerInfo& b)
254
4.03M
{
255
4.03M
    return std::tie(a.m_total, a.m_completed, a.m_requested) ==
256
4.03M
           std::tie(b.m_total, b.m_completed, b.m_requested);
257
4.03M
};
258
259
/** (Re)compute the PeerInfo map from the index. Only used for sanity checking. */
260
std::unordered_map<NodeId, PeerInfo> RecomputePeerInfo(const Index& index)
261
41.5k
{
262
41.5k
    std::unordered_map<NodeId, PeerInfo> ret;
263
4.41M
    for (const Announcement& ann : index) {
264
4.41M
        PeerInfo& info = ret[ann.m_peer];
265
4.41M
        ++info.m_total;
266
4.41M
        info.m_requested += (ann.GetState() == State::REQUESTED);
267
4.41M
        info.m_completed += (ann.GetState() == State::COMPLETED);
268
4.41M
    }
269
41.5k
    return ret;
270
41.5k
}
271
272
/** Compute the TxHashInfo map. Only used for sanity checking. */
273
std::map<uint256, TxHashInfo> ComputeTxHashInfo(const Index& index, const PriorityComputer& computer)
274
41.5k
{
275
41.5k
    std::map<uint256, TxHashInfo> ret;
276
4.41M
    for (const Announcement& ann : index) {
277
4.41M
        TxHashInfo& info = ret[ann.m_gtxid.ToUint256()];
278
        // Classify how many announcements of each state we have for this txhash.
279
4.41M
        info.m_candidate_delayed += (ann.GetState() == State::CANDIDATE_DELAYED);
280
4.41M
        info.m_candidate_ready += (ann.GetState() == State::CANDIDATE_READY);
281
4.41M
        info.m_candidate_best += (ann.GetState() == State::CANDIDATE_BEST);
282
4.41M
        info.m_requested += (ann.GetState() == State::REQUESTED);
283
        // And track the priority of the best CANDIDATE_READY/CANDIDATE_BEST announcements.
284
4.41M
        if (ann.GetState() == State::CANDIDATE_BEST) {
285
1.27M
            info.m_priority_candidate_best = computer(ann);
286
1.27M
        }
287
4.41M
        if (ann.GetState() == State::CANDIDATE_READY) {
288
1.06M
            info.m_priority_best_candidate_ready = std::max(info.m_priority_best_candidate_ready, computer(ann));
289
1.06M
        }
290
        // Also keep track of which peers this txhash has an announcement for (so we can detect duplicates).
291
4.41M
        info.m_peers.push_back(ann.m_peer);
292
4.41M
    }
293
41.5k
    return ret;
294
41.5k
}
295
296
}  // namespace
297
298
/** Actual implementation for TxRequestTracker's data structure. */
299
class TxRequestTracker::Impl {
300
    //! The current sequence number. Increases for every announcement. This is used to sort txhashes returned by
301
    //! GetRequestable in announcement order.
302
    SequenceNumber m_current_sequence{0};
303
304
    //! This tracker's priority computer.
305
    const PriorityComputer m_computer;
306
307
    //! This tracker's main data structure. See SanityCheck() for the invariants that apply to it.
308
    Index m_index;
309
310
    //! Map with this tracker's per-peer statistics.
311
    std::unordered_map<NodeId, PeerInfo> m_peerinfo;
312
313
public:
314
    void SanityCheck() const
315
41.5k
    {
316
        // Recompute m_peerdata from m_index. This verifies the data in it as it should just be caching statistics
317
        // on m_index. It also verifies the invariant that no PeerInfo announcements with m_total==0 exist.
318
41.5k
        assert(m_peerinfo == RecomputePeerInfo(m_index));
319
320
        // Calculate per-txhash statistics from m_index, and validate invariants.
321
1.93M
        for (auto& item : ComputeTxHashInfo(m_index, m_computer)) {
322
1.93M
            TxHashInfo& info = item.second;
323
324
            // Cannot have only COMPLETED peer (txhash should have been forgotten already)
325
1.93M
            assert(info.m_candidate_delayed + info.m_candidate_ready + info.m_candidate_best + info.m_requested > 0);
326
327
            // Can have at most 1 CANDIDATE_BEST/REQUESTED peer
328
1.93M
            assert(info.m_candidate_best + info.m_requested <= 1);
329
330
            // If there are any CANDIDATE_READY announcements, there must be exactly one CANDIDATE_BEST or REQUESTED
331
            // announcement.
332
1.93M
            if (info.m_candidate_ready > 0) {
333
587k
                assert(info.m_candidate_best + info.m_requested == 1);
334
587k
            }
335
336
            // If there is both a CANDIDATE_READY and a CANDIDATE_BEST announcement, the CANDIDATE_BEST one must be
337
            // at least as good (equal or higher priority) as the best CANDIDATE_READY.
338
1.93M
            if (info.m_candidate_ready && info.m_candidate_best) {
339
341k
                assert(info.m_priority_candidate_best >= info.m_priority_best_candidate_ready);
340
341k
            }
341
342
            // No txhash can have been announced by the same peer twice.
343
1.93M
            std::sort(info.m_peers.begin(), info.m_peers.end());
344
1.93M
            assert(std::adjacent_find(info.m_peers.begin(), info.m_peers.end()) == info.m_peers.end());
345
1.93M
        }
346
41.5k
    }
347
348
    void PostGetRequestableSanityCheck(std::chrono::microseconds now) const
349
29.3k
    {
350
3.12M
        for (const Announcement& ann : m_index) {
351
3.12M
            if (ann.IsWaiting()) {
352
                // REQUESTED and CANDIDATE_DELAYED must have a time in the future (they should have been converted
353
                // to COMPLETED/CANDIDATE_READY respectively).
354
1.19M
                assert(ann.m_time > now);
355
1.93M
            } else if (ann.IsSelectable()) {
356
                // CANDIDATE_READY and CANDIDATE_BEST cannot have a time in the future (they should have remained
357
                // CANDIDATE_DELAYED, or should have been converted back to it if time went backwards).
358
1.66M
                assert(ann.m_time <= now);
359
1.66M
            }
360
3.12M
        }
361
29.3k
    }
362
363
private:
364
    //! Wrapper around Index::...::erase that keeps m_peerinfo up to date.
365
    template<typename Tag>
366
    Iter<Tag> Erase(Iter<Tag> it)
367
30.8k
    {
368
30.8k
        auto peerit = m_peerinfo.find(it->m_peer);
369
30.8k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
370
30.8k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
371
30.8k
        if (--peerit->second.m_total == 0) m_peerinfo.erase(peerit);
372
30.8k
        return m_index.get<Tag>().erase(it);
373
30.8k
    }
txrequest.cpp:boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator TxRequestTracker::Impl::Erase<(anonymous namespace)::ByTxHash>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator)
Line
Count
Source
367
30.1k
    {
368
30.1k
        auto peerit = m_peerinfo.find(it->m_peer);
369
30.1k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
370
30.1k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
371
30.1k
        if (--peerit->second.m_total == 0) m_peerinfo.erase(peerit);
372
30.1k
        return m_index.get<Tag>().erase(it);
373
30.1k
    }
txrequest.cpp:boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByPeer>::type::iterator TxRequestTracker::Impl::Erase<(anonymous namespace)::ByPeer>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByPeer>::type::iterator)
Line
Count
Source
367
747
    {
368
747
        auto peerit = m_peerinfo.find(it->m_peer);
369
747
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
370
747
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
371
747
        if (--peerit->second.m_total == 0) m_peerinfo.erase(peerit);
372
747
        return m_index.get<Tag>().erase(it);
373
747
    }
374
375
    //! Wrapper around Index::...::modify that keeps m_peerinfo up to date.
376
    template<typename Tag, typename Modifier>
377
    void Modify(Iter<Tag> it, Modifier modifier)
378
93.3k
    {
379
93.3k
        auto peerit = m_peerinfo.find(it->m_peer);
380
93.3k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
93.3k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
93.3k
        m_index.get<Tag>().modify(it, std::move(modifier));
383
93.3k
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
93.3k
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
93.3k
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByTxHash, TxRequestTracker::Impl::ChangeAndReselect(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>, (anonymous namespace)::State)::'lambda'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator, TxRequestTracker::Impl::ChangeAndReselect(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>, (anonymous namespace)::State)::'lambda'((anonymous namespace)::Announcement&))
Line
Count
Source
378
3.61k
    {
379
3.61k
        auto peerit = m_peerinfo.find(it->m_peer);
380
3.61k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
3.61k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
3.61k
        m_index.get<Tag>().modify(it, std::move(modifier));
383
3.61k
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
3.61k
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
3.61k
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByTxHash, TxRequestTracker::Impl::ChangeAndReselect(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>, (anonymous namespace)::State)::'lambda0'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator, TxRequestTracker::Impl::ChangeAndReselect(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>, (anonymous namespace)::State)::'lambda0'((anonymous namespace)::Announcement&))
Line
Count
Source
378
4.74k
    {
379
4.74k
        auto peerit = m_peerinfo.find(it->m_peer);
380
4.74k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
4.74k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
4.74k
        m_index.get<Tag>().modify(it, std::move(modifier));
383
4.74k
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
4.74k
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
4.74k
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByTxHash, TxRequestTracker::Impl::RequestedTx(long, uint256 const&, std::chrono::duration<long, std::ratio<1l, 1000000l>>)::'lambda'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator, TxRequestTracker::Impl::RequestedTx(long, uint256 const&, std::chrono::duration<long, std::ratio<1l, 1000000l>>)::'lambda'((anonymous namespace)::Announcement&))
Line
Count
Source
378
640
    {
379
640
        auto peerit = m_peerinfo.find(it->m_peer);
380
640
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
640
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
640
        m_index.get<Tag>().modify(it, std::move(modifier));
383
640
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
640
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
640
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByTxHash, TxRequestTracker::Impl::RequestedTx(long, uint256 const&, std::chrono::duration<long, std::ratio<1l, 1000000l>>)::'lambda0'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator, TxRequestTracker::Impl::RequestedTx(long, uint256 const&, std::chrono::duration<long, std::ratio<1l, 1000000l>>)::'lambda0'((anonymous namespace)::Announcement&))
Line
Count
Source
378
320
    {
379
320
        auto peerit = m_peerinfo.find(it->m_peer);
380
320
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
320
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
320
        m_index.get<Tag>().modify(it, std::move(modifier));
383
320
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
320
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
320
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByPeer, TxRequestTracker::Impl::RequestedTx(long, uint256 const&, std::chrono::duration<long, std::ratio<1l, 1000000l>>)::'lambda1'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByPeer>::type::iterator, TxRequestTracker::Impl::RequestedTx(long, uint256 const&, std::chrono::duration<long, std::ratio<1l, 1000000l>>)::'lambda1'((anonymous namespace)::Announcement&))
Line
Count
Source
378
24.7k
    {
379
24.7k
        auto peerit = m_peerinfo.find(it->m_peer);
380
24.7k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
24.7k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
24.7k
        m_index.get<Tag>().modify(it, std::move(modifier));
383
24.7k
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
24.7k
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
24.7k
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByTxHash, TxRequestTracker::Impl::PromoteCandidateReady(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>)::'lambda'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator, TxRequestTracker::Impl::PromoteCandidateReady(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>)::'lambda'((anonymous namespace)::Announcement&))
Line
Count
Source
378
30.3k
    {
379
30.3k
        auto peerit = m_peerinfo.find(it->m_peer);
380
30.3k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
30.3k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
30.3k
        m_index.get<Tag>().modify(it, std::move(modifier));
383
30.3k
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
30.3k
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
30.3k
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByTxHash, TxRequestTracker::Impl::PromoteCandidateReady(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>)::'lambda0'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator, TxRequestTracker::Impl::PromoteCandidateReady(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>)::'lambda0'((anonymous namespace)::Announcement&))
Line
Count
Source
378
25.9k
    {
379
25.9k
        auto peerit = m_peerinfo.find(it->m_peer);
380
25.9k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
25.9k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
25.9k
        m_index.get<Tag>().modify(it, std::move(modifier));
383
25.9k
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
25.9k
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
25.9k
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByTxHash, TxRequestTracker::Impl::PromoteCandidateReady(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>)::'lambda1'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator, TxRequestTracker::Impl::PromoteCandidateReady(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>)::'lambda1'((anonymous namespace)::Announcement&))
Line
Count
Source
378
1.45k
    {
379
1.45k
        auto peerit = m_peerinfo.find(it->m_peer);
380
1.45k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
1.45k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
1.45k
        m_index.get<Tag>().modify(it, std::move(modifier));
383
1.45k
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
1.45k
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
1.45k
    }
txrequest.cpp:void TxRequestTracker::Impl::Modify<(anonymous namespace)::ByTxHash, TxRequestTracker::Impl::PromoteCandidateReady(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>)::'lambda2'((anonymous namespace)::Announcement&)>(boost::multi_index::multi_index_container<(anonymous namespace)::Announcement, boost::multi_index::indexed_by<boost::multi_index::ordered_unique<boost::multi_index::tag<(anonymous namespace)::ByPeer, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByPeerViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTxHash, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTxHashViewExtractor, mpl_::na>, boost::multi_index::ordered_non_unique<boost::multi_index::tag<(anonymous namespace)::ByTime, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, (anonymous namespace)::ByTimeViewExtractor, mpl_::na>, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na, mpl_::na>, std::allocator<(anonymous namespace)::Announcement>>::index<(anonymous namespace)::ByTxHash>::type::iterator, TxRequestTracker::Impl::PromoteCandidateReady(boost::multi_index::detail::bidir_node_iterator<boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::ordered_index_node<boost::multi_index::detail::null_augment_policy, boost::multi_index::detail::index_node_base<(anonymous namespace)::Announcement, std::allocator<(anonymous namespace)::Announcement>>>>>)::'lambda2'((anonymous namespace)::Announcement&))
Line
Count
Source
378
1.45k
    {
379
1.45k
        auto peerit = m_peerinfo.find(it->m_peer);
380
1.45k
        peerit->second.m_completed -= it->GetState() == State::COMPLETED;
381
1.45k
        peerit->second.m_requested -= it->GetState() == State::REQUESTED;
382
1.45k
        m_index.get<Tag>().modify(it, std::move(modifier));
383
1.45k
        peerit->second.m_completed += it->GetState() == State::COMPLETED;
384
1.45k
        peerit->second.m_requested += it->GetState() == State::REQUESTED;
385
1.45k
    }
386
387
    //! Convert a CANDIDATE_DELAYED announcement into a CANDIDATE_READY. If this makes it the new best
388
    //! CANDIDATE_READY (and no REQUESTED exists) and better than the CANDIDATE_BEST (if any), it becomes the new
389
    //! CANDIDATE_BEST.
390
    void PromoteCandidateReady(Iter<ByTxHash> it)
391
30.3k
    {
392
30.3k
        assert(it != m_index.get<ByTxHash>().end());
393
30.3k
        assert(it->GetState() == State::CANDIDATE_DELAYED);
394
        // Convert CANDIDATE_DELAYED to CANDIDATE_READY first.
395
30.3k
        Modify<ByTxHash>(it, [](Announcement& ann){ ann.SetState(State::CANDIDATE_READY); });
396
        // The following code relies on the fact that the ByTxHash is sorted by txhash, and then by state (first
397
        // _DELAYED, then _READY, then _BEST/REQUESTED). Within the _READY announcements, the best one (highest
398
        // priority) comes last. Thus, if an existing _BEST exists for the same txhash that this announcement may
399
        // be preferred over, it must immediately follow the newly created _READY.
400
30.3k
        auto it_next = std::next(it);
401
30.3k
        if (it_next == m_index.get<ByTxHash>().end() || it_next->m_gtxid.ToUint256() != it->m_gtxid.ToUint256() ||
402
30.3k
            it_next->GetState() == State::COMPLETED) {
403
            // This is the new best CANDIDATE_READY, and there is no IsSelected() announcement for this txhash
404
            // already.
405
25.9k
            Modify<ByTxHash>(it, [](Announcement& ann){ ann.SetState(State::CANDIDATE_BEST); });
406
25.9k
        } else if (it_next->GetState() == State::CANDIDATE_BEST) {
407
2.09k
            Priority priority_old = m_computer(*it_next);
408
2.09k
            Priority priority_new = m_computer(*it);
409
2.09k
            if (priority_new > priority_old) {
410
                // There is a CANDIDATE_BEST announcement already, but this one is better.
411
1.45k
                Modify<ByTxHash>(it_next, [](Announcement& ann){ ann.SetState(State::CANDIDATE_READY); });
412
1.45k
                Modify<ByTxHash>(it, [](Announcement& ann){ ann.SetState(State::CANDIDATE_BEST); });
413
1.45k
            }
414
2.09k
        }
415
30.3k
    }
416
417
    //! Change the state of an announcement to something non-IsSelected(). If it was IsSelected(), the next best
418
    //! announcement will be marked CANDIDATE_BEST.
419
    void ChangeAndReselect(Iter<ByTxHash> it, State new_state)
420
4.74k
    {
421
4.74k
        assert(new_state == State::COMPLETED || new_state == State::CANDIDATE_DELAYED);
422
4.74k
        assert(it != m_index.get<ByTxHash>().end());
423
4.74k
        if (it->IsSelected() && it != m_index.get<ByTxHash>().begin()) {
424
3.92k
            auto it_prev = std::prev(it);
425
            // The next best CANDIDATE_READY, if any, immediately precedes the REQUESTED or CANDIDATE_BEST
426
            // announcement in the ByTxHash index.
427
3.92k
            if (it_prev->m_gtxid.ToUint256() == it->m_gtxid.ToUint256() && it_prev->GetState() == State::CANDIDATE_READY) {
428
                // If one such CANDIDATE_READY exists (for this txhash), convert it to CANDIDATE_BEST.
429
3.61k
                Modify<ByTxHash>(it_prev, [](Announcement& ann){ ann.SetState(State::CANDIDATE_BEST); });
430
3.61k
            }
431
3.92k
        }
432
4.74k
        Modify<ByTxHash>(it, [new_state](Announcement& ann){ ann.SetState(new_state); });
433
4.74k
    }
434
435
    //! Check if 'it' is the only announcement for a given txhash that isn't COMPLETED.
436
    bool IsOnlyNonCompleted(Iter<ByTxHash> it)
437
28.0k
    {
438
28.0k
        assert(it != m_index.get<ByTxHash>().end());
439
28.0k
        assert(it->GetState() != State::COMPLETED); // Not allowed to call this on COMPLETED announcements.
440
441
        // This announcement has a predecessor that belongs to the same txhash. Due to ordering, and the
442
        // fact that 'it' is not COMPLETED, its predecessor cannot be COMPLETED here.
443
28.0k
        if (it != m_index.get<ByTxHash>().begin() && std::prev(it)->m_gtxid.ToUint256() == it->m_gtxid.ToUint256()) return false;
444
445
        // This announcement has a successor that belongs to the same txhash, and is not COMPLETED.
446
23.9k
        if (std::next(it) != m_index.get<ByTxHash>().end() && std::next(it)->m_gtxid.ToUint256() == it->m_gtxid.ToUint256() &&
447
23.9k
            std::next(it)->GetState() != State::COMPLETED) return false;
448
449
23.6k
        return true;
450
23.9k
    }
451
452
    /** Convert any announcement to a COMPLETED one. If there are no non-COMPLETED announcements left for this
453
     *  txhash, they are deleted. If this was a REQUESTED announcement, and there are other CANDIDATEs left, the
454
     *  best one is made CANDIDATE_BEST. Returns whether the announcement still exists. */
455
    bool MakeCompleted(Iter<ByTxHash> it)
456
28.0k
    {
457
28.0k
        assert(it != m_index.get<ByTxHash>().end());
458
459
        // Nothing to be done if it's already COMPLETED.
460
28.0k
        if (it->GetState() == State::COMPLETED) return true;
461
462
28.0k
        if (IsOnlyNonCompleted(it)) {
463
            // This is the last non-COMPLETED announcement for this txhash. Delete all.
464
23.6k
            uint256 txhash = it->m_gtxid.ToUint256();
465
25.2k
            do {
466
25.2k
                it = Erase<ByTxHash>(it);
467
25.2k
            } while (it != m_index.get<ByTxHash>().end() && it->m_gtxid.ToUint256() == txhash);
468
23.6k
            return false;
469
23.6k
        }
470
471
        // Mark the announcement COMPLETED, and select the next best announcement (the first CANDIDATE_READY) if
472
        // needed.
473
4.42k
        ChangeAndReselect(it, State::COMPLETED);
474
475
4.42k
        return true;
476
28.0k
    }
477
478
    //! Make the data structure consistent with a given point in time:
479
    //! - REQUESTED announcements with expiry <= now are turned into COMPLETED.
480
    //! - CANDIDATE_DELAYED announcements with reqtime <= now are turned into CANDIDATE_{READY,BEST}.
481
    //! - CANDIDATE_{READY,BEST} announcements with reqtime > now are turned into CANDIDATE_DELAYED.
482
    void SetTimePoint(std::chrono::microseconds now, std::vector<std::pair<NodeId, GenTxid>>* expired)
483
402k
    {
484
402k
        if (expired) expired->clear();
485
486
        // Iterate over all CANDIDATE_DELAYED and REQUESTED from old to new, as long as they're in the past,
487
        // and convert them to CANDIDATE_READY and COMPLETED respectively.
488
434k
        while (!m_index.empty()) {
489
133k
            auto it = m_index.get<ByTime>().begin();
490
133k
            if (it->GetState() == State::CANDIDATE_DELAYED && it->m_time <= now) {
491
30.3k
                PromoteCandidateReady(m_index.project<ByTxHash>(it));
492
102k
            } else if (it->GetState() == State::REQUESTED && it->m_time <= now) {
493
1.06k
                if (expired) expired->emplace_back(it->m_peer, it->m_gtxid);
494
1.06k
                MakeCompleted(m_index.project<ByTxHash>(it));
495
101k
            } else {
496
101k
                break;
497
101k
            }
498
133k
        }
499
500
403k
        while (!m_index.empty()) {
501
            // If time went backwards, we may need to demote CANDIDATE_BEST and CANDIDATE_READY announcements back
502
            // to CANDIDATE_DELAYED. This is an unusual edge case, and unlikely to matter in production. However,
503
            // it makes it much easier to specify and test TxRequestTracker::Impl's behaviour.
504
101k
            auto it = std::prev(m_index.get<ByTime>().end());
505
101k
            if (it->IsSelectable() && it->m_time > now) {
506
320
                ChangeAndReselect(m_index.project<ByTxHash>(it), State::CANDIDATE_DELAYED);
507
101k
            } else {
508
101k
                break;
509
101k
            }
510
101k
        }
511
402k
    }
512
513
public:
514
    explicit Impl(bool deterministic) :
515
1.23k
        m_computer(deterministic),
516
        // Explicitly initialize m_index as we need to pass a reference to m_computer to ByTxHashViewExtractor.
517
1.23k
        m_index(boost::make_tuple(
518
1.23k
            boost::make_tuple(ByPeerViewExtractor(), std::less<ByPeerView>()),
519
1.23k
            boost::make_tuple(ByTxHashViewExtractor(m_computer), std::less<ByTxHashView>()),
520
1.23k
            boost::make_tuple(ByTimeViewExtractor(), std::less<ByTimeView>())
521
1.23k
        )) {}
522
523
    // Disable copying and assigning (a default copy won't work due the stateful ByTxHashViewExtractor).
524
    Impl(const Impl&) = delete;
525
    Impl& operator=(const Impl&) = delete;
526
527
    void DisconnectedPeer(NodeId peer)
528
3.85k
    {
529
3.85k
        auto& index = m_index.get<ByPeer>();
530
3.85k
        auto it = index.lower_bound(ByPeerView{peer, false, uint256::ZERO});
531
17.3k
        while (it != index.end() && it->m_peer == peer) {
532
            // Check what to continue with after this iteration. 'it' will be deleted in what follows, so we need to
533
            // decide what to continue with afterwards. There are a number of cases to consider:
534
            // - std::next(it) is end() or belongs to a different peer. In that case, this is the last iteration
535
            //   of the loop (denote this by setting it_next to end()).
536
            // - 'it' is not the only non-COMPLETED announcement for its txhash. This means it will be deleted, but
537
            //   no other Announcement objects will be modified. Continue with std::next(it) if it belongs to the
538
            //   same peer, but decide this ahead of time (as 'it' may change position in what follows).
539
            // - 'it' is the only non-COMPLETED announcement for its txhash. This means it will be deleted along
540
            //   with all other announcements for the same txhash - which may include std::next(it). However, other
541
            //   than 'it', no announcements for the same peer can be affected (due to (peer, txhash) uniqueness).
542
            //   In other words, the situation where std::next(it) is deleted can only occur if std::next(it)
543
            //   belongs to a different peer but the same txhash as 'it'. This is covered by the first bulletpoint
544
            //   already, and we'll have set it_next to end().
545
13.4k
            auto it_next = (std::next(it) == index.end() || std::next(it)->m_peer != peer) ? index.end() :
546
13.4k
                std::next(it);
547
            // If the announcement isn't already COMPLETED, first make it COMPLETED (which will mark other
548
            // CANDIDATEs as CANDIDATE_BEST, or delete all of a txhash's announcements if no non-COMPLETED ones are
549
            // left).
550
13.4k
            if (MakeCompleted(m_index.project<ByTxHash>(it))) {
551
                // Then actually delete the announcement (unless it was already deleted by MakeCompleted).
552
747
                Erase<ByPeer>(it);
553
747
            }
554
13.4k
            it = it_next;
555
13.4k
        }
556
3.85k
    }
557
558
    void ForgetTxHash(const uint256& txhash)
559
283k
    {
560
283k
        auto it = m_index.get<ByTxHash>().lower_bound(ByTxHashView{txhash, State::CANDIDATE_DELAYED, 0});
561
288k
        while (it != m_index.get<ByTxHash>().end() && it->m_gtxid.ToUint256() == txhash) {
562
4.85k
            it = Erase<ByTxHash>(it);
563
4.85k
        }
564
283k
    }
565
566
    void GetCandidatePeers(const uint256& txhash, std::vector<NodeId>& result_peers) const
567
1.05k
    {
568
1.05k
        auto it = m_index.get<ByTxHash>().lower_bound(ByTxHashView{txhash, State::CANDIDATE_DELAYED, 0});
569
1.06k
        while (it != m_index.get<ByTxHash>().end() && it->m_gtxid.ToUint256() == txhash && it->GetState() != State::COMPLETED) {
570
3
            result_peers.push_back(it->m_peer);
571
3
            ++it;
572
3
        }
573
1.05k
    }
574
575
    void ReceivedInv(NodeId peer, const GenTxid& gtxid, bool preferred,
576
                     std::chrono::microseconds reqtime)
577
30.9k
    {
578
        // Bail out if we already have a CANDIDATE_BEST announcement for this (txhash, peer) combination. The case
579
        // where there is a non-CANDIDATE_BEST announcement already will be caught by the uniqueness property of the
580
        // ByPeer index when we try to emplace the new object below.
581
30.9k
        if (m_index.get<ByPeer>().count(ByPeerView{peer, true, gtxid.ToUint256()})) return;
582
583
        // Try creating the announcement with CANDIDATE_DELAYED state (which will fail due to the uniqueness
584
        // of the ByPeer index if a non-CANDIDATE_BEST announcement already exists with the same txhash and peer).
585
        // Bail out in that case.
586
30.9k
        auto ret = m_index.get<ByPeer>().emplace(gtxid, peer, preferred, reqtime, m_current_sequence);
587
30.9k
        if (!ret.second) return;
588
589
        // Update accounting metadata.
590
30.9k
        ++m_peerinfo[peer].m_total;
591
30.9k
        ++m_current_sequence;
592
30.9k
    }
593
594
    //! Find the GenTxids to request now from peer.
595
    std::vector<GenTxid> GetRequestable(NodeId peer, std::chrono::microseconds now,
596
                                        std::vector<std::pair<NodeId, GenTxid>>* expired)
597
402k
    {
598
        // Move time.
599
402k
        SetTimePoint(now, expired);
600
601
        // Find all CANDIDATE_BEST announcements for this peer.
602
402k
        std::vector<const Announcement*> selected;
603
402k
        auto it_peer = m_index.get<ByPeer>().lower_bound(ByPeerView{peer, true, uint256::ZERO});
604
438k
        while (it_peer != m_index.get<ByPeer>().end() && it_peer->m_peer == peer &&
605
438k
            it_peer->GetState() == State::CANDIDATE_BEST) {
606
35.9k
            selected.emplace_back(&*it_peer);
607
35.9k
            ++it_peer;
608
35.9k
        }
609
610
        // Sort by sequence number.
611
402k
        std::sort(selected.begin(), selected.end(), [](const Announcement* a, const Announcement* b) {
612
179k
            return a->m_sequence < b->m_sequence;
613
179k
        });
614
615
        // Convert to GenTxid and return.
616
402k
        std::vector<GenTxid> ret;
617
402k
        ret.reserve(selected.size());
618
402k
        std::transform(selected.begin(), selected.end(), std::back_inserter(ret), [](const Announcement* ann) {
619
35.9k
            return ann->m_gtxid;
620
35.9k
        });
621
402k
        return ret;
622
402k
    }
623
624
    void RequestedTx(NodeId peer, const uint256& txhash, std::chrono::microseconds expiry)
625
25.7k
    {
626
25.7k
        auto it = m_index.get<ByPeer>().find(ByPeerView{peer, true, txhash});
627
25.7k
        if (it == m_index.get<ByPeer>().end()) {
628
            // There is no CANDIDATE_BEST announcement, look for a _READY or _DELAYED instead. If the caller only
629
            // ever invokes RequestedTx with the values returned by GetRequestable, and no other non-const functions
630
            // other than ForgetTxHash and GetRequestable in between, this branch will never execute (as txhashes
631
            // returned by GetRequestable always correspond to CANDIDATE_BEST announcements).
632
633
1.92k
            it = m_index.get<ByPeer>().find(ByPeerView{peer, false, txhash});
634
1.92k
            if (it == m_index.get<ByPeer>().end() || (it->GetState() != State::CANDIDATE_DELAYED &&
635
1.60k
                                                      it->GetState() != State::CANDIDATE_READY)) {
636
                // There is no CANDIDATE announcement tracked for this peer, so we have nothing to do. Either this
637
                // txhash wasn't tracked at all (and the caller should have called ReceivedInv), or it was already
638
                // requested and/or completed for other reasons and this is just a superfluous RequestedTx call.
639
960
                return;
640
960
            }
641
642
            // Look for an existing CANDIDATE_BEST or REQUESTED with the same txhash. We only need to do this if the
643
            // found announcement had a different state than CANDIDATE_BEST. If it did, invariants guarantee that no
644
            // other CANDIDATE_BEST or REQUESTED can exist.
645
960
            auto it_old = m_index.get<ByTxHash>().lower_bound(ByTxHashView{txhash, State::CANDIDATE_BEST, 0});
646
960
            if (it_old != m_index.get<ByTxHash>().end() && it_old->m_gtxid.ToUint256() == txhash) {
647
960
                if (it_old->GetState() == State::CANDIDATE_BEST) {
648
                    // The data structure's invariants require that there can be at most one CANDIDATE_BEST or one
649
                    // REQUESTED announcement per txhash (but not both simultaneously), so we have to convert any
650
                    // existing CANDIDATE_BEST to another CANDIDATE_* when constructing another REQUESTED.
651
                    // It doesn't matter whether we pick CANDIDATE_READY or _DELAYED here, as SetTimePoint()
652
                    // will correct it at GetRequestable() time. If time only goes forward, it will always be
653
                    // _READY, so pick that to avoid extra work in SetTimePoint().
654
640
                    Modify<ByTxHash>(it_old, [](Announcement& ann) { ann.SetState(State::CANDIDATE_READY); });
655
640
                } else if (it_old->GetState() == State::REQUESTED) {
656
                    // As we're no longer waiting for a response to the previous REQUESTED announcement, convert it
657
                    // to COMPLETED. This also helps guaranteeing progress.
658
320
                    Modify<ByTxHash>(it_old, [](Announcement& ann) { ann.SetState(State::COMPLETED); });
659
320
                }
660
960
            }
661
960
        }
662
663
24.7k
        Modify<ByPeer>(it, [expiry](Announcement& ann) {
664
24.7k
            ann.SetState(State::REQUESTED);
665
24.7k
            ann.m_time = expiry;
666
24.7k
        });
667
24.7k
    }
668
669
    void ReceivedResponse(NodeId peer, const uint256& txhash)
670
32.7k
    {
671
        // We need to search the ByPeer index for both (peer, false, txhash) and (peer, true, txhash).
672
32.7k
        auto it = m_index.get<ByPeer>().find(ByPeerView{peer, false, txhash});
673
32.7k
        if (it == m_index.get<ByPeer>().end()) {
674
19.7k
            it = m_index.get<ByPeer>().find(ByPeerView{peer, true, txhash});
675
19.7k
        }
676
32.7k
        if (it != m_index.get<ByPeer>().end()) MakeCompleted(m_index.project<ByTxHash>(it));
677
32.7k
    }
678
679
    size_t CountInFlight(NodeId peer) const
680
49.7k
    {
681
49.7k
        auto it = m_peerinfo.find(peer);
682
49.7k
        if (it != m_peerinfo.end()) return it->second.m_requested;
683
9.35k
        return 0;
684
49.7k
    }
685
686
    size_t CountCandidates(NodeId peer) const
687
29.3k
    {
688
29.3k
        auto it = m_peerinfo.find(peer);
689
29.3k
        if (it != m_peerinfo.end()) return it->second.m_total - it->second.m_requested - it->second.m_completed;
690
5.67k
        return 0;
691
29.3k
    }
692
693
    size_t Count(NodeId peer) const
694
51.3k
    {
695
51.3k
        auto it = m_peerinfo.find(peer);
696
51.3k
        if (it != m_peerinfo.end()) return it->second.m_total;
697
10.9k
        return 0;
698
51.3k
    }
699
700
    //! Count how many announcements are being tracked in total across all peers and transactions.
701
845
    size_t Size() const { return m_index.size(); }
702
703
    uint64_t ComputePriority(const uint256& txhash, NodeId peer, bool preferred) const
704
2.23M
    {
705
        // Return Priority as a uint64_t as Priority is internal.
706
2.23M
        return uint64_t{m_computer(txhash, peer, preferred)};
707
2.23M
    }
708
709
};
710
711
TxRequestTracker::TxRequestTracker(bool deterministic) :
712
1.23k
    m_impl{std::make_unique<TxRequestTracker::Impl>(deterministic)} {}
713
714
1.23k
TxRequestTracker::~TxRequestTracker() = default;
715
716
283k
void TxRequestTracker::ForgetTxHash(const uint256& txhash) { m_impl->ForgetTxHash(txhash); }
717
3.85k
void TxRequestTracker::DisconnectedPeer(NodeId peer) { m_impl->DisconnectedPeer(peer); }
718
49.7k
size_t TxRequestTracker::CountInFlight(NodeId peer) const { return m_impl->CountInFlight(peer); }
719
29.3k
size_t TxRequestTracker::CountCandidates(NodeId peer) const { return m_impl->CountCandidates(peer); }
720
51.3k
size_t TxRequestTracker::Count(NodeId peer) const { return m_impl->Count(peer); }
721
845
size_t TxRequestTracker::Size() const { return m_impl->Size(); }
722
1.05k
void TxRequestTracker::GetCandidatePeers(const uint256& txhash, std::vector<NodeId>& result_peers) const { return m_impl->GetCandidatePeers(txhash, result_peers); }
723
41.5k
void TxRequestTracker::SanityCheck() const { m_impl->SanityCheck(); }
724
725
void TxRequestTracker::PostGetRequestableSanityCheck(std::chrono::microseconds now) const
726
29.3k
{
727
29.3k
    m_impl->PostGetRequestableSanityCheck(now);
728
29.3k
}
729
730
void TxRequestTracker::ReceivedInv(NodeId peer, const GenTxid& gtxid, bool preferred,
731
                                   std::chrono::microseconds reqtime)
732
30.9k
{
733
30.9k
    m_impl->ReceivedInv(peer, gtxid, preferred, reqtime);
734
30.9k
}
735
736
void TxRequestTracker::RequestedTx(NodeId peer, const uint256& txhash, std::chrono::microseconds expiry)
737
25.7k
{
738
25.7k
    m_impl->RequestedTx(peer, txhash, expiry);
739
25.7k
}
740
741
void TxRequestTracker::ReceivedResponse(NodeId peer, const uint256& txhash)
742
32.7k
{
743
32.7k
    m_impl->ReceivedResponse(peer, txhash);
744
32.7k
}
745
746
std::vector<GenTxid> TxRequestTracker::GetRequestable(NodeId peer, std::chrono::microseconds now,
747
                                                      std::vector<std::pair<NodeId, GenTxid>>* expired)
748
402k
{
749
402k
    return m_impl->GetRequestable(peer, now, expired);
750
402k
}
751
752
uint64_t TxRequestTracker::ComputePriority(const uint256& txhash, NodeId peer, bool preferred) const
753
2.23M
{
754
2.23M
    return m_impl->ComputePriority(txhash, peer, preferred);
755
2.23M
}