log: Construct global logger on first use

This commit is contained in:
MarcoFalke 2019-01-25 15:54:49 -05:00
parent d14ef5721f
commit 77777c5624
No known key found for this signature in database
GPG key ID: CE2B75697E69A548
6 changed files with 34 additions and 28 deletions

View file

@ -364,8 +364,8 @@ bool InitHTTPServer()
// Update libevent's log handling. Returns false if our version of // Update libevent's log handling. Returns false if our version of
// libevent doesn't support debug logging, in which case we should // libevent doesn't support debug logging, in which case we should
// clear the BCLog::LIBEVENT flag. // clear the BCLog::LIBEVENT flag.
if (!UpdateHTTPServerLogging(g_logger->WillLogCategory(BCLog::LIBEVENT))) { if (!UpdateHTTPServerLogging(LogInstance().WillLogCategory(BCLog::LIBEVENT))) {
g_logger->DisableCategory(BCLog::LIBEVENT); LogInstance().DisableCategory(BCLog::LIBEVENT);
} }
#ifdef WIN32 #ifdef WIN32

View file

@ -289,7 +289,7 @@ static void HandleSIGTERM(int)
static void HandleSIGHUP(int) static void HandleSIGHUP(int)
{ {
g_logger->m_reopen_file = true; LogInstance().m_reopen_file = true;
} }
#else #else
static BOOL WINAPI consoleCtrlHandler(DWORD dwCtrlType) static BOOL WINAPI consoleCtrlHandler(DWORD dwCtrlType)
@ -833,17 +833,17 @@ static std::string ResolveErrMsg(const char * const optname, const std::string&
*/ */
void InitLogging() void InitLogging()
{ {
g_logger->m_print_to_file = !gArgs.IsArgNegated("-debuglogfile"); LogInstance().m_print_to_file = !gArgs.IsArgNegated("-debuglogfile");
g_logger->m_file_path = AbsPathForConfigVal(gArgs.GetArg("-debuglogfile", DEFAULT_DEBUGLOGFILE)); LogInstance().m_file_path = AbsPathForConfigVal(gArgs.GetArg("-debuglogfile", DEFAULT_DEBUGLOGFILE));
// Add newlines to the logfile to distinguish this execution from the last // Add newlines to the logfile to distinguish this execution from the last
// one; called before console logging is set up, so this is only sent to // one; called before console logging is set up, so this is only sent to
// debug.log. // debug.log.
LogPrintf("\n\n\n\n\n"); LogPrintf("\n\n\n\n\n");
g_logger->m_print_to_console = gArgs.GetBoolArg("-printtoconsole", !gArgs.GetBoolArg("-daemon", false)); LogInstance().m_print_to_console = gArgs.GetBoolArg("-printtoconsole", !gArgs.GetBoolArg("-daemon", false));
g_logger->m_log_timestamps = gArgs.GetBoolArg("-logtimestamps", DEFAULT_LOGTIMESTAMPS); LogInstance().m_log_timestamps = gArgs.GetBoolArg("-logtimestamps", DEFAULT_LOGTIMESTAMPS);
g_logger->m_log_time_micros = gArgs.GetBoolArg("-logtimemicros", DEFAULT_LOGTIMEMICROS); LogInstance().m_log_time_micros = gArgs.GetBoolArg("-logtimemicros", DEFAULT_LOGTIMEMICROS);
fLogIPs = gArgs.GetBoolArg("-logips", DEFAULT_LOGIPS); fLogIPs = gArgs.GetBoolArg("-logips", DEFAULT_LOGIPS);
@ -981,7 +981,7 @@ bool AppInitParameterInteraction()
if (std::none_of(categories.begin(), categories.end(), if (std::none_of(categories.begin(), categories.end(),
[](std::string cat){return cat == "0" || cat == "none";})) { [](std::string cat){return cat == "0" || cat == "none";})) {
for (const auto& cat : categories) { for (const auto& cat : categories) {
if (!g_logger->EnableCategory(cat)) { if (!LogInstance().EnableCategory(cat)) {
InitWarning(strprintf(_("Unsupported logging category %s=%s."), "-debug", cat)); InitWarning(strprintf(_("Unsupported logging category %s=%s."), "-debug", cat));
} }
} }
@ -990,7 +990,7 @@ bool AppInitParameterInteraction()
// Now remove the logging categories which were explicitly excluded // Now remove the logging categories which were explicitly excluded
for (const std::string& cat : gArgs.GetArgs("-debugexclude")) { for (const std::string& cat : gArgs.GetArgs("-debugexclude")) {
if (!g_logger->DisableCategory(cat)) { if (!LogInstance().DisableCategory(cat)) {
InitWarning(strprintf(_("Unsupported logging category %s=%s."), "-debugexclude", cat)); InitWarning(strprintf(_("Unsupported logging category %s=%s."), "-debugexclude", cat));
} }
} }
@ -1197,19 +1197,19 @@ bool AppInitMain(InitInterfaces& interfaces)
#ifndef WIN32 #ifndef WIN32
CreatePidFile(GetPidFile(), getpid()); CreatePidFile(GetPidFile(), getpid());
#endif #endif
if (g_logger->m_print_to_file) { if (LogInstance().m_print_to_file) {
if (gArgs.GetBoolArg("-shrinkdebugfile", g_logger->DefaultShrinkDebugFile())) { if (gArgs.GetBoolArg("-shrinkdebugfile", LogInstance().DefaultShrinkDebugFile())) {
// Do this first since it both loads a bunch of debug.log into memory, // Do this first since it both loads a bunch of debug.log into memory,
// and because this needs to happen before any other debug.log printing // and because this needs to happen before any other debug.log printing
g_logger->ShrinkDebugFile(); LogInstance().ShrinkDebugFile();
} }
if (!g_logger->OpenDebugLog()) { if (!LogInstance().OpenDebugLog()) {
return InitError(strprintf("Could not open debug log file %s", return InitError(strprintf("Could not open debug log file %s",
g_logger->m_file_path.string())); LogInstance().m_file_path.string()));
} }
} }
if (!g_logger->m_log_timestamps) if (!LogInstance().m_log_timestamps)
LogPrintf("Startup time: %s\n", FormatISO8601DateTime(GetTime())); LogPrintf("Startup time: %s\n", FormatISO8601DateTime(GetTime()));
LogPrintf("Default data directory %s\n", GetDefaultDataDir().string()); LogPrintf("Default data directory %s\n", GetDefaultDataDir().string());
LogPrintf("Using data directory %s\n", GetDataDir().string()); LogPrintf("Using data directory %s\n", GetDataDir().string());

View file

@ -67,7 +67,7 @@ public:
void initLogging() override { InitLogging(); } void initLogging() override { InitLogging(); }
void initParameterInteraction() override { InitParameterInteraction(); } void initParameterInteraction() override { InitParameterInteraction(); }
std::string getWarnings(const std::string& type) override { return GetWarnings(type); } std::string getWarnings(const std::string& type) override { return GetWarnings(type); }
uint32_t getLogCategories() override { return g_logger->GetCategoryMask(); } uint32_t getLogCategories() override { return LogInstance().GetCategoryMask(); }
bool baseInitialize() override bool baseInitialize() override
{ {
return AppInitBasicSetup() && AppInitParameterInteraction() && AppInitSanityChecks() && return AppInitBasicSetup() && AppInitParameterInteraction() && AppInitSanityChecks() &&

View file

@ -8,6 +8,8 @@
const char * const DEFAULT_DEBUGLOGFILE = "debug.log"; const char * const DEFAULT_DEBUGLOGFILE = "debug.log";
BCLog::Logger& LogInstance()
{
/** /**
* NOTE: the logger instances is leaked on exit. This is ugly, but will be * NOTE: the logger instances is leaked on exit. This is ugly, but will be
* cleaned up by the OS/libc. Defining a logger as a global object doesn't work * cleaned up by the OS/libc. Defining a logger as a global object doesn't work
@ -17,11 +19,15 @@ const char * const DEFAULT_DEBUGLOGFILE = "debug.log";
* access the logger. When the shutdown sequence is fully audited and tested, * access the logger. When the shutdown sequence is fully audited and tested,
* explicit destruction of these objects can be implemented by changing this * explicit destruction of these objects can be implemented by changing this
* from a raw pointer to a std::unique_ptr. * from a raw pointer to a std::unique_ptr.
* Since the destructor is never called, the logger and all its members must
* have a trivial destructor.
* *
* This method of initialization was originally introduced in * This method of initialization was originally introduced in
* ee3374234c60aba2cc4c5cd5cac1c0aefc2d817c. * ee3374234c60aba2cc4c5cd5cac1c0aefc2d817c.
*/ */
BCLog::Logger* const g_logger = new BCLog::Logger(); static BCLog::Logger* g_logger{new BCLog::Logger()};
return *g_logger;
}
bool fLogIPs = DEFAULT_LOGIPS; bool fLogIPs = DEFAULT_LOGIPS;

View file

@ -108,12 +108,12 @@ namespace BCLog {
} // namespace BCLog } // namespace BCLog
extern BCLog::Logger* const g_logger; BCLog::Logger& LogInstance();
/** Return true if log accepts specified category */ /** Return true if log accepts specified category */
static inline bool LogAcceptCategory(BCLog::LogFlags category) static inline bool LogAcceptCategory(BCLog::LogFlags category)
{ {
return g_logger->WillLogCategory(category); return LogInstance().WillLogCategory(category);
} }
/** Returns a string with the log categories. */ /** Returns a string with the log categories. */
@ -132,7 +132,7 @@ bool GetLogCategory(BCLog::LogFlags& flag, const std::string& str);
template <typename... Args> template <typename... Args>
static inline void LogPrintf(const char* fmt, const Args&... args) static inline void LogPrintf(const char* fmt, const Args&... args)
{ {
if (g_logger->Enabled()) { if (LogInstance().Enabled()) {
std::string log_msg; std::string log_msg;
try { try {
log_msg = tfm::format(fmt, args...); log_msg = tfm::format(fmt, args...);
@ -140,7 +140,7 @@ static inline void LogPrintf(const char* fmt, const Args&... args)
/* Original format string will have newline so don't add one here */ /* Original format string will have newline so don't add one here */
log_msg = "Error \"" + std::string(fmterr.what()) + "\" while formatting log message: " + fmt; log_msg = "Error \"" + std::string(fmterr.what()) + "\" while formatting log message: " + fmt;
} }
g_logger->LogPrintStr(log_msg); LogInstance().LogPrintStr(log_msg);
} }
} }

View file

@ -357,9 +357,9 @@ static void EnableOrDisableLogCategories(UniValue cats, bool enable) {
bool success; bool success;
if (enable) { if (enable) {
success = g_logger->EnableCategory(cat); success = LogInstance().EnableCategory(cat);
} else { } else {
success = g_logger->DisableCategory(cat); success = LogInstance().DisableCategory(cat);
} }
if (!success) { if (!success) {
@ -405,14 +405,14 @@ UniValue logging(const JSONRPCRequest& request)
); );
} }
uint32_t original_log_categories = g_logger->GetCategoryMask(); uint32_t original_log_categories = LogInstance().GetCategoryMask();
if (request.params[0].isArray()) { if (request.params[0].isArray()) {
EnableOrDisableLogCategories(request.params[0], true); EnableOrDisableLogCategories(request.params[0], true);
} }
if (request.params[1].isArray()) { if (request.params[1].isArray()) {
EnableOrDisableLogCategories(request.params[1], false); EnableOrDisableLogCategories(request.params[1], false);
} }
uint32_t updated_log_categories = g_logger->GetCategoryMask(); uint32_t updated_log_categories = LogInstance().GetCategoryMask();
uint32_t changed_log_categories = original_log_categories ^ updated_log_categories; uint32_t changed_log_categories = original_log_categories ^ updated_log_categories;
// Update libevent logging if BCLog::LIBEVENT has changed. // Update libevent logging if BCLog::LIBEVENT has changed.
@ -421,8 +421,8 @@ UniValue logging(const JSONRPCRequest& request)
// Throw an error if the user has explicitly asked to change only the libevent // Throw an error if the user has explicitly asked to change only the libevent
// flag and it failed. // flag and it failed.
if (changed_log_categories & BCLog::LIBEVENT) { if (changed_log_categories & BCLog::LIBEVENT) {
if (!UpdateHTTPServerLogging(g_logger->WillLogCategory(BCLog::LIBEVENT))) { if (!UpdateHTTPServerLogging(LogInstance().WillLogCategory(BCLog::LIBEVENT))) {
g_logger->DisableCategory(BCLog::LIBEVENT); LogInstance().DisableCategory(BCLog::LIBEVENT);
if (changed_log_categories == BCLog::LIBEVENT) { if (changed_log_categories == BCLog::LIBEVENT) {
throw JSONRPCError(RPC_INVALID_PARAMETER, "libevent logging cannot be updated when using libevent before v2.1.1."); throw JSONRPCError(RPC_INVALID_PARAMETER, "libevent logging cannot be updated when using libevent before v2.1.1.");
} }