Coverage Report

Created: 2026-06-16 16:41

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/tmp/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
1.29k
        : m_task_runner{std::move(Assert(task_runner))} {}
51
52
    void Register(std::shared_ptr<CValidationInterface> callbacks) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
53
259k
    {
54
259k
        LOCK(m_mutex);
55
259k
        auto inserted = m_map.emplace(callbacks.get(), m_list.end());
56
259k
        if (inserted.second) inserted.first->second = m_list.emplace(m_list.end());
57
259k
        inserted.first->second->callbacks = std::move(callbacks);
58
259k
    }
59
60
    void Unregister(CValidationInterface* callbacks) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
61
260k
    {
62
260k
        LOCK(m_mutex);
63
260k
        auto it = m_map.find(callbacks);
64
260k
        if (it != m_map.end()) {
65
259k
            if (!--it->second->count) m_list.erase(it->second);
66
259k
            m_map.erase(it);
67
259k
        }
68
260k
    }
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
1.13k
    {
76
1.13k
        LOCK(m_mutex);
77
1.13k
        for (const auto& entry : m_map) {
78
1
            if (!--entry.second->count) m_list.erase(entry.second);
79
1
        }
80
1.13k
        m_map.clear();
81
1.13k
    }
82
83
    template<typename F> void Iterate(F&& f) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
84
701k
    {
85
701k
        WAIT_LOCK(m_mutex, lock);
86
2.20M
        for (auto it = m_list.begin(); it != m_list.end();) {
87
1.50M
            ++it->count;
88
1.50M
            {
89
1.50M
                REVERSE_LOCK(lock, m_mutex);
90
1.50M
                f(*it->callbacks);
91
1.50M
            }
92
1.50M
            it = --it->count ? std::next(it) : m_list.erase(it);
93
1.50M
        }
94
701k
    }
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&)&&)
Line
Count
Source
84
94.2k
    {
85
94.2k
        WAIT_LOCK(m_mutex, lock);
86
309k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
215k
            ++it->count;
88
215k
            {
89
215k
                REVERSE_LOCK(lock, m_mutex);
90
215k
                f(*it->callbacks);
91
215k
            }
92
215k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
215k
        }
94
94.2k
    }
validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::ActiveTipChange(CBlockIndex const&, bool)::$_0>(ValidationSignals::ActiveTipChange(CBlockIndex const&, bool)::$_0&&)
Line
Count
Source
84
97.1k
    {
85
97.1k
        WAIT_LOCK(m_mutex, lock);
86
325k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
228k
            ++it->count;
88
228k
            {
89
228k
                REVERSE_LOCK(lock, m_mutex);
90
228k
                f(*it->callbacks);
91
228k
            }
92
228k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
228k
        }
94
97.1k
    }
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&)&&)
Line
Count
Source
84
50.2k
    {
85
50.2k
        WAIT_LOCK(m_mutex, lock);
86
133k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
83.4k
            ++it->count;
88
83.4k
            {
89
83.4k
                REVERSE_LOCK(lock, m_mutex);
90
83.4k
                f(*it->callbacks);
91
83.4k
            }
92
83.4k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
83.4k
        }
94
50.2k
    }
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&)&&)
Line
Count
Source
84
2.02k
    {
85
2.02k
        WAIT_LOCK(m_mutex, lock);
86
6.17k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
4.14k
            ++it->count;
88
4.14k
            {
89
4.14k
                REVERSE_LOCK(lock, m_mutex);
90
4.14k
                f(*it->callbacks);
91
4.14k
            }
92
4.14k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
4.14k
        }
94
2.02k
    }
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&)&&)
Line
Count
Source
84
107k
    {
85
107k
        WAIT_LOCK(m_mutex, lock);
86
349k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
242k
            ++it->count;
88
242k
            {
89
242k
                REVERSE_LOCK(lock, m_mutex);
90
242k
                f(*it->callbacks);
91
242k
            }
92
242k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
242k
        }
94
107k
    }
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&)&&)
Line
Count
Source
84
106k
    {
85
106k
        WAIT_LOCK(m_mutex, lock);
86
347k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
240k
            ++it->count;
88
240k
            {
89
240k
                REVERSE_LOCK(lock, m_mutex);
90
240k
                f(*it->callbacks);
91
240k
            }
92
240k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
240k
        }
94
106k
    }
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&)&&)
Line
Count
Source
84
14.0k
    {
85
14.0k
        WAIT_LOCK(m_mutex, lock);
86
42.5k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
28.5k
            ++it->count;
88
28.5k
            {
89
28.5k
                REVERSE_LOCK(lock, m_mutex);
90
28.5k
                f(*it->callbacks);
91
28.5k
            }
92
28.5k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
28.5k
        }
94
14.0k
    }
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&)&&)
Line
Count
Source
84
2.37k
    {
85
2.37k
        WAIT_LOCK(m_mutex, lock);
86
5.23k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
2.85k
            ++it->count;
88
2.85k
            {
89
2.85k
                REVERSE_LOCK(lock, m_mutex);
90
2.85k
                f(*it->callbacks);
91
2.85k
            }
92
2.85k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
2.85k
        }
94
2.37k
    }
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&&)
Line
Count
Source
84
147k
    {
85
147k
        WAIT_LOCK(m_mutex, lock);
86
422k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
274k
            ++it->count;
88
274k
            {
89
274k
                REVERSE_LOCK(lock, m_mutex);
90
274k
                f(*it->callbacks);
91
274k
            }
92
274k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
274k
        }
94
147k
    }
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&&)
Line
Count
Source
84
79.7k
    {
85
79.7k
        WAIT_LOCK(m_mutex, lock);
86
266k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
186k
            ++it->count;
88
186k
            {
89
186k
                REVERSE_LOCK(lock, m_mutex);
90
186k
                f(*it->callbacks);
91
186k
            }
92
186k
            it = --it->count ? std::next(it) : m_list.erase(it);
93
186k
        }
94
79.7k
    }
95
};
96
97
ValidationSignals::ValidationSignals(std::unique_ptr<util::TaskRunnerInterface> task_runner)
98
1.29k
    : m_internals{std::make_unique<ValidationSignalsImpl>(std::move(task_runner))} {}
99
100
1.29k
ValidationSignals::~ValidationSignals() = default;
101
102
void ValidationSignals::FlushBackgroundCallbacks()
103
1.29k
{
104
1.29k
    m_internals->m_task_runner->flush();
105
1.29k
}
106
107
size_t ValidationSignals::CallbacksPending()
108
129k
{
109
129k
    return m_internals->m_task_runner->size();
110
129k
}
111
112
void ValidationSignals::RegisterSharedValidationInterface(std::shared_ptr<CValidationInterface> callbacks)
113
259k
{
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
259k
    m_internals->Register(std::move(callbacks));
117
259k
}
118
119
void ValidationSignals::RegisterValidationInterface(CValidationInterface* callbacks)
120
2.23k
{
121
    // Create a shared_ptr with a no-op deleter - CValidationInterface lifecycle
122
    // is managed by the caller.
123
2.23k
    RegisterSharedValidationInterface({callbacks, [](CValidationInterface*){}});
124
2.23k
}
125
126
void ValidationSignals::UnregisterSharedValidationInterface(std::shared_ptr<CValidationInterface> callbacks)
127
257k
{
128
257k
    UnregisterValidationInterface(callbacks.get());
129
257k
}
130
131
void ValidationSignals::UnregisterValidationInterface(CValidationInterface* callbacks)
132
260k
{
133
260k
    m_internals->Unregister(callbacks);
134
260k
}
135
136
void ValidationSignals::UnregisterAllValidationInterfaces()
137
1.13k
{
138
1.13k
    m_internals->Clear();
139
1.13k
}
140
141
void ValidationSignals::CallFunctionInValidationInterfaceQueue(std::function<void()> func)
142
16.5k
{
143
16.5k
    m_internals->m_task_runner->insert(std::move(func));
144
16.5k
}
145
146
void ValidationSignals::SyncWithValidationInterfaceQueue()
147
16.5k
{
148
16.5k
    AssertLockNotHeld(cs_main);
149
    // Block until the validation queue drains
150
16.5k
    std::promise<void> promise;
151
16.5k
    CallFunctionInValidationInterfaceQueue([&promise] {
152
16.5k
        promise.set_value();
153
16.5k
    });
154
16.5k
    promise.get_future().wait();
155
16.5k
}
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
377k
    do {                                                                                                         \
161
377k
        static_assert(std::is_rvalue_reference_v<decltype((event))>,                                             \
162
377k
                      "event must be passed as an rvalue");                                                      \
163
377k
        static_assert(std::is_rvalue_reference_v<decltype((log_msg))>,                                           \
164
377k
                      "log_msg must be passed as an rvalue");                                                    \
165
377k
        auto enqueue_log_msg = (log_msg);                                                                        \
166
377k
        LOG_EVENT("Enqueuing %s", enqueue_log_msg);                                                              \
167
377k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
376k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
376k
            local_event();                                                                                       \
170
376k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::UpdatedBlockTip(CBlockIndex const*, CBlockIndex const*, bool)::$_1::operator()() const
Line
Count
Source
167
94.2k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
94.2k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
94.2k
            local_event();                                                                                       \
170
94.2k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::TransactionAddedToMempool(NewMempoolTransactionInfo const&, unsigned long)::$_1::operator()() const
Line
Count
Source
167
50.2k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
50.2k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
50.2k
            local_event();                                                                                       \
170
50.2k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::TransactionRemovedFromMempool(std::shared_ptr<CTransaction const> const&, MemPoolRemovalReason, unsigned long)::$_1::operator()() const
Line
Count
Source
167
2.02k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
2.02k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
2.02k
            local_event();                                                                                       \
170
2.02k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::BlockConnected(kernel::ChainstateRole const&, std::shared_ptr<CBlock const>, CBlockIndex const*)::$_1::operator()() const
Line
Count
Source
167
107k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
107k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
107k
            local_event();                                                                                       \
170
107k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::MempoolTransactionsRemovedForBlock(std::vector<RemovedMempoolTransactionInfo, std::allocator<RemovedMempoolTransactionInfo>> const&, unsigned int)::$_1::operator()() const
Line
Count
Source
167
106k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
106k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
106k
            local_event();                                                                                       \
170
106k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::BlockDisconnected(std::shared_ptr<CBlock const>, CBlockIndex const*)::$_1::operator()() const
Line
Count
Source
167
14.0k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
14.0k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
14.0k
            local_event();                                                                                       \
170
14.0k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::ChainStateFlushed(kernel::ChainstateRole const&, CBlockLocator const&)::$_1::operator()() const
Line
Count
Source
167
2.37k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
2.37k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
2.37k
            local_event();                                                                                       \
170
2.37k
        });                                                                                                      \
171
377k
    } while (0)
172
173
#define LOG_MSG(fmt, ...) \
174
377k
    (util::log::ShouldDebugLog(BCLog::VALIDATION) ? tfm::format((fmt), __VA_ARGS__) : std::string{})
175
176
#define LOG_EVENT(fmt, ...) \
177
1.07M
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
178
179
94.2k
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
94.2k
    auto log_msg = LOG_MSG("%s: new block hash=%s fork block hash=%s (in IBD=%s)", __func__,
185
94.2k
                          pindexNew->GetBlockHash().ToString(),
186
94.2k
                          pindexFork ? pindexFork->GetBlockHash().ToString() : "null",
187
94.2k
                          fInitialDownload);
188
94.2k
    auto event = [pindexNew, pindexFork, fInitialDownload, this] {
189
215k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.UpdatedBlockTip(pindexNew, pindexFork, fInitialDownload); });
190
94.2k
    };
191
94.2k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
192
94.2k
}
193
194
void ValidationSignals::ActiveTipChange(const CBlockIndex& new_tip, bool is_ibd)
195
97.1k
{
196
97.1k
    LOG_EVENT("%s: new block hash=%s block height=%d", __func__, new_tip.GetBlockHash().ToString(), new_tip.nHeight);
197
228k
    m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.ActiveTipChange(new_tip, is_ibd); });
198
97.1k
}
199
200
void ValidationSignals::TransactionAddedToMempool(const NewMempoolTransactionInfo& tx, uint64_t mempool_sequence)
201
50.2k
{
202
50.2k
    auto log_msg = LOG_MSG("%s: txid=%s wtxid=%s", __func__,
203
50.2k
                          tx.info.m_tx->GetHash().ToString(),
204
50.2k
                          tx.info.m_tx->GetWitnessHash().ToString());
205
50.2k
    auto event = [tx, mempool_sequence, this] {
206
83.4k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.TransactionAddedToMempool(tx, mempool_sequence); });
207
50.2k
    };
208
50.2k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
209
50.2k
}
210
211
2.02k
void ValidationSignals::TransactionRemovedFromMempool(const CTransactionRef& tx, MemPoolRemovalReason reason, uint64_t mempool_sequence) {
212
2.02k
    auto log_msg = LOG_MSG("%s: txid=%s wtxid=%s reason=%s", __func__,
213
2.02k
                          tx->GetHash().ToString(),
214
2.02k
                          tx->GetWitnessHash().ToString(),
215
2.02k
                          RemovalReasonToString(reason));
216
2.02k
    auto event = [tx, reason, mempool_sequence, this] {
217
4.14k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.TransactionRemovedFromMempool(tx, reason, mempool_sequence); });
218
2.02k
    };
219
2.02k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
220
2.02k
}
221
222
void ValidationSignals::BlockConnected(const ChainstateRole& role, std::shared_ptr<const CBlock> pblock, const CBlockIndex* pindex)
223
107k
{
224
107k
    auto log_msg = LOG_MSG("%s: block hash=%s block height=%d", __func__,
225
107k
                          pblock->GetHash().ToString(),
226
107k
                          pindex->nHeight);
227
107k
    auto event = [role, pblock = std::move(pblock), pindex, this] {
228
242k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockConnected(role, pblock, pindex); });
229
107k
    };
230
107k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
231
107k
}
232
233
void ValidationSignals::MempoolTransactionsRemovedForBlock(const std::vector<RemovedMempoolTransactionInfo>& txs_removed_for_block, unsigned int nBlockHeight)
234
106k
{
235
106k
    auto log_msg = LOG_MSG("%s: block height=%s txs removed=%s", __func__,
236
106k
                          nBlockHeight,
237
106k
                          txs_removed_for_block.size());
238
106k
    auto event = [txs_removed_for_block, nBlockHeight, this] {
239
240k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.MempoolTransactionsRemovedForBlock(txs_removed_for_block, nBlockHeight); });
240
106k
    };
241
106k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
242
106k
}
243
244
void ValidationSignals::BlockDisconnected(std::shared_ptr<const CBlock> pblock, const CBlockIndex* pindex)
245
14.0k
{
246
14.0k
    auto log_msg = LOG_MSG("%s: block hash=%s block height=%d", __func__,
247
14.0k
                          pblock->GetHash().ToString(),
248
14.0k
                          pindex->nHeight);
249
14.0k
    auto event = [pblock = std::move(pblock), pindex, this] {
250
28.5k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockDisconnected(pblock, pindex); });
251
14.0k
    };
252
14.0k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
253
14.0k
}
254
255
void ValidationSignals::ChainStateFlushed(const ChainstateRole& role, const CBlockLocator& locator)
256
3.42k
{
257
3.42k
    auto log_msg = LOG_MSG("%s: block hash=%s", __func__,
258
3.42k
                          locator.IsNull() ? "null" : locator.vHave.front().ToString());
259
3.42k
    auto event = [role, locator, this] {
260
2.85k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.ChainStateFlushed(role, locator); });
261
2.37k
    };
262
3.42k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
263
3.42k
}
264
265
void ValidationSignals::BlockChecked(const std::shared_ptr<const CBlock>& block, const BlockValidationState& state)
266
147k
{
267
147k
    LOG_EVENT("%s: block hash=%s state=%s", __func__,
268
147k
              block->GetHash().ToString(), state.ToString());
269
274k
    m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockChecked(block, state); });
270
147k
}
271
272
79.7k
void ValidationSignals::NewPoWValidBlock(const CBlockIndex *pindex, const std::shared_ptr<const CBlock> &block) {
273
79.7k
    LOG_EVENT("%s: block hash=%s", __func__, block->GetHash().ToString());
274
186k
    m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.NewPoWValidBlock(pindex, block); });
275
79.7k
}