Bitcoin Core 22.99.0
P2P Digital Currency
validationinterface.cpp
Go to the documentation of this file.
1// Copyright (c) 2009-2010 Satoshi Nakamoto
2// Copyright (c) 2009-2020 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
7
8#include <chain.h>
10#include <logging.h>
11#include <primitives/block.h>
13#include <scheduler.h>
14
15#include <future>
16#include <unordered_map>
17#include <utility>
18
27private:
33 struct ListEntry { std::shared_ptr<CValidationInterface> callbacks; int count = 1; };
34 std::list<ListEntry> m_list GUARDED_BY(m_mutex);
35 std::unordered_map<CValidationInterface*, std::list<ListEntry>::iterator> m_map GUARDED_BY(m_mutex);
36
37public:
38 // We are not allowed to assume the scheduler only runs in one thread,
39 // but must ensure all callbacks happen in-order, so we end up creating
40 // our own queue here :(
42
43 explicit MainSignalsInstance(CScheduler *pscheduler) : m_schedulerClient(pscheduler) {}
44
45 void Register(std::shared_ptr<CValidationInterface> callbacks)
46 {
48 auto inserted = m_map.emplace(callbacks.get(), m_list.end());
49 if (inserted.second) inserted.first->second = m_list.emplace(m_list.end());
50 inserted.first->second->callbacks = std::move(callbacks);
51 }
52
54 {
56 auto it = m_map.find(callbacks);
57 if (it != m_map.end()) {
58 if (!--it->second->count) m_list.erase(it->second);
59 m_map.erase(it);
60 }
61 }
62
67 void Clear()
68 {
70 for (const auto& entry : m_map) {
71 if (!--entry.second->count) m_list.erase(entry.second);
72 }
73 m_map.clear();
74 }
75
76 template<typename F> void Iterate(F&& f)
77 {
78 WAIT_LOCK(m_mutex, lock);
79 for (auto it = m_list.begin(); it != m_list.end();) {
80 ++it->count;
81 {
82 REVERSE_LOCK(lock);
83 f(*it->callbacks);
84 }
85 it = --it->count ? std::next(it) : m_list.erase(it);
86 }
87 }
88};
89
91
93{
95 m_internals.reset(new MainSignalsInstance(&scheduler));
96}
97
99{
100 m_internals.reset(nullptr);
101}
102
104{
105 if (m_internals) {
106 m_internals->m_schedulerClient.EmptyQueue();
107 }
108}
109
111{
112 if (!m_internals) return 0;
113 return m_internals->m_schedulerClient.CallbacksPending();
114}
115
117{
118 return g_signals;
119}
120
121void RegisterSharedValidationInterface(std::shared_ptr<CValidationInterface> callbacks)
122{
123 // Each connection captures the shared_ptr to ensure that each callback is
124 // executed before the subscriber is destroyed. For more details see #18338.
125 g_signals.m_internals->Register(std::move(callbacks));
126}
127
129{
130 // Create a shared_ptr with a no-op deleter - CValidationInterface lifecycle
131 // is managed by the caller.
133}
134
135void UnregisterSharedValidationInterface(std::shared_ptr<CValidationInterface> callbacks)
136{
137 UnregisterValidationInterface(callbacks.get());
138}
139
141{
143 g_signals.m_internals->Unregister(callbacks);
144 }
145}
146
148{
149 if (!g_signals.m_internals) {
150 return;
151 }
152 g_signals.m_internals->Clear();
153}
154
155void CallFunctionInValidationInterfaceQueue(std::function<void()> func)
156{
157 g_signals.m_internals->m_schedulerClient.AddToProcessQueue(std::move(func));
158}
159
161{
163 // Block until the validation queue drains
164 std::promise<void> promise;
166 promise.set_value();
167 });
168 promise.get_future().wait();
169}
170
171// Use a macro instead of a function for conditional logging to prevent
172// evaluating arguments when logging is not enabled.
173//
174// NOTE: The lambda captures all local variables by value.
175#define ENQUEUE_AND_LOG_EVENT(event, fmt, name, ...) \
176 do { \
177 auto local_name = (name); \
178 LOG_EVENT("Enqueuing " fmt, local_name, __VA_ARGS__); \
179 m_internals->m_schedulerClient.AddToProcessQueue([=] { \
180 LOG_EVENT(fmt, local_name, __VA_ARGS__); \
181 event(); \
182 }); \
183 } while (0)
184
185#define LOG_EVENT(fmt, ...) \
186 LogPrint(BCLog::VALIDATION, fmt "\n", __VA_ARGS__)
187
188void CMainSignals::UpdatedBlockTip(const CBlockIndex *pindexNew, const CBlockIndex *pindexFork, bool fInitialDownload) {
189 // Dependencies exist that require UpdatedBlockTip events to be delivered in the order in which
190 // the chain actually updates. One way to ensure this is for the caller to invoke this signal
191 // in the same critical section where the chain is updated
192
193 auto event = [pindexNew, pindexFork, fInitialDownload, this] {
194 m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.UpdatedBlockTip(pindexNew, pindexFork, fInitialDownload); });
195 };
196 ENQUEUE_AND_LOG_EVENT(event, "%s: new block hash=%s fork block hash=%s (in IBD=%s)", __func__,
197 pindexNew->GetBlockHash().ToString(),
198 pindexFork ? pindexFork->GetBlockHash().ToString() : "null",
199 fInitialDownload);
200}
201
202void CMainSignals::TransactionAddedToMempool(const CTransactionRef& tx, uint64_t mempool_sequence) {
203 auto event = [tx, mempool_sequence, this] {
204 m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.TransactionAddedToMempool(tx, mempool_sequence); });
205 };
206 ENQUEUE_AND_LOG_EVENT(event, "%s: txid=%s wtxid=%s", __func__,
207 tx->GetHash().ToString(),
208 tx->GetWitnessHash().ToString());
209}
210
211void CMainSignals::TransactionRemovedFromMempool(const CTransactionRef& tx, MemPoolRemovalReason reason, uint64_t mempool_sequence) {
212 auto event = [tx, reason, mempool_sequence, this] {
213 m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.TransactionRemovedFromMempool(tx, reason, mempool_sequence); });
214 };
215 ENQUEUE_AND_LOG_EVENT(event, "%s: txid=%s wtxid=%s", __func__,
216 tx->GetHash().ToString(),
217 tx->GetWitnessHash().ToString());
218}
219
220void CMainSignals::BlockConnected(const std::shared_ptr<const CBlock> &pblock, const CBlockIndex *pindex) {
221 auto event = [pblock, pindex, this] {
222 m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockConnected(pblock, pindex); });
223 };
224 ENQUEUE_AND_LOG_EVENT(event, "%s: block hash=%s block height=%d", __func__,
225 pblock->GetHash().ToString(),
226 pindex->nHeight);
227}
228
229void CMainSignals::BlockDisconnected(const std::shared_ptr<const CBlock>& pblock, const CBlockIndex* pindex)
230{
231 auto event = [pblock, pindex, this] {
232 m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockDisconnected(pblock, pindex); });
233 };
234 ENQUEUE_AND_LOG_EVENT(event, "%s: block hash=%s block height=%d", __func__,
235 pblock->GetHash().ToString(),
236 pindex->nHeight);
237}
238
240 auto event = [locator, this] {
241 m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.ChainStateFlushed(locator); });
242 };
243 ENQUEUE_AND_LOG_EVENT(event, "%s: block hash=%s", __func__,
244 locator.IsNull() ? "null" : locator.vHave.front().ToString());
245}
246
248 LOG_EVENT("%s: block hash=%s state=%s", __func__,
249 block.GetHash().ToString(), state.ToString());
250 m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.BlockChecked(block, state); });
251}
252
253void CMainSignals::NewPoWValidBlock(const CBlockIndex *pindex, const std::shared_ptr<const CBlock> &block) {
254 LOG_EVENT("%s: block hash=%s", __func__, block->GetHash().ToString());
255 m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.NewPoWValidBlock(pindex, block); });
256}
RecursiveMutex cs_main
Mutex to guard access to validation specific variables, such as reading or changing the chainstate.
Definition: validation.cpp:118
uint256 GetHash() const
Definition: block.cpp:11
Definition: block.h:63
The block chain is a tree shaped structure starting with the genesis block at the root,...
Definition: chain.h:146
uint256 GetBlockHash() const
Definition: chain.h:254
int nHeight
height of the entry in the chain. The genesis block has height 0
Definition: chain.h:158
std::unique_ptr< MainSignalsInstance > m_internals
void TransactionAddedToMempool(const CTransactionRef &, uint64_t mempool_sequence)
void BlockConnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
void BlockDisconnected(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)
void BlockChecked(const CBlock &, const BlockValidationState &)
void UnregisterBackgroundSignalScheduler()
Unregister a CScheduler to give callbacks which should run in the background - these callbacks will n...
void TransactionRemovedFromMempool(const CTransactionRef &, MemPoolRemovalReason, uint64_t mempool_sequence)
void NewPoWValidBlock(const CBlockIndex *, const std::shared_ptr< const CBlock > &)
void RegisterBackgroundSignalScheduler(CScheduler &scheduler)
Register a CScheduler to give callbacks which should run in the background (may only be called once)
void ChainStateFlushed(const CBlockLocator &)
void FlushBackgroundCallbacks()
Call any remaining callbacks on the calling thread.
Simple class for background tasks that should be run periodically or once "after a while".
Definition: scheduler.h:34
Implement this to subscribe to events generated in validation.
virtual void NewPoWValidBlock(const CBlockIndex *pindex, const std::shared_ptr< const CBlock > &block)
Notifies listeners that a block which builds directly on our current tip has been received and connec...
virtual void TransactionAddedToMempool(const CTransactionRef &tx, uint64_t mempool_sequence)
Notifies listeners of a transaction having been added to mempool.
virtual void ChainStateFlushed(const CBlockLocator &locator)
Notifies listeners of the new active block chain on-disk.
virtual void BlockChecked(const CBlock &, const BlockValidationState &)
Notifies listeners of a block validation result.
virtual void TransactionRemovedFromMempool(const CTransactionRef &tx, MemPoolRemovalReason reason, uint64_t mempool_sequence)
Notifies listeners of a transaction leaving mempool.
virtual void UpdatedBlockTip(const CBlockIndex *pindexNew, const CBlockIndex *pindexFork, bool fInitialDownload)
Notifies listeners when the block chain tip advances.
virtual void BlockConnected(const std::shared_ptr< const CBlock > &block, const CBlockIndex *pindex)
Notifies listeners of a block being connected.
virtual void BlockDisconnected(const std::shared_ptr< const CBlock > &block, const CBlockIndex *pindex)
Notifies listeners of a block being disconnected.
Class used by CScheduler clients which may schedule multiple jobs which are required to be run serial...
Definition: scheduler.h:118
std::string ToString() const
Definition: validation.h:125
std::string ToString() const
Definition: uint256.cpp:64
std::shared_ptr< const CTransaction > CTransactionRef
Definition: transaction.h:386
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
Definition: block.h:115
std::vector< uint256 > vHave
Definition: block.h:116
bool IsNull() const
Definition: block.h:135
List entries consist of a callback pointer and reference count.
int count
std::shared_ptr< CValidationInterface > callbacks
The MainSignalsInstance manages a list of shared_ptr<CValidationInterface> callbacks.
std::list< ListEntry > m_list GUARDED_BY(m_mutex)
std::unordered_map< CValidationInterface *, std::list< ListEntry >::iterator > m_map GUARDED_BY(m_mutex)
SingleThreadedSchedulerClient m_schedulerClient
void Unregister(CValidationInterface *callbacks)
void Register(std::shared_ptr< CValidationInterface > callbacks)
void Clear()
Clear unregisters every previously registered callback, erasing every map entry.
MainSignalsInstance(CScheduler *pscheduler)
#define WAIT_LOCK(cs, name)
Definition: sync.h:231
#define AssertLockNotHeld(cs)
Definition: sync.h:84
#define LOCK(cs)
Definition: sync.h:226
#define REVERSE_LOCK(g)
Definition: sync.h:221
MemPoolRemovalReason
Reason why a transaction was removed from the mempool, this is passed to the notification signal.
Definition: txmempool.h:341
assert(!tx.IsCoinBase())
#define LOG_EVENT(fmt,...)
static CMainSignals g_signals
void CallFunctionInValidationInterfaceQueue(std::function< void()> func)
Pushes a function to callback onto the notification queue, guaranteeing any callbacks generated prior...
CMainSignals & GetMainSignals()
void UnregisterSharedValidationInterface(std::shared_ptr< CValidationInterface > callbacks)
Unregister subscriber.
void UnregisterAllValidationInterfaces()
Unregister all subscribers.
void UnregisterValidationInterface(CValidationInterface *callbacks)
Unregister subscriber.
void RegisterValidationInterface(CValidationInterface *callbacks)
Register subscriber.
void SyncWithValidationInterfaceQueue()
This is a synonym for the following, which asserts certain locks are not held: std::promise<void> pro...
#define ENQUEUE_AND_LOG_EVENT(event, fmt, name,...)
void RegisterSharedValidationInterface(std::shared_ptr< CValidationInterface > callbacks)
Register subscriber.