fuzz coverage

Coverage Report

Created: 2026-05-08 05:52

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/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
116
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
186k
    {
54
186k
        LOCK(m_mutex);
Line
Count
Source
268
186k
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
186k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
186k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
186k
#define PASTE(x, y) x ## y
55
186k
        auto inserted = m_map.emplace(callbacks.get(), m_list.end());
56
186k
        if (inserted.second) inserted.first->second = m_list.emplace(m_list.end());
57
186k
        inserted.first->second->callbacks = std::move(callbacks);
58
186k
    }
59
60
    void Unregister(CValidationInterface* callbacks) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
61
185k
    {
62
185k
        LOCK(m_mutex);
Line
Count
Source
268
185k
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
63
185k
        auto it = m_map.find(callbacks);
64
185k
        if (it != m_map.end()) {
65
185k
            if (!--it->second->count) m_list.erase(it->second);
66
185k
            m_map.erase(it);
67
185k
        }
68
185k
    }
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.43k
    {
76
1.43k
        LOCK(m_mutex);
Line
Count
Source
268
1.43k
#define LOCK(cs) UniqueLock UNIQUE_NAME(criticalblock)(MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__)
Line
Count
Source
11
1.43k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
1.43k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
1.43k
#define PASTE(x, y) x ## y
77
1.43k
        for (const auto& entry : m_map) {
78
1.43k
            if (!--entry.second->count) m_list.erase(entry.second);
79
1.43k
        }
80
1.43k
        m_map.clear();
81
1.43k
    }
82
83
    template<typename F> void Iterate(F&& f) EXCLUSIVE_LOCKS_REQUIRED(!m_mutex)
84
1.13M
    {
85
1.13M
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
7.42k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
7.42k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
5.22k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
5.22k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
0
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
0
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
0
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
0
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
185k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
185k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
2.26M
        for (auto it = m_list.begin(); it != m_list.end();) {
87
1.12M
            ++it->count;
88
1.12M
            {
89
1.12M
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
185k
#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
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
186k
#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
186k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
186k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
186k
#define PASTE(x, y) x ## y
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
7.42k
#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
7.42k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
7.42k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
7.42k
#define PASTE(x, y) x ## y
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
5.22k
#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
5.22k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
5.22k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
5.22k
#define PASTE(x, y) x ## y
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
185k
#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
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
185k
#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
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
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
254
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
254
186k
#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
186k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
186k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
186k
#define PASTE(x, y) x ## y
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
185k
#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
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
90
1.12M
                f(*it->callbacks);
91
1.12M
            }
92
1.12M
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
1.12M
        }
94
1.13M
    }
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
186k
    {
85
186k
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
372k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
185k
            ++it->count;
88
185k
            {
89
185k
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
185k
#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
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
90
185k
                f(*it->callbacks);
91
185k
            }
92
185k
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
185k
        }
94
186k
    }
validationinterface.cpp:void ValidationSignalsImpl::Iterate<ValidationSignals::ActiveTipChange(CBlockIndex const&, bool)::$_0>(ValidationSignals::ActiveTipChange(CBlockIndex const&, bool)::$_0&&)
Line
Count
Source
84
186k
    {
85
186k
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
372k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
186k
            ++it->count;
88
186k
            {
89
186k
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
186k
#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
186k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
186k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
186k
#define PASTE(x, y) x ## y
90
186k
                f(*it->callbacks);
91
186k
            }
92
186k
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
186k
        }
94
186k
    }
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
7.42k
    {
85
7.42k
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
7.42k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
7.42k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
14.8k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
7.42k
            ++it->count;
88
7.42k
            {
89
7.42k
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
7.42k
#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
7.42k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
7.42k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
7.42k
#define PASTE(x, y) x ## y
90
7.42k
                f(*it->callbacks);
91
7.42k
            }
92
7.42k
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
7.42k
        }
94
7.42k
    }
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
5.22k
    {
85
5.22k
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
5.22k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
5.22k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
10.4k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
5.22k
            ++it->count;
88
5.22k
            {
89
5.22k
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
5.22k
#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
5.22k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
5.22k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
5.22k
#define PASTE(x, y) x ## y
90
5.22k
                f(*it->callbacks);
91
5.22k
            }
92
5.22k
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
5.22k
        }
94
5.22k
    }
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
186k
    {
85
186k
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
372k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
185k
            ++it->count;
88
185k
            {
89
185k
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
185k
#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
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
90
185k
                f(*it->callbacks);
91
185k
            }
92
185k
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
185k
        }
94
186k
    }
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
186k
    {
85
186k
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
372k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
185k
            ++it->count;
88
185k
            {
89
185k
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
185k
#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
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
90
185k
                f(*it->callbacks);
91
185k
            }
92
185k
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
185k
        }
94
186k
    }
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&)&&)
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
186k
    {
85
186k
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
186k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
186k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
372k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
186k
            ++it->count;
88
186k
            {
89
186k
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
186k
#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
186k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
186k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
186k
#define PASTE(x, y) x ## y
90
186k
                f(*it->callbacks);
91
186k
            }
92
186k
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
186k
        }
94
186k
    }
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
185k
    {
85
185k
        WAIT_LOCK(m_mutex, lock);
Line
Count
Source
274
185k
#define WAIT_LOCK(cs, name) UniqueLock name(LOCK_ARGS(cs))
Line
Count
Source
272
185k
#define LOCK_ARGS(cs) MaybeCheckNotHeld(cs), #cs, __FILE__, __LINE__
86
370k
        for (auto it = m_list.begin(); it != m_list.end();) {
87
185k
            ++it->count;
88
185k
            {
89
185k
                REVERSE_LOCK(lock, m_mutex);
Line
Count
Source
254
185k
#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
185k
#define UNIQUE_NAME(name) PASTE2(name, __COUNTER__)
Line
Count
Source
9
185k
#define PASTE2(x, y) PASTE(x, y)
Line
Count
Source
8
185k
#define PASTE(x, y) x ## y
90
185k
                f(*it->callbacks);
91
185k
            }
92
185k
            it = --it->count ? std::next(it) : 
m_list.erase(it)0
;
93
185k
        }
94
185k
    }
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
1
ValidationSignals::~ValidationSignals() = default;
101
102
void ValidationSignals::FlushBackgroundCallbacks()
103
1
{
104
1
    m_internals->m_task_runner->flush();
105
1
}
106
107
size_t ValidationSignals::CallbacksPending()
108
187k
{
109
187k
    return m_internals->m_task_runner->size();
110
187k
}
111
112
void ValidationSignals::RegisterSharedValidationInterface(std::shared_ptr<CValidationInterface> callbacks)
113
186k
{
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
186k
    m_internals->Register(std::move(callbacks));
117
186k
}
118
119
void ValidationSignals::RegisterValidationInterface(CValidationInterface* callbacks)
120
186k
{
121
    // Create a shared_ptr with a no-op deleter - CValidationInterface lifecycle
122
    // is managed by the caller.
123
186k
    RegisterSharedValidationInterface({callbacks, [](CValidationInterface*){}});
124
186k
}
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
185k
{
133
185k
    m_internals->Unregister(callbacks);
134
185k
}
135
136
void ValidationSignals::UnregisterAllValidationInterfaces()
137
1.43k
{
138
1.43k
    m_internals->Clear();
139
1.43k
}
140
141
void ValidationSignals::CallFunctionInValidationInterfaceQueue(std::function<void()> func)
142
187k
{
143
187k
    m_internals->m_task_runner->insert(std::move(func));
144
187k
}
145
146
void ValidationSignals::SyncWithValidationInterfaceQueue()
147
187k
{
148
187k
    AssertLockNotHeld(cs_main);
Line
Count
Source
149
187k
#define AssertLockNotHeld(cs) AssertLockNotHeldInline(#cs, __FILE__, __LINE__, &cs)
149
    // Block until the validation queue drains
150
187k
    std::promise<void> promise;
151
187k
    CallFunctionInValidationInterfaceQueue([&promise] {
152
187k
        promise.set_value();
153
187k
    });
154
187k
    promise.get_future().wait();
155
187k
}
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
573k
    do {                                                                                                         \
161
573k
        static_assert(std::is_rvalue_reference_v<decltype((event))>,                                             \
162
573k
                      "event must be passed as an rvalue");                                                      \
163
573k
        static_assert(std::is_rvalue_reference_v<decltype((log_msg))>,                                           \
164
573k
                      "log_msg must be passed as an rvalue");                                                    \
165
573k
        auto enqueue_log_msg = (log_msg);                                                                        \
166
573k
        LOG_EVENT("Enqueuing %s", enqueue_log_msg);                                                              \
167
573k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
573k
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
7.42k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
7.42k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
7.42k
    do {                                                               \
115
7.42k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
7.42k
    } while (0)
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
5.22k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
5.22k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
5.22k
    do {                                                               \
115
5.22k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
5.22k
    } while (0)
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
0
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
0
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
0
    do {                                                               \
115
0
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
0
    } while (0)
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
0
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
0
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
0
    do {                                                               \
115
0
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
0
    } while (0)
169
573k
            local_event();                                                                                       \
170
573k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::UpdatedBlockTip(CBlockIndex const*, CBlockIndex const*, bool)::$_1::operator()() const
Line
Count
Source
167
186k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
186k
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
169
186k
            local_event();                                                                                       \
170
186k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::TransactionAddedToMempool(NewMempoolTransactionInfo const&, unsigned long)::$_1::operator()() const
Line
Count
Source
167
7.42k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
7.42k
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
7.42k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
7.42k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
7.42k
    do {                                                               \
115
7.42k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
7.42k
    } while (0)
169
7.42k
            local_event();                                                                                       \
170
7.42k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::TransactionRemovedFromMempool(std::shared_ptr<CTransaction const> const&, MemPoolRemovalReason, unsigned long)::$_1::operator()() const
Line
Count
Source
167
5.22k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
5.22k
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
5.22k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
5.22k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
5.22k
    do {                                                               \
115
5.22k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
5.22k
    } while (0)
169
5.22k
            local_event();                                                                                       \
170
5.22k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::BlockConnected(kernel::ChainstateRole const&, std::shared_ptr<CBlock const>, CBlockIndex const*)::$_1::operator()() const
Line
Count
Source
167
186k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
186k
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
169
186k
            local_event();                                                                                       \
170
186k
        });                                                                                                      \
validationinterface.cpp:ValidationSignals::MempoolTransactionsRemovedForBlock(std::vector<RemovedMempoolTransactionInfo, std::allocator<RemovedMempoolTransactionInfo>> const&, unsigned int)::$_1::operator()() const
Line
Count
Source
167
186k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
186k
            LOG_EVENT("%s", local_log_msg);                                                                      \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
169
186k
            local_event();                                                                                       \
170
186k
        });                                                                                                      \
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
573k
    } while (0)
172
173
#define LOG_MSG(fmt, ...) \
174
573k
    (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? 
tfm::format((fmt), 0
__VA_ARGS__0
) : std::string{})
175
176
#define LOG_EVENT(fmt, ...) \
177
1.70M
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
178
179
186k
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
186k
    auto log_msg = LOG_MSG("%s: new block hash=%s fork block hash=%s (in IBD=%s)", __func__,
Line
Count
Source
174
186k
    (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? 
tfm::format((fmt), 0
__VA_ARGS__0
) : std::string{})
185
186k
                          pindexNew->GetBlockHash().ToString(),
186
186k
                          pindexFork ? pindexFork->GetBlockHash().ToString() : "null",
187
186k
                          fInitialDownload);
188
186k
    auto event = [pindexNew, pindexFork, fInitialDownload, this] {
189
186k
        m_internals->Iterate([&](CValidationInterface& callbacks) 
{ callbacks.UpdatedBlockTip(pindexNew, pindexFork, fInitialDownload); }185k
);
190
186k
    };
191
186k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
Line
Count
Source
160
186k
    do {                                                                                                         \
161
186k
        static_assert(std::is_rvalue_reference_v<decltype((event))>,                                             \
162
186k
                      "event must be passed as an rvalue");                                                      \
163
186k
        static_assert(std::is_rvalue_reference_v<decltype((log_msg))>,                                           \
164
186k
                      "log_msg must be passed as an rvalue");                                                    \
165
186k
        auto enqueue_log_msg = (log_msg);                                                                        \
166
186k
        LOG_EVENT("Enqueuing %s", enqueue_log_msg);                                                              \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
167
186k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
186k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
186k
            local_event();                                                                                       \
170
186k
        });                                                                                                      \
171
186k
    } while (0)
192
186k
}
193
194
void ValidationSignals::ActiveTipChange(const CBlockIndex& new_tip, bool is_ibd)
195
186k
{
196
186k
    LOG_EVENT("%s: new block hash=%s block height=%d", __func__, new_tip.GetBlockHash().ToString(), new_tip.nHeight);
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
197
186k
    m_internals->Iterate([&](CValidationInterface& callbacks) 
{ callbacks.ActiveTipChange(new_tip, is_ibd); }186k
);
198
186k
}
199
200
void ValidationSignals::TransactionAddedToMempool(const NewMempoolTransactionInfo& tx, uint64_t mempool_sequence)
201
7.42k
{
202
7.42k
    auto log_msg = LOG_MSG("%s: txid=%s wtxid=%s", __func__,
Line
Count
Source
174
7.42k
    (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? 
tfm::format((fmt), __VA_ARGS__)0
: std::string{})
203
7.42k
                          tx.info.m_tx->GetHash().ToString(),
204
7.42k
                          tx.info.m_tx->GetWitnessHash().ToString());
205
7.42k
    auto event = [tx, mempool_sequence, this] {
206
7.42k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.TransactionAddedToMempool(tx, mempool_sequence); });
207
7.42k
    };
208
7.42k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
Line
Count
Source
160
7.42k
    do {                                                                                                         \
161
7.42k
        static_assert(std::is_rvalue_reference_v<decltype((event))>,                                             \
162
7.42k
                      "event must be passed as an rvalue");                                                      \
163
7.42k
        static_assert(std::is_rvalue_reference_v<decltype((log_msg))>,                                           \
164
7.42k
                      "log_msg must be passed as an rvalue");                                                    \
165
7.42k
        auto enqueue_log_msg = (log_msg);                                                                        \
166
7.42k
        LOG_EVENT("Enqueuing %s", enqueue_log_msg);                                                              \
Line
Count
Source
177
7.42k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
7.42k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
7.42k
    do {                                                               \
115
7.42k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
7.42k
    } while (0)
167
7.42k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
7.42k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
7.42k
            local_event();                                                                                       \
170
7.42k
        });                                                                                                      \
171
7.42k
    } while (0)
209
7.42k
}
210
211
5.22k
void ValidationSignals::TransactionRemovedFromMempool(const CTransactionRef& tx, MemPoolRemovalReason reason, uint64_t mempool_sequence) {
212
5.22k
    auto log_msg = LOG_MSG("%s: txid=%s wtxid=%s reason=%s", __func__,
Line
Count
Source
174
5.22k
    (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? 
tfm::format((fmt), __VA_ARGS__)0
: std::string{})
213
5.22k
                          tx->GetHash().ToString(),
214
5.22k
                          tx->GetWitnessHash().ToString(),
215
5.22k
                          RemovalReasonToString(reason));
216
5.22k
    auto event = [tx, reason, mempool_sequence, this] {
217
5.22k
        m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.TransactionRemovedFromMempool(tx, reason, mempool_sequence); });
218
5.22k
    };
219
5.22k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
Line
Count
Source
160
5.22k
    do {                                                                                                         \
161
5.22k
        static_assert(std::is_rvalue_reference_v<decltype((event))>,                                             \
162
5.22k
                      "event must be passed as an rvalue");                                                      \
163
5.22k
        static_assert(std::is_rvalue_reference_v<decltype((log_msg))>,                                           \
164
5.22k
                      "log_msg must be passed as an rvalue");                                                    \
165
5.22k
        auto enqueue_log_msg = (log_msg);                                                                        \
166
5.22k
        LOG_EVENT("Enqueuing %s", enqueue_log_msg);                                                              \
Line
Count
Source
177
5.22k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
5.22k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
5.22k
    do {                                                               \
115
5.22k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
5.22k
    } while (0)
167
5.22k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
5.22k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
5.22k
            local_event();                                                                                       \
170
5.22k
        });                                                                                                      \
171
5.22k
    } while (0)
220
5.22k
}
221
222
void ValidationSignals::BlockConnected(const ChainstateRole& role, std::shared_ptr<const CBlock> pblock, const CBlockIndex* pindex)
223
186k
{
224
186k
    auto log_msg = LOG_MSG("%s: block hash=%s block height=%d", __func__,
Line
Count
Source
174
186k
    (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? 
tfm::format((fmt), __VA_ARGS__)0
: std::string{})
225
186k
                          pblock->GetHash().ToString(),
226
186k
                          pindex->nHeight);
227
186k
    auto event = [role, pblock = std::move(pblock), pindex, this] {
228
186k
        m_internals->Iterate([&](CValidationInterface& callbacks) 
{ callbacks.BlockConnected(role, pblock, pindex); }185k
);
229
186k
    };
230
186k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
Line
Count
Source
160
186k
    do {                                                                                                         \
161
186k
        static_assert(std::is_rvalue_reference_v<decltype((event))>,                                             \
162
186k
                      "event must be passed as an rvalue");                                                      \
163
186k
        static_assert(std::is_rvalue_reference_v<decltype((log_msg))>,                                           \
164
186k
                      "log_msg must be passed as an rvalue");                                                    \
165
186k
        auto enqueue_log_msg = (log_msg);                                                                        \
166
186k
        LOG_EVENT("Enqueuing %s", enqueue_log_msg);                                                              \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
167
186k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
186k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
186k
            local_event();                                                                                       \
170
186k
        });                                                                                                      \
171
186k
    } while (0)
231
186k
}
232
233
void ValidationSignals::MempoolTransactionsRemovedForBlock(const std::vector<RemovedMempoolTransactionInfo>& txs_removed_for_block, unsigned int nBlockHeight)
234
186k
{
235
186k
    auto log_msg = LOG_MSG("%s: block height=%s txs removed=%s", __func__,
Line
Count
Source
174
186k
    (ShouldLog(BCLog::VALIDATION, BCLog::Level::Debug) ? 
tfm::format((fmt), __VA_ARGS__)0
: std::string{})
236
186k
                          nBlockHeight,
237
186k
                          txs_removed_for_block.size());
238
186k
    auto event = [txs_removed_for_block, nBlockHeight, this] {
239
186k
        m_internals->Iterate([&](CValidationInterface& callbacks) 
{ callbacks.MempoolTransactionsRemovedForBlock(txs_removed_for_block, nBlockHeight); }185k
);
240
186k
    };
241
186k
    ENQUEUE_AND_LOG_EVENT(std::move(event), std::move(log_msg));
Line
Count
Source
160
186k
    do {                                                                                                         \
161
186k
        static_assert(std::is_rvalue_reference_v<decltype((event))>,                                             \
162
186k
                      "event must be passed as an rvalue");                                                      \
163
186k
        static_assert(std::is_rvalue_reference_v<decltype((log_msg))>,                                           \
164
186k
                      "log_msg must be passed as an rvalue");                                                    \
165
186k
        auto enqueue_log_msg = (log_msg);                                                                        \
166
186k
        LOG_EVENT("Enqueuing %s", enqueue_log_msg);                                                              \
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
167
186k
        m_internals->m_task_runner->insert([local_log_msg = std::move(enqueue_log_msg), local_event = (event)] { \
168
186k
            LOG_EVENT("%s", local_log_msg);                                                                      \
169
186k
            local_event();                                                                                       \
170
186k
        });                                                                                                      \
171
186k
    } while (0)
242
186k
}
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
123
0
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
0
    do {                                                               \
115
0
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
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
123
0
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
0
    do {                                                               \
115
0
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
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
186k
{
267
186k
    LOG_EVENT("%s: block hash=%s state=%s", __func__,
Line
Count
Source
177
186k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
186k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
186k
    do {                                                               \
115
186k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
186k
    } while (0)
268
186k
              block->GetHash().ToString(), state.ToString());
269
186k
    m_internals->Iterate([&](CValidationInterface& callbacks) 
{ callbacks.BlockChecked(block, state); }186k
);
270
186k
}
271
272
185k
void ValidationSignals::NewPoWValidBlock(const CBlockIndex *pindex, const std::shared_ptr<const CBlock> &block) {
273
185k
    LOG_EVENT("%s: block hash=%s", __func__, block->GetHash().ToString());
Line
Count
Source
177
185k
    LogDebug(BCLog::VALIDATION, fmt, __VA_ARGS__)
Line
Count
Source
123
185k
#define LogDebug(category, ...) detail_LogIfCategoryAndLevelEnabled(category, BCLog::Level::Debug, __VA_ARGS__)
Line
Count
Source
114
185k
    do {                                                               \
115
185k
        if (util::log::ShouldLog((category), (level))) {               \
116
0
            bool rate_limit{level >= BCLog::Level::Info};              \
117
0
            Assume(!rate_limit); /*Only called with the levels below*/ \
Line
Count
Source
128
0
#define Assume(val) inline_assertion_check<false>(val, std::source_location::current(), #val)
118
0
            LogPrintLevel_(category, level, rate_limit, __VA_ARGS__);  \
Line
Count
Source
97
0
#define LogPrintLevel_(category, level, should_ratelimit, ...) LogPrintFormatInternal(SourceLocation{__func__}, category, level, should_ratelimit, __VA_ARGS__)
119
0
        }                                                              \
120
185k
    } while (0)
274
185k
    m_internals->Iterate([&](CValidationInterface& callbacks) { callbacks.NewPoWValidBlock(pindex, block); });
275
185k
}