PIVX Core  5.6.99
P2P Digital Currency
validationinterface.cpp
Go to the documentation of this file.
1 // Copyright (c) 2009-2010 Satoshi Nakamoto
2 // Copyright (c) 2009-2014 The Bitcoin Core developers
3 // Copyright (c) 2017-2021 The PIVX Core developers
4 // Distributed under the MIT software license, see the accompanying
5 // file COPYING or http://www.opensource.org/licenses/mit-license.php.
6 
7 #include "validationinterface.h"
8 
9 #include "chain.h"
10 #include "consensus/validation.h"
11 #include "evo/deterministicmns.h"
12 #include "logging.h"
13 #include "scheduler.h"
14 #include "util/validation.h"
15 #include "validation.h" // cs_main
16 
17 #include <future>
18 #include <list>
19 #include <unordered_map>
20 #include <boost/signals2/signal.hpp>
21 
23  boost::signals2::scoped_connection AcceptedBlockHeader;
24  boost::signals2::scoped_connection UpdatedBlockTip;
25  boost::signals2::scoped_connection TransactionAddedToMempool;
26  boost::signals2::scoped_connection BlockConnected;
27  boost::signals2::scoped_connection BlockDisconnected;
28  boost::signals2::scoped_connection TransactionRemovedFromMempool;
29  boost::signals2::scoped_connection SetBestChain;
30  boost::signals2::scoped_connection Broadcast;
31  boost::signals2::scoped_connection BlockChecked;
32  boost::signals2::scoped_connection NotifyMasternodeListChanged;
33 };
34 
37  boost::signals2::signal<void(const CBlockIndex*)> AcceptedBlockHeader;
39  boost::signals2::signal<void (const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)> UpdatedBlockTip;
41  boost::signals2::signal<void (const CTransactionRef &)> TransactionAddedToMempool;
46  boost::signals2::signal<void (const std::shared_ptr<const CBlock> &, const CBlockIndex *pindex)> BlockConnected;
48  boost::signals2::signal<void (const std::shared_ptr<const CBlock> &, const uint256& blockHash, int nBlockHeight, int64_t blockTime)> BlockDisconnected;
50  boost::signals2::signal<void (const CTransactionRef &, MemPoolRemovalReason reason)> TransactionRemovedFromMempool;
52  boost::signals2::signal<void (const CBlockLocator &)> SetBestChain;
54  boost::signals2::signal<void (CConnman* connman)> Broadcast;
56  boost::signals2::signal<void (const CBlock&, const CValidationState&)> BlockChecked;
58  boost::signals2::signal<void (bool undo, const CDeterministicMNList& oldMNList, const CDeterministicMNListDiff& diff)> NotifyMasternodeListChanged;
59 
60  std::unordered_map<CValidationInterface*, ValidationInterfaceConnections> m_connMainSignals;
61 
62  // We are not allowed to assume the scheduler only runs in one thread,
63  // but must ensure all callbacks happen in-order, so we end up creating
64  // our own queue here :(
66 
67  explicit MainSignalsInstance(CScheduler *pscheduler) : m_schedulerClient(pscheduler) {}
68 };
69 
70 static CMainSignals g_signals;
71 
73  assert(!m_internals);
75 }
76 
78  m_internals.reset(nullptr);
79 }
80 
82  if (m_internals) {
83  m_internals->m_schedulerClient.EmptyQueue();
84  }
85 }
86 
88  if (!m_internals) return 0;
89  return m_internals->m_schedulerClient.CallbacksPending();
90 }
91 
93 {
94  return g_signals;
95 }
96 void RegisterSharedValidationInterface(std::shared_ptr<CValidationInterface> pwalletIn)
97 {
98  // Each connection captures pwalletIn to ensure that each callback is
99  // executed before pwalletIn is destroyed. For more details see bitcoin #18338
100  ValidationInterfaceConnections& conns = g_signals.m_internals->m_connMainSignals[pwalletIn.get()];
101  conns.AcceptedBlockHeader = g_signals.m_internals->AcceptedBlockHeader.connect(std::bind(&CValidationInterface::AcceptedBlockHeader, pwalletIn, std::placeholders::_1));
102  conns.UpdatedBlockTip = g_signals.m_internals->UpdatedBlockTip.connect(std::bind(&CValidationInterface::UpdatedBlockTip, pwalletIn, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3));
103  conns.TransactionAddedToMempool = g_signals.m_internals->TransactionAddedToMempool.connect(std::bind(&CValidationInterface::TransactionAddedToMempool, pwalletIn, std::placeholders::_1));
104  conns.BlockConnected = g_signals.m_internals->BlockConnected.connect(std::bind(&CValidationInterface::BlockConnected, pwalletIn, std::placeholders::_1, std::placeholders::_2));
105  conns.BlockDisconnected = g_signals.m_internals->BlockDisconnected.connect(std::bind(&CValidationInterface::BlockDisconnected, pwalletIn, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3, std::placeholders::_4));
106  conns.TransactionRemovedFromMempool = g_signals.m_internals->TransactionRemovedFromMempool.connect(std::bind(&CValidationInterface::TransactionRemovedFromMempool, pwalletIn, std::placeholders::_1, std::placeholders::_2));
107  conns.SetBestChain = g_signals.m_internals->SetBestChain.connect(std::bind(&CValidationInterface::SetBestChain, pwalletIn, std::placeholders::_1));
108  conns.Broadcast = g_signals.m_internals->Broadcast.connect(std::bind(&CValidationInterface::ResendWalletTransactions, pwalletIn, std::placeholders::_1));
109  conns.BlockChecked = g_signals.m_internals->BlockChecked.connect(std::bind(&CValidationInterface::BlockChecked, pwalletIn, std::placeholders::_1, std::placeholders::_2));
110  conns.NotifyMasternodeListChanged = g_signals.m_internals->NotifyMasternodeListChanged.connect(std::bind(&CValidationInterface::NotifyMasternodeListChanged, pwalletIn, std::placeholders::_1, std::placeholders::_2, std::placeholders::_3));
111 }
113 {
114  // Create a shared_ptr with a no-op deleter - CValidationInterface lifecycle
115  // is managed by the caller.
117 }
118 
120 {
121  if (g_signals.m_internals) {
122  g_signals.m_internals->m_connMainSignals.erase(pwalletIn);
123  }
124 }
125 
126 void UnregisterSharedValidationInterface(std::shared_ptr<CValidationInterface> pwalletIn)
127 {
128  UnregisterValidationInterface(pwalletIn.get());
129 }
130 
132 {
133  if (!g_signals.m_internals) {
134  return;
135  }
136  g_signals.m_internals->m_connMainSignals.clear();
137 }
138 
139 void CallFunctionInValidationInterfaceQueue(std::function<void ()> func) {
140  g_signals.m_internals->m_schedulerClient.AddToProcessQueue(std::move(func));
141 }
142 
145  // if queue is empty, do not wait for nothing.s
146  if (g_signals.CallbacksPending() == 0) return;
147 
148  // Block until the validation queue drains
149  std::promise<void> promise;
151  promise.set_value();
152  });
153  promise.get_future().wait();
154 }
155 
156 // Use a macro instead of a function for conditional logging to prevent
157 // evaluating arguments when logging is not enabled.
158 //
159 // NOTE: The lambda captures all local variables by value.
160 #define ENQUEUE_AND_LOG_EVENT(event, fmt, name, ...) \
161  do { \
162  auto local_name = (name); \
163  LOG_EVENT("Enqueuing " fmt, local_name, __VA_ARGS__); \
164  m_internals->m_schedulerClient.AddToProcessQueue([=] { \
165  LOG_EVENT(fmt, local_name, __VA_ARGS__); \
166  event(); \
167  }); \
168  } while (0)
169 
170 #define LOG_EVENT(fmt, ...) \
171  LogPrint(BCLog::VALIDATION, fmt "\n", __VA_ARGS__)
172 
174 {
175  auto event = [pindexNew, this] {
176  m_internals->AcceptedBlockHeader(pindexNew);
177  };
178  ENQUEUE_AND_LOG_EVENT(event, "%s: new block header=%s", __func__, pindexNew->GetBlockHash().ToString());
179 }
180 
181 void CMainSignals::UpdatedBlockTip(const CBlockIndex* pindexNew, const CBlockIndex* pindexFork, bool fInitialDownload) {
182  // Dependencies exist that require UpdatedBlockTip events to be delivered in the order in which
183  // the chain actually updates. One way to ensure this is for the caller to invoke this signal
184  // in the same critical section where the chain is updated
185 
186  auto event = [pindexNew, pindexFork, fInitialDownload, this] {
187  m_internals->UpdatedBlockTip(pindexNew, pindexFork, fInitialDownload);
188  };
189  ENQUEUE_AND_LOG_EVENT(event, "%s: new block hash=%s, fork block hash=%s (in IBD=%s)", __func__,
190  pindexNew->GetBlockHash().ToString(),
191  pindexFork ? pindexFork->GetBlockHash().ToString() : "null",
192  fInitialDownload);
193 }
194 
196  auto event = [ptx, this] {
197  m_internals->TransactionAddedToMempool(ptx);
198  };
199  ENQUEUE_AND_LOG_EVENT(event, "%s: txid=%s", __func__, ptx->GetHash().ToString());
200 }
201 
203  auto event = [ptx, reason, this] {
204  m_internals->TransactionRemovedFromMempool(ptx, reason);
205  };
206  ENQUEUE_AND_LOG_EVENT(event, "%s: txid=%s", __func__, ptx->GetHash().ToString());
207 }
208 
209 void CMainSignals::BlockConnected(const std::shared_ptr<const CBlock> &pblock, const CBlockIndex *pindex) {
210  auto event = [pblock, pindex, this] {
211  m_internals->BlockConnected(pblock, pindex);
212  };
213  ENQUEUE_AND_LOG_EVENT(event, "%s: block hash=%s, block height=%d", __func__,
214  pblock->GetHash().ToString(), pindex->nHeight);
215 }
216 
217 void CMainSignals::BlockDisconnected(const std::shared_ptr<const CBlock> &pblock, const uint256& blockHash, int nBlockHeight, int64_t blockTime) {
218  auto event = [pblock, blockHash, nBlockHeight, blockTime, this] {
219  m_internals->BlockDisconnected(pblock, blockHash, nBlockHeight, blockTime);
220  };
221  ENQUEUE_AND_LOG_EVENT(event, "%s: block hash=%s, block height=%d, block time=%d", __func__,
222  blockHash.ToString(), nBlockHeight, blockTime);
223 }
224 
226  auto event = [locator, this] {
227  m_internals->SetBestChain(locator);
228  };
229  ENQUEUE_AND_LOG_EVENT(event, "%s: block hash=%s", __func__,
230  locator.IsNull() ? "null" : locator.vHave.front().ToString());
231 }
232 
234  m_internals->Broadcast(connman);
235 }
236 
237 void CMainSignals::BlockChecked(const CBlock& block, const CValidationState& state) {
238  m_internals->BlockChecked(block, state);
239  LOG_EVENT("%s: block hash=%s (state=%s)", __func__,
240  block.GetHash().ToString(), FormatStateMessage(state));
241 }
242 
244  m_internals->NotifyMasternodeListChanged(undo, oldMNList, diff);
245  LOG_EVENT("%s: (undo=%d) old list for=%s, added=%d, updated=%d, removed=%d", __func__,
246  undo,
247  oldMNList.GetBlockHash().ToString(),
248  diff.addedMNs.size(),
249  diff.updatedMNs.size(),
250  diff.removedMns.size());
251 }
uint256 GetHash() const
Definition: block.cpp:15
Definition: block.h:80
The block chain is a tree shaped structure starting with the genesis block at the root,...
Definition: chain.h:139
uint256 GetBlockHash() const
Definition: chain.h:215
int nHeight
height of the entry in the chain. The genesis block has height 0
Definition: chain.h:151
Definition: net.h:145
std::set< uint64_t > removedMns
std::map< uint64_t, CDeterministicMNStateDiff > updatedMNs
std::vector< CDeterministicMNCPtr > addedMNs
const uint256 & GetBlockHash() const
std::unique_ptr< MainSignalsInstance > m_internals
void SetBestChain(const CBlockLocator &)
void UpdatedBlockTip(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)
void NotifyMasternodeListChanged(bool undo, const CDeterministicMNList &oldMNList, const CDeterministicMNListDiff &diff)
void UnregisterBackgroundSignalScheduler()
Unregister a CScheduler to give callbacks which should run in the background - these callbacks will n...
void AcceptedBlockHeader(const CBlockIndex *pindexNew)
void Broadcast(CConnman *connman)
void BlockChecked(const CBlock &, const CValidationState &)
void RegisterBackgroundSignalScheduler(CScheduler &scheduler)
Register a CScheduler to give callbacks which should run in the background (may only be called once)
void BlockDisconnected(const std::shared_ptr< const CBlock > &block, const uint256 &blockHash, int nBlockHeight, int64_t blockTime)
void TransactionAddedToMempool(const CTransactionRef &ptxn)
void BlockConnected(const std::shared_ptr< const CBlock > &block, const CBlockIndex *pindex)
void FlushBackgroundCallbacks()
Call any remaining callbacks on the calling thread.
void TransactionRemovedFromMempool(const CTransactionRef &, MemPoolRemovalReason)
Implement this to subscribe to events generated in validation.
virtual void BlockDisconnected(const std::shared_ptr< const CBlock > &block, const uint256 &blockHash, int nBlockHeight, int64_t blockTime)
Notifies listeners of a block being disconnected.
virtual void ResendWalletTransactions(CConnman *connman)
Tells listeners to broadcast their data.
virtual void TransactionAddedToMempool(const CTransactionRef &ptxn)
Notifies listeners of a transaction having been added to mempool.
virtual void UpdatedBlockTip(const CBlockIndex *pindexNew, const CBlockIndex *pindexFork, bool fInitialDownload)
Notifies listeners when the block chain tip advances.
virtual void TransactionRemovedFromMempool(const CTransactionRef &ptx, MemPoolRemovalReason reason)
Notifies listeners of a transaction leaving mempool.
virtual void NotifyMasternodeListChanged(bool undo, const CDeterministicMNList &oldMNList, const CDeterministicMNListDiff &diff)
Notifies listeners of updated deterministic masternode list.
virtual void SetBestChain(const CBlockLocator &locator)
Notifies listeners of the new active block chain on-disk.
virtual void BlockConnected(const std::shared_ptr< const CBlock > &block, const CBlockIndex *pindex)
Notifies listeners of a block being connected.
virtual void AcceptedBlockHeader(const CBlockIndex *pindexNew)
virtual void BlockChecked(const CBlock &, const CValidationState &)
Capture information about block/transaction validation.
Definition: validation.h:24
Class used by CScheduler clients which may schedule multiple jobs which are required to be run serial...
Definition: scheduler.h:99
std::string ToString() const
Definition: uint256.cpp:65
256-bit opaque blob.
Definition: uint256.h:138
RecursiveMutex cs_main
Global state.
Definition: validation.cpp:80
Describes a place in the block chain to another node such that if the other node doesn't have the sam...
Definition: block.h:154
std::vector< uint256 > vHave
Definition: block.h:155
bool IsNull() const
Definition: block.h:177
boost::signals2::signal< void(const CBlockIndex *, const CBlockIndex *, bool fInitialDownload)> UpdatedBlockTip
Notifies listeners of updated block chain tip.
boost::signals2::signal< void(const std::shared_ptr< const CBlock > &, const CBlockIndex *pindex)> BlockConnected
Notifies listeners of a block being connected.
boost::signals2::signal< void(const CBlock &, const CValidationState &)> BlockChecked
Notifies listeners of a block validation result.
SingleThreadedSchedulerClient m_schedulerClient
boost::signals2::signal< void(const CTransactionRef &, MemPoolRemovalReason reason)> TransactionRemovedFromMempool
Notifies listeners of a transaction removal from the mempool.
std::unordered_map< CValidationInterface *, ValidationInterfaceConnections > m_connMainSignals
boost::signals2::signal< void(CConnman *connman)> Broadcast
Tells listeners to broadcast their data.
boost::signals2::signal< void(const std::shared_ptr< const CBlock > &, const uint256 &blockHash, int nBlockHeight, int64_t blockTime)> BlockDisconnected
Notifies listeners of a block being disconnected.
boost::signals2::signal< void(const CBlockLocator &)> SetBestChain
Notifies listeners of a new active block chain.
MainSignalsInstance(CScheduler *pscheduler)
boost::signals2::signal< void(bool undo, const CDeterministicMNList &oldMNList, const CDeterministicMNListDiff &diff)> NotifyMasternodeListChanged
Notifies listeners of updated deterministic masternode list.
boost::signals2::signal< void(const CTransactionRef &)> TransactionAddedToMempool
Notifies listeners of a transaction having been added to mempool.
boost::signals2::signal< void(const CBlockIndex *)> AcceptedBlockHeader
Notifies listeners of accepted block header.
boost::signals2::scoped_connection TransactionAddedToMempool
boost::signals2::scoped_connection UpdatedBlockTip
boost::signals2::scoped_connection SetBestChain
boost::signals2::scoped_connection BlockDisconnected
boost::signals2::scoped_connection AcceptedBlockHeader
boost::signals2::scoped_connection Broadcast
boost::signals2::scoped_connection BlockChecked
boost::signals2::scoped_connection TransactionRemovedFromMempool
boost::signals2::scoped_connection BlockConnected
boost::signals2::scoped_connection NotifyMasternodeListChanged
#define AssertLockNotHeld(cs)
Definition: sync.h:76
std::shared_ptr< const CTransaction > CTransactionRef
Definition: transaction.h:456
MemPoolRemovalReason
Reason why a transaction was removed from the mempool, this is passed to the notification signal.
Definition: txmempool.h:295
std::string FormatStateMessage(const CValidationState &state)
Convert CValidationState to a human-readable message for logging.
Definition: validation.cpp:13
CMainSignals & GetMainSignals()
#define LOG_EVENT(fmt,...)
void CallFunctionInValidationInterfaceQueue(std::function< void()> func)
Pushes a function to callback onto the notification queue, guaranteeing any callbacks generated prior...
void UnregisterValidationInterface(CValidationInterface *pwalletIn)
Unregister a wallet from core.
void RegisterSharedValidationInterface(std::shared_ptr< CValidationInterface > pwalletIn)
void UnregisterAllValidationInterfaces()
Unregister all wallets from core.
void UnregisterSharedValidationInterface(std::shared_ptr< CValidationInterface > pwalletIn)
void RegisterValidationInterface(CValidationInterface *pwalletIn)
Register a wallet to receive updates from core.
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,...)
CScheduler scheduler