Branch data Line data Source code
1 : : // Copyright (c) 2009-2010 Satoshi Nakamoto
2 : : // Copyright (c) 2009-2022 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 <logging.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 : 1 : 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 : 1 : : m_prefix(std::move(prefix)),
34 : 1 : m_title(std::move(end_msg)),
35 : 1 : m_log_category(log_category),
36 : 1 : m_message_on_completion(msg_on_completion)
37 : : {
38 [ + - ][ - + ]: 1 : this->Log(strprintf("%s started", m_title));
39 : 1 : m_start_t = std::chrono::steady_clock::now();
40 : 1 : }
41 : :
42 : 1 : ~Timer()
43 : : {
44 [ + - ]: 1 : if (m_message_on_completion) {
45 [ + - ][ + - ]: 1 : this->Log(strprintf("%s completed", m_title));
46 : 1 : } else {
47 [ # # ][ # # ]: 0 : this->Log("completed");
48 : : }
49 : 1 : }
50 : :
51 : 2 : void Log(const std::string& msg)
52 : : {
53 : 2 : const std::string full_msg = this->LogMsg(msg);
54 : :
55 [ - + ]: 2 : if (m_log_category == BCLog::LogFlags::ALL) {
56 [ + - ][ + - ]: 2 : LogPrintf("%s\n", full_msg);
[ + - ]
57 : 2 : } else {
58 [ # # ][ # # ]: 0 : LogPrint(m_log_category, "%s\n", full_msg);
[ # # ][ # # ]
[ # # ]
59 : : }
60 : 2 : }
61 : :
62 : 2 : std::string LogMsg(const std::string& msg)
63 : : {
64 : 2 : const auto end_time{std::chrono::steady_clock::now()};
65 [ + + ]: 2 : if (!m_start_t) {
66 : 1 : return strprintf("%s: %s", m_prefix, msg);
67 : : }
68 : 1 : const auto duration{end_time - *m_start_t};
69 : :
70 : : if constexpr (std::is_same<TimeType, std::chrono::microseconds>::value) {
71 : : return strprintf("%s: %s (%iμs)", m_prefix, msg, Ticks<std::chrono::microseconds>(duration));
72 : : } else if constexpr (std::is_same<TimeType, std::chrono::milliseconds>::value) {
73 : 0 : return strprintf("%s: %s (%.2fms)", m_prefix, msg, Ticks<MillisecondsDouble>(duration));
74 : : } else if constexpr (std::is_same<TimeType, std::chrono::seconds>::value) {
75 : 1 : 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 : 2 : }
80 : :
81 : : private:
82 : 1 : 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 LogPrint 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 : : BCLog::Timer<std::chrono::microseconds> UNIQUE_NAME(logging_timer)(__func__, end_msg, log_category)
103 : : #define LOG_TIME_MILLIS_WITH_CATEGORY(end_msg, log_category) \
104 : : BCLog::Timer<std::chrono::milliseconds> UNIQUE_NAME(logging_timer)(__func__, end_msg, log_category)
105 : : #define LOG_TIME_MILLIS_WITH_CATEGORY_MSG_ONCE(end_msg, log_category) \
106 : : BCLog::Timer<std::chrono::milliseconds> UNIQUE_NAME(logging_timer)(__func__, end_msg, log_category, /* msg_on_completion=*/false)
107 : : #define LOG_TIME_SECONDS(end_msg) \
108 : : BCLog::Timer<std::chrono::seconds> UNIQUE_NAME(logging_timer)(__func__, end_msg)
109 : :
110 : :
111 : : #endif // BITCOIN_LOGGING_TIMER_H
|