Update Log class.

This commit is contained in:
XMRig 2022-06-10 02:15:10 +07:00
parent d6a1c98314
commit 774f630e14
No known key found for this signature in database
GPG key ID: 446A53638BE94409
2 changed files with 202 additions and 121 deletions

View file

@ -1,7 +1,7 @@
/* XMRig /* XMRig
* Copyright (c) 2019 Spudz76 <https://github.com/Spudz76> * Copyright (c) 2019 Spudz76 <https://github.com/Spudz76>
* Copyright (c) 2018-2021 SChernykh <https://github.com/SChernykh> * Copyright (c) 2018-2022 SChernykh <https://github.com/SChernykh>
* Copyright (c) 2016-2021 XMRig <https://github.com/xmrig>, <support@xmrig.com> * Copyright (c) 2016-2022 XMRig <https://github.com/xmrig>, <support@xmrig.com>
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@ -15,9 +15,16 @@
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>. * along with this program. If not, see <http://www.gnu.org/licenses/>.
*
* Additional permission under GNU GPL version 3 section 7
*
* If you modify this Program, or any covered work, by linking or combining
* it with OpenSSL (or a modified version of that library), containing parts
* covered by the terms of OpenSSL License and SSLeay License, the licensors
* of this Program grant you additional permission to convey the resulting work.
*/ */
#ifdef WIN32 #ifdef XMRIG_OS_WIN
# include <winsock2.h> # include <winsock2.h>
# include <windows.h> # include <windows.h>
#endif #endif
@ -25,103 +32,93 @@
#include <algorithm> #include <algorithm>
#include <cassert> #include <cassert>
#include <cstdarg>
#include <cstring> #include <cstring>
#include <ctime> #include <ctime>
#include <mutex> #include <mutex>
#include <string>
#include <uv.h>
#include <vector> #include <vector>
#include "base/io/log/Log.h" #include "base/io/log/Log.h"
#include "base/kernel/interfaces/ILogBackend.h"
#include "base/tools/Chrono.h" #include "base/tools/Chrono.h"
#include "base/tools/Object.h"
#ifdef XMRIG_FEATURE_EVENTS
# include "base/kernel/Events.h"
# include "base/kernel/events/LogEvent.h"
# include "base/kernel/private/LogConfig.h"
# include "base/kernel/Process.h"
#else
# include "base/kernel/interfaces/ILogBackend.h"
#endif
namespace xmrig { namespace xmrig {
bool Log::m_background = false;
bool Log::m_colors = true;
LogPrivate *Log::d = nullptr;
uint32_t Log::m_verbose = 0;
static char buf[Log::kMaxBufferSize]{};
static std::mutex mutex;
static const char *colors_map[] = { static const char *colors_map[] = {
RED_BOLD_S, // EMERG RED_BOLD_S "E ", // EMERG
RED_BOLD_S, // ALERT RED_BOLD_S "A ", // ALERT
RED_BOLD_S, // CRIT RED_BOLD_S "C ", // CRIT
RED_S, // ERR RED_S "E ", // ERR
YELLOW_S, // WARNING YELLOW_S "W ", // WARNING
WHITE_BOLD_S, // NOTICE WHITE_BOLD_S "N ", // NOTICE
nullptr, // INFO "I ", // INFO
# ifdef WIN32 "1 ", // V1
BLACK_BOLD_S // DEBUG "2 ", // V2
"3 ", // V3
"4 ", // V4
# ifdef XMRIG_OS_WIN
BLACK_BOLD_S "5 ", // V5
BLACK_BOLD_S "D " // DEBUG
# else # else
BRIGHT_BLACK_S // DEBUG BRIGHT_BLACK_S "5 ", // V5
BRIGHT_BLACK_S "D " // DEBUG
# endif # endif
}; };
static void log_endl(size_t &size)
class LogPrivate
{ {
public: # ifdef XMRIG_OS_WIN
XMRIG_DISABLE_COPY_MOVE(LogPrivate) memcpy(buf + size, CLEAR "\r\n", 7);
size += 6;
# else
LogPrivate() = default; memcpy(buf + size, CLEAR "\n", 6);
size += 5;
# endif
inline ~LogPrivate()
{
for (auto backend : m_backends) {
delete backend;
}
} }
inline void add(ILogBackend *backend) { m_backends.push_back(backend); } static void log_color(Log::Level level, size_t &size)
void print(Log::Level level, const char *fmt, va_list args)
{ {
size_t size = 0; if (level == Log::NONE) {
size_t offset = 0;
std::lock_guard<std::mutex> lock(m_mutex);
if (Log::isBackground() && m_backends.empty()) {
return; return;
} }
const uint64_t ts = timestamp(level, size, offset); const char *color = colors_map[level];
color(level, size); if (color == nullptr) {
const int rc = vsnprintf(m_buf + size, sizeof (m_buf) - offset - 32, fmt, args);
if (rc < 0) {
return; return;
} }
size += std::min(static_cast<size_t>(rc), sizeof (m_buf) - offset - 32); const size_t s = strlen(color);
endl(size); memcpy(buf + size, color, s); // NOLINT(bugprone-not-null-terminated-result)
std::string txt(m_buf); size += s;
size_t i = 0;
while ((i = txt.find(CSI)) != std::string::npos) {
txt.erase(i, txt.find('m', i) - i + 1);
}
if (!m_backends.empty()) {
for (auto backend : m_backends) {
backend->print(ts, level, m_buf, offset, size, true);
backend->print(ts, level, txt.c_str(), offset ? (offset - 11) : 0, txt.size(), false);
}
}
else {
fputs(txt.c_str(), stdout);
fflush(stdout);
}
} }
private: static uint64_t log_timestamp(Log::Level level, size_t &size, size_t &offset)
inline uint64_t timestamp(Log::Level level, size_t &size, size_t &offset)
{ {
const uint64_t ms = Chrono::currentMSecsSinceEpoch(); const uint64_t ms = Chrono::currentMSecsSinceEpoch();
@ -132,13 +129,13 @@ private:
time_t now = ms / 1000; time_t now = ms / 1000;
tm stime{}; tm stime{};
# ifdef _WIN32 # ifdef XMRIG_OS_WIN
localtime_s(&stime, &now); localtime_s(&stime, &now);
# else # else
localtime_r(&now, &stime); localtime_r(&now, &stime);
# endif # endif
const int rc = snprintf(m_buf, sizeof(m_buf) - 1, "[%d-%02d-%02d %02d:%02d:%02d" BLACK_BOLD(".%03d") "] ", const int rc = snprintf(buf, sizeof(buf) - 1, "[%d-%02d-%02d %02d:%02d:%02d" BLACK_BOLD(".%03d") "] ",
stime.tm_year + 1900, stime.tm_year + 1900,
stime.tm_mon + 1, stime.tm_mon + 1,
stime.tm_mday, stime.tm_mday,
@ -156,58 +153,101 @@ private:
} }
inline void color(Log::Level level, size_t &size) #ifdef XMRIG_FEATURE_EVENTS
static void log_print(Log::Level level, const char *fmt, va_list args)
{ {
if (level == Log::NONE) { size_t size = 0;
size_t offset = 0;
std::lock_guard<std::mutex> lock(mutex);
const uint64_t ts = log_timestamp(level, size, offset);
log_color(level, size);
const int rc = vsnprintf(buf + size, sizeof(buf) - offset - 32, fmt, args);
if (rc < 0) {
return; return;
} }
const char *color = colors_map[level]; size += std::min(static_cast<size_t>(rc), sizeof(buf) - offset - 32);
if (color == nullptr) { log_endl(size);
return;
Process::events().post<LogEvent>(ts, level, buf, offset, size);
} }
const size_t s = strlen(color);
memcpy(m_buf + size, color, s);
size += s;
}
inline void endl(size_t &size)
{
# ifdef _WIN32
memcpy(m_buf + size, CLEAR "\r\n", 7);
size += 6;
#else #else
memcpy(m_buf + size, CLEAR "\n", 6); class LogPrivate
size += 5; {
# endif public:
XMRIG_DISABLE_COPY_MOVE(LogPrivate)
LogPrivate() = default;
inline ~LogPrivate()
{
for (auto backend : backends) {
delete backend;
}
} }
char m_buf[Log::kMaxBufferSize]{}; void print(Log::Level level, const char *fmt, va_list args)
std::mutex m_mutex; {
std::vector<ILogBackend*> m_backends; size_t size = 0;
size_t offset = 0;
std::lock_guard<std::mutex> lock(mutex);
if (Log::isBackground() && backends.empty()) {
return;
}
const uint64_t ts = log_timestamp(level, size, offset);
log_color(level, size);
const int rc = vsnprintf(buf + size, sizeof (buf) - offset - 32, fmt, args);
if (rc < 0) {
return;
}
size += std::min(static_cast<size_t>(rc), sizeof (buf) - offset - 32);
log_endl(size);
std::string txt(buf);
size_t i = 0;
while ((i = txt.find(CSI)) != std::string::npos) {
txt.erase(i, txt.find('m', i) - i + 1);
}
if (!backends.empty()) {
for (auto backend : backends) {
backend->print(ts, level, buf, offset, size, true);
backend->print(ts, level, txt.c_str(), offset ? (offset - 11) : 0, txt.size(), false);
}
}
else {
fputs(txt.c_str(), stdout);
fflush(stdout);
}
}
std::vector<ILogBackend*> backends;
}; };
#endif
bool Log::m_background = false; } // namespace xmrig
bool Log::m_colors = true;
LogPrivate *Log::d = nullptr;
uint32_t Log::m_verbose = 0;
} /* namespace xmrig */
#ifndef XMRIG_FEATURE_EVENTS
void xmrig::Log::add(ILogBackend *backend) void xmrig::Log::add(ILogBackend *backend)
{ {
assert(d != nullptr); assert(d != nullptr);
if (d) { if (d) {
d->add(backend); d->backends.push_back(backend);
} }
} }
@ -223,18 +263,25 @@ void xmrig::Log::init()
{ {
d = new LogPrivate(); d = new LogPrivate();
} }
#endif
void xmrig::Log::print(const char *fmt, ...) void xmrig::Log::print(const char *fmt, ...)
{ {
# ifndef XMRIG_FEATURE_EVENTS
if (!d) { if (!d) {
return; return;
} }
# endif
va_list args{}; va_list args{};
va_start(args, fmt); va_start(args, fmt);
# ifdef XMRIG_FEATURE_EVENTS
log_print(NONE, fmt, args);
# else
d->print(NONE, fmt, args); d->print(NONE, fmt, args);
# endif
va_end(args); va_end(args);
} }
@ -242,14 +289,33 @@ void xmrig::Log::print(const char *fmt, ...)
void xmrig::Log::print(Level level, const char *fmt, ...) void xmrig::Log::print(Level level, const char *fmt, ...)
{ {
# ifndef XMRIG_FEATURE_EVENTS
if (!d) { if (!d) {
return; return;
} }
# endif
va_list args{}; va_list args{};
va_start(args, fmt); va_start(args, fmt);
# ifdef XMRIG_FEATURE_EVENTS
log_print(level, fmt, args);
# else
d->print(level, fmt, args); d->print(level, fmt, args);
# endif
va_end(args); va_end(args);
} }
void xmrig::Log::setVerbose(uint32_t verbose)
{
static constexpr uint32_t kMaxVerbose =
# ifdef XMRIG_FEATURE_EVENTS
LogConfig::kMaxVerbose;
# else
5U;
# endif
m_verbose = std::min(verbose, kMaxVerbose);
}

View file

@ -1,7 +1,7 @@
/* XMRig /* XMRig
* Copyright (c) 2019 Spudz76 <https://github.com/Spudz76> * Copyright (c) 2019 Spudz76 <https://github.com/Spudz76>
* Copyright (c) 2018-2021 SChernykh <https://github.com/SChernykh> * Copyright (c) 2018-2022 SChernykh <https://github.com/SChernykh>
* Copyright (c) 2016-2021 XMRig <https://github.com/xmrig>, <support@xmrig.com> * Copyright (c) 2016-2022 XMRig <https://github.com/xmrig>, <support@xmrig.com>
* *
* This program is free software: you can redistribute it and/or modify * This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by * it under the terms of the GNU General Public License as published by
@ -15,6 +15,13 @@
* *
* You should have received a copy of the GNU General Public License * You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>. * along with this program. If not, see <http://www.gnu.org/licenses/>.
*
* Additional permission under GNU GPL version 3 section 7
*
* If you modify this Program, or any covered work, by linking or combining
* it with OpenSSL (or a modified version of that library), containing parts
* covered by the terms of OpenSSL License and SSLeay License, the licensors
* of this Program grant you additional permission to convey the resulting work.
*/ */
#ifndef XMRIG_LOG_H #ifndef XMRIG_LOG_H
@ -44,16 +51,25 @@ public:
WARNING, // warning conditions WARNING, // warning conditions
NOTICE, // normal but significant condition NOTICE, // normal but significant condition
INFO, // informational INFO, // informational
V1, // Verbose
V2, // Verbose
V3, // Verbose
V4, // Verbose
V5, // Verbose
DEBUG, // debug-level messages DEBUG, // debug-level messages
}; };
constexpr static size_t kMaxBufferSize = 16384; constexpr static size_t kMaxBufferSize = 16384;
# ifndef XMRIG_FEATURE_EVENTS
static void add(ILogBackend *backend); static void add(ILogBackend *backend);
static void destroy(); static void destroy();
static void init(); static void init();
# endif
static void print(const char *fmt, ...); static void print(const char *fmt, ...);
static void print(Level level, const char *fmt, ...); static void print(Level level, const char *fmt, ...);
static void setVerbose(uint32_t verbose);
static inline bool isBackground() { return m_background; } static inline bool isBackground() { return m_background; }
static inline bool isColors() { return m_colors; } static inline bool isColors() { return m_colors; }
@ -61,7 +77,6 @@ public:
static inline uint32_t verbose() { return m_verbose; } static inline uint32_t verbose() { return m_verbose; }
static inline void setBackground(bool background) { m_background = background; } static inline void setBackground(bool background) { m_background = background; }
static inline void setColors(bool colors) { m_colors = colors; } static inline void setColors(bool colors) { m_colors = colors; }
static inline void setVerbose(uint32_t verbose) { m_verbose = verbose; }
private: private:
static bool m_background; static bool m_background;
@ -143,12 +158,12 @@ private:
#define LOG_WARN(x, ...) xmrig::Log::print(xmrig::Log::WARNING, x, ##__VA_ARGS__) #define LOG_WARN(x, ...) xmrig::Log::print(xmrig::Log::WARNING, x, ##__VA_ARGS__)
#define LOG_NOTICE(x, ...) xmrig::Log::print(xmrig::Log::NOTICE, x, ##__VA_ARGS__) #define LOG_NOTICE(x, ...) xmrig::Log::print(xmrig::Log::NOTICE, x, ##__VA_ARGS__)
#define LOG_INFO(x, ...) xmrig::Log::print(xmrig::Log::INFO, x, ##__VA_ARGS__) #define LOG_INFO(x, ...) xmrig::Log::print(xmrig::Log::INFO, x, ##__VA_ARGS__)
#define LOG_VERBOSE(x, ...) if (xmrig::Log::verbose() > 0) { xmrig::Log::print(xmrig::Log::INFO, x, ##__VA_ARGS__); } #define LOG_VERBOSE(x, ...) if (xmrig::Log::verbose() > 0) { xmrig::Log::print(xmrig::Log::V1, x, ##__VA_ARGS__); }
#define LOG_V1(x, ...) if (xmrig::Log::verbose() > 0) { xmrig::Log::print(xmrig::Log::INFO, x, ##__VA_ARGS__); } #define LOG_V1(x, ...) if (xmrig::Log::verbose() > 0) { xmrig::Log::print(xmrig::Log::V1, x, ##__VA_ARGS__); }
#define LOG_V2(x, ...) if (xmrig::Log::verbose() > 1) { xmrig::Log::print(xmrig::Log::INFO, x, ##__VA_ARGS__); } #define LOG_V2(x, ...) if (xmrig::Log::verbose() > 1) { xmrig::Log::print(xmrig::Log::V2, x, ##__VA_ARGS__); }
#define LOG_V3(x, ...) if (xmrig::Log::verbose() > 2) { xmrig::Log::print(xmrig::Log::INFO, x, ##__VA_ARGS__); } #define LOG_V3(x, ...) if (xmrig::Log::verbose() > 2) { xmrig::Log::print(xmrig::Log::V3, x, ##__VA_ARGS__); }
#define LOG_V4(x, ...) if (xmrig::Log::verbose() > 3) { xmrig::Log::print(xmrig::Log::INFO, x, ##__VA_ARGS__); } #define LOG_V4(x, ...) if (xmrig::Log::verbose() > 3) { xmrig::Log::print(xmrig::Log::V4, x, ##__VA_ARGS__); }
#define LOG_V5(x, ...) if (xmrig::Log::verbose() > 4) { xmrig::Log::print(xmrig::Log::INFO, x, ##__VA_ARGS__); } #define LOG_V5(x, ...) if (xmrig::Log::verbose() > 4) { xmrig::Log::print(xmrig::Log::V5, x, ##__VA_ARGS__); }
#ifdef APP_DEBUG #ifdef APP_DEBUG
# define LOG_DEBUG(x, ...) xmrig::Log::print(xmrig::Log::DEBUG, x, ##__VA_ARGS__) # define LOG_DEBUG(x, ...) xmrig::Log::print(xmrig::Log::DEBUG, x, ##__VA_ARGS__)
@ -165,7 +180,7 @@ private:
#endif #endif
} /* namespace xmrig */ } // namespace xmrig
#endif /* XMRIG_LOG_H */ #endif // XMRIG_LOG_H