@@ -57,9 +57,20 @@ class DepGraph
57
57
/* * Data for each transaction, in the same order as the Cluster it was constructed from. */
58
58
std::vector<Entry> entries;
59
59
60
+ /* * Which positions are used. */
61
+ SetType m_used;
62
+
60
63
public:
61
64
/* * Equality operator (primarily for testing purposes). */
62
- friend bool operator ==(const DepGraph&, const DepGraph&) noexcept = default ;
65
+ friend bool operator ==(const DepGraph& a, const DepGraph& b) noexcept
66
+ {
67
+ if (a.m_used != b.m_used ) return false ;
68
+ // Only compare the used positions within the entries vector.
69
+ for (auto idx : a.m_used ) {
70
+ if (a.entries [idx] != b.entries [idx]) return false ;
71
+ }
72
+ return true ;
73
+ }
63
74
64
75
// Default constructors.
65
76
DepGraph () noexcept = default ;
@@ -80,6 +91,7 @@ class DepGraph
80
91
entries[i].ancestors = SetType::Singleton (i);
81
92
entries[i].descendants = SetType::Singleton (i);
82
93
}
94
+ m_used = SetType::Fill (ntx);
83
95
}
84
96
85
97
/* * Construct a DepGraph object given a cluster.
@@ -97,24 +109,50 @@ class DepGraph
97
109
}
98
110
99
111
/* * Construct a DepGraph object given another DepGraph and a mapping from old to new.
112
+ *
113
+ * @param depgraph The original DepGraph that is being remapped.
114
+ *
115
+ * @param mapping A Span such that mapping[i] gives the position in the new DepGraph
116
+ * for position i in the old depgraph. Its size must be equal to
117
+ * depgraph.PositionRange(). The value of mapping[i] is ignored if
118
+ * position i is a hole in depgraph (i.e., if !depgraph.Positions()[i]).
119
+ *
120
+ * @param pos_range The PositionRange() for the new DepGraph. It must equal the largest
121
+ * value in mapping for any used position in depgraph plus 1, or 0 if
122
+ * depgraph.TxCount() == 0.
100
123
*
101
124
* Complexity: O(N^2) where N=depgraph.TxCount().
102
125
*/
103
- DepGraph (const DepGraph<SetType>& depgraph, Span<const ClusterIndex> mapping) noexcept : DepGraph(depgraph.TxCount() )
126
+ DepGraph (const DepGraph<SetType>& depgraph, Span<const ClusterIndex> mapping, ClusterIndex pos_range ) noexcept : entries(pos_range )
104
127
{
105
- Assert (mapping.size () == depgraph.TxCount ());
106
- for (ClusterIndex i = 0 ; i < depgraph.TxCount (); ++i) {
128
+ Assume (mapping.size () == depgraph.PositionRange ());
129
+ Assume ((pos_range == 0 ) == (depgraph.TxCount () == 0 ));
130
+ for (ClusterIndex i : depgraph.Positions ()) {
131
+ auto new_idx = mapping[i];
132
+ Assume (new_idx < pos_range);
133
+ // Add transaction.
134
+ entries[new_idx].ancestors = SetType::Singleton (new_idx);
135
+ entries[new_idx].descendants = SetType::Singleton (new_idx);
136
+ m_used.Set (new_idx);
107
137
// Fill in fee and size.
108
- entries[mapping[i]].feerate = depgraph.entries [i].feerate ;
138
+ entries[new_idx].feerate = depgraph.entries [i].feerate ;
139
+ }
140
+ for (ClusterIndex i : depgraph.Positions ()) {
109
141
// Fill in dependencies by mapping direct parents.
110
142
SetType parents;
111
143
for (auto j : depgraph.GetReducedParents (i)) parents.Set (mapping[j]);
112
144
AddDependencies (parents, mapping[i]);
113
145
}
146
+ // Verify that the provided pos_range was correct (no unused positions at the end).
147
+ Assume (m_used.None () ? (pos_range == 0 ) : (pos_range == m_used.Last () + 1 ));
114
148
}
115
149
150
+ /* * Get the set of transactions positions in use. Complexity: O(1). */
151
+ const SetType& Positions () const noexcept { return m_used; }
152
+ /* * Get the range of positions in this DepGraph. All entries in Positions() are in [0, PositionRange() - 1]. */
153
+ ClusterIndex PositionRange () const noexcept { return entries.size (); }
116
154
/* * Get the number of transactions in the graph. Complexity: O(1). */
117
- auto TxCount () const noexcept { return entries. size (); }
155
+ auto TxCount () const noexcept { return m_used. Count (); }
118
156
/* * Get the feerate of a given transaction i. Complexity: O(1). */
119
157
const FeeFrac& FeeRate (ClusterIndex i) const noexcept { return entries[i].feerate ; }
120
158
/* * Get the mutable feerate of a given transaction i. Complexity: O(1). */
@@ -124,25 +162,59 @@ class DepGraph
124
162
/* * Get the descendants of a given transaction i. Complexity: O(1). */
125
163
const SetType& Descendants (ClusterIndex i) const noexcept { return entries[i].descendants ; }
126
164
127
- /* * Add a new unconnected transaction to this transaction graph (at the end), and return its
128
- * ClusterIndex.
165
+ /* * Add a new unconnected transaction to this transaction graph (in the first available
166
+ * position), and return its ClusterIndex.
129
167
*
130
168
* Complexity: O(1) (amortized, due to resizing of backing vector).
131
169
*/
132
170
ClusterIndex AddTransaction (const FeeFrac& feefrac) noexcept
133
171
{
134
- Assume (TxCount () < SetType::Size ());
135
- ClusterIndex new_idx = TxCount ();
136
- entries.emplace_back (feefrac, SetType::Singleton (new_idx), SetType::Singleton (new_idx));
172
+ static constexpr auto ALL_POSITIONS = SetType::Fill (SetType::Size ());
173
+ auto available = ALL_POSITIONS - m_used;
174
+ Assume (available.Any ());
175
+ ClusterIndex new_idx = available.First ();
176
+ if (new_idx == entries.size ()) {
177
+ entries.emplace_back (feefrac, SetType::Singleton (new_idx), SetType::Singleton (new_idx));
178
+ } else {
179
+ entries[new_idx] = Entry (feefrac, SetType::Singleton (new_idx), SetType::Singleton (new_idx));
180
+ }
181
+ m_used.Set (new_idx);
137
182
return new_idx;
138
183
}
139
184
185
+ /* * Remove the specified positions from this DepGraph.
186
+ *
187
+ * The specified positions will no longer be part of Positions(), and dependencies with them are
188
+ * removed. Note that due to DepGraph only tracking ancestors/descendants (and not direct
189
+ * dependencies), if a parent is removed while a grandparent remains, the grandparent will
190
+ * remain an ancestor.
191
+ *
192
+ * Complexity: O(N) where N=TxCount().
193
+ */
194
+ void RemoveTransactions (const SetType& del) noexcept
195
+ {
196
+ m_used -= del;
197
+ // Remove now-unused trailing entries.
198
+ while (!entries.empty () && !m_used[entries.size () - 1 ]) {
199
+ entries.pop_back ();
200
+ }
201
+ // Remove the deleted transactions from ancestors/descendants of other transactions. Note
202
+ // that the deleted positions will retain old feerate and dependency information. This does
203
+ // not matter as they will be overwritten by AddTransaction if they get used again.
204
+ for (auto & entry : entries) {
205
+ entry.ancestors &= m_used;
206
+ entry.descendants &= m_used;
207
+ }
208
+ }
209
+
140
210
/* * Modify this transaction graph, adding multiple parents to a specified child.
141
211
*
142
212
* Complexity: O(N) where N=TxCount().
143
213
*/
144
214
void AddDependencies (const SetType& parents, ClusterIndex child) noexcept
145
215
{
216
+ Assume (m_used[child]);
217
+ Assume (parents.IsSubsetOf (m_used));
146
218
// Compute the ancestors of parents that are not already ancestors of child.
147
219
SetType par_anc;
148
220
for (auto par : parents - Ancestors (child)) {
@@ -257,7 +329,7 @@ class DepGraph
257
329
*
258
330
* Complexity: O(TxCount()).
259
331
*/
260
- bool IsConnected () const noexcept { return IsConnected (SetType::Fill ( TxCount ()) ); }
332
+ bool IsConnected () const noexcept { return IsConnected (m_used ); }
261
333
262
334
/* * Append the entries of select to list in a topologically valid order.
263
335
*
@@ -507,11 +579,11 @@ class AncestorCandidateFinder
507
579
*/
508
580
AncestorCandidateFinder (const DepGraph<SetType>& depgraph LIFETIMEBOUND) noexcept :
509
581
m_depgraph (depgraph),
510
- m_todo{SetType::Fill ( depgraph.TxCount () )},
511
- m_ancestor_set_feerates (depgraph.TxCount ())
582
+ m_todo{depgraph.Positions ( )},
583
+ m_ancestor_set_feerates (depgraph.PositionRange ())
512
584
{
513
585
// Precompute ancestor-set feerates.
514
- for (ClusterIndex i = 0 ; i < depgraph. TxCount (); ++i ) {
586
+ for (ClusterIndex i : m_depgraph. Positions () ) {
515
587
/* * The remaining ancestors for transaction i. */
516
588
SetType anc_to_add = m_depgraph.Ancestors (i);
517
589
FeeFrac anc_feerate;
@@ -634,22 +706,26 @@ class SearchCandidateFinder
634
706
SearchCandidateFinder (const DepGraph<SetType>& depgraph, uint64_t rng_seed) noexcept :
635
707
m_rng (rng_seed),
636
708
m_sorted_to_original (depgraph.TxCount()),
637
- m_original_to_sorted (depgraph.TxCount()),
638
- m_todo (SetType::Fill(depgraph.TxCount()))
709
+ m_original_to_sorted (depgraph.PositionRange())
639
710
{
640
- // Determine reordering mapping, by sorting by decreasing feerate.
641
- std::iota (m_sorted_to_original.begin (), m_sorted_to_original.end (), ClusterIndex{0 });
711
+ // Determine reordering mapping, by sorting by decreasing feerate. Unusued positions are
712
+ // not included, as they will never be looked up anyway.
713
+ ClusterIndex sorted_pos{0 };
714
+ for (auto i : depgraph.Positions ()) {
715
+ m_sorted_to_original[sorted_pos++] = i;
716
+ }
642
717
std::sort (m_sorted_to_original.begin (), m_sorted_to_original.end (), [&](auto a, auto b) {
643
718
auto feerate_cmp = depgraph.FeeRate (a) <=> depgraph.FeeRate (b);
644
719
if (feerate_cmp == 0 ) return a < b;
645
720
return feerate_cmp > 0 ;
646
721
});
647
722
// Compute reverse mapping.
648
- for (ClusterIndex i = 0 ; i < depgraph. TxCount (); ++i) {
723
+ for (ClusterIndex i = 0 ; i < m_sorted_to_original. size (); ++i) {
649
724
m_original_to_sorted[m_sorted_to_original[i]] = i;
650
725
}
651
726
// Compute reordered dependency graph.
652
- m_sorted_depgraph = DepGraph (depgraph, m_original_to_sorted);
727
+ m_sorted_depgraph = DepGraph (depgraph, m_original_to_sorted, m_sorted_to_original.size ());
728
+ m_todo = m_sorted_depgraph.Positions ();
653
729
}
654
730
655
731
/* * Check whether any unlinearized transactions remain. */
@@ -1161,7 +1237,7 @@ void PostLinearize(const DepGraph<SetType>& depgraph, Span<ClusterIndex> lineari
1161
1237
// During an even pass, the diagram above would correspond to linearization [2,3,0,1], with
1162
1238
// groups [2] and [3,0,1].
1163
1239
1164
- std::vector<TxEntry> entries (linearization. size () + 1 );
1240
+ std::vector<TxEntry> entries (depgraph. PositionRange () + 1 );
1165
1241
1166
1242
// Perform two passes over the linearization.
1167
1243
for (int pass = 0 ; pass < 2 ; ++pass) {
0 commit comments