Coverage Report

Created: 2026-04-29 19:21

next uncovered line (L), next uncovered region (R), next uncovered branch (B)
/tmp/bitcoin/src/logging.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 <logging.h>
7
#include <memusage.h>
8
#include <util/check.h>
9
#include <util/fs.h>
10
#include <util/string.h>
11
#include <util/threadnames.h>
12
#include <util/time.h>
13
14
#include <array>
15
#include <cstring>
16
#include <map>
17
#include <optional>
18
#include <utility>
19
20
using util::Join;
21
using util::RemovePrefixView;
22
23
const char * const DEFAULT_DEBUGLOGFILE = "debug.log";
24
constexpr auto MAX_USER_SETABLE_SEVERITY_LEVEL{BCLog::Level::Info};
25
26
BCLog::Logger& LogInstance()
27
11.6M
{
28
/**
29
 * NOTE: the logger instances is leaked on exit. This is ugly, but will be
30
 * cleaned up by the OS/libc. Defining a logger as a global object doesn't work
31
 * since the order of destruction of static/global objects is undefined.
32
 * Consider if the logger gets destroyed, and then some later destructor calls
33
 * LogInfo, maybe indirectly, and you get a core dump at shutdown trying to
34
 * access the logger. When the shutdown sequence is fully audited and tested,
35
 * explicit destruction of these objects can be implemented by changing this
36
 * from a raw pointer to a std::unique_ptr.
37
 * Since the ~Logger() destructor is never called, the Logger class and all
38
 * its subclasses must have implicitly-defined destructors.
39
 *
40
 * This method of initialization was originally introduced in
41
 * ee3374234c60aba2cc4c5cd5cac1c0aefc2d817c.
42
 */
43
11.6M
    static BCLog::Logger* g_logger{new BCLog::Logger()};
44
11.6M
    return *g_logger;
45
11.6M
}
46
47
bool fLogIPs = DEFAULT_LOGIPS;
48
49
static int FileWriteStr(std::string_view str, FILE *fp)
50
5.23M
{
51
5.23M
    return fwrite(str.data(), 1, str.size(), fp);
52
5.23M
}
53
54
bool BCLog::Logger::StartLogging()
55
1.78k
{
56
1.78k
    STDLOCK(m_cs);
57
58
1.78k
    assert(m_buffering);
59
1.78k
    assert(m_fileout == nullptr);
60
61
1.78k
    if (m_print_to_file) {
62
1.78k
        assert(!m_file_path.empty());
63
1.78k
        m_fileout = fsbridge::fopen(m_file_path, "a");
64
1.78k
        if (!m_fileout) {
65
2
            return false;
66
2
        }
67
68
1.77k
        setbuf(m_fileout, nullptr); // unbuffered
69
70
        // Add newlines to the logfile to distinguish this execution from the
71
        // last one.
72
1.77k
        FileWriteStr("\n\n\n\n\n", m_fileout);
73
1.77k
    }
74
75
    // dump buffered messages from before we opened the log
76
1.78k
    m_buffering = false;
77
1.78k
    if (m_buffer_lines_discarded > 0) {
78
0
        LogPrint_({
79
0
            .category = BCLog::ALL,
80
0
            .level = Level::Info,
81
0
            .should_ratelimit = false,
82
0
            .source_loc = SourceLocation{__func__},
83
0
            .message = strprintf("Early logging buffer overflowed, %d log lines discarded.", m_buffer_lines_discarded),
84
0
        });
85
0
    }
86
8.30k
    while (!m_msgs_before_open.empty()) {
87
6.52k
        const auto& buflog = m_msgs_before_open.front();
88
6.52k
        std::string s{Format(buflog)};
89
6.52k
        m_msgs_before_open.pop_front();
90
91
6.52k
        if (m_print_to_file) FileWriteStr(s, m_fileout);
92
6.52k
        if (m_print_to_console) fwrite(s.data(), 1, s.size(), stdout);
93
6.52k
        for (const auto& cb : m_print_callbacks) {
94
0
            cb(s);
95
0
        }
96
6.52k
    }
97
1.78k
    m_cur_buffer_memusage = 0;
98
1.78k
    if (m_print_to_console) fflush(stdout);
99
100
1.78k
    return true;
101
1.78k
}
102
103
void BCLog::Logger::DisconnectTestLogger()
104
668
{
105
668
    STDLOCK(m_cs);
106
668
    m_buffering = true;
107
668
    if (m_fileout != nullptr) fclose(m_fileout);
108
668
    m_fileout = nullptr;
109
668
    m_print_callbacks.clear();
110
668
    m_max_buffer_memusage = DEFAULT_MAX_LOG_BUFFER;
111
668
    m_cur_buffer_memusage = 0;
112
668
    m_buffer_lines_discarded = 0;
113
668
    m_msgs_before_open.clear();
114
668
}
115
116
void BCLog::Logger::DisableLogging()
117
0
{
118
0
    {
119
0
        STDLOCK(m_cs);
120
0
        assert(m_buffering);
121
0
        assert(m_print_callbacks.empty());
122
0
    }
123
0
    m_print_to_file = false;
124
0
    m_print_to_console = false;
125
0
    StartLogging();
126
0
}
127
128
void BCLog::Logger::EnableCategory(BCLog::LogFlags flag)
129
1.81k
{
130
1.81k
    m_categories |= flag;
131
1.81k
}
132
133
bool BCLog::Logger::EnableCategory(std::string_view str)
134
1.80k
{
135
1.80k
    if (const auto flag{GetLogCategory(str)}) {
136
1.80k
        EnableCategory(*flag);
137
1.80k
        return true;
138
1.80k
    }
139
1
    return false;
140
1.80k
}
141
142
void BCLog::Logger::DisableCategory(BCLog::LogFlags flag)
143
4.74k
{
144
4.74k
    m_categories &= ~flag;
145
4.74k
}
146
147
bool BCLog::Logger::DisableCategory(std::string_view str)
148
4.72k
{
149
4.72k
    if (const auto flag{GetLogCategory(str)}) {
150
4.72k
        DisableCategory(*flag);
151
4.72k
        return true;
152
4.72k
    }
153
1
    return false;
154
4.72k
}
155
156
bool BCLog::Logger::WillLogCategory(BCLog::LogFlags category) const
157
6.49M
{
158
6.49M
    return (m_categories.load(std::memory_order_relaxed) & category) != 0;
159
6.49M
}
160
161
bool BCLog::Logger::WillLogCategoryLevel(BCLog::LogFlags category, BCLog::Level level) const
162
6.30M
{
163
    // Log messages at Info, Warning and Error level unconditionally, so that
164
    // important troubleshooting information doesn't get lost.
165
6.30M
    if (level >= BCLog::Level::Info) return true;
166
167
6.30M
    if (!WillLogCategory(category)) return false;
168
169
5.24M
    STDLOCK(m_cs);
170
5.24M
    const auto it{m_category_log_levels.find(category)};
171
18.4E
    return level >= (it == m_category_log_levels.end() ? LogLevel() : it->second);
172
6.30M
}
173
174
bool BCLog::Logger::DefaultShrinkDebugFile() const
175
1.11k
{
176
1.11k
    return m_categories == BCLog::NONE;
177
1.11k
}
178
179
static const std::map<std::string, BCLog::LogFlags, std::less<>> LOG_CATEGORIES_BY_STR{
180
    {"net", BCLog::NET},
181
    {"tor", BCLog::TOR},
182
    {"mempool", BCLog::MEMPOOL},
183
    {"http", BCLog::HTTP},
184
    {"bench", BCLog::BENCH},
185
    {"zmq", BCLog::ZMQ},
186
    {"walletdb", BCLog::WALLETDB},
187
    {"rpc", BCLog::RPC},
188
    {"estimatefee", BCLog::ESTIMATEFEE},
189
    {"addrman", BCLog::ADDRMAN},
190
    {"selectcoins", BCLog::SELECTCOINS},
191
    {"reindex", BCLog::REINDEX},
192
    {"cmpctblock", BCLog::CMPCTBLOCK},
193
    {"rand", BCLog::RAND},
194
    {"prune", BCLog::PRUNE},
195
    {"proxy", BCLog::PROXY},
196
    {"mempoolrej", BCLog::MEMPOOLREJ},
197
    {"libevent", BCLog::LIBEVENT},
198
    {"coindb", BCLog::COINDB},
199
    {"qt", BCLog::QT},
200
    {"leveldb", BCLog::LEVELDB},
201
    {"validation", BCLog::VALIDATION},
202
    {"i2p", BCLog::I2P},
203
    {"ipc", BCLog::IPC},
204
#ifdef DEBUG_LOCKCONTENTION
205
    {"lock", BCLog::LOCK},
206
#endif
207
    {"blockstorage", BCLog::BLOCKSTORAGE},
208
    {"txreconciliation", BCLog::TXRECONCILIATION},
209
    {"scan", BCLog::SCAN},
210
    {"txpackages", BCLog::TXPACKAGES},
211
    {"kernel", BCLog::KERNEL},
212
    {"privatebroadcast", BCLog::PRIVBROADCAST},
213
};
214
215
static const std::unordered_map<BCLog::LogFlags, std::string> LOG_CATEGORIES_BY_FLAG{
216
    // Swap keys and values from LOG_CATEGORIES_BY_STR.
217
2.58k
    [](const auto& in) {
218
2.58k
        std::unordered_map<BCLog::LogFlags, std::string> out;
219
80.0k
        for (const auto& [k, v] : in) {
220
80.0k
            const bool inserted{out.emplace(v, k).second};
221
80.0k
            assert(inserted);
222
80.0k
        }
223
2.58k
        return out;
224
2.58k
    }(LOG_CATEGORIES_BY_STR)
225
};
226
227
std::optional<BCLog::LogFlags> GetLogCategory(std::string_view str)
228
6.57k
{
229
6.57k
    if (str.empty() || str == "1" || str == "all") {
230
1.79k
        return BCLog::ALL;
231
1.79k
    }
232
4.77k
    auto it = LOG_CATEGORIES_BY_STR.find(str);
233
4.77k
    if (it != LOG_CATEGORIES_BY_STR.end()) {
234
4.77k
        return it->second;
235
4.77k
    }
236
2
    return std::nullopt;
237
4.77k
}
238
239
std::string BCLog::Logger::LogLevelToStr(BCLog::Level level)
240
456k
{
241
456k
    switch (level) {
242
445k
    case BCLog::Level::Trace:
243
445k
        return "trace";
244
3.69k
    case BCLog::Level::Debug:
245
3.69k
        return "debug";
246
1.84k
    case BCLog::Level::Info:
247
1.84k
        return "info";
248
1.76k
    case BCLog::Level::Warning:
249
1.76k
        return "warning";
250
3.43k
    case BCLog::Level::Error:
251
3.43k
        return "error";
252
456k
    }
253
456k
    assert(false);
254
0
}
255
256
static std::string LogCategoryToStr(BCLog::LogFlags category)
257
4.84M
{
258
4.84M
    if (category == BCLog::ALL) {
259
0
        return "all";
260
0
    }
261
4.84M
    auto it = LOG_CATEGORIES_BY_FLAG.find(category);
262
4.84M
    assert(it != LOG_CATEGORIES_BY_FLAG.end());
263
4.84M
    return it->second;
264
4.84M
}
265
266
static std::optional<BCLog::Level> GetLogLevel(std::string_view level_str)
267
1.80k
{
268
1.80k
    if (level_str == "trace") {
269
1.79k
        return BCLog::Level::Trace;
270
1.79k
    } else if (level_str == "debug") {
271
2
        return BCLog::Level::Debug;
272
4
    } else if (level_str == "info") {
273
2
        return BCLog::Level::Info;
274
2
    } else if (level_str == "warning") {
275
0
        return BCLog::Level::Warning;
276
2
    } else if (level_str == "error") {
277
0
        return BCLog::Level::Error;
278
2
    } else {
279
2
        return std::nullopt;
280
2
    }
281
1.80k
}
282
283
std::vector<LogCategory> BCLog::Logger::LogCategoriesList() const
284
6.02k
{
285
6.02k
    std::vector<LogCategory> ret;
286
6.02k
    ret.reserve(LOG_CATEGORIES_BY_STR.size());
287
186k
    for (const auto& [category, flag] : LOG_CATEGORIES_BY_STR) {
288
186k
        ret.push_back(LogCategory{.category = category, .active = WillLogCategory(flag)});
289
186k
    }
290
6.02k
    return ret;
291
6.02k
}
292
293
/** Log severity levels that can be selected by the user. */
294
static constexpr std::array<BCLog::Level, 3> LogLevelsList()
295
1.84k
{
296
1.84k
    return {BCLog::Level::Info, BCLog::Level::Debug, BCLog::Level::Trace};
297
1.84k
}
298
299
std::string BCLog::Logger::LogLevelsString() const
300
1.84k
{
301
1.84k
    const auto& levels = LogLevelsList();
302
5.54k
    return Join(std::vector<BCLog::Level>{levels.begin(), levels.end()}, ", ", [](BCLog::Level level) { return LogLevelToStr(level); });
303
1.84k
}
304
305
std::string BCLog::Logger::LogTimestampStr(SystemClock::time_point now, std::chrono::seconds mocktime) const
306
5.23M
{
307
5.23M
    std::string strStamped;
308
309
5.23M
    if (!m_log_timestamps)
310
91
        return strStamped;
311
312
5.23M
    const auto now_seconds{std::chrono::time_point_cast<std::chrono::seconds>(now)};
313
5.23M
    strStamped = FormatISO8601DateTime(TicksSinceEpoch<std::chrono::seconds>(now_seconds));
314
5.23M
    if (m_log_time_micros && !strStamped.empty()) {
315
5.23M
        strStamped.pop_back();
316
5.23M
        strStamped += strprintf(".%06dZ", Ticks<std::chrono::microseconds>(now - now_seconds));
317
5.23M
    }
318
5.23M
    if (mocktime > 0s) {
319
1.12M
        strStamped += " (mocktime: " + FormatISO8601DateTime(count_seconds(mocktime)) + ")";
320
1.12M
    }
321
5.23M
    strStamped += ' ';
322
323
5.23M
    return strStamped;
324
5.23M
}
325
326
namespace BCLog {
327
    /** Belts and suspenders: make sure outgoing log messages don't contain
328
     * potentially suspicious characters, such as terminal control codes.
329
     *
330
     * This escapes control characters except newline ('\n') in C syntax.
331
     * It escapes instead of removes them to still allow for troubleshooting
332
     * issues where they accidentally end up in strings.
333
     */
334
5.23M
    std::string LogEscapeMessage(std::string_view str) {
335
5.23M
        std::string ret;
336
455M
        for (char ch_in : str) {
337
455M
            uint8_t ch = (uint8_t)ch_in;
338
455M
            if ((ch >= 32 || ch == '\n') && ch != '\x7f') {
339
455M
                ret += ch_in;
340
455M
            } else {
341
6.82k
                ret += strprintf("\\x%02x", ch);
342
6.82k
            }
343
455M
        }
344
5.23M
        return ret;
345
5.23M
    }
346
} // namespace BCLog
347
348
std::string BCLog::Logger::GetLogPrefix(BCLog::LogFlags category, BCLog::Level level) const
349
5.23M
{
350
5.23M
    if (category == LogFlags::NONE) category = LogFlags::ALL;
351
352
5.23M
    const bool has_category{m_always_print_category_level || category != LogFlags::ALL};
353
354
    // If there is no category, Info is implied
355
5.23M
    if (!has_category && level == Level::Info) return {};
356
357
4.84M
    std::string s{"["};
358
4.84M
    if (has_category) {
359
4.84M
        s += LogCategoryToStr(category);
360
4.84M
    }
361
362
4.84M
    if (m_always_print_category_level || !has_category || level != Level::Debug) {
363
        // If there is a category, Debug is implied, so don't add the level
364
365
        // Only add separator if we have a category
366
448k
        if (has_category) s += ":";
367
448k
        s += Logger::LogLevelToStr(level);
368
448k
    }
369
370
4.84M
    s += "] ";
371
4.84M
    return s;
372
5.23M
}
373
374
static size_t MemUsage(const util::log::Entry& log)
375
29.1k
{
376
29.1k
    return memusage::DynamicUsage(log.message) +
377
29.1k
           memusage::DynamicUsage(log.thread_name) +
378
29.1k
           memusage::MallocUsage(sizeof(memusage::list_node<util::log::Entry>));
379
29.1k
}
380
381
BCLog::LogRateLimiter::LogRateLimiter(uint64_t max_bytes, std::chrono::seconds reset_window)
382
3
    : m_max_bytes{max_bytes}, m_reset_window{reset_window} {}
383
384
std::shared_ptr<BCLog::LogRateLimiter> BCLog::LogRateLimiter::Create(
385
    SchedulerFunction&& scheduler_func, uint64_t max_bytes, std::chrono::seconds reset_window)
386
3
{
387
3
    auto limiter{std::shared_ptr<LogRateLimiter>(new LogRateLimiter(max_bytes, reset_window))};
388
3
    std::weak_ptr<LogRateLimiter> weak_limiter{limiter};
389
3
    auto reset = [weak_limiter] {
390
2
        if (auto shared_limiter{weak_limiter.lock()}) shared_limiter->Reset();
391
2
    };
392
3
    scheduler_func(reset, limiter->m_reset_window);
393
3
    return limiter;
394
3
}
395
396
BCLog::LogRateLimiter::Status BCLog::LogRateLimiter::Consume(
397
    const SourceLocation& source_loc,
398
    const std::string& str)
399
89
{
400
89
    STDLOCK(m_mutex);
401
89
    auto& stats{m_source_locations.try_emplace(source_loc, m_max_bytes).first->second};
402
89
    Status status{stats.m_dropped_bytes > 0 ? Status::STILL_SUPPRESSED : Status::UNSUPPRESSED};
403
404
89
    if (!stats.Consume(str.size()) && status == Status::UNSUPPRESSED) {
405
3
        status = Status::NEWLY_SUPPRESSED;
406
3
        m_suppression_active = true;
407
3
    }
408
409
89
    return status;
410
89
}
411
412
std::string BCLog::Logger::Format(const util::log::Entry& entry) const
413
5.23M
{
414
5.23M
    std::string result{LogTimestampStr(entry.timestamp, entry.mocktime)};
415
416
5.23M
    if (m_log_threadnames) {
417
5.23M
        result += strprintf("[%s] ", (entry.thread_name.empty() ? "unknown" : entry.thread_name));
418
5.23M
    }
419
420
5.23M
    if (m_log_sourcelocations) {
421
5.23M
        result += strprintf("[%s:%d] [%s] ", RemovePrefixView(entry.source_loc.file_name(), "./"), entry.source_loc.line(), entry.source_loc.function_name_short());
422
5.23M
    }
423
424
5.23M
    result += GetLogPrefix(static_cast<LogFlags>(entry.category), entry.level);
425
5.23M
    result += LogEscapeMessage(entry.message);
426
427
5.23M
    if (!result.ends_with('\n')) result += '\n';
428
5.23M
    return result;
429
5.23M
}
430
431
void BCLog::Logger::LogPrint(util::log::Entry entry)
432
5.25M
{
433
5.25M
    STDLOCK(m_cs);
434
5.25M
    return LogPrint_(std::move(entry));
435
5.25M
}
436
437
// NOLINTNEXTLINE(misc-no-recursion)
438
void BCLog::Logger::LogPrint_(util::log::Entry entry)
439
5.25M
{
440
5.25M
    if (m_buffering) {
441
29.1k
        {
442
29.1k
            m_cur_buffer_memusage += MemUsage(entry);
443
29.1k
            m_msgs_before_open.push_back(std::move(entry));
444
29.1k
        }
445
446
29.1k
        while (m_cur_buffer_memusage > m_max_buffer_memusage) {
447
0
            if (m_msgs_before_open.empty()) {
448
0
                m_cur_buffer_memusage = 0;
449
0
                break;
450
0
            }
451
0
            m_cur_buffer_memusage -= MemUsage(m_msgs_before_open.front());
452
0
            m_msgs_before_open.pop_front();
453
0
            ++m_buffer_lines_discarded;
454
0
        }
455
456
29.1k
        return;
457
29.1k
    }
458
459
5.22M
    std::string str_prefixed{Format(entry)};
460
5.22M
    bool ratelimit{false};
461
5.22M
    if (entry.should_ratelimit && m_limiter) {
462
81
        auto status{m_limiter->Consume(entry.source_loc, str_prefixed)};
463
81
        if (status == LogRateLimiter::Status::NEWLY_SUPPRESSED) {
464
            // NOLINTNEXTLINE(misc-no-recursion)
465
1
            LogPrint_({
466
1
                .category = LogFlags::ALL,
467
1
                .level = Level::Warning,
468
1
                .should_ratelimit = false, // with should_ratelimit=false, this cannot lead to infinite recursion
469
1
                .source_loc = SourceLocation{__func__},
470
1
                .message = strprintf(
471
1
                    "Excessive logging detected from %s:%d (%s): >%d bytes logged during "
472
1
                    "the last time window of %is. Suppressing logging to disk from this "
473
1
                    "source location until time window resets. Console logging "
474
1
                    "unaffected. Last log entry.",
475
1
                    entry.source_loc.file_name(), entry.source_loc.line(), entry.source_loc.function_name_short(),
476
1
                    m_limiter->m_max_bytes,
477
1
                    Ticks<std::chrono::seconds>(m_limiter->m_reset_window)),
478
1
            });
479
80
        } else if (status == LogRateLimiter::Status::STILL_SUPPRESSED) {
480
1
            ratelimit = true;
481
1
        }
482
81
    }
483
484
    // To avoid confusion caused by dropped log messages when debugging an issue,
485
    // we prefix log lines with "[*]" when there are any suppressed source locations.
486
5.22M
    if (m_limiter && m_limiter->SuppressionsActive()) {
487
4
        str_prefixed.insert(0, "[*] ");
488
4
    }
489
490
5.22M
    if (m_print_to_console) {
491
        // print to console
492
419k
        fwrite(str_prefixed.data(), 1, str_prefixed.size(), stdout);
493
419k
        fflush(stdout);
494
419k
    }
495
5.22M
    for (const auto& cb : m_print_callbacks) {
496
1.05k
        cb(str_prefixed);
497
1.05k
    }
498
5.22M
    if (m_print_to_file && !ratelimit) {
499
5.22M
        assert(m_fileout != nullptr);
500
501
        // reopen the log file, if requested
502
5.22M
        if (m_reopen_file) {
503
6
            m_reopen_file = false;
504
6
            FILE* new_fileout = fsbridge::fopen(m_file_path, "a");
505
6
            if (new_fileout) {
506
6
                setbuf(new_fileout, nullptr); // unbuffered
507
6
                fclose(m_fileout);
508
6
                m_fileout = new_fileout;
509
6
            }
510
6
        }
511
5.22M
        FileWriteStr(str_prefixed, m_fileout);
512
5.22M
    }
513
5.22M
}
514
515
void BCLog::Logger::ShrinkDebugFile()
516
0
{
517
    // Amount of debug.log to save at end when shrinking (must fit in memory)
518
0
    constexpr size_t RECENT_DEBUG_HISTORY_SIZE = 10 * 1000000;
519
520
0
    assert(!m_file_path.empty());
521
522
    // Scroll debug.log if it's getting too big
523
0
    FILE* file = fsbridge::fopen(m_file_path, "r");
524
525
    // Special files (e.g. device nodes) may not have a size.
526
0
    size_t log_size = 0;
527
0
    try {
528
0
        log_size = fs::file_size(m_file_path);
529
0
    } catch (const fs::filesystem_error&) {}
530
531
    // If debug.log file is more than 10% bigger the RECENT_DEBUG_HISTORY_SIZE
532
    // trim it down by saving only the last RECENT_DEBUG_HISTORY_SIZE bytes
533
0
    if (file && log_size > 11 * (RECENT_DEBUG_HISTORY_SIZE / 10))
534
0
    {
535
        // Restart the file with some of the end
536
0
        std::vector<char> vch(RECENT_DEBUG_HISTORY_SIZE, 0);
537
0
        if (fseek(file, -((long)vch.size()), SEEK_END)) {
538
0
            LogWarning("Failed to shrink debug log file: fseek(...) failed");
539
0
            fclose(file);
540
0
            return;
541
0
        }
542
0
        int nBytes = fread(vch.data(), 1, vch.size(), file);
543
0
        fclose(file);
544
545
0
        file = fsbridge::fopen(m_file_path, "w");
546
0
        if (file)
547
0
        {
548
0
            fwrite(vch.data(), 1, nBytes, file);
549
0
            fclose(file);
550
0
        }
551
0
    }
552
0
    else if (file != nullptr)
553
0
        fclose(file);
554
0
}
555
556
void BCLog::LogRateLimiter::Reset()
557
3
{
558
3
    decltype(m_source_locations) source_locations;
559
3
    {
560
3
        STDLOCK(m_mutex);
561
3
        source_locations.swap(m_source_locations);
562
3
        m_suppression_active = false;
563
3
    }
564
4
    for (const auto& [source_loc, stats] : source_locations) {
565
4
        if (stats.m_dropped_bytes == 0) continue;
566
3
        LogPrintLevel_(
567
3
            LogFlags::ALL, Level::Warning, /*should_ratelimit=*/false,
568
3
            "Restarting logging from %s:%d (%s): %d bytes were dropped during the last %ss.",
569
3
            source_loc.file_name(), source_loc.line(), source_loc.function_name_short(),
570
3
            stats.m_dropped_bytes, Ticks<std::chrono::seconds>(m_reset_window));
571
3
    }
572
3
}
573
574
bool BCLog::LogRateLimiter::Stats::Consume(uint64_t bytes)
575
92
{
576
92
    if (bytes > m_available_bytes) {
577
6
        m_dropped_bytes += bytes;
578
6
        m_available_bytes = 0;
579
6
        return false;
580
6
    }
581
582
86
    m_available_bytes -= bytes;
583
86
    return true;
584
92
}
585
586
bool BCLog::Logger::SetLogLevel(std::string_view level_str)
587
1.79k
{
588
1.79k
    const auto level = GetLogLevel(level_str);
589
1.79k
    if (!level.has_value() || level.value() > MAX_USER_SETABLE_SEVERITY_LEVEL) return false;
590
1.79k
    m_log_level = level.value();
591
1.79k
    return true;
592
1.79k
}
593
594
bool BCLog::Logger::SetCategoryLogLevel(std::string_view category_str, std::string_view level_str)
595
5
{
596
5
    const auto flag{GetLogCategory(category_str)};
597
5
    if (!flag) return false;
598
599
5
    const auto level = GetLogLevel(level_str);
600
5
    if (!level.has_value() || level.value() > MAX_USER_SETABLE_SEVERITY_LEVEL) return false;
601
602
4
    STDLOCK(m_cs);
603
4
    m_category_log_levels[*flag] = level.value();
604
4
    return true;
605
5
}
606
607
bool util::log::ShouldLog(Category category, Level level)
608
6.24M
{
609
6.24M
    return LogInstance().WillLogCategoryLevel(static_cast<BCLog::LogFlags>(category), level);
610
6.24M
}
611
612
void util::log::Log(util::log::Entry entry)
613
5.25M
{
614
5.25M
    BCLog::Logger& logger{LogInstance()};
615
5.25M
    if (logger.Enabled()) {
616
5.25M
        logger.LogPrint(std::move(entry));
617
5.25M
    }
618
5.25M
}