24static constexpr int MAX_LEVELS{2};
30using LinearizationIndex = uint32_t;
32using ClusterSetIndex = uint32_t;
35enum class QualityLevel
40 NEEDS_SPLIT_ACCEPTABLE,
55 friend class TxGraphImpl;
63 std::vector<GraphIndex> m_mapping;
66 std::vector<DepGraphIndex> m_linearization;
70 ClusterSetIndex m_setindex{ClusterSetIndex(-1)};
76 Cluster() noexcept = default;
78 explicit Cluster(TxGraphImpl& graph, const
FeePerWeight& feerate, GraphIndex graph_index) noexcept;
81 Cluster(const Cluster&) = delete;
82 Cluster& operator=(const Cluster&) = delete;
83 Cluster(Cluster&&) = delete;
84 Cluster& operator=(Cluster&&) = delete;
89 bool IsAcceptable(
bool after_split = false) const noexcept
91 return m_quality == QualityLevel::ACCEPTABLE || m_quality == QualityLevel::OPTIMAL ||
92 (after_split && m_quality == QualityLevel::NEEDS_SPLIT_ACCEPTABLE);
95 bool IsOptimal() const noexcept
97 return m_quality == QualityLevel::OPTIMAL;
100 bool NeedsSplitting() const noexcept
102 return m_quality == QualityLevel::NEEDS_SPLIT ||
103 m_quality == QualityLevel::NEEDS_SPLIT_ACCEPTABLE;
106 LinearizationIndex GetTxCount() const noexcept {
return m_linearization.size(); }
108 GraphIndex GetClusterEntry(
DepGraphIndex index)
const noexcept {
return m_mapping[index]; }
110 void UpdateMapping(
DepGraphIndex cluster_idx, GraphIndex graph_idx)
noexcept { m_mapping[cluster_idx] = graph_idx; }
112 void Updated(TxGraphImpl& graph)
noexcept;
114 Cluster* CopyToStaging(TxGraphImpl& graph)
const noexcept;
116 void GetConflicts(
const TxGraphImpl& graph, std::vector<Cluster*>&
out)
const noexcept;
119 void MakeStagingTransactionsMissing(TxGraphImpl& graph)
noexcept;
121 void Clear(TxGraphImpl& graph)
noexcept;
123 void MoveToMain(TxGraphImpl& graph)
noexcept;
129 void ApplyRemovals(TxGraphImpl& graph, std::span<GraphIndex>& to_remove)
noexcept;
132 [[nodiscard]]
bool Split(TxGraphImpl& graph)
noexcept;
134 void Merge(TxGraphImpl& graph, Cluster& cluster)
noexcept;
136 void ApplyDependencies(TxGraphImpl& graph, std::span<std::pair<GraphIndex, GraphIndex>> to_apply)
noexcept;
138 void Relinearize(TxGraphImpl& graph, uint64_t max_iters)
noexcept;
144 void GetAncestorRefs(
const TxGraphImpl& graph, std::span<std::pair<Cluster*, DepGraphIndex>>&
args, std::vector<TxGraph::Ref*>& output)
noexcept;
147 void GetDescendantRefs(
const TxGraphImpl& graph, std::span<std::pair<Cluster*, DepGraphIndex>>&
args, std::vector<TxGraph::Ref*>& output)
noexcept;
149 std::vector<TxGraph::Ref*> GetClusterRefs(
const TxGraphImpl& graph)
noexcept;
157 void SanityCheck(
const TxGraphImpl& graph,
int level)
const;
183class TxGraphImpl final :
public TxGraph
185 friend class Cluster;
196 uint32_t m_cluster_offset;
198 uint32_t m_cluster_count;
200 uint32_t m_deps_offset;
202 uint32_t m_deps_count;
209 std::vector<GroupEntry> m_groups;
211 std::vector<Cluster*> m_group_clusters;
214 bool m_group_oversized;
221 std::array<std::vector<std::unique_ptr<Cluster>>, int(
QualityLevel::NONE)> m_clusters;
223 std::vector<GraphIndex> m_to_remove;
226 std::vector<std::pair<GraphIndex, GraphIndex>> m_deps_to_add;
228 std::optional<GroupData> m_group_data = GroupData{};
232 std::vector<GraphIndex> m_removed;
235 GraphIndex m_txcount{0};
238 std::optional<bool> m_oversized{
false};
240 ClusterSet() noexcept = default;
244 ClusterSet m_main_clusterset;
246 std::optional<ClusterSet> m_staging_clusterset;
281 Cluster* cluster{
nullptr};
286 void SetMissing() noexcept { cluster =
nullptr; index = 0; }
288 void SetRemoved() noexcept { cluster =
nullptr; index =
DepGraphIndex(-1); }
290 void SetPresent(Cluster* c,
DepGraphIndex i)
noexcept { cluster = c; index = i; }
292 bool IsMissing() const noexcept {
return cluster ==
nullptr && index == 0; }
294 bool IsRemoved() const noexcept {
return cluster ==
nullptr && index ==
DepGraphIndex(-1); }
296 bool IsPresent() const noexcept {
return cluster !=
nullptr; }
305 Locator m_locator[MAX_LEVELS];
309 LinearizationIndex m_main_lin_index;
313 std::vector<Entry> m_entries;
316 std::vector<GraphIndex> m_unlinked;
320 explicit TxGraphImpl(
DepGraphIndex max_cluster_count) noexcept :
321 m_max_cluster_count(max_cluster_count)
323 Assume(max_cluster_count >= 1);
328 ~TxGraphImpl() noexcept;
331 TxGraphImpl(const TxGraphImpl&) = delete;
332 TxGraphImpl& operator=(const TxGraphImpl&) = delete;
333 TxGraphImpl(TxGraphImpl&&) = delete;
334 TxGraphImpl& operator=(TxGraphImpl&&) = delete;
339 void SwapIndexes(GraphIndex a, GraphIndex b) noexcept;
342 Cluster* FindCluster(GraphIndex idx,
int level) const noexcept;
344 std::unique_ptr<Cluster> ExtractCluster(
int level, QualityLevel quality, ClusterSetIndex setindex) noexcept;
346 void DeleteCluster(Cluster& cluster) noexcept;
348 ClusterSetIndex InsertCluster(
int level,
std::unique_ptr<Cluster>&& cluster, QualityLevel quality) noexcept;
350 void SetClusterQuality(
int level, QualityLevel old_quality, ClusterSetIndex old_index, QualityLevel new_quality) noexcept;
352 int GetTopLevel() const noexcept {
return m_staging_clusterset.has_value(); }
354 int GetSpecifiedLevel(
bool main_only)
const noexcept {
return m_staging_clusterset.has_value() && !main_only; }
356 ClusterSet& GetClusterSet(
int level)
noexcept;
357 const ClusterSet& GetClusterSet(
int level)
const noexcept;
359 void ClearLocator(
int level, GraphIndex index)
noexcept;
361 std::vector<Cluster*> GetConflicts() const noexcept;
366 void UpdateRef(GraphIndex idx, Ref& new_location) noexcept final
368 auto& entry = m_entries[idx];
369 Assume(entry.m_ref !=
nullptr);
370 entry.m_ref = &new_location;
374 void UnlinkRef(GraphIndex idx)
noexcept final
376 auto& entry = m_entries[idx];
377 Assume(entry.m_ref !=
nullptr);
378 entry.m_ref =
nullptr;
381 bool exists_anywhere{
false};
383 for (
int level = 0; level <= GetTopLevel(); ++level) {
384 if (entry.m_locator[level].IsPresent()) {
385 exists_anywhere =
true;
387 }
else if (entry.m_locator[level].IsRemoved()) {
391 auto& clusterset = GetClusterSet(level);
392 clusterset.m_to_remove.push_back(idx);
394 clusterset.m_group_data = std::nullopt;
400 if (level == GetTopLevel() && clusterset.m_oversized ==
true) {
401 clusterset.m_oversized = std::nullopt;
405 m_unlinked.push_back(idx);
406 if (!exists_anywhere) Compact();
413 void Compact() noexcept;
417 Cluster* PullIn(Cluster* cluster) noexcept;
420 void ApplyRemovals(
int up_to_level) noexcept;
422 void Split(Cluster& cluster) noexcept;
424 void SplitAll(
int up_to_level) noexcept;
426 void GroupClusters(
int level) noexcept;
428 void Merge(
std::span<Cluster*> to_merge) noexcept;
430 void ApplyDependencies(
int level) noexcept;
432 void MakeAcceptable(Cluster& cluster) noexcept;
434 void MakeAllAcceptable(
int level) noexcept;
438 Ref AddTransaction(const
FeePerWeight& feerate) noexcept final;
439 void RemoveTransaction(const Ref& arg) noexcept final;
440 void AddDependency(const Ref& parent, const Ref& child) noexcept final;
441 void SetTransactionFee(const Ref&, int64_t
fee) noexcept final;
443 void DoWork() noexcept final;
445 void StartStaging() noexcept final;
446 void CommitStaging() noexcept final;
447 void AbortStaging() noexcept final;
448 bool HaveStaging() const noexcept final {
return m_staging_clusterset.has_value(); }
450 bool Exists(
const Ref& arg,
bool main_only =
false) noexcept final;
451 FeePerWeight GetMainChunkFeerate(const Ref& arg) noexcept final;
452 FeePerWeight GetIndividualFeerate(const Ref& arg) noexcept final;
453 std::vector<Ref*> GetCluster(const Ref& arg,
bool main_only = false) noexcept final;
454 std::vector<Ref*> GetAncestors(const Ref& arg,
bool main_only = false) noexcept final;
455 std::vector<Ref*> GetDescendants(const Ref& arg,
bool main_only = false) noexcept final;
456 std::vector<Ref*> GetAncestorsUnion(
std::span<const Ref* const>
args,
bool main_only = false) noexcept final;
457 std::vector<Ref*> GetDescendantsUnion(
std::span<const Ref* const>
args,
bool main_only = false) noexcept final;
458 GraphIndex GetTransactionCount(
bool main_only = false) noexcept final;
459 bool IsOversized(
bool main_only = false) noexcept final;
460 std::strong_ordering CompareMainOrder(const Ref& a, const Ref& b) noexcept final;
461 GraphIndex CountDistinctClusters(
std::span<const Ref* const> refs,
bool main_only = false) noexcept final;
463 void SanityCheck() const final;
466TxGraphImpl::ClusterSet& TxGraphImpl::GetClusterSet(
int level) noexcept
468 if (level == 0)
return m_main_clusterset;
470 Assume(m_staging_clusterset.has_value());
471 return *m_staging_clusterset;
474const TxGraphImpl::ClusterSet& TxGraphImpl::GetClusterSet(
int level)
const noexcept
476 if (level == 0)
return m_main_clusterset;
478 Assume(m_staging_clusterset.has_value());
479 return *m_staging_clusterset;
482void TxGraphImpl::ClearLocator(
int level, GraphIndex idx)
noexcept
484 auto& entry = m_entries[idx];
485 auto& clusterset = GetClusterSet(level);
486 Assume(entry.m_locator[level].IsPresent());
488 if (level == 0 || !entry.m_locator[level - 1].IsPresent()) {
489 entry.m_locator[level].SetMissing();
491 entry.m_locator[level].SetRemoved();
492 clusterset.m_removed.push_back(idx);
495 --clusterset.m_txcount;
497 if (level == 0 && GetTopLevel() == 1) {
498 if (entry.m_locator[1].IsRemoved()) {
499 entry.m_locator[1].SetMissing();
500 }
else if (!entry.m_locator[1].IsPresent()) {
501 --m_staging_clusterset->m_txcount;
506void Cluster::Updated(TxGraphImpl& graph)
noexcept
510 auto& entry = graph.m_entries[m_mapping[idx]];
511 entry.m_locator[m_level].SetPresent(
this, idx);
518 if (m_level == 0 && IsAcceptable()) {
520 LinearizationIndex lin_idx{0};
522 for (
unsigned chunk_idx = 0; chunk_idx < chunking.NumChunksLeft(); ++chunk_idx) {
523 auto chunk = chunking.GetChunk(chunk_idx);
524 Assume(chunk.transactions.Any());
528 GraphIndex graph_idx = m_mapping[idx];
529 auto& entry = graph.m_entries[graph_idx];
530 entry.m_main_lin_index = lin_idx++;
532 Assume(chunk.transactions[idx]);
533 chunk.transactions.Reset(idx);
534 }
while(chunk.transactions.Any());
539void Cluster::GetConflicts(
const TxGraphImpl& graph, std::vector<Cluster*>&
out)
const noexcept
542 for (
auto i : m_linearization) {
543 auto& entry = graph.m_entries[m_mapping[i]];
546 if (entry.m_locator[0].IsPresent()) {
547 out.push_back(entry.m_locator[0].cluster);
552std::vector<Cluster*> TxGraphImpl::GetConflicts() const noexcept
554 Assume(GetTopLevel() == 1);
555 auto& clusterset = GetClusterSet(1);
556 std::vector<Cluster*>
ret;
558 for (
auto i : clusterset.m_removed) {
559 auto& entry = m_entries[i];
560 if (entry.m_locator[0].IsPresent()) {
561 ret.push_back(entry.m_locator[0].cluster);
566 auto& clusters = clusterset.m_clusters[quality];
567 for (
const auto& cluster : clusters) {
568 cluster->GetConflicts(*
this,
ret);
572 std::sort(
ret.begin(),
ret.end());
573 ret.erase(std::unique(
ret.begin(),
ret.end()),
ret.end());
577Cluster* Cluster::CopyToStaging(TxGraphImpl& graph)
const noexcept
580 auto ret = std::make_unique<Cluster>();
581 auto ptr =
ret.get();
583 ptr->m_depgraph = m_depgraph;
584 ptr->m_mapping = m_mapping;
585 ptr->m_linearization = m_linearization;
587 graph.InsertCluster(1, std::move(
ret), m_quality);
593void Cluster::ApplyRemovals(TxGraphImpl& graph, std::span<GraphIndex>& to_remove)
noexcept
596 Assume(!to_remove.empty());
599 GraphIndex idx = to_remove.front();
600 Assume(idx < graph.m_entries.size());
601 auto& entry = graph.m_entries[idx];
602 auto& locator = entry.m_locator[m_level];
604 if (locator.cluster !=
this)
break;
606 todo.Set(locator.index);
610 m_mapping[locator.index] = GraphIndex(-1);
613 entry.m_main_lin_index = LinearizationIndex(-1);
616 graph.ClearLocator(m_level, idx);
617 to_remove = to_remove.subspan(1);
618 }
while(!to_remove.empty());
620 auto quality = m_quality;
628 while (!m_linearization.empty() && todo[m_linearization.back()]) {
629 todo.Reset(m_linearization.back());
630 m_linearization.pop_back();
635 if (IsAcceptable(
true)) {
636 quality = QualityLevel::NEEDS_SPLIT_ACCEPTABLE;
638 quality = QualityLevel::NEEDS_SPLIT;
642 m_linearization.erase(std::remove_if(
643 m_linearization.begin(),
644 m_linearization.end(),
645 [&](
auto pos) { return todo[pos]; }), m_linearization.end());
646 quality = QualityLevel::NEEDS_SPLIT;
648 graph.SetClusterQuality(m_level, m_quality, m_setindex, quality);
652void Cluster::Clear(TxGraphImpl& graph)
noexcept
654 for (
auto i : m_linearization) {
655 graph.ClearLocator(m_level, m_mapping[i]);
658 m_linearization.clear();
662void Cluster::MoveToMain(TxGraphImpl& graph)
noexcept
665 for (
auto i : m_linearization) {
666 GraphIndex idx = m_mapping[i];
667 auto& entry = graph.m_entries[idx];
668 entry.m_locator[1].SetMissing();
670 auto quality = m_quality;
671 auto cluster = graph.ExtractCluster(1, quality, m_setindex);
672 graph.InsertCluster(0, std::move(cluster), quality);
681 QualityLevel new_quality = IsAcceptable(
true) ? QualityLevel::ACCEPTABLE
682 : QualityLevel::NEEDS_RELINEARIZE;
689 if (new_quality == QualityLevel::ACCEPTABLE) {
696 std::vector<std::pair<Cluster*, DepGraphIndex>> remap(m_depgraph.
PositionRange());
697 std::vector<Cluster*> new_clusters;
702 if (first && component == todo) {
705 graph.SetClusterQuality(m_level, m_quality, m_setindex, new_quality);
713 auto new_cluster = std::make_unique<Cluster>();
714 new_clusters.push_back(new_cluster.get());
717 for (
auto i : component) {
720 graph.InsertCluster(m_level, std::move(new_cluster), new_quality);
724 for (
auto i : m_linearization) {
726 Cluster* new_cluster = remap[i].first;
728 remap[i].second = new_cluster->m_depgraph.AddTransaction(m_depgraph.
FeeRate(i));
730 new_cluster->m_mapping.push_back(m_mapping[i]);
733 new_cluster->m_linearization.push_back(remap[i].second);
736 for (
auto i : m_linearization) {
738 Cluster* new_cluster = remap[i].first;
741 for (
auto par : m_depgraph.
GetReducedParents(i)) new_parents.Set(remap[par].second);
742 new_cluster->m_depgraph.AddDependencies(new_parents, remap[i].second);
745 for (Cluster* new_cluster : new_clusters) {
746 new_cluster->Updated(graph);
751 m_linearization.clear();
755void Cluster::Merge(TxGraphImpl& graph, Cluster& other)
noexcept
758 std::vector<DepGraphIndex> remap(other.m_depgraph.PositionRange());
760 for (
auto pos : other.m_linearization) {
761 auto idx = other.m_mapping[pos];
763 auto new_pos = m_depgraph.
AddTransaction(other.m_depgraph.FeeRate(pos));
764 remap[pos] = new_pos;
765 if (new_pos == m_mapping.size()) {
766 m_mapping.push_back(idx);
768 m_mapping[new_pos] = idx;
770 m_linearization.push_back(new_pos);
775 for (
auto par : other.m_depgraph.GetReducedParents(pos)) {
776 parents.Set(remap[par]);
782 graph.m_entries[idx].m_locator[m_level].SetPresent(
this, new_pos);
786 other.m_linearization.clear();
787 other.m_mapping.clear();
790void Cluster::ApplyDependencies(TxGraphImpl& graph, std::span<std::pair<GraphIndex, GraphIndex>> to_apply)
noexcept
801 std::sort(to_apply.begin(), to_apply.end(), [](
auto& a,
auto& b) { return a.second < b.second; });
803 auto it = to_apply.begin();
804 while (it != to_apply.end()) {
805 auto& first_child = graph.m_entries[it->second].m_locator[m_level];
806 const auto child_idx = first_child.index;
810 while (it != to_apply.end()) {
811 auto& child = graph.m_entries[it->second].m_locator[m_level];
812 auto& parent = graph.m_entries[it->first].m_locator[m_level];
813 Assume(child.cluster ==
this && parent.cluster ==
this);
814 if (child.index != child_idx)
break;
815 parents.Set(parent.index);
833TxGraphImpl::~TxGraphImpl() noexcept
837 for (
auto& entry : m_entries) {
838 if (entry.m_ref !=
nullptr) {
839 GetRefGraph(*entry.m_ref) =
nullptr;
844std::unique_ptr<Cluster> TxGraphImpl::ExtractCluster(
int level, QualityLevel quality, ClusterSetIndex setindex)
noexcept
848 auto& clusterset = GetClusterSet(level);
849 auto& quality_clusters = clusterset.m_clusters[int(quality)];
850 Assume(setindex < quality_clusters.size());
853 std::unique_ptr<Cluster>
ret = std::move(quality_clusters[setindex]);
855 ret->m_setindex = ClusterSetIndex(-1);
859 auto max_setindex = quality_clusters.size() - 1;
860 if (setindex != max_setindex) {
862 quality_clusters.back()->m_setindex = setindex;
863 quality_clusters.back()->m_level = level;
864 quality_clusters[setindex] = std::move(quality_clusters.back());
867 quality_clusters.pop_back();
872ClusterSetIndex TxGraphImpl::InsertCluster(
int level, std::unique_ptr<Cluster>&& cluster, QualityLevel quality)
noexcept
880 auto& clusterset = GetClusterSet(level);
881 auto& quality_clusters = clusterset.m_clusters[int(quality)];
882 ClusterSetIndex
ret = quality_clusters.size();
883 cluster->m_quality = quality;
884 cluster->m_setindex =
ret;
885 cluster->m_level = level;
886 quality_clusters.push_back(std::move(cluster));
890void TxGraphImpl::SetClusterQuality(
int level, QualityLevel old_quality, ClusterSetIndex old_index, QualityLevel new_quality)
noexcept
895 if (old_quality == new_quality)
return;
897 auto cluster_ptr = ExtractCluster(level, old_quality, old_index);
899 InsertCluster(level, std::move(cluster_ptr), new_quality);
902void TxGraphImpl::DeleteCluster(Cluster& cluster)
noexcept
905 auto cluster_ptr = ExtractCluster(cluster.m_level, cluster.m_quality, cluster.m_setindex);
910Cluster* TxGraphImpl::FindCluster(GraphIndex idx,
int level)
const noexcept
912 Assume(level >= 0 && level <= GetTopLevel());
913 auto& entry = m_entries[idx];
915 for (
int l = level; l >= 0; --l) {
918 if (entry.m_locator[l].IsMissing())
continue;
920 if (entry.m_locator[l].IsRemoved())
break;
922 return entry.m_locator[l].cluster;
928Cluster* TxGraphImpl::PullIn(Cluster* cluster)
noexcept
930 int to_level = GetTopLevel();
931 if (to_level == 0)
return cluster;
932 int level = cluster->m_level;
933 Assume(level <= to_level);
938 MakeAcceptable(*cluster);
939 cluster = cluster->CopyToStaging(*
this);
944void TxGraphImpl::ApplyRemovals(
int up_to_level)
noexcept
946 Assume(up_to_level >= 0 && up_to_level <= GetTopLevel());
947 for (
int level = 0; level <= up_to_level; ++level) {
948 auto& clusterset = GetClusterSet(level);
949 auto& to_remove = clusterset.m_to_remove;
951 if (to_remove.empty())
continue;
954 for (GraphIndex index : to_remove) {
955 auto cluster = FindCluster(index, level);
956 if (cluster !=
nullptr) PullIn(cluster);
960 std::sort(to_remove.begin(), to_remove.end(), [&](GraphIndex a, GraphIndex b)
noexcept {
961 return std::less{}(m_entries[a].m_locator[level].cluster, m_entries[b].m_locator[level].cluster);
964 std::span to_remove_span{to_remove};
965 while (!to_remove_span.empty()) {
966 Cluster* cluster = m_entries[to_remove_span.front()].m_locator[level].cluster;
967 if (cluster !=
nullptr) {
970 cluster->ApplyRemovals(*
this, to_remove_span);
974 to_remove_span = to_remove_span.subspan(1);
982void TxGraphImpl::SwapIndexes(GraphIndex a, GraphIndex b)
noexcept
984 Assume(a < m_entries.size());
985 Assume(b < m_entries.size());
987 std::swap(m_entries[a], m_entries[b]);
989 for (
int i = 0; i < 2; ++i) {
990 GraphIndex idx = i ? b : a;
991 Entry& entry = m_entries[idx];
993 if (entry.m_ref) GetRefIndex(*entry.m_ref) = idx;
996 for (
int level = 0; level < MAX_LEVELS; ++level) {
997 Locator& locator = entry.m_locator[level];
998 if (locator.IsPresent()) {
999 locator.cluster->UpdateMapping(locator.index, idx);
1005void TxGraphImpl::Compact() noexcept
1009 if (!m_main_clusterset.m_deps_to_add.empty())
return;
1010 if (!m_main_clusterset.m_to_remove.empty())
return;
1011 if (!m_main_clusterset.m_removed.empty())
return;
1012 if (m_staging_clusterset.has_value()) {
1013 if (!m_staging_clusterset->m_deps_to_add.empty())
return;
1014 if (!m_staging_clusterset->m_to_remove.empty())
return;
1015 if (!m_staging_clusterset->m_removed.empty())
return;
1022 std::sort(m_unlinked.begin(), m_unlinked.end(), std::greater{});
1024 auto last = GraphIndex(-1);
1025 for (GraphIndex idx : m_unlinked) {
1032 Entry& entry = m_entries[idx];
1033 Assume(entry.m_ref ==
nullptr);
1035 for (
int level = 0; level < MAX_LEVELS; ++level) {
1036 Assume(!entry.m_locator[level].IsPresent());
1040 if (idx != m_entries.size() - 1) SwapIndexes(idx, m_entries.size() - 1);
1042 m_entries.pop_back();
1051 ApplyRemovals(cluster.m_level);
1052 bool del = cluster.Split(*
this);
1055 DeleteCluster(cluster);
1059void TxGraphImpl::SplitAll(
int up_to_level)
noexcept
1061 Assume(up_to_level >= 0 && up_to_level <= GetTopLevel());
1063 ApplyRemovals(up_to_level);
1064 for (
int level = 0; level <= up_to_level; ++level) {
1065 for (
auto quality : {QualityLevel::NEEDS_SPLIT, QualityLevel::NEEDS_SPLIT_ACCEPTABLE}) {
1066 auto& queue = GetClusterSet(level).m_clusters[int(quality)];
1067 while (!queue.empty()) {
1068 Split(*queue.back().get());
1074void TxGraphImpl::GroupClusters(
int level)
noexcept
1076 auto& clusterset = GetClusterSet(level);
1078 if (clusterset.m_group_data.has_value())
return;
1087 std::vector<std::pair<Cluster*, Cluster*>> an_clusters;
1091 std::vector<std::pair<std::pair<GraphIndex, GraphIndex>, Cluster*>> an_deps;
1094 for (
const auto& [par, chl] : clusterset.m_deps_to_add) {
1095 auto par_cluster = FindCluster(par, level);
1096 auto chl_cluster = FindCluster(chl, level);
1098 if (par_cluster ==
nullptr || chl_cluster ==
nullptr)
continue;
1099 an_clusters.emplace_back(par_cluster,
nullptr);
1102 if (chl_cluster != par_cluster) an_clusters.emplace_back(chl_cluster,
nullptr);
1106 std::sort(an_clusters.begin(), an_clusters.end());
1107 an_clusters.erase(std::unique(an_clusters.begin(), an_clusters.end()), an_clusters.end());
1110 std::sort(clusterset.m_deps_to_add.begin(), clusterset.m_deps_to_add.end(), [&](
auto& a,
auto& b)
noexcept {
1111 auto [_a_par, a_chl] = a;
1112 auto [_b_par, b_chl] = b;
1113 auto a_chl_cluster = FindCluster(a_chl, level);
1114 auto b_chl_cluster = FindCluster(b_chl, level);
1115 return std::less{}(a_chl_cluster, b_chl_cluster);
1122 struct PartitionData
1130 PartitionData* parent;
1136 std::vector<PartitionData> partition_data;
1139 auto locate_fn = [&](Cluster* arg)
noexcept -> PartitionData* {
1140 auto it = std::lower_bound(partition_data.begin(), partition_data.end(), arg,
1141 [](
auto& a, Cluster* ptr)
noexcept { return a.cluster < ptr; });
1142 Assume(it != partition_data.end());
1143 Assume(it->cluster == arg);
1148 static constexpr auto find_root_fn = [](PartitionData*
data)
noexcept -> PartitionData* {
1152 auto par =
data->parent;
1153 data->parent = par->parent;
1161 static constexpr auto union_fn = [](PartitionData* arg1, PartitionData* arg2)
noexcept {
1164 auto rep1 = find_root_fn(arg1);
1165 auto rep2 = find_root_fn(arg2);
1166 if (rep1 == rep2)
return rep1;
1169 if (rep1->rank < rep2->rank) std::swap(rep1, rep2);
1170 rep2->parent = rep1;
1171 rep1->rank += (rep1->rank == rep2->rank);
1176 partition_data.resize(an_clusters.size());
1177 for (
size_t i = 0; i < an_clusters.size(); ++i) {
1178 partition_data[i].cluster = an_clusters[i].first;
1179 partition_data[i].parent = &partition_data[i];
1180 partition_data[i].rank = 0;
1185 Cluster* last_chl_cluster{
nullptr};
1186 PartitionData* last_partition{
nullptr};
1187 for (
const auto& [par, chl] : clusterset.m_deps_to_add) {
1188 auto par_cluster = FindCluster(par, level);
1189 auto chl_cluster = FindCluster(chl, level);
1191 if (par_cluster == chl_cluster)
continue;
1193 if (par_cluster ==
nullptr || chl_cluster ==
nullptr)
continue;
1195 if (chl_cluster == last_chl_cluster) {
1199 last_partition = union_fn(locate_fn(par_cluster), last_partition);
1201 last_chl_cluster = chl_cluster;
1202 last_partition = union_fn(locate_fn(par_cluster), locate_fn(chl_cluster));
1209 an_deps.reserve(clusterset.m_deps_to_add.size());
1210 auto deps_it = clusterset.m_deps_to_add.begin();
1211 for (
size_t i = 0; i < partition_data.size(); ++i) {
1212 auto&
data = partition_data[i];
1215 auto rep = find_root_fn(&
data)->cluster;
1216 Assume(an_clusters[i].second ==
nullptr);
1217 an_clusters[i].second = rep;
1219 while (deps_it != clusterset.m_deps_to_add.end()) {
1220 auto [par, chl] = *deps_it;
1221 auto chl_cluster = FindCluster(chl, level);
1222 if (std::greater{}(chl_cluster,
data.cluster))
break;
1225 if (chl_cluster ==
data.cluster) {
1226 auto par_cluster = FindCluster(par, level);
1228 if (par_cluster !=
nullptr) an_deps.emplace_back(*deps_it, rep);
1237 std::sort(an_deps.begin(), an_deps.end(), [](
auto& a,
auto& b)
noexcept { return a.second < b.second; });
1238 std::sort(an_clusters.begin(), an_clusters.end(), [](
auto& a,
auto& b)
noexcept { return a.second < b.second; });
1242 clusterset.m_group_data = GroupData{};
1243 clusterset.m_group_data->m_group_clusters.reserve(an_clusters.size());
1244 clusterset.m_group_data->m_group_oversized =
false;
1245 clusterset.m_deps_to_add.clear();
1246 clusterset.m_deps_to_add.reserve(an_deps.size());
1247 auto an_deps_it = an_deps.begin();
1248 auto an_clusters_it = an_clusters.begin();
1249 while (an_clusters_it != an_clusters.end()) {
1251 auto rep = an_clusters_it->second;
1253 auto& new_entry = clusterset.m_group_data->m_groups.emplace_back();
1254 new_entry.m_cluster_offset = clusterset.m_group_data->m_group_clusters.size();
1255 new_entry.m_cluster_count = 0;
1256 new_entry.m_deps_offset = clusterset.m_deps_to_add.size();
1257 new_entry.m_deps_count = 0;
1258 uint32_t total_count{0};
1260 while (an_clusters_it != an_clusters.end() && an_clusters_it->second == rep) {
1261 clusterset.m_group_data->m_group_clusters.push_back(an_clusters_it->first);
1262 total_count += an_clusters_it->first->GetTxCount();
1264 ++new_entry.m_cluster_count;
1267 while (an_deps_it != an_deps.end() && an_deps_it->second == rep) {
1268 clusterset.m_deps_to_add.push_back(an_deps_it->first);
1270 ++new_entry.m_deps_count;
1273 if (total_count > m_max_cluster_count) {
1274 clusterset.m_group_data->m_group_oversized =
true;
1277 Assume(an_deps_it == an_deps.end());
1278 Assume(an_clusters_it == an_clusters.end());
1279 clusterset.m_oversized = clusterset.m_group_data->m_group_oversized;
1283void TxGraphImpl::Merge(std::span<Cluster*> to_merge)
noexcept
1285 Assume(!to_merge.empty());
1287 if (to_merge.size() == 1)
return;
1292 size_t max_size_pos{0};
1293 DepGraphIndex max_size = to_merge[max_size_pos]->GetTxCount();
1294 for (
size_t i = 1; i < to_merge.size(); ++i) {
1296 if (size > max_size) {
1301 if (max_size_pos != 0) std::swap(to_merge[0], to_merge[max_size_pos]);
1304 for (
size_t i = 1; i < to_merge.size(); ++i) {
1305 to_merge[0]->Merge(*
this, *to_merge[i]);
1306 DeleteCluster(*to_merge[i]);
1310void TxGraphImpl::ApplyDependencies(
int level)
noexcept
1312 auto& clusterset = GetClusterSet(level);
1314 if (clusterset.m_oversized ==
true)
return;
1316 GroupClusters(level);
1317 Assume(clusterset.m_group_data.has_value());
1319 if (clusterset.m_deps_to_add.empty())
return;
1321 if (clusterset.m_group_data->m_group_oversized)
return;
1324 for (
const auto& group_entry : clusterset.m_group_data->m_groups) {
1325 auto cluster_span = std::span{clusterset.m_group_data->m_group_clusters}
1326 .subspan(group_entry.m_cluster_offset, group_entry.m_cluster_count);
1329 for (Cluster*& cluster : cluster_span) {
1330 cluster = PullIn(cluster);
1334 Merge(cluster_span);
1337 auto deps_span = std::span{clusterset.m_deps_to_add}
1338 .subspan(group_entry.m_deps_offset, group_entry.m_deps_count);
1339 Assume(!deps_span.empty());
1340 const auto& loc = m_entries[deps_span[0].second].m_locator[level];
1342 loc.cluster->ApplyDependencies(*
this, deps_span);
1346 clusterset.m_deps_to_add.clear();
1350 clusterset.m_group_data = GroupData{};
1353void Cluster::Relinearize(TxGraphImpl& graph, uint64_t max_iters)
noexcept
1356 Assume(!NeedsSplitting());
1358 if (IsOptimal())
return;
1360 uint64_t rng_seed = graph.m_rng.rand64();
1361 auto [linearization, optimal] =
Linearize(m_depgraph, max_iters, rng_seed, m_linearization);
1366 m_linearization = std::move(linearization);
1368 auto new_quality = optimal ? QualityLevel::OPTIMAL : QualityLevel::ACCEPTABLE;
1369 graph.SetClusterQuality(m_level, m_quality, m_setindex, new_quality);
1374void TxGraphImpl::MakeAcceptable(Cluster& cluster)
noexcept
1377 if (!cluster.NeedsSplitting() && !cluster.IsAcceptable()) {
1378 cluster.Relinearize(*
this, 10000);
1382void TxGraphImpl::MakeAllAcceptable(
int level)
noexcept
1384 ApplyDependencies(level);
1385 auto& clusterset = GetClusterSet(level);
1386 if (clusterset.m_oversized ==
true)
return;
1387 auto& queue = clusterset.m_clusters[int(QualityLevel::NEEDS_RELINEARIZE)];
1388 while (!queue.empty()) {
1389 MakeAcceptable(*queue.back().get());
1393Cluster::Cluster(TxGraphImpl& graph,
const FeePerWeight& feerate, GraphIndex graph_index)
noexcept
1397 m_mapping.push_back(graph_index);
1398 m_linearization.push_back(cluster_idx);
1406 auto idx = m_entries.size();
1407 m_entries.emplace_back();
1408 auto& entry = m_entries.back();
1410 GetRefGraph(
ret) =
this;
1411 GetRefIndex(
ret) = idx;
1413 auto cluster = std::make_unique<Cluster>(*
this, feerate, idx);
1414 auto cluster_ptr = cluster.get();
1415 int level = GetTopLevel();
1416 auto& clusterset = GetClusterSet(level);
1417 InsertCluster(level, std::move(cluster), QualityLevel::OPTIMAL);
1418 cluster_ptr->Updated(*
this);
1419 ++clusterset.m_txcount;
1424void TxGraphImpl::RemoveTransaction(
const Ref& arg)
noexcept
1428 if (GetRefGraph(arg) ==
nullptr)
return;
1429 Assume(GetRefGraph(arg) ==
this);
1431 int level = GetTopLevel();
1432 auto cluster = FindCluster(GetRefIndex(arg), level);
1433 if (cluster ==
nullptr)
return;
1435 auto& clusterset = GetClusterSet(level);
1436 clusterset.m_to_remove.push_back(GetRefIndex(arg));
1438 clusterset.m_group_data.reset();
1439 if (clusterset.m_oversized ==
true) clusterset.m_oversized = std::nullopt;
1442void TxGraphImpl::AddDependency(
const Ref& parent,
const Ref& child)
noexcept
1446 if (GetRefGraph(parent) ==
nullptr || GetRefGraph(child) ==
nullptr)
return;
1447 Assume(GetRefGraph(parent) ==
this && GetRefGraph(child) ==
this);
1449 if (GetRefIndex(parent) == GetRefIndex(child))
return;
1452 int level = GetTopLevel();
1453 auto par_cluster = FindCluster(GetRefIndex(parent), level);
1454 if (par_cluster ==
nullptr)
return;
1455 auto chl_cluster = FindCluster(GetRefIndex(child), level);
1456 if (chl_cluster ==
nullptr)
return;
1458 auto& clusterset = GetClusterSet(level);
1459 clusterset.m_deps_to_add.emplace_back(GetRefIndex(parent), GetRefIndex(child));
1461 clusterset.m_group_data.reset();
1462 if (clusterset.m_oversized ==
false) clusterset.m_oversized = std::nullopt;
1465bool TxGraphImpl::Exists(
const Ref& arg,
bool main_only)
noexcept
1467 if (GetRefGraph(arg) ==
nullptr)
return false;
1468 Assume(GetRefGraph(arg) ==
this);
1469 size_t level = GetSpecifiedLevel(main_only);
1471 ApplyRemovals(level);
1472 auto cluster = FindCluster(GetRefIndex(arg), level);
1473 return cluster !=
nullptr;
1476void Cluster::GetAncestorRefs(
const TxGraphImpl& graph, std::span<std::pair<Cluster*, DepGraphIndex>>&
args, std::vector<TxGraph::Ref*>& output)
noexcept
1479 SetType ancestors_union;
1481 while (!
args.empty()) {
1482 if (
args.front().first !=
this)
break;
1483 ancestors_union |= m_depgraph.
Ancestors(
args.front().second);
1486 Assume(ancestors_union.Any());
1488 for (
auto idx : ancestors_union) {
1489 const auto& entry = graph.m_entries[m_mapping[idx]];
1490 Assume(entry.m_ref !=
nullptr);
1491 output.push_back(entry.m_ref);
1495void Cluster::GetDescendantRefs(
const TxGraphImpl& graph, std::span<std::pair<Cluster*, DepGraphIndex>>&
args, std::vector<TxGraph::Ref*>& output)
noexcept
1498 SetType descendants_union;
1500 while (!
args.empty()) {
1501 if (
args.front().first !=
this)
break;
1505 Assume(descendants_union.Any());
1507 for (
auto idx : descendants_union) {
1508 const auto& entry = graph.m_entries[m_mapping[idx]];
1509 Assume(entry.m_ref !=
nullptr);
1510 output.push_back(entry.m_ref);
1514std::vector<TxGraph::Ref*> Cluster::GetClusterRefs(
const TxGraphImpl& graph)
noexcept
1516 std::vector<TxGraph::Ref*>
ret;
1517 ret.reserve(m_linearization.size());
1519 for (
auto idx : m_linearization) {
1520 const auto& entry = graph.m_entries[m_mapping[idx]];
1521 Assume(entry.m_ref !=
nullptr);
1522 ret.push_back(entry.m_ref);
1532void Cluster::MakeStagingTransactionsMissing(TxGraphImpl& graph)
noexcept
1537 for (
auto ci : m_linearization) {
1538 GraphIndex idx = m_mapping[ci];
1539 auto& entry = graph.m_entries[idx];
1540 entry.m_locator[1].SetMissing();
1544std::vector<TxGraph::Ref*> TxGraphImpl::GetAncestors(
const Ref& arg,
bool main_only)
noexcept
1547 if (GetRefGraph(arg) ==
nullptr)
return {};
1548 Assume(GetRefGraph(arg) ==
this);
1550 size_t level = GetSpecifiedLevel(main_only);
1551 ApplyDependencies(level);
1553 Assume(GetClusterSet(level).m_deps_to_add.empty());
1555 auto cluster = FindCluster(GetRefIndex(arg), level);
1556 if (cluster ==
nullptr)
return {};
1558 std::pair<Cluster*, DepGraphIndex> match = {cluster, m_entries[GetRefIndex(arg)].m_locator[cluster->m_level].index};
1559 auto matches = std::span(&match, 1);
1560 std::vector<TxGraph::Ref*>
ret;
1561 cluster->GetAncestorRefs(*
this, matches,
ret);
1565std::vector<TxGraph::Ref*> TxGraphImpl::GetDescendants(
const Ref& arg,
bool main_only)
noexcept
1568 if (GetRefGraph(arg) ==
nullptr)
return {};
1569 Assume(GetRefGraph(arg) ==
this);
1571 size_t level = GetSpecifiedLevel(main_only);
1572 ApplyDependencies(level);
1574 Assume(GetClusterSet(level).m_deps_to_add.empty());
1576 auto cluster = FindCluster(GetRefIndex(arg), level);
1577 if (cluster ==
nullptr)
return {};
1579 std::pair<Cluster*, DepGraphIndex> match = {cluster, m_entries[GetRefIndex(arg)].m_locator[cluster->m_level].index};
1580 auto matches = std::span(&match, 1);
1581 std::vector<TxGraph::Ref*>
ret;
1582 cluster->GetDescendantRefs(*
this, matches,
ret);
1586std::vector<TxGraph::Ref*> TxGraphImpl::GetAncestorsUnion(std::span<const Ref* const>
args,
bool main_only)
noexcept
1589 size_t level = GetSpecifiedLevel(main_only);
1590 ApplyDependencies(level);
1592 Assume(GetClusterSet(level).m_deps_to_add.empty());
1595 std::vector<std::pair<Cluster*, DepGraphIndex>> matches;
1596 matches.reserve(
args.size());
1597 for (
auto arg :
args) {
1599 if (GetRefGraph(*arg) ==
nullptr)
continue;
1600 Assume(GetRefGraph(*arg) ==
this);
1602 auto cluster = FindCluster(GetRefIndex(*arg), level);
1603 if (cluster ==
nullptr)
continue;
1605 matches.emplace_back(cluster, m_entries[GetRefIndex(*arg)].m_locator[cluster->m_level].index);
1608 std::sort(matches.begin(), matches.end(), [](
auto& a,
auto& b)
noexcept { return std::less{}(a.first, b.first); });
1610 std::span match_span(matches);
1611 std::vector<TxGraph::Ref*>
ret;
1612 while (!match_span.empty()) {
1613 match_span.front().first->GetAncestorRefs(*
this, match_span,
ret);
1618std::vector<TxGraph::Ref*> TxGraphImpl::GetDescendantsUnion(std::span<const Ref* const>
args,
bool main_only)
noexcept
1621 size_t level = GetSpecifiedLevel(main_only);
1622 ApplyDependencies(level);
1624 Assume(GetClusterSet(level).m_deps_to_add.empty());
1627 std::vector<std::pair<Cluster*, DepGraphIndex>> matches;
1628 matches.reserve(
args.size());
1629 for (
auto arg :
args) {
1631 if (GetRefGraph(*arg) ==
nullptr)
continue;
1632 Assume(GetRefGraph(*arg) ==
this);
1634 auto cluster = FindCluster(GetRefIndex(*arg), level);
1635 if (cluster ==
nullptr)
continue;
1637 matches.emplace_back(cluster, m_entries[GetRefIndex(*arg)].m_locator[cluster->m_level].index);
1640 std::sort(matches.begin(), matches.end(), [](
auto& a,
auto& b)
noexcept { return std::less{}(a.first, b.first); });
1642 std::span match_span(matches);
1643 std::vector<TxGraph::Ref*>
ret;
1644 while (!match_span.empty()) {
1645 match_span.front().first->GetDescendantRefs(*
this, match_span,
ret);
1650std::vector<TxGraph::Ref*> TxGraphImpl::GetCluster(
const Ref& arg,
bool main_only)
noexcept
1654 if (GetRefGraph(arg) ==
nullptr)
return {};
1655 Assume(GetRefGraph(arg) ==
this);
1657 size_t level = GetSpecifiedLevel(main_only);
1658 ApplyDependencies(level);
1660 Assume(GetClusterSet(level).m_deps_to_add.empty());
1662 auto cluster = FindCluster(GetRefIndex(arg), level);
1663 if (cluster ==
nullptr)
return {};
1665 MakeAcceptable(*cluster);
1666 return cluster->GetClusterRefs(*
this);
1671 size_t level = GetSpecifiedLevel(main_only);
1672 ApplyRemovals(level);
1673 return GetClusterSet(level).m_txcount;
1676FeePerWeight TxGraphImpl::GetIndividualFeerate(
const Ref& arg)
noexcept
1679 if (GetRefGraph(arg) ==
nullptr)
return {};
1680 Assume(GetRefGraph(arg) ==
this);
1683 Cluster* cluster{
nullptr};
1684 for (
int level = 0; level <= GetTopLevel(); ++level) {
1687 ApplyRemovals(level);
1688 if (m_entries[GetRefIndex(arg)].m_locator[level].IsPresent()) {
1689 cluster = m_entries[GetRefIndex(arg)].m_locator[level].cluster;
1693 if (cluster ==
nullptr)
return {};
1695 return cluster->GetIndividualFeerate(m_entries[GetRefIndex(arg)].m_locator[cluster->m_level].index);
1698FeePerWeight TxGraphImpl::GetMainChunkFeerate(
const Ref& arg)
noexcept
1701 if (GetRefGraph(arg) ==
nullptr)
return {};
1702 Assume(GetRefGraph(arg) ==
this);
1704 ApplyDependencies(0);
1706 Assume(m_main_clusterset.m_deps_to_add.empty());
1708 auto cluster = FindCluster(GetRefIndex(arg), 0);
1709 if (cluster ==
nullptr)
return {};
1712 MakeAcceptable(*cluster);
1713 const auto& entry = m_entries[GetRefIndex(arg)];
1714 return entry.m_main_chunk_feerate;
1717bool TxGraphImpl::IsOversized(
bool main_only)
noexcept
1719 size_t level = GetSpecifiedLevel(main_only);
1720 auto& clusterset = GetClusterSet(level);
1721 if (clusterset.m_oversized.has_value()) {
1723 return *clusterset.m_oversized;
1727 GroupClusters(level);
1728 Assume(clusterset.m_group_data.has_value());
1729 clusterset.m_oversized = clusterset.m_group_data->m_group_oversized;
1730 return *clusterset.m_oversized;
1733void TxGraphImpl::StartStaging() noexcept
1736 Assume(!m_staging_clusterset.has_value());
1743 ApplyDependencies(0);
1745 m_staging_clusterset.emplace();
1748 m_staging_clusterset->m_txcount = m_main_clusterset.m_txcount;
1749 m_staging_clusterset->m_deps_to_add = m_main_clusterset.m_deps_to_add;
1750 m_staging_clusterset->m_group_data = m_main_clusterset.m_group_data;
1751 m_staging_clusterset->m_oversized = m_main_clusterset.m_oversized;
1752 Assume(m_staging_clusterset->m_oversized.has_value());
1755void TxGraphImpl::AbortStaging() noexcept
1758 Assume(m_staging_clusterset.has_value());
1761 for (
auto idx : m_staging_clusterset->m_removed) {
1762 m_entries[idx].m_locator[1].SetMissing();
1766 for (
auto& cluster : m_staging_clusterset->m_clusters[quality]) {
1767 cluster->MakeStagingTransactionsMissing(*
this);
1771 m_staging_clusterset.reset();
1773 if (!m_main_clusterset.m_group_data.has_value()) {
1776 m_main_clusterset.m_oversized = std::nullopt;
1780void TxGraphImpl::CommitStaging() noexcept
1783 Assume(m_staging_clusterset.has_value());
1786 auto conflicts = GetConflicts();
1787 for (Cluster* conflict : conflicts) {
1788 conflict->Clear(*
this);
1789 DeleteCluster(*conflict);
1793 for (
auto idx : m_staging_clusterset->m_removed) {
1794 m_entries[idx].m_locator[1].SetMissing();
1798 auto& stage_sets = m_staging_clusterset->m_clusters[quality];
1799 while (!stage_sets.empty()) {
1800 stage_sets.back()->MoveToMain(*
this);
1804 m_main_clusterset.m_deps_to_add = std::move(m_staging_clusterset->m_deps_to_add);
1805 m_main_clusterset.m_to_remove = std::move(m_staging_clusterset->m_to_remove);
1806 m_main_clusterset.m_group_data = std::move(m_staging_clusterset->m_group_data);
1807 m_main_clusterset.m_oversized = std::move(m_staging_clusterset->m_oversized);
1808 m_main_clusterset.m_txcount = std::move(m_staging_clusterset->m_txcount);
1810 m_staging_clusterset.reset();
1814void Cluster::SetFee(TxGraphImpl& graph,
DepGraphIndex idx, int64_t
fee)
noexcept
1823 if (!NeedsSplitting()) {
1824 graph.SetClusterQuality(m_level, m_quality, m_setindex, QualityLevel::NEEDS_RELINEARIZE);
1826 graph.SetClusterQuality(m_level, m_quality, m_setindex, QualityLevel::NEEDS_SPLIT);
1831void TxGraphImpl::SetTransactionFee(
const Ref& ref, int64_t
fee)
noexcept
1834 if (GetRefGraph(ref) ==
nullptr)
return;
1835 Assume(GetRefGraph(ref) ==
this);
1837 auto& entry = m_entries[GetRefIndex(ref)];
1838 for (
int level = 0; level < MAX_LEVELS; ++level) {
1839 auto& locator = entry.m_locator[level];
1840 if (locator.IsPresent()) {
1841 locator.cluster->SetFee(*
this, locator.index,
fee);
1846std::strong_ordering TxGraphImpl::CompareMainOrder(
const Ref& a,
const Ref& b)
noexcept
1849 Assume(GetRefGraph(a) ==
this);
1850 Assume(GetRefGraph(b) ==
this);
1852 ApplyDependencies(0);
1853 Assume(m_main_clusterset.m_deps_to_add.empty());
1855 const auto& entry_a = m_entries[GetRefIndex(a)];
1856 const auto& entry_b = m_entries[GetRefIndex(b)];
1857 const auto& locator_a = entry_a.m_locator[0];
1858 const auto& locator_b = entry_b.m_locator[0];
1859 Assume(locator_a.IsPresent());
1860 Assume(locator_b.IsPresent());
1861 MakeAcceptable(*locator_a.cluster);
1862 MakeAcceptable(*locator_b.cluster);
1864 auto feerate_cmp = FeeRateCompare(entry_b.m_main_chunk_feerate, entry_a.m_main_chunk_feerate);
1865 if (feerate_cmp < 0)
return std::strong_ordering::less;
1866 if (feerate_cmp > 0)
return std::strong_ordering::greater;
1868 if (locator_a.cluster != locator_b.cluster)
return locator_a.cluster <=> locator_b.cluster;
1870 return entry_a.m_main_lin_index <=> entry_b.m_main_lin_index;
1873TxGraph::GraphIndex TxGraphImpl::CountDistinctClusters(std::span<const Ref* const> refs,
bool main_only)
noexcept
1875 size_t level = GetSpecifiedLevel(main_only);
1876 ApplyDependencies(level);
1877 auto& clusterset = GetClusterSet(level);
1878 Assume(clusterset.m_deps_to_add.empty());
1880 std::vector<Cluster*> clusters;
1881 clusters.reserve(refs.size());
1882 for (
const Ref* ref : refs) {
1883 if (ref ==
nullptr)
continue;
1884 if (GetRefGraph(*ref) ==
nullptr)
continue;
1885 Assume(GetRefGraph(*ref) ==
this);
1886 auto cluster = FindCluster(GetRefIndex(*ref), level);
1887 if (cluster !=
nullptr) clusters.push_back(cluster);
1890 std::sort(clusters.begin(), clusters.end());
1891 Cluster* last{
nullptr};
1893 for (Cluster* cluster : clusters) {
1894 ret += (cluster != last);
1900void Cluster::SanityCheck(
const TxGraphImpl& graph,
int level)
const
1907 assert(m_linearization.size() <= graph.m_max_cluster_count);
1909 assert(m_level == level);
1917 LinearizationIndex linindex{0};
1919 for (
auto lin_pos : m_linearization) {
1920 assert(lin_pos < m_mapping.size());
1921 const auto& entry = graph.m_entries[m_mapping[lin_pos]];
1923 m_done.Set(lin_pos);
1926 assert(entry.m_locator[level].cluster ==
this);
1927 assert(entry.m_locator[level].index == lin_pos);
1929 if (level == 0 && IsAcceptable()) {
1930 assert(entry.m_main_lin_index == linindex);
1932 if (!linchunking.GetChunk(0).transactions[lin_pos]) {
1933 linchunking.MarkDone(linchunking.GetChunk(0).transactions);
1935 assert(entry.m_main_chunk_feerate == linchunking.GetChunk(0).feerate);
1944void TxGraphImpl::SanityCheck()
const
1947 std::set<GraphIndex> expected_unlinked;
1949 std::set<const Cluster*> expected_clusters[MAX_LEVELS];
1951 std::set<GraphIndex> expected_removed[MAX_LEVELS];
1953 bool compact_possible{
true};
1956 for (GraphIndex idx = 0; idx < m_entries.size(); ++idx) {
1957 const auto& entry = m_entries[idx];
1958 if (entry.m_ref ==
nullptr) {
1960 expected_unlinked.insert(idx);
1963 assert(GetRefGraph(*entry.m_ref) ==
this);
1964 assert(GetRefIndex(*entry.m_ref) == idx);
1967 bool was_present{
false}, was_removed{
false};
1968 for (
int level = 0; level < MAX_LEVELS; ++level) {
1969 const auto& locator = entry.m_locator[level];
1971 assert(locator.IsMissing() + locator.IsRemoved() + locator.IsPresent() == 1);
1972 if (locator.IsPresent()) {
1976 assert(locator.cluster->GetClusterEntry(locator.index) == idx);
1978 expected_clusters[level].insert(locator.cluster);
1980 }
else if (locator.IsRemoved()) {
1984 assert(was_present && !was_removed);
1986 expected_removed[level].insert(idx);
1993 for (
int level = 0; level <= GetTopLevel(); ++level) {
1994 assert(level < MAX_LEVELS);
1995 auto& clusterset = GetClusterSet(level);
1996 std::set<const Cluster*> actual_clusters;
2000 QualityLevel quality{qual};
2001 const auto& quality_clusters = clusterset.m_clusters[qual];
2003 for (ClusterSetIndex setindex = 0; setindex < quality_clusters.size(); ++setindex) {
2004 const auto& cluster = *quality_clusters[setindex];
2007 if (cluster.GetTxCount() != 0) {
2008 actual_clusters.insert(&cluster);
2011 cluster.SanityCheck(*
this, level);
2013 assert(cluster.m_quality == quality);
2014 assert(cluster.m_setindex == setindex);
2019 for (GraphIndex idx : clusterset.m_to_remove) {
2020 assert(idx < m_entries.size());
2028 for (
auto [par_idx, chl_idx] : clusterset.m_deps_to_add) {
2029 assert(par_idx != chl_idx);
2030 assert(par_idx < m_entries.size());
2031 assert(chl_idx < m_entries.size());
2035 assert(actual_clusters == expected_clusters[level]);
2038 std::set<GraphIndex> actual_removed(clusterset.m_removed.begin(), clusterset.m_removed.end());
2039 for (
auto i : expected_unlinked) {
2044 actual_removed.erase(i);
2045 expected_removed[level].erase(i);
2048 assert(actual_removed == expected_removed[level]);
2051 if (!clusterset.m_deps_to_add.empty()) compact_possible =
false;
2052 if (!clusterset.m_to_remove.empty()) compact_possible =
false;
2053 if (!clusterset.m_removed.empty()) compact_possible =
false;
2056 if (clusterset.m_group_data.has_value()) {
2057 assert(clusterset.m_oversized == clusterset.m_group_data->m_group_oversized);
2062 if (level < GetTopLevel())
assert(clusterset.m_oversized.has_value());
2066 std::set<GraphIndex> actual_unlinked(m_unlinked.begin(), m_unlinked.end());
2067 assert(actual_unlinked == expected_unlinked);
2071 if (compact_possible) {
2072 assert(actual_unlinked.empty());
2076void TxGraphImpl::DoWork() noexcept
2078 for (
int level = 0; level <= GetTopLevel(); ++level) {
2079 MakeAllAcceptable(level);
2097 if (m_graph) m_graph->UnlinkRef(m_index);
2101 m_graph = other.m_graph;
2102 m_index = other.m_index;
2103 other.m_graph =
nullptr;
2111 if (other.m_graph) other.m_graph->UpdateRef(other.m_index, *
this);
2113 std::swap(m_graph, other.m_graph);
2114 std::swap(m_index, other.m_index);
2117std::unique_ptr<TxGraph>
MakeTxGraph(
unsigned max_cluster_count)
noexcept
2119 return std::make_unique<TxGraphImpl>(max_cluster_count);
std::conditional_t<(BITS<=32), bitset_detail::IntBitSet< uint32_t >, std::conditional_t<(BITS<=std::numeric_limits< size_t >::digits), bitset_detail::IntBitSet< size_t >, bitset_detail::MultiIntBitSet< size_t,(BITS+std::numeric_limits< size_t >::digits - 1)/std::numeric_limits< size_t >::digits > > > BitSet
#define Assume(val)
Assume is the identity function.
TxGraph * m_graph
Which Graph the Entry lives in.
virtual ~Ref()
Destroy this Ref.
Ref & operator=(Ref &&other) noexcept
GraphIndex m_index
Index into the Graph's m_entries.
Ref() noexcept=default
Construct an empty Ref.
Data structure to encapsulate fees, sizes, and dependencies for a set of transactions.
virtual void UpdateRef(GraphIndex index, Ref &new_location) noexcept=0
Inform the TxGraph implementation that a TxGraph::Ref has moved.
uint32_t GraphIndex
Internal identifier for a transaction within a TxGraph.
virtual void UnlinkRef(GraphIndex index) noexcept=0
Inform the TxGraph implementation that a TxGraph::Ref was destroyed.
Data structure that holds a transaction graph's preprocessed data (fee, size, ancestors,...
const SetType & Ancestors(DepGraphIndex i) const noexcept
Get the ancestors of a given transaction i.
SetType GetReducedParents(DepGraphIndex i) const noexcept
Compute the (reduced) set of parents of node i in this graph.
const FeeFrac & FeeRate(DepGraphIndex i) const noexcept
Get the feerate of a given transaction i.
bool IsConnected(const SetType &subset) const noexcept
Determine if a subset is connected.
DepGraphIndex PositionRange() const noexcept
Get the range of positions in this DepGraph.
SetType FindConnectedComponent(const SetType &todo) const noexcept
Find some connected component within the subset "todo" of this graph.
DepGraphIndex AddTransaction(const FeeFrac &feefrac) noexcept
Add a new unconnected transaction to this transaction graph (in the first available position),...
void RemoveTransactions(const SetType &del) noexcept
Remove the specified positions from this DepGraph.
auto TxCount() const noexcept
Get the number of transactions in the graph.
const SetType & Descendants(DepGraphIndex i) const noexcept
Get the descendants of a given transaction i.
const SetType & Positions() const noexcept
Get the set of transactions positions in use.
void AddDependencies(const SetType &parents, DepGraphIndex child) noexcept
Modify this transaction graph, adding multiple parents to a specified child.
bool IsAcyclic() const noexcept
Check if this graph is acyclic.
Data structure encapsulating the chunking of a linearization, permitting removal of subsets.
std::pair< std::vector< DepGraphIndex >, bool > Linearize(const DepGraph< SetType > &depgraph, uint64_t max_iterations, uint64_t rng_seed, std::span< const DepGraphIndex > old_linearization={}) noexcept
Find or improve a linearization for a cluster.
void FixLinearization(const DepGraph< SetType > &depgraph, std::span< DepGraphIndex > linearization) noexcept
Make linearization topological, retaining its ordering where possible.
uint32_t DepGraphIndex
Data type to represent transaction indices in DepGraphs and the clusters they represent.
void PostLinearize(const DepGraph< SetType > &depgraph, std::span< DepGraphIndex > linearization)
Improve a given linearization.
static bool exists(const path &p)
std::vector< T > Split(const std::span< const char > &sp, std::string_view separators)
Split a string on any char found in separators, returning a vector.
Tagged wrapper around FeeFrac to avoid unit confusion.
static FeePerUnit FromFeeFrac(const FeeFrac &feefrac) noexcept
Convert a FeeFrac to a FeePerUnit.
std::unique_ptr< TxGraph > MakeTxGraph(unsigned max_cluster_count) noexcept
Construct a new TxGraph with the specified limit on transactions within a cluster.
static constexpr unsigned MAX_CLUSTER_COUNT_LIMIT