/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 | } |