| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | // Copyright 2014 Citra Emulator Project
 | 
					
						
							| 
									
										
										
										
											2014-12-16 21:38:14 -08:00
										 |  |  | // Licensed under GPLv2 or any later version
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | // Refer to the license.txt file included.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-20 15:31:25 -04:00
										 |  |  | #include <atomic>
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | #include <chrono>
 | 
					
						
							| 
									
										
										
										
											2018-07-14 12:47:14 -06:00
										 |  |  | #include <climits>
 | 
					
						
							| 
									
										
										
										
											2021-08-24 01:32:38 -04:00
										 |  |  | #include <exception>
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | #include <thread>
 | 
					
						
							| 
									
										
										
										
											2018-07-20 15:31:25 -04:00
										 |  |  | #include <vector>
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2018-12-07 16:21:18 +01:00
										 |  |  | #include <windows.h> // For OutputDebugStringW
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-13 07:52:02 -04:00
										 |  |  | #include "common/fs/file.h"
 | 
					
						
							| 
									
										
										
										
											2021-05-25 19:32:56 -04:00
										 |  |  | #include "common/fs/fs.h"
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | #include "common/fs/fs_paths.h"
 | 
					
						
							|  |  |  | #include "common/fs/path_util.h"
 | 
					
						
							| 
									
										
										
										
											2021-06-23 14:18:27 -07:00
										 |  |  | #include "common/literals.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-20 23:52:38 -07:00
										 |  |  | #include "common/logging/backend.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | #include "common/logging/log.h"
 | 
					
						
							|  |  |  | #include "common/logging/text_formatter.h"
 | 
					
						
							| 
									
										
										
										
											2021-04-14 16:07:40 -07:00
										 |  |  | #include "common/settings.h"
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | #ifdef _WIN32
 | 
					
						
							| 
									
										
										
										
											2018-03-22 18:21:29 +08:00
										 |  |  | #include "common/string_util.h"
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | #include "common/threadsafe_queue.h"
 | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-14 20:19:52 -04:00
										 |  |  | namespace Common::Log { | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |  * Interface for logging backends. | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | class Backend { | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | public: | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     virtual ~Backend() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual void Write(const Entry& entry) = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual void EnableForStacktrace() = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     virtual void Flush() = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Backend that writes to stderr and with color | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class ColorConsoleBackend final : public Backend { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit ColorConsoleBackend() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ~ColorConsoleBackend() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Write(const Entry& entry) override { | 
					
						
							|  |  |  |         if (enabled.load(std::memory_order_relaxed)) { | 
					
						
							|  |  |  |             PrintColoredMessage(entry); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     void Flush() override { | 
					
						
							|  |  |  |         // stderr shouldn't be buffered
 | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-04-14 23:05:42 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     void EnableForStacktrace() override { | 
					
						
							|  |  |  |         enabled = true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     void SetEnabled(bool enabled_) { | 
					
						
							|  |  |  |         enabled = enabled_; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							|  |  |  |     std::atomic_bool enabled{false}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Backend that writes to a file passed into the constructor | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class FileBackend final : public Backend { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit FileBackend(const std::filesystem::path& filename) { | 
					
						
							|  |  |  |         auto old_filename = filename; | 
					
						
							|  |  |  |         old_filename += ".old.txt"; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // Existence checks are done within the functions themselves.
 | 
					
						
							|  |  |  |         // We don't particularly care if these succeed or not.
 | 
					
						
							|  |  |  |         static_cast<void>(FS::RemoveFile(old_filename)); | 
					
						
							|  |  |  |         static_cast<void>(FS::RenameFile(filename, old_filename)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         file = std::make_unique<FS::IOFile>(filename, FS::FileAccessMode::Write, | 
					
						
							|  |  |  |                                             FS::FileType::TextFile); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ~FileBackend() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Write(const Entry& entry) override { | 
					
						
							|  |  |  |         if (!enabled) { | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         bytes_written += file->WriteString(FormatLogMessage(entry).append(1, '\n')); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         using namespace Common::Literals; | 
					
						
							|  |  |  |         // Prevent logs from exceeding a set maximum size in the event that log entries are spammed.
 | 
					
						
							|  |  |  |         const auto write_limit = Settings::values.extended_logging ? 1_GiB : 100_MiB; | 
					
						
							|  |  |  |         const bool write_limit_exceeded = bytes_written > write_limit; | 
					
						
							|  |  |  |         if (entry.log_level >= Level::Error || write_limit_exceeded) { | 
					
						
							|  |  |  |             if (write_limit_exceeded) { | 
					
						
							|  |  |  |                 // Stop writing after the write limit is exceeded.
 | 
					
						
							|  |  |  |                 // Don't close the file so we can print a stacktrace if necessary
 | 
					
						
							|  |  |  |                 enabled = false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             file->Flush(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Flush() override { | 
					
						
							|  |  |  |         file->Flush(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void EnableForStacktrace() override { | 
					
						
							|  |  |  |         enabled = true; | 
					
						
							|  |  |  |         bytes_written = 0; | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | private: | 
					
						
							|  |  |  |     std::unique_ptr<FS::IOFile> file; | 
					
						
							|  |  |  |     bool enabled = true; | 
					
						
							|  |  |  |     std::size_t bytes_written = 0; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * Backend that writes to Visual Studio's output window | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class DebuggerBackend final : public Backend { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit DebuggerBackend() = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ~DebuggerBackend() override = default; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void Write(const Entry& entry) override { | 
					
						
							|  |  |  | #ifdef _WIN32
 | 
					
						
							|  |  |  |         ::OutputDebugStringW(UTF8ToUTF16W(FormatLogMessage(entry).append(1, '\n')).c_str()); | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     void Flush() override {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void EnableForStacktrace() override {} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-24 01:32:38 -04:00
										 |  |  | bool initialization_in_progress_suppress_logging = true; | 
					
						
							| 
									
										
										
										
											2021-04-20 12:53:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Static state as a singleton. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class Impl { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     static Impl& Instance() { | 
					
						
							|  |  |  |         if (!instance) { | 
					
						
							| 
									
										
										
										
											2021-08-24 01:32:38 -04:00
										 |  |  |             throw std::runtime_error("Using Logging instance before its initialization"); | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         return *instance; | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     static void Initialize() { | 
					
						
							|  |  |  |         if (instance) { | 
					
						
							| 
									
										
										
										
											2021-08-24 01:32:38 -04:00
										 |  |  |             LOG_WARNING(Log, "Reinitializing logging backend"); | 
					
						
							|  |  |  |             return; | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |         } | 
					
						
							|  |  |  |         using namespace Common::FS; | 
					
						
							|  |  |  |         const auto& log_dir = GetYuzuPath(YuzuPath::LogDir); | 
					
						
							|  |  |  |         void(CreateDir(log_dir)); | 
					
						
							|  |  |  |         Filter filter; | 
					
						
							|  |  |  |         filter.ParseFilterString(Settings::values.log_filter.GetValue()); | 
					
						
							|  |  |  |         instance = std::unique_ptr<Impl, decltype(&Deleter)>(new Impl(log_dir / LOG_FILE, filter), | 
					
						
							|  |  |  |                                                              Deleter); | 
					
						
							|  |  |  |         initialization_in_progress_suppress_logging = false; | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     Impl(const Impl&) = delete; | 
					
						
							|  |  |  |     Impl& operator=(const Impl&) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Impl(Impl&&) = delete; | 
					
						
							|  |  |  |     Impl& operator=(Impl&&) = delete; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     void SetGlobalFilter(const Filter& f) { | 
					
						
							|  |  |  |         filter = f; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     void SetColorConsoleBackendEnabled(bool enabled) { | 
					
						
							|  |  |  |         color_console_backend.SetEnabled(enabled); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void PushEntry(Class log_class, Level log_level, const char* filename, unsigned int line_num, | 
					
						
							|  |  |  |                    const char* function, std::string message) { | 
					
						
							|  |  |  |         if (!filter.CheckMessage(log_class, log_level)) | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |         const Entry& entry = | 
					
						
							|  |  |  |             CreateEntry(log_class, log_level, filename, line_num, function, std::move(message)); | 
					
						
							|  |  |  |         message_queue.Push(entry); | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | private: | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     Impl(const std::filesystem::path& file_backend_filename, const Filter& filter_) | 
					
						
							|  |  |  |         : filter{filter_}, file_backend{file_backend_filename}, backend_thread{std::thread([this] { | 
					
						
							|  |  |  |               Common::SetCurrentThreadName("yuzu:Log"); | 
					
						
							|  |  |  |               Entry entry; | 
					
						
							|  |  |  |               const auto write_logs = [this, &entry]() { | 
					
						
							|  |  |  |                   ForEachBackend([&entry](Backend& backend) { backend.Write(entry); }); | 
					
						
							|  |  |  |               }; | 
					
						
							|  |  |  |               while (true) { | 
					
						
							|  |  |  |                   entry = message_queue.PopWait(); | 
					
						
							|  |  |  |                   if (entry.final_entry) { | 
					
						
							|  |  |  |                       break; | 
					
						
							|  |  |  |                   } | 
					
						
							|  |  |  |                   write_logs(); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               // Drain the logging queue. Only writes out up to MAX_LOGS_TO_WRITE to prevent a
 | 
					
						
							|  |  |  |               // case where a system is repeatedly spamming logs even on close.
 | 
					
						
							|  |  |  |               int max_logs_to_write = filter.IsDebug() ? INT_MAX : 100; | 
					
						
							|  |  |  |               while (max_logs_to_write-- && message_queue.Pop(entry)) { | 
					
						
							|  |  |  |                   write_logs(); | 
					
						
							|  |  |  |               } | 
					
						
							| 
									
										
										
										
											2021-08-27 04:01:22 -04:00
										 |  |  |           })} {} | 
					
						
							| 
									
										
										
										
											2018-09-09 13:08:57 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     ~Impl() { | 
					
						
							|  |  |  |         StopBackendThread(); | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     void StopBackendThread() { | 
					
						
							|  |  |  |         Entry stop_entry{}; | 
					
						
							|  |  |  |         stop_entry.final_entry = true; | 
					
						
							|  |  |  |         message_queue.Push(stop_entry); | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  |         backend_thread.join(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-02 14:25:50 -05:00
										 |  |  |     Entry CreateEntry(Class log_class, Level log_level, const char* filename, unsigned int line_nr, | 
					
						
							|  |  |  |                       const char* function, std::string message) const { | 
					
						
							|  |  |  |         using std::chrono::duration_cast; | 
					
						
							| 
									
										
										
										
											2020-08-03 10:31:57 -04:00
										 |  |  |         using std::chrono::microseconds; | 
					
						
							| 
									
										
										
										
											2019-03-02 14:25:50 -05:00
										 |  |  |         using std::chrono::steady_clock; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-03 10:31:57 -04:00
										 |  |  |         return { | 
					
						
							|  |  |  |             .timestamp = duration_cast<microseconds>(steady_clock::now() - time_origin), | 
					
						
							|  |  |  |             .log_class = log_class, | 
					
						
							|  |  |  |             .log_level = log_level, | 
					
						
							|  |  |  |             .filename = filename, | 
					
						
							|  |  |  |             .line_num = line_nr, | 
					
						
							|  |  |  |             .function = function, | 
					
						
							|  |  |  |             .message = std::move(message), | 
					
						
							|  |  |  |             .final_entry = false, | 
					
						
							|  |  |  |         }; | 
					
						
							| 
									
										
										
										
											2019-03-02 14:25:50 -05:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     void ForEachBackend(auto lambda) { | 
					
						
							|  |  |  |         lambda(static_cast<Backend&>(debugger_backend)); | 
					
						
							|  |  |  |         lambda(static_cast<Backend&>(color_console_backend)); | 
					
						
							|  |  |  |         lambda(static_cast<Backend&>(file_backend)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-06-13 07:52:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     static void Deleter(Impl* ptr) { | 
					
						
							|  |  |  |         delete ptr; | 
					
						
							| 
									
										
										
										
											2021-07-05 12:54:06 -04:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     static inline std::unique_ptr<Impl, decltype(&Deleter)> instance{nullptr, Deleter}; | 
					
						
							| 
									
										
										
										
											2020-07-29 10:25:37 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     Filter filter; | 
					
						
							|  |  |  |     DebuggerBackend debugger_backend{}; | 
					
						
							|  |  |  |     ColorConsoleBackend color_console_backend{}; | 
					
						
							|  |  |  |     FileBackend file_backend; | 
					
						
							| 
									
										
										
										
											2020-07-29 10:25:37 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     std::thread backend_thread; | 
					
						
							|  |  |  |     MPSCQueue<Entry> message_queue{}; | 
					
						
							|  |  |  |     std::chrono::steady_clock::time_point time_origin{std::chrono::steady_clock::now()}; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | } // namespace
 | 
					
						
							| 
									
										
										
										
											2020-07-29 10:25:37 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | void Initialize() { | 
					
						
							|  |  |  |     Impl::Initialize(); | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | void DisableLoggingInTests() { | 
					
						
							|  |  |  |     initialization_in_progress_suppress_logging = true; | 
					
						
							| 
									
										
										
										
											2018-10-05 12:52:49 +09:30
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-07-02 11:10:41 -06:00
										 |  |  | void SetGlobalFilter(const Filter& filter) { | 
					
						
							|  |  |  |     Impl::Instance().SetGlobalFilter(filter); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  | void SetColorConsoleBackendEnabled(bool enabled) { | 
					
						
							|  |  |  |     Impl::Instance().SetColorConsoleBackendEnabled(enabled); | 
					
						
							| 
									
										
										
										
											2015-03-06 19:15:02 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-04-05 22:42:09 -06:00
										 |  |  | void FmtLogMessageImpl(Class log_class, Level log_level, const char* filename, | 
					
						
							|  |  |  |                        unsigned int line_num, const char* function, const char* format, | 
					
						
							|  |  |  |                        const fmt::format_args& args) { | 
					
						
							| 
									
										
										
										
											2021-08-13 18:39:45 +00:00
										 |  |  |     if (!initialization_in_progress_suppress_logging) { | 
					
						
							|  |  |  |         Impl::Instance().PushEntry(log_class, log_level, filename, line_num, function, | 
					
						
							|  |  |  |                                    fmt::vformat(format, args)); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2014-10-28 05:36:00 -02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-04-14 20:19:52 -04:00
										 |  |  | } // namespace Common::Log
 |