#include #include #include #include #include #include #include #include namespace mbgl { namespace { std::unique_ptr currentObserver; std::atomic useThread(true); std::mutex mutex; } // namespace class Log::Impl { public: Impl() : scheduler(Scheduler::GetSequenced()) {} void record(EventSeverity severity, Event event, int64_t code, const std::string& msg) { if (useThread) { auto threadName = platform::getCurrentThreadName(); scheduler->schedule([=]() { Log::record(severity, event, code, msg, threadName); }); } else { Log::record(severity, event, code, msg, {}); } } private: const std::shared_ptr scheduler; }; Log::Log() : impl(std::make_unique()) {} Log::~Log() = default; Log* Log::get() noexcept { static Log instance; return &instance; } void Log::useLogThread(bool enable) { useThread = enable; } void Log::setObserver(std::unique_ptr observer) { std::lock_guard lock(mutex); currentObserver = std::move(observer); } std::unique_ptr Log::removeObserver() { std::lock_guard lock(mutex); std::unique_ptr observer; std::swap(observer, currentObserver); return observer; } void Log::record(EventSeverity severity, Event event, const std::string &msg) { get()->impl->record(severity, event, -1, msg); } void Log::record(EventSeverity severity, Event event, const char* format, ...) { va_list args; va_start(args, format); char msg[4096]; vsnprintf(msg, sizeof(msg), format, args); va_end(args); get()->impl->record(severity, event, -1, std::string{msg}); } void Log::record(EventSeverity severity, Event event, int64_t code, const char* format, ...) { va_list args; va_start(args, format); char msg[4096]; vsnprintf(msg, sizeof(msg), format, args); va_end(args); get()->impl->record(severity, event, code, std::string{msg}); } void Log::record(EventSeverity severity, Event event, int64_t code, const std::string& msg, const optional& threadName) { std::lock_guard lock(mutex); if (currentObserver && severity != EventSeverity::Debug && currentObserver->onRecord(severity, event, code, msg)) { return; } std::stringstream logStream; logStream << "{" << threadName.value_or(platform::getCurrentThreadName()) << "}"; logStream << "[" << Enum::toString(event) << "]"; if (code >= 0) { logStream << "(" << code << ")"; } if (!msg.empty()) { logStream << ": " << msg; } platformRecord(severity, logStream.str()); } } // namespace mbgl