Coverage Report

Created: 2026-06-16 16:41

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/tmp/bitcoin/src/logging/timer.h
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
#ifndef BITCOIN_LOGGING_TIMER_H
7
#define BITCOIN_LOGGING_TIMER_H
8
9
#include <util/log.h>
10
#include <util/macros.h>
11
#include <util/time.h>
12
#include <util/types.h>
13
14
#include <chrono>
15
#include <optional>
16
#include <string>
17
18
19
namespace BCLog {
20
21
//! RAII-style object that outputs timing information to logs.
22
template <typename TimeType>
23
class Timer
24
{
25
public:
26
    //! If log_category is left as the default, end_msg will log unconditionally
27
    //! (instead of being filtered by category).
28
    Timer(
29
        std::string prefix,
30
        std::string end_msg,
31
        BCLog::LogFlags log_category = BCLog::LogFlags::ALL,
32
        bool msg_on_completion = true)
33
783k
        : m_prefix(std::move(prefix)),
34
783k
          m_title(std::move(end_msg)),
35
783k
          m_log_category(log_category),
36
783k
          m_message_on_completion(msg_on_completion)
37
783k
    {
38
783k
        this->Log(strprintf("%s started", m_title));
39
783k
        m_start_t = std::chrono::steady_clock::now();
40
783k
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1000000l>>>::Timer(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>>, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>>, BCLog::LogFlags, bool)
Line
Count
Source
33
772k
        : m_prefix(std::move(prefix)),
34
772k
          m_title(std::move(end_msg)),
35
772k
          m_log_category(log_category),
36
772k
          m_message_on_completion(msg_on_completion)
37
772k
    {
38
772k
        this->Log(strprintf("%s started", m_title));
39
772k
        m_start_t = std::chrono::steady_clock::now();
40
772k
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1l>>>::Timer(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>>, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>>, BCLog::LogFlags, bool)
Line
Count
Source
33
77
        : m_prefix(std::move(prefix)),
34
77
          m_title(std::move(end_msg)),
35
77
          m_log_category(log_category),
36
77
          m_message_on_completion(msg_on_completion)
37
77
    {
38
77
        this->Log(strprintf("%s started", m_title));
39
77
        m_start_t = std::chrono::steady_clock::now();
40
77
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1000l>>>::Timer(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>>, std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>>, BCLog::LogFlags, bool)
Line
Count
Source
33
11.6k
        : m_prefix(std::move(prefix)),
34
11.6k
          m_title(std::move(end_msg)),
35
11.6k
          m_log_category(log_category),
36
11.6k
          m_message_on_completion(msg_on_completion)
37
11.6k
    {
38
11.6k
        this->Log(strprintf("%s started", m_title));
39
11.6k
        m_start_t = std::chrono::steady_clock::now();
40
11.6k
    }
41
42
    ~Timer()
43
784k
    {
44
784k
        if (m_message_on_completion) {
45
783k
            this->Log(strprintf("%s completed", m_title));
46
783k
        } else {
47
939
            this->Log("completed");
48
939
        }
49
784k
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1000000l>>>::~Timer()
Line
Count
Source
43
772k
    {
44
772k
        if (m_message_on_completion) {
45
772k
            this->Log(strprintf("%s completed", m_title));
46
772k
        } else {
47
0
            this->Log("completed");
48
0
        }
49
772k
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1l>>>::~Timer()
Line
Count
Source
43
77
    {
44
77
        if (m_message_on_completion) {
45
77
            this->Log(strprintf("%s completed", m_title));
46
77
        } else {
47
0
            this->Log("completed");
48
0
        }
49
77
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1000l>>>::~Timer()
Line
Count
Source
43
11.6k
    {
44
11.6k
        if (m_message_on_completion) {
45
10.7k
            this->Log(strprintf("%s completed", m_title));
46
10.7k
        } else {
47
939
            this->Log("completed");
48
939
        }
49
11.6k
    }
50
51
    void Log(const std::string& msg)
52
1.56M
    {
53
1.56M
        const std::string full_msg = this->LogMsg(msg);
54
55
1.56M
        if (m_log_category == BCLog::LogFlags::ALL) {
56
200
            LogInfo("%s\n", full_msg);
57
1.56M
        } else {
58
1.56M
            LogDebug(m_log_category, "%s\n", full_msg);
59
1.56M
        }
60
1.56M
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1000000l>>>::Log(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>> const&)
Line
Count
Source
52
1.54M
    {
53
1.54M
        const std::string full_msg = this->LogMsg(msg);
54
55
1.54M
        if (m_log_category == BCLog::LogFlags::ALL) {
56
2
            LogInfo("%s\n", full_msg);
57
1.54M
        } else {
58
1.54M
            LogDebug(m_log_category, "%s\n", full_msg);
59
1.54M
        }
60
1.54M
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1l>>>::Log(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>> const&)
Line
Count
Source
52
154
    {
53
154
        const std::string full_msg = this->LogMsg(msg);
54
55
154
        if (m_log_category == BCLog::LogFlags::ALL) {
56
154
            LogInfo("%s\n", full_msg);
57
154
        } else {
58
0
            LogDebug(m_log_category, "%s\n", full_msg);
59
0
        }
60
154
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1000l>>>::Log(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>> const&)
Line
Count
Source
52
23.3k
    {
53
23.3k
        const std::string full_msg = this->LogMsg(msg);
54
55
23.3k
        if (m_log_category == BCLog::LogFlags::ALL) {
56
44
            LogInfo("%s\n", full_msg);
57
23.3k
        } else {
58
23.3k
            LogDebug(m_log_category, "%s\n", full_msg);
59
23.3k
        }
60
23.3k
    }
61
62
    std::string LogMsg(const std::string& msg)
63
1.56M
    {
64
1.56M
        const auto end_time{std::chrono::steady_clock::now()};
65
1.56M
        if (!m_start_t) {
66
783k
            return strprintf("%s: %s", m_prefix, msg);
67
783k
        }
68
783k
        const auto duration{end_time - *m_start_t};
69
70
783k
        if constexpr (std::is_same_v<TimeType, std::chrono::microseconds>) {
71
771k
            return strprintf("%s: %s (%iμs)", m_prefix, msg, Ticks<std::chrono::microseconds>(duration));
72
771k
        } else if constexpr (std::is_same_v<TimeType, std::chrono::milliseconds>) {
73
11.6k
            return strprintf("%s: %s (%.2fms)", m_prefix, msg, Ticks<MillisecondsDouble>(duration));
74
11.6k
        } else if constexpr (std::is_same_v<TimeType, std::chrono::seconds>) {
75
77
            return strprintf("%s: %s (%.2fs)", m_prefix, msg, Ticks<SecondsDouble>(duration));
76
        } else {
77
            static_assert(ALWAYS_FALSE<TimeType>, "Error: unexpected time type");
78
        }
79
783k
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1000000l>>>::LogMsg(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>> const&)
Line
Count
Source
63
1.54M
    {
64
1.54M
        const auto end_time{std::chrono::steady_clock::now()};
65
1.54M
        if (!m_start_t) {
66
772k
            return strprintf("%s: %s", m_prefix, msg);
67
772k
        }
68
771k
        const auto duration{end_time - *m_start_t};
69
70
771k
        if constexpr (std::is_same_v<TimeType, std::chrono::microseconds>) {
71
771k
            return strprintf("%s: %s (%iμs)", m_prefix, msg, Ticks<std::chrono::microseconds>(duration));
72
        } else if constexpr (std::is_same_v<TimeType, std::chrono::milliseconds>) {
73
            return strprintf("%s: %s (%.2fms)", m_prefix, msg, Ticks<MillisecondsDouble>(duration));
74
        } else if constexpr (std::is_same_v<TimeType, std::chrono::seconds>) {
75
            return strprintf("%s: %s (%.2fs)", m_prefix, msg, Ticks<SecondsDouble>(duration));
76
        } else {
77
            static_assert(ALWAYS_FALSE<TimeType>, "Error: unexpected time type");
78
        }
79
771k
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1l>>>::LogMsg(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>> const&)
Line
Count
Source
63
154
    {
64
154
        const auto end_time{std::chrono::steady_clock::now()};
65
154
        if (!m_start_t) {
66
77
            return strprintf("%s: %s", m_prefix, msg);
67
77
        }
68
77
        const auto duration{end_time - *m_start_t};
69
70
        if constexpr (std::is_same_v<TimeType, std::chrono::microseconds>) {
71
            return strprintf("%s: %s (%iμs)", m_prefix, msg, Ticks<std::chrono::microseconds>(duration));
72
        } else if constexpr (std::is_same_v<TimeType, std::chrono::milliseconds>) {
73
            return strprintf("%s: %s (%.2fms)", m_prefix, msg, Ticks<MillisecondsDouble>(duration));
74
77
        } else if constexpr (std::is_same_v<TimeType, std::chrono::seconds>) {
75
77
            return strprintf("%s: %s (%.2fs)", m_prefix, msg, Ticks<SecondsDouble>(duration));
76
        } else {
77
            static_assert(ALWAYS_FALSE<TimeType>, "Error: unexpected time type");
78
        }
79
77
    }
BCLog::Timer<std::chrono::duration<long, std::ratio<1l, 1000l>>>::LogMsg(std::__cxx11::basic_string<char, std::char_traits<char>, std::allocator<char>> const&)
Line
Count
Source
63
23.3k
    {
64
23.3k
        const auto end_time{std::chrono::steady_clock::now()};
65
23.3k
        if (!m_start_t) {
66
11.6k
            return strprintf("%s: %s", m_prefix, msg);
67
11.6k
        }
68
11.6k
        const auto duration{end_time - *m_start_t};
69
70
        if constexpr (std::is_same_v<TimeType, std::chrono::microseconds>) {
71
            return strprintf("%s: %s (%iμs)", m_prefix, msg, Ticks<std::chrono::microseconds>(duration));
72
11.6k
        } else if constexpr (std::is_same_v<TimeType, std::chrono::milliseconds>) {
73
11.6k
            return strprintf("%s: %s (%.2fms)", m_prefix, msg, Ticks<MillisecondsDouble>(duration));
74
        } else if constexpr (std::is_same_v<TimeType, std::chrono::seconds>) {
75
            return strprintf("%s: %s (%.2fs)", m_prefix, msg, Ticks<SecondsDouble>(duration));
76
        } else {
77
            static_assert(ALWAYS_FALSE<TimeType>, "Error: unexpected time type");
78
        }
79
11.6k
    }
80
81
private:
82
    std::optional<std::chrono::steady_clock::time_point> m_start_t{};
83
84
    //! Log prefix; usually the name of the function this was created in.
85
    const std::string m_prefix;
86
87
    //! A descriptive message of what is being timed.
88
    const std::string m_title;
89
90
    //! Forwarded on to LogDebug if specified - has the effect of only
91
    //! outputting the timing log when a particular debug= category is specified.
92
    const BCLog::LogFlags m_log_category;
93
94
    //! Whether to output the message again on completion.
95
    const bool m_message_on_completion;
96
};
97
98
} // namespace BCLog
99
100
101
#define LOG_TIME_MICROS_WITH_CATEGORY(end_msg, log_category) \
102
772k
    BCLog::Timer<std::chrono::microseconds> BITCOIN_UNIQUE_NAME(logging_timer)(__func__, end_msg, log_category)
103
#define LOG_TIME_MILLIS_WITH_CATEGORY(end_msg, log_category) \
104
10.7k
    BCLog::Timer<std::chrono::milliseconds> BITCOIN_UNIQUE_NAME(logging_timer)(__func__, end_msg, log_category)
105
#define LOG_TIME_MILLIS_WITH_CATEGORY_MSG_ONCE(end_msg, log_category) \
106
939
    BCLog::Timer<std::chrono::milliseconds> BITCOIN_UNIQUE_NAME(logging_timer)(__func__, end_msg, log_category, /* msg_on_completion=*/false)
107
#define LOG_TIME_SECONDS(end_msg) \
108
77
    BCLog::Timer<std::chrono::seconds> BITCOIN_UNIQUE_NAME(logging_timer)(__func__, end_msg)
109
110
111
#endif // BITCOIN_LOGGING_TIMER_H