/root/bitcoin/src/validationinterface.cpp
Line | Count | Source |
1 | | // Copyright (c) 2009-2010 Satoshi Nakamoto |
2 | | // Copyright (c) 2009-present The Bitcoin Core developers |
3 | | // Distributed under the MIT software license, see the accompanying |
4 | | // file COPYING or http://www.opensource.org/licenses/mit-license.php. |
5 | | |
6 | | #include <validationinterface.h> |
7 | | |
8 | | #include <chain.h> |
9 | | #include <consensus/validation.h> |
10 | | #include <kernel/mempool_entry.h> |
11 | | #include <kernel/mempool_removal_reason.h> |
12 | | #include <kernel/types.h> |
13 | | #include <primitives/block.h> |
14 | | #include <primitives/transaction.h> |
15 | | #include <util/check.h> |
16 | | #include <util/log.h> |
17 | | #include <util/task_runner.h> |
18 | | |
19 | | #include <future> |
20 | | #include <memory> |
21 | | #include <unordered_map> |
22 | | #include <utility> |
23 | | |
24 | | using kernel::ChainstateRole; |
25 | | |
26 | | /** |
27 | | * ValidationSignalsImpl manages a list of shared_ptr<CValidationInterface> callbacks. |
28 | | * |
29 | | * A std::unordered_map is used to track what callbacks are currently |
30 | | * registered, and a std::list is used to store the callbacks that are |
31 | | * currently registered as well as any callbacks that are just unregistered |
32 | | * and about to be deleted when they are done executing. |
33 | | */ |
34 | | class ValidationSignalsImpl |
35 | | { |
36 | | private: |
37 | | Mutex m_mutex; |
38 | | //! List entries consist of a callback pointer and reference count. The |
39 | | //! count is equal to the number of current executions of that entry, plus 1 |
40 | | //! if it's registered. It cannot be 0 because that would imply it is |
41 | | //! unregistered and also not being executed (so shouldn't exist). |
42 | | struct ListEntry { std::shared_ptr<CValidationInterface> callbacks; int count = 1; }; |
43 | | std::list<ListEntry> m_list GUARDED_BY(m_mutex); |
44 | | std::unordered_map<CValidationInterface*, std::list<ListEntry>::iterator> m_map GUARDED_BY(m_mutex); |
45 | | |
46 | | public: |
47 | | std::unique_ptr<util::TaskRunnerInterface> m_task_runner; |
48 | | |
49 | | explicit ValidationSignalsImpl(std::unique_ptr<util::TaskRunnerInterface> task_runner) |
50 | 0 | : m_task_runner{std::move(Assert(task_runner))} {}Line | Count | Source | 113 | 0 | #define Assert(val) inline_assertion_check<true>(val, std::source_location::current(), #val) |
|
51 | | |
52 | | void Register(std::shared_ptr<CValidationInterface> callbacks) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex) |
53 | 0 | { |
54 | 0 | LOCK(m_mutex); Line | Count | Source | 266 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
|
55 | 0 | auto inserted = m_map.emplace(callbacks.get(), m_list.end()); |
56 | 0 | if (inserted.second) inserted.first->second = m_list.emplace(m_list.end()); |
57 | 0 | inserted.first->second->callbacks = std::move(callbacks); |
58 | 0 | } |
59 | | |
60 | | void Unregister(CValidationInterface* callbacks) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex) |
61 | 0 | { |
62 | 0 | LOCK(m_mutex); Line | Count | Source | 266 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
|
63 | 0 | auto it = m_map.find(callbacks); |
64 | 0 | if (it != m_map.end()) { |
65 | 0 | if (!--it->second->count) m_list.erase(it->second); |
66 | 0 | m_map.erase(it); |
67 | 0 | } |
68 | 0 | } |
69 | | |
70 | | //! Clear unregisters every previously registered callback, erasing every |
71 | | //! map entry. After this call, the list may still contain callbacks that |
72 | | //! are currently executing, but it will be cleared when they are done |
73 | | //! executing. |
74 | | void Clear() EXCLUSIVE_LOCKS_REQUIRED(!m_mutex) |
75 | 0 | { |
76 | 0 | LOCK(m_mutex); Line | Count | Source | 266 | 0 | #define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
|
77 | 0 | for (const auto& entry : m_map) { |
78 | 0 | if (!--entry.second->count) m_list.erase(entry.second); |
79 | 0 | } |
80 | 0 | m_map.clear(); |
81 | 0 | } |
82 | | |
83 | | template<typename F> void Iterate(F&& f) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex) |
84 | 0 | { |
85 | 0 | WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
| WAIT_LOCK(m_mutex, lock); Line | Count | Source | 272 | 0 | #define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs)) Line | Count | Source | 270 | 0 | #define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__ |
|
|
86 | 0 | for (auto it = m_list.begin(); it != m_list.end();) { |
87 | 0 | ++it->count; |
88 | 0 | { |
89 | 0 | REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
| REVERSE_LOCK(lock, m_mutex); Line | Count | Source | 252 | 0 | #define REVERSE_LOCK(g, cs) typename std::decay<decltype(g)>::type::reverse_lock UNIQUE_NAME(revlock)(g, cs, #cs, __FILE__, __LINE__) Line | Count | Source | 11 | 0 | #define UNIQUE_NAME(name) PASTE2(name, __COUNTER__) Line | Count | Source | 9 | 0 | #define PASTE2(x, y) PASTE(x, y) Line | Count | Source | 8 | 0 | #define PASTE(x, y) x ## y |
|
|
|
|
90 | 0 | f(*it->callbacks); |
91 | 0 | } |
92 | 0 | it = --it->count ? std::next(it) : m_list.erase(it); |
93 | 0 | } |
94 | 0 | } Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::UpdatedBlockTip(CBlockIndex const*, CBlockIndex const*, bool)::$_0::operator()() const::'lambda'(CValidationInterface&)>(ValidationSignals::UpdatedBlockTip(CBlockIndex const*, CBlockIndex const*, bool)::$_0::operator()() const::'lambda'(CValidationInterface&)&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::ActiveTipChange(CBlockIndex const&, bool)::$_0>(ValidationSignals::ActiveTipChange(CBlockIndex const&, bool)::$_0&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::TransactionAddedToMempool(NewMempoolTransactionInfo const&, unsigned long)::$_0::operator()() const::'lambda'(CValidationInterface&)>(ValidationSignals::TransactionAddedToMempool(NewMempoolTransactionInfo const&, unsigned long)::$_0::operator()() const::'lambda'(CValidationInterface&)&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::TransactionRemovedFromMempool(std::shared_ptr<CTransaction const> const&, MemPoolRemovalReason, unsigned long)::$_0::operator()() const::'lambda'(CValidationInterface&)>(ValidationSignals::TransactionRemovedFromMempool(std::shared_ptr<CTransaction const> const&, MemPoolRemovalReason, unsigned long)::$_0::operator()() const::'lambda'(CValidationInterface&)&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::BlockConnected(kernel::ChainstateRole const&, std::shared_ptr<CBlock const>, CBlockIndex const*)::$_0::operator()() const::'lambda'(CValidationInterface&)>(ValidationSignals::BlockConnected(kernel::ChainstateRole const&, std::shared_ptr<CBlock const>, CBlockIndex const*)::$_0::operator()() const::'lambda'(CValidationInterface&)&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::MempoolTransactionsRemovedForBlock(std::vector<RemovedMempoolTransactionInfo, std::allocator<RemovedMempoolTransactionInfo> > const&, unsigned int)::$_0::operator()() const::'lambda'(CValidationInterface&)>(ValidationSignals::MempoolTransactionsRemovedForBlock(std::vector<RemovedMempoolTransactionInfo, std::allocator<RemovedMempoolTransactionInfo> > const&, unsigned int)::$_0::operator()() const::'lambda'(CValidationInterface&)&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::BlockDisconnected(std::shared_ptr<CBlock const>, CBlockIndex const*)::$_0::operator()() const::'lambda'(CValidationInterface&)>(ValidationSignals::BlockDisconnected(std::shared_ptr<CBlock const>, CBlockIndex const*)::$_0::operator()() const::'lambda'(CValidationInterface&)&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::ChainStateFlushed(kernel::ChainstateRole const&, CBlockLocator const&)::$_0::operator()() const::'lambda'(CValidationInterface&)>(ValidationSignals::ChainStateFlushed(kernel::ChainstateRole const&, CBlockLocator const&)::$_0::operator()() const::'lambda'(CValidationInterface&)&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::BlockChecked(std::shared_ptr<CBlock const> const&, BlockValidationState const&)::$_0>(ValidationSignals::BlockChecked(std::shared_ptr<CBlock const> const&, BlockValidationState const&)::$_0&&) Unexecuted instantiation: validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::NewPoWValidBlock(CBlockIndex const*, std::shared_ptr<CBlock const> const&)::$_0>(ValidationSignals::NewPoWValidBlock(CBlockIndex const*, std::shared_ptr<CBlock const> const&)::$_0&&) |
95 | | }; |
96 | | |
97 | | ValidationSignals::ValidationSignals(std::unique_ptr<util::TaskRunnerInterface> task_runner) |
98 | 0 | : m_internals{std::make_unique<ValidationSignalsImpl>(std::move(task_runner))} {} |
99 | | |
100 | 0 | ValidationSignals::~ValidationSignals() = default; |
101 | | |
102 | | void ValidationSignals::FlushBackgroundCallbacks() |
103 | 0 | { |
104 | 0 | m_internals->m_task_runner->flush(); |
105 | 0 | } |
106 | | |
107 | | size_t ValidationSignals::CallbacksPending() |
108 | 0 | { |
109 | 0 | return m_internals->m_task_runner->size(); |
110 | 0 | } |
111 | | |
112 | | void ValidationSignals::RegisterSharedValidationInterface(std::shared_ptr<CValidationInterface> callbacks) |
113 | 0 | { |
114 | | // Each connection captures the shared_ptr to ensure that each callback is |
115 | | // executed before the subscriber is destroyed. For more details see #18338. |
116 | 0 | m_internals->Register(std::move(callbacks)); |
117 | 0 | } |
118 | | |
119 | | void ValidationSignals::RegisterValidationInterface(CValidationInterface* callbacks) |
120 | 0 | { |
121 | | // Create a shared_ptr with a no-op deleter - CValidationInterface lifecycle |
122 | | // is managed by the caller. |
123 | 0 | RegisterSharedValidationInterface({callbacks, [](CValidationInterface*){}}); |
124 | 0 | } |
125 | | |
126 | | void ValidationSignals::UnregisterSharedValidationInterface(std::shared_ptr<CValidationInterface> callbacks) |
127 | 0 | { |
128 | 0 | UnregisterValidationInterface(callbacks.get()); |
129 | 0 | } |
130 | | |
131 | | void ValidationSignals::UnregisterValidationInterface(CValidationInterface* callbacks) |
132 | 0 | { |
133 | 0 | m_internals->Unregister(callbacks); |
134 | 0 | } |
135 | | |
136 | | void ValidationSignals::UnregisterAllValidationInterfaces() |
137 | 0 | { |
138 | 0 | m_internals->Clear(); |
139 | 0 | } |
140 | | |
141 | | void ValidationSignals::CallFunctionInValidationInterfaceQueue(std::function<void()> func) |
142 | 0 | { |
143 | 0 | m_internals->m_task_runner->insert(std::move(func)); |
144 | 0 | } |
145 | | |
146 | | void ValidationSignals::SyncWithValidationInterfaceQueue() |
147 | 0 | { |
148 | 0 | AssertLockNotHeld(cs_main); Line | Count | Source | 147 | 0 | #define AssertLockNotHeld(cs) AssertLockNotHeldInline(#cs, __FILE__, __LINE__, &cs) |
|
149 | | // Block until the validation queue drains |
150 | 0 | std::promise<void> promise; |
151 | 0 | CallFunctionInValidationInterfaceQueue([&promise] { |
152 | 0 | promise.set_value(); |
153 | 0 | }); |
154 | 0 | promise.get_future().wait(); |
155 | 0 | } |
156 | | |
157 | | // Use a macro instead of a function for conditional logging to prevent |
158 | | // evaluating arguments when logging is not enabled. |
159 | | #define ENQUEUE_AND_LOG_EVENT(event, log_msg) \ |
160 | 0 | do { \ |
161 | 0 | static_assert(std::is_rvalue_reference_v<decltype((event))>, \ |
162 | 0 | "event must be passed as an rvalue"); \ |
163 | 0 | static_assert(std::is_rvalue_reference_v<decltype((log_msg))>, \ |
164 | 0 | "log_msg must be passed as an rvalue"); \ |
165 | 0 | auto enqueue_log_msg = (log_msg); \ |
166 | 0 | LOG_EVENT("Enqueuing %s", enqueue_log_msg); \ |
167 | 0 | m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \ |
168 | 0 | LOG_EVENT("%s", local_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| LOG_EVENT("%s", local_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| LOG_EVENT("%s", local_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| LOG_EVENT("%s", local_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| LOG_EVENT("%s", local_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| LOG_EVENT("%s", local_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| LOG_EVENT("%s", local_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
|
169 | 0 | local_event(); \ |
170 | 0 | }); \ Unexecuted instantiation: validationinterface.cpp:ValidationSignals::UpdatedBlockTip(CBlockIndex const*, CBlockIndex const*, bool)::$_1::operator()() const Unexecuted instantiation: validationinterface.cpp:ValidationSignals::TransactionAddedToMempool(NewMempoolTransactionInfo const&, unsigned long)::$_1::operator()() const Unexecuted instantiation: validationinterface.cpp:ValidationSignals::TransactionRemovedFromMempool(std::shared_ptr<CTransaction const> const&, MemPoolRemovalReason, unsigned long)::$_1::operator()() const Unexecuted instantiation: validationinterface.cpp:ValidationSignals::BlockConnected(kernel::ChainstateRole const&, std::shared_ptr<CBlock const>, CBlockIndex const*)::$_1::operator()() const Unexecuted instantiation: validationinterface.cpp:ValidationSignals::MempoolTransactionsRemovedForBlock(std::vector<RemovedMempoolTransactionInfo, std::allocator<RemovedMempoolTransactionInfo> > const&, unsigned int)::$_1::operator()() const Unexecuted instantiation: validationinterface.cpp:ValidationSignals::BlockDisconnected(std::shared_ptr<CBlock const>, CBlockIndex const*)::$_1::operator()() const Unexecuted instantiation: validationinterface.cpp:ValidationSignals::ChainStateFlushed(kernel::ChainstateRole const&, CBlockLocator const&)::$_1::operator()() const |
171 | 0 | } while (0) |
172 | | |
173 | | #define LOG_MSG(fmt, ...) \ |
174 | 0 | (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? tfm::format((fmt), __VA_ARGS__) : std::string{}) |
175 | | |
176 | | #define LOG_EVENT(fmt, ...) \ |
177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) |
178 | | |
179 | 0 | void ValidationSignals::UpdatedBlockTip(const CBlockIndex *pindexNew, const CBlockIndex *pindexFork, bool fInitialDownload) { |
180 | | // Dependencies exist that require UpdatedBlockTip events to be delivered in the order in which |
181 | | // the chain actually updates. One way to ensure this is for the caller to invoke this signal |
182 | | // in the same critical section where the chain is updated |
183 | |
|
184 | 0 | auto log_msg = LOG_MSG("%s: new block hash=%s fork block hash=%s (in IBD=%s)", __func__,Line | Count | Source | 174 | 0 | (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? tfm::format((fmt), __VA_ARGS__) : std::string{}) |
|
185 | 0 | pindexNew->GetBlockHash().ToString(), |
186 | 0 | pindexFork ? pindexFork->GetBlockHash().ToString() : "null", |
187 | 0 | fInitialDownload); |
188 | 0 | auto event = [pindexNew, pindexFork, fInitialDownload, this] { |
189 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.UpdatedBlockTip(pindexNew, pindexFork, fInitialDownload); }); |
190 | 0 | }; |
191 | 0 | ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg)); Line | Count | Source | 160 | 0 | do { \ | 161 | 0 | static_assert(std::is_rvalue_reference_v<decltype((event))>, \ | 162 | 0 | "event must be passed as an rvalue"); \ | 163 | 0 | static_assert(std::is_rvalue_reference_v<decltype((log_msg))>, \ | 164 | 0 | "log_msg must be passed as an rvalue"); \ | 165 | 0 | auto enqueue_log_msg = (log_msg); \ | 166 | 0 | LOG_EVENT("Enqueuing %s", enqueue_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| 167 | 0 | m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \ | 168 | 0 | LOG_EVENT("%s", local_log_msg); \ | 169 | 0 | local_event(); \ | 170 | 0 | }); \ | 171 | 0 | } while (0) |
|
192 | 0 | } |
193 | | |
194 | | void ValidationSignals::ActiveTipChange(const CBlockIndex& new_tip, bool is_ibd) |
195 | 0 | { |
196 | 0 | LOG_EVENT("%s: new block hash=%s block height=%d", __func__, new_tip.GetBlockHash().ToString(), new_tip.nHeight);Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
|
197 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.ActiveTipChange(new_tip, is_ibd); }); |
198 | 0 | } |
199 | | |
200 | | void ValidationSignals::TransactionAddedToMempool(const NewMempoolTransactionInfo& tx, uint64_t mempool_sequence) |
201 | 0 | { |
202 | 0 | auto log_msg = LOG_MSG("%s: txid=%s wtxid=%s", __func__,Line | Count | Source | 174 | 0 | (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? tfm::format((fmt), __VA_ARGS__) : std::string{}) |
|
203 | 0 | tx.info.m_tx->GetHash().ToString(), |
204 | 0 | tx.info.m_tx->GetWitnessHash().ToString()); |
205 | 0 | auto event = [tx, mempool_sequence, this] { |
206 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.TransactionAddedToMempool(tx, mempool_sequence); }); |
207 | 0 | }; |
208 | 0 | ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg)); Line | Count | Source | 160 | 0 | do { \ | 161 | 0 | static_assert(std::is_rvalue_reference_v<decltype((event))>, \ | 162 | 0 | "event must be passed as an rvalue"); \ | 163 | 0 | static_assert(std::is_rvalue_reference_v<decltype((log_msg))>, \ | 164 | 0 | "log_msg must be passed as an rvalue"); \ | 165 | 0 | auto enqueue_log_msg = (log_msg); \ | 166 | 0 | LOG_EVENT("Enqueuing %s", enqueue_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| 167 | 0 | m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \ | 168 | 0 | LOG_EVENT("%s", local_log_msg); \ | 169 | 0 | local_event(); \ | 170 | 0 | }); \ | 171 | 0 | } while (0) |
|
209 | 0 | } |
210 | | |
211 | 0 | void ValidationSignals::TransactionRemovedFromMempool(const CTransactionRef& tx, MemPoolRemovalReason reason, uint64_t mempool_sequence) { |
212 | 0 | auto log_msg = LOG_MSG("%s: txid=%s wtxid=%s reason=%s", __func__,Line | Count | Source | 174 | 0 | (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? tfm::format((fmt), __VA_ARGS__) : std::string{}) |
|
213 | 0 | tx->GetHash().ToString(), |
214 | 0 | tx->GetWitnessHash().ToString(), |
215 | 0 | RemovalReasonToString(reason)); |
216 | 0 | auto event = [tx, reason, mempool_sequence, this] { |
217 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.TransactionRemovedFromMempool(tx, reason, mempool_sequence); }); |
218 | 0 | }; |
219 | 0 | ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg)); Line | Count | Source | 160 | 0 | do { \ | 161 | 0 | static_assert(std::is_rvalue_reference_v<decltype((event))>, \ | 162 | 0 | "event must be passed as an rvalue"); \ | 163 | 0 | static_assert(std::is_rvalue_reference_v<decltype((log_msg))>, \ | 164 | 0 | "log_msg must be passed as an rvalue"); \ | 165 | 0 | auto enqueue_log_msg = (log_msg); \ | 166 | 0 | LOG_EVENT("Enqueuing %s", enqueue_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| 167 | 0 | m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \ | 168 | 0 | LOG_EVENT("%s", local_log_msg); \ | 169 | 0 | local_event(); \ | 170 | 0 | }); \ | 171 | 0 | } while (0) |
|
220 | 0 | } |
221 | | |
222 | | void ValidationSignals::BlockConnected(const ChainstateRole& role, std::shared_ptr<const CBlock> pblock, const CBlockIndex* pindex) |
223 | 0 | { |
224 | 0 | auto log_msg = LOG_MSG("%s: block hash=%s block height=%d", __func__,Line | Count | Source | 174 | 0 | (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? tfm::format((fmt), __VA_ARGS__) : std::string{}) |
|
225 | 0 | pblock->GetHash().ToString(), |
226 | 0 | pindex->nHeight); |
227 | 0 | auto event = [role, pblock = std::move(pblock), pindex, this] { |
228 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockConnected(role, pblock, pindex); }); |
229 | 0 | }; |
230 | 0 | ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg)); Line | Count | Source | 160 | 0 | do { \ | 161 | 0 | static_assert(std::is_rvalue_reference_v<decltype((event))>, \ | 162 | 0 | "event must be passed as an rvalue"); \ | 163 | 0 | static_assert(std::is_rvalue_reference_v<decltype((log_msg))>, \ | 164 | 0 | "log_msg must be passed as an rvalue"); \ | 165 | 0 | auto enqueue_log_msg = (log_msg); \ | 166 | 0 | LOG_EVENT("Enqueuing %s", enqueue_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| 167 | 0 | m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \ | 168 | 0 | LOG_EVENT("%s", local_log_msg); \ | 169 | 0 | local_event(); \ | 170 | 0 | }); \ | 171 | 0 | } while (0) |
|
231 | 0 | } |
232 | | |
233 | | void ValidationSignals::MempoolTransactionsRemovedForBlock(const std::vector<RemovedMempoolTransactionInfo>& txs_removed_for_block, unsigned int nBlockHeight) |
234 | 0 | { |
235 | 0 | auto log_msg = LOG_MSG("%s: block height=%s txs removed=%s", __func__,Line | Count | Source | 174 | 0 | (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? tfm::format((fmt), __VA_ARGS__) : std::string{}) |
|
236 | 0 | nBlockHeight, |
237 | 0 | txs_removed_for_block.size()); |
238 | 0 | auto event = [txs_removed_for_block, nBlockHeight, this] { |
239 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.MempoolTransactionsRemovedForBlock(txs_removed_for_block, nBlockHeight); }); |
240 | 0 | }; |
241 | 0 | ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg)); Line | Count | Source | 160 | 0 | do { \ | 161 | 0 | static_assert(std::is_rvalue_reference_v<decltype((event))>, \ | 162 | 0 | "event must be passed as an rvalue"); \ | 163 | 0 | static_assert(std::is_rvalue_reference_v<decltype((log_msg))>, \ | 164 | 0 | "log_msg must be passed as an rvalue"); \ | 165 | 0 | auto enqueue_log_msg = (log_msg); \ | 166 | 0 | LOG_EVENT("Enqueuing %s", enqueue_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| 167 | 0 | m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \ | 168 | 0 | LOG_EVENT("%s", local_log_msg); \ | 169 | 0 | local_event(); \ | 170 | 0 | }); \ | 171 | 0 | } while (0) |
|
242 | 0 | } |
243 | | |
244 | | void ValidationSignals::BlockDisconnected(std::shared_ptr<const CBlock> pblock, const CBlockIndex* pindex) |
245 | 0 | { |
246 | 0 | auto log_msg = LOG_MSG("%s: block hash=%s block height=%d", __func__,Line | Count | Source | 174 | 0 | (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? tfm::format((fmt), __VA_ARGS__) : std::string{}) |
|
247 | 0 | pblock->GetHash().ToString(), |
248 | 0 | pindex->nHeight); |
249 | 0 | auto event = [pblock = std::move(pblock), pindex, this] { |
250 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockDisconnected(pblock, pindex); }); |
251 | 0 | }; |
252 | 0 | ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg)); Line | Count | Source | 160 | 0 | do { \ | 161 | 0 | static_assert(std::is_rvalue_reference_v<decltype((event))>, \ | 162 | 0 | "event must be passed as an rvalue"); \ | 163 | 0 | static_assert(std::is_rvalue_reference_v<decltype((log_msg))>, \ | 164 | 0 | "log_msg must be passed as an rvalue"); \ | 165 | 0 | auto enqueue_log_msg = (log_msg); \ | 166 | 0 | LOG_EVENT("Enqueuing %s", enqueue_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| 167 | 0 | m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \ | 168 | 0 | LOG_EVENT("%s", local_log_msg); \ | 169 | 0 | local_event(); \ | 170 | 0 | }); \ | 171 | 0 | } while (0) |
|
253 | 0 | } |
254 | | |
255 | | void ValidationSignals::ChainStateFlushed(const ChainstateRole& role, const CBlockLocator& locator) |
256 | 0 | { |
257 | 0 | auto log_msg = LOG_MSG("%s: block hash=%s", __func__,Line | Count | Source | 174 | 0 | (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? tfm::format((fmt), __VA_ARGS__) : std::string{}) |
|
258 | 0 | locator.IsNull() ? "null" : locator.vHave.front().ToString()); |
259 | 0 | auto event = [role, locator, this] { |
260 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.ChainStateFlushed(role, locator); }); |
261 | 0 | }; |
262 | 0 | ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg)); Line | Count | Source | 160 | 0 | do { \ | 161 | 0 | static_assert(std::is_rvalue_reference_v<decltype((event))>, \ | 162 | 0 | "event must be passed as an rvalue"); \ | 163 | 0 | static_assert(std::is_rvalue_reference_v<decltype((log_msg))>, \ | 164 | 0 | "log_msg must be passed as an rvalue"); \ | 165 | 0 | auto enqueue_log_msg = (log_msg); \ | 166 | 0 | LOG_EVENT("Enqueuing %s", enqueue_log_msg); \Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
| 167 | 0 | m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \ | 168 | 0 | LOG_EVENT("%s", local_log_msg); \ | 169 | 0 | local_event(); \ | 170 | 0 | }); \ | 171 | 0 | } while (0) |
|
263 | 0 | } |
264 | | |
265 | | void ValidationSignals::BlockChecked(const std::shared_ptr<const CBlock>& block, const BlockValidationState& state) |
266 | 0 | { |
267 | 0 | LOG_EVENT("%s: block hash=%s state=%s", __func__,Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
|
268 | 0 | block->GetHash().ToString(), state.ToString()); |
269 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockChecked(block, state); }); |
270 | 0 | } |
271 | | |
272 | 0 | void ValidationSignals::NewPoWValidBlock(const CBlockIndex *pindex, const std::shared_ptr<const CBlock> &block) { |
273 | 0 | LOG_EVENT("%s: block hash=%s", __func__, block->GetHash().ToString());Line | Count | Source | 177 | 0 | LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__) Line | Count | Source | 115 | 0 | #define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__) Line | Count | Source | 106 | 0 | do { \ | 107 | 0 | if (util::log::ShouldLog((category), (level))) { \ | 108 | 0 | bool rate_limit{level >= BCLog::Level::Info}; \ | 109 | 0 | Assume(!rate_limit); /*Only called with the levels below*/ \ Line | Count | Source | 125 | 0 | #define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val) |
| 110 | 0 | LogPrintLevel_(category, level, rate_limit, __VA_ARGS__); \ Line | Count | Source | 89 | 0 | #define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__) |
| 111 | 0 | } \ | 112 | 0 | } while (0) |
|
|
|
274 | 0 | m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.NewPoWValidBlock(pindex, block); }); |
275 | 0 | } |