|  | /* | 
|  | *  Copyright 2004 The WebRTC Project Authors. All rights reserved. | 
|  | * | 
|  | *  Use of this source code is governed by a BSD-style license | 
|  | *  that can be found in the LICENSE file in the root of the source | 
|  | *  tree. An additional intellectual property rights grant can be found | 
|  | *  in the file PATENTS.  All contributing project authors may | 
|  | *  be found in the AUTHORS file in the root of the source tree. | 
|  | */ | 
|  |  | 
|  | #include "rtc_base/logging.h" | 
|  |  | 
|  | #include <algorithm> | 
|  | #include <atomic> | 
|  | #include <cinttypes> | 
|  | #include <cstdarg> | 
|  | #include <cstdint> | 
|  | #include <cstdio> | 
|  | #include <cstring> | 
|  | #include <ctime> | 
|  | #include <string> | 
|  | #include <vector> | 
|  |  | 
|  | #include "absl/base/attributes.h" | 
|  | #include "absl/strings/string_view.h" | 
|  | #include "api/units/timestamp.h" | 
|  | #include "rtc_base/checks.h" | 
|  | #include "rtc_base/platform_thread_types.h" | 
|  | #include "rtc_base/string_encode.h" | 
|  | #include "rtc_base/string_utils.h" | 
|  | #include "rtc_base/strings/string_builder.h" | 
|  | #include "rtc_base/synchronization/mutex.h" | 
|  | #include "rtc_base/thread_annotations.h" | 
|  | #include "rtc_base/time_utils.h" | 
|  |  | 
|  | #if RTC_LOG_ENABLED() | 
|  |  | 
|  | #if defined(WEBRTC_WIN) | 
|  | #include <windows.h> | 
|  | #undef ERROR  // wingdi.h | 
|  | #endif | 
|  |  | 
|  | #if defined(WEBRTC_MAC) && !defined(WEBRTC_IOS) | 
|  | #include <CoreServices/CoreServices.h> | 
|  | #elif defined(WEBRTC_ANDROID) | 
|  | #include <android/log.h> | 
|  |  | 
|  | // Android has a 1024 limit on log inputs. We use 60 chars as an | 
|  | // approx for the header/tag portion. | 
|  | // See android/system/core/liblog/logd_write.c | 
|  | static const int kMaxLogLineSize = 1024 - 60; | 
|  | #endif  // WEBRTC_MAC && !defined(WEBRTC_IOS) || WEBRTC_ANDROID | 
|  |  | 
|  | namespace webrtc { | 
|  | namespace { | 
|  |  | 
|  | // By default, release builds don't log, debug builds at info level | 
|  | #if !defined(NDEBUG) | 
|  | constexpr LoggingSeverity kDefaultLoggingSeverity = LS_INFO; | 
|  | #else | 
|  | constexpr LoggingSeverity kDefaultLoggingSeverity = LS_NONE; | 
|  | #endif | 
|  |  | 
|  | // Note: `g_min_sev` and `g_dbg_sev` can be changed while running. | 
|  | LoggingSeverity g_min_sev = kDefaultLoggingSeverity; | 
|  | LoggingSeverity g_dbg_sev = kDefaultLoggingSeverity; | 
|  |  | 
|  | // Return the filename portion of the string (that following the last slash). | 
|  | const char* FilenameFromPath(const char* file) { | 
|  | const char* end1 = ::strrchr(file, '/'); | 
|  | const char* end2 = ::strrchr(file, '\\'); | 
|  | if (!end1 && !end2) | 
|  | return file; | 
|  | else | 
|  | return (end1 > end2) ? end1 + 1 : end2 + 1; | 
|  | } | 
|  |  | 
|  | // Global lock for log subsystem, only needed to serialize access to streams_. | 
|  | Mutex& GetLoggingLock() { | 
|  | static Mutex& mutex = *new Mutex(); | 
|  | return mutex; | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | std::string LogLineRef::DefaultLogLine() const { | 
|  | StringBuilder log_output; | 
|  | if (timestamp_ != Timestamp::MinusInfinity()) { | 
|  | // TODO(kwiberg): Switch to absl::StrFormat, if binary size is ok. | 
|  | char timestamp[50];  // Maximum string length of an int64_t is 20. | 
|  | int len = | 
|  | snprintf(timestamp, sizeof(timestamp), "[%03" PRId64 ":%03" PRId64 "]", | 
|  | timestamp_.ms() / 1000, timestamp_.ms() % 1000); | 
|  | RTC_DCHECK_LT(len, sizeof(timestamp)); | 
|  | log_output << timestamp; | 
|  | } | 
|  | if (thread_id_.has_value()) { | 
|  | log_output << "[" << *thread_id_ << "] "; | 
|  | } | 
|  | if (!filename_.empty()) { | 
|  | #if defined(WEBRTC_ANDROID) | 
|  | log_output << "(line " << line_ << "): "; | 
|  | #else | 
|  | log_output << "(" << filename_ << ":" << line_ << "): "; | 
|  | #endif | 
|  | } | 
|  | log_output << message_; | 
|  | return log_output.Release(); | 
|  | } | 
|  |  | 
|  | ///////////////////////////////////////////////////////////////////////////// | 
|  | // LogMessage | 
|  | ///////////////////////////////////////////////////////////////////////////// | 
|  |  | 
|  | bool LogMessage::log_to_stderr_ = true; | 
|  |  | 
|  | // The list of logging streams currently configured. | 
|  | // Note: we explicitly do not clean this up, because of the uncertain ordering | 
|  | // of destructors at program exit.  Let the person who sets the stream trigger | 
|  | // cleanup by setting to null, or let it leak (safe at program exit). | 
|  | ABSL_CONST_INIT LogSink* LogMessage::streams_ RTC_GUARDED_BY(GetLoggingLock()) = | 
|  | nullptr; | 
|  | ABSL_CONST_INIT std::atomic<bool> LogMessage::streams_empty_ = {true}; | 
|  |  | 
|  | // Boolean options default to false. | 
|  | ABSL_CONST_INIT bool LogMessage::log_thread_ = false; | 
|  | ABSL_CONST_INIT bool LogMessage::log_timestamp_ = false; | 
|  |  | 
|  | LogMessage::LogMessage(const char* file, int line, LoggingSeverity sev) | 
|  | : LogMessage(file, line, sev, ERRCTX_NONE, 0) {} | 
|  |  | 
|  | LogMessage::LogMessage(const char* file, | 
|  | int line, | 
|  | LoggingSeverity sev, | 
|  | LogErrorContext err_ctx, | 
|  | int err) { | 
|  | log_line_.set_severity(sev); | 
|  | if (log_timestamp_) { | 
|  | int64_t log_start_time = LogStartTime(); | 
|  | // Use SystemTimeMillis so that even if tests use fake clocks, the timestamp | 
|  | // in log messages represents the real system time. | 
|  | int64_t time = TimeDiff(SystemTimeMillis(), log_start_time); | 
|  | // Also ensure WallClockStartTime is initialized, so that it matches | 
|  | // LogStartTime. | 
|  | WallClockStartTime(); | 
|  | log_line_.set_timestamp(Timestamp::Millis(time)); | 
|  | } | 
|  |  | 
|  | if (log_thread_) { | 
|  | log_line_.set_thread_id(CurrentThreadId()); | 
|  | } | 
|  |  | 
|  | if (file != nullptr) { | 
|  | log_line_.set_filename(FilenameFromPath(file)); | 
|  | log_line_.set_line(line); | 
|  | #if defined(WEBRTC_ANDROID) | 
|  | log_line_.set_tag(log_line_.filename()); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | if (err_ctx != ERRCTX_NONE) { | 
|  | char tmp_buf[1024]; | 
|  | SimpleStringBuilder tmp(tmp_buf); | 
|  | tmp.AppendFormat("[0x%08X]", err); | 
|  | switch (err_ctx) { | 
|  | case ERRCTX_ERRNO: | 
|  | tmp << " " << strerror(err); | 
|  | break; | 
|  | #ifdef WEBRTC_WIN | 
|  | case ERRCTX_HRESULT: { | 
|  | char msgbuf[256]; | 
|  | DWORD flags = | 
|  | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS; | 
|  | if (DWORD len = FormatMessageA( | 
|  | flags, nullptr, err, MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), | 
|  | msgbuf, sizeof(msgbuf) / sizeof(msgbuf[0]), nullptr)) { | 
|  | while ((len > 0) && | 
|  | isspace(static_cast<unsigned char>(msgbuf[len - 1]))) { | 
|  | msgbuf[--len] = 0; | 
|  | } | 
|  | tmp << " " << msgbuf; | 
|  | } | 
|  | break; | 
|  | } | 
|  | #endif  // WEBRTC_WIN | 
|  | default: | 
|  | break; | 
|  | } | 
|  | extra_ = tmp.str(); | 
|  | } | 
|  | } | 
|  |  | 
|  | #if defined(WEBRTC_ANDROID) | 
|  | LogMessage::LogMessage(const char* file, | 
|  | int line, | 
|  | LoggingSeverity sev, | 
|  | const char* tag) | 
|  | : LogMessage(file, line, sev, ERRCTX_NONE, /*err=*/0) { | 
|  | log_line_.set_tag(tag); | 
|  | print_stream_ << tag << ": "; | 
|  | } | 
|  | #endif | 
|  |  | 
|  | LogMessage::~LogMessage() { | 
|  | FinishPrintStream(); | 
|  |  | 
|  | log_line_.set_message(print_stream_.Release()); | 
|  |  | 
|  | if (log_line_.severity() >= g_dbg_sev) { | 
|  | OutputToDebug(log_line_); | 
|  | } | 
|  |  | 
|  | MutexLock lock(&GetLoggingLock()); | 
|  | for (LogSink* entry = streams_; entry != nullptr; entry = entry->next_) { | 
|  | if (log_line_.severity() >= entry->min_severity_) { | 
|  | entry->OnLogMessage(log_line_); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | void LogMessage::AddTag([[maybe_unused]] const char* tag) { | 
|  | #ifdef WEBRTC_ANDROID | 
|  | log_line_.set_tag(tag); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | StringBuilder& LogMessage::stream() { | 
|  | return print_stream_; | 
|  | } | 
|  |  | 
|  | int LogMessage::GetMinLogSeverity() { | 
|  | return g_min_sev; | 
|  | } | 
|  |  | 
|  | LoggingSeverity LogMessage::GetLogToDebug() { | 
|  | return g_dbg_sev; | 
|  | } | 
|  | int64_t LogMessage::LogStartTime() { | 
|  | static const int64_t g_start = SystemTimeMillis(); | 
|  | return g_start; | 
|  | } | 
|  |  | 
|  | uint32_t LogMessage::WallClockStartTime() { | 
|  | static const uint32_t g_start_wallclock = time(nullptr); | 
|  | return g_start_wallclock; | 
|  | } | 
|  |  | 
|  | void LogMessage::LogThreads(bool on) { | 
|  | log_thread_ = on; | 
|  | } | 
|  |  | 
|  | void LogMessage::LogTimestamps(bool on) { | 
|  | log_timestamp_ = on; | 
|  | } | 
|  |  | 
|  | void LogMessage::LogToDebug(LoggingSeverity min_sev) { | 
|  | g_dbg_sev = min_sev; | 
|  | MutexLock lock(&GetLoggingLock()); | 
|  | UpdateMinLogSeverity(); | 
|  | } | 
|  |  | 
|  | void LogMessage::SetLogToStderr(bool log_to_stderr) { | 
|  | log_to_stderr_ = log_to_stderr; | 
|  | } | 
|  |  | 
|  | int LogMessage::GetLogToStream(LogSink* stream) { | 
|  | MutexLock lock(&GetLoggingLock()); | 
|  | LoggingSeverity sev = LS_NONE; | 
|  | for (LogSink* entry = streams_; entry != nullptr; entry = entry->next_) { | 
|  | if (stream == nullptr || stream == entry) { | 
|  | sev = std::min(sev, entry->min_severity_); | 
|  | } | 
|  | } | 
|  | return sev; | 
|  | } | 
|  |  | 
|  | void LogMessage::AddLogToStream(LogSink* stream, LoggingSeverity min_sev) { | 
|  | MutexLock lock(&GetLoggingLock()); | 
|  | stream->min_severity_ = min_sev; | 
|  | stream->next_ = streams_; | 
|  | streams_ = stream; | 
|  | streams_empty_.store(false, std::memory_order_relaxed); | 
|  | UpdateMinLogSeverity(); | 
|  | } | 
|  |  | 
|  | void LogMessage::RemoveLogToStream(LogSink* stream) { | 
|  | MutexLock lock(&GetLoggingLock()); | 
|  | for (LogSink** entry = &streams_; *entry != nullptr; | 
|  | entry = &(*entry)->next_) { | 
|  | if (*entry == stream) { | 
|  | *entry = (*entry)->next_; | 
|  | break; | 
|  | } | 
|  | } | 
|  | streams_empty_.store(streams_ == nullptr, std::memory_order_relaxed); | 
|  | UpdateMinLogSeverity(); | 
|  | } | 
|  |  | 
|  | void LogMessage::ConfigureLogging(absl::string_view params) { | 
|  | LoggingSeverity current_level = LS_VERBOSE; | 
|  | LoggingSeverity debug_level = GetLogToDebug(); | 
|  |  | 
|  | std::vector<std::string> tokens; | 
|  | tokenize(params, ' ', &tokens); | 
|  |  | 
|  | for (const std::string& token : tokens) { | 
|  | if (token.empty()) | 
|  | continue; | 
|  |  | 
|  | // Logging features | 
|  | if (token == "tstamp") { | 
|  | LogTimestamps(); | 
|  | } else if (token == "thread") { | 
|  | LogThreads(); | 
|  |  | 
|  | // Logging levels | 
|  | } else if (token == "verbose") { | 
|  | current_level = LS_VERBOSE; | 
|  | } else if (token == "info") { | 
|  | current_level = LS_INFO; | 
|  | } else if (token == "warning") { | 
|  | current_level = LS_WARNING; | 
|  | } else if (token == "error") { | 
|  | current_level = LS_ERROR; | 
|  | } else if (token == "none") { | 
|  | current_level = LS_NONE; | 
|  |  | 
|  | // Logging targets | 
|  | } else if (token == "debug") { | 
|  | debug_level = current_level; | 
|  | } | 
|  | } | 
|  |  | 
|  | #if defined(WEBRTC_WIN) && !defined(WINUWP) | 
|  | if ((LS_NONE != debug_level) && !::IsDebuggerPresent()) { | 
|  | // First, attempt to attach to our parent's console... so if you invoke | 
|  | // from the command line, we'll see the output there.  Otherwise, create | 
|  | // our own console window. | 
|  | // Note: These methods fail if a console already exists, which is fine. | 
|  | if (!AttachConsole(ATTACH_PARENT_PROCESS)) | 
|  | ::AllocConsole(); | 
|  | } | 
|  | #endif  // defined(WEBRTC_WIN) && !defined(WINUWP) | 
|  |  | 
|  | LogToDebug(debug_level); | 
|  | } | 
|  |  | 
|  | void LogMessage::UpdateMinLogSeverity() | 
|  | RTC_EXCLUSIVE_LOCKS_REQUIRED(GetLoggingLock()) { | 
|  | LoggingSeverity min_sev = g_dbg_sev; | 
|  | for (LogSink* entry = streams_; entry != nullptr; entry = entry->next_) { | 
|  | min_sev = std::min(min_sev, entry->min_severity_); | 
|  | } | 
|  | g_min_sev = min_sev; | 
|  | } | 
|  |  | 
|  | void LogMessage::OutputToDebug(const LogLineRef& log_line) { | 
|  | std::string msg_str = log_line.DefaultLogLine(); | 
|  | bool log_to_stderr = log_to_stderr_; | 
|  | #if defined(WEBRTC_MAC) && !defined(WEBRTC_IOS) && defined(NDEBUG) | 
|  | // On the Mac, all stderr output goes to the Console log and causes clutter. | 
|  | // So in opt builds, don't log to stderr unless the user specifically sets | 
|  | // a preference to do so. | 
|  | CFStringRef domain = CFBundleGetIdentifier(CFBundleGetMainBundle()); | 
|  | if (domain != nullptr) { | 
|  | Boolean exists_and_is_valid; | 
|  | Boolean should_log = CFPreferencesGetAppBooleanValue( | 
|  | CFSTR("logToStdErr"), domain, &exists_and_is_valid); | 
|  | // If the key doesn't exist or is invalid or is false, we will not log to | 
|  | // stderr. | 
|  | log_to_stderr = exists_and_is_valid && should_log; | 
|  | } | 
|  | #endif  // defined(WEBRTC_MAC) && !defined(WEBRTC_IOS) && defined(NDEBUG) | 
|  |  | 
|  | #if defined(WEBRTC_WIN) | 
|  | // Always log to the debugger. | 
|  | // Perhaps stderr should be controlled by a preference, as on Mac? | 
|  | OutputDebugStringA(msg_str.c_str()); | 
|  | if (log_to_stderr) { | 
|  | // This handles dynamically allocated consoles, too. | 
|  | if (HANDLE error_handle = ::GetStdHandle(STD_ERROR_HANDLE)) { | 
|  | log_to_stderr = false; | 
|  | DWORD written = 0; | 
|  | ::WriteFile(error_handle, msg_str.c_str(), | 
|  | static_cast<DWORD>(msg_str.size()), &written, 0); | 
|  | } | 
|  | } | 
|  | #endif  // WEBRTC_WIN | 
|  |  | 
|  | #if defined(WEBRTC_ANDROID) | 
|  | // Android's logging facility uses severity to log messages but we | 
|  | // need to map libjingle's severity levels to Android ones first. | 
|  | // Also write to stderr which maybe available to executable started | 
|  | // from the shell. | 
|  | int prio; | 
|  | switch (log_line.severity()) { | 
|  | case LS_VERBOSE: | 
|  | prio = ANDROID_LOG_VERBOSE; | 
|  | break; | 
|  | case LS_INFO: | 
|  | prio = ANDROID_LOG_INFO; | 
|  | break; | 
|  | case LS_WARNING: | 
|  | prio = ANDROID_LOG_WARN; | 
|  | break; | 
|  | case LS_ERROR: | 
|  | prio = ANDROID_LOG_ERROR; | 
|  | break; | 
|  | default: | 
|  | prio = ANDROID_LOG_UNKNOWN; | 
|  | } | 
|  |  | 
|  | int size = msg_str.size(); | 
|  | int current_line = 0; | 
|  | int idx = 0; | 
|  | const int max_lines = size / kMaxLogLineSize + 1; | 
|  | if (max_lines == 1) { | 
|  | __android_log_print(prio, log_line.tag().data(), "%.*s", size, | 
|  | msg_str.c_str()); | 
|  | } else { | 
|  | while (size > 0) { | 
|  | const int len = std::min(size, kMaxLogLineSize); | 
|  | // Use the size of the string in the format (msg may have \0 in the | 
|  | // middle). | 
|  | __android_log_print(prio, log_line.tag().data(), "[%d/%d] %.*s", | 
|  | current_line + 1, max_lines, len, | 
|  | msg_str.c_str() + idx); | 
|  | idx += len; | 
|  | size -= len; | 
|  | ++current_line; | 
|  | } | 
|  | } | 
|  | #endif  // WEBRTC_ANDROID | 
|  | if (log_to_stderr) { | 
|  | fprintf(stderr, "%s", msg_str.c_str()); | 
|  | fflush(stderr); | 
|  | } | 
|  | } | 
|  |  | 
|  | // static | 
|  | bool LogMessage::IsNoop(LoggingSeverity severity) { | 
|  | if (severity >= g_dbg_sev || severity >= g_min_sev) | 
|  | return false; | 
|  | return streams_empty_.load(std::memory_order_relaxed); | 
|  | } | 
|  |  | 
|  | void LogMessage::FinishPrintStream() { | 
|  | if (!extra_.empty()) | 
|  | print_stream_ << " : " << extra_; | 
|  | print_stream_ << "\n"; | 
|  | } | 
|  |  | 
|  | namespace webrtc_logging_impl { | 
|  |  | 
|  | void Log(const LogArgType* fmt, ...) { | 
|  | va_list args; | 
|  | va_start(args, fmt); | 
|  |  | 
|  | LogMetadataErr meta; | 
|  | const char* tag = nullptr; | 
|  | switch (*fmt) { | 
|  | case LogArgType::kLogMetadata: { | 
|  | meta = { | 
|  | .meta = va_arg(args, LogMetadata), .err_ctx = ERRCTX_NONE, .err = 0}; | 
|  | break; | 
|  | } | 
|  | case LogArgType::kLogMetadataErr: { | 
|  | meta = va_arg(args, LogMetadataErr); | 
|  | break; | 
|  | } | 
|  | #ifdef WEBRTC_ANDROID | 
|  | case LogArgType::kLogMetadataTag: { | 
|  | const LogMetadataTag tag_meta = va_arg(args, LogMetadataTag); | 
|  | meta = {{nullptr, 0, tag_meta.severity}, ERRCTX_NONE, 0}; | 
|  | tag = tag_meta.tag; | 
|  | break; | 
|  | } | 
|  | #endif | 
|  | default: { | 
|  | RTC_DCHECK_NOTREACHED(); | 
|  | va_end(args); | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | LogMessage log_message(meta.meta.File(), meta.meta.Line(), | 
|  | meta.meta.Severity(), meta.err_ctx, meta.err); | 
|  | if (tag) { | 
|  | log_message.AddTag(tag); | 
|  | } | 
|  |  | 
|  | for (++fmt; *fmt != LogArgType::kEnd; ++fmt) { | 
|  | switch (*fmt) { | 
|  | case LogArgType::kInt: | 
|  | log_message.stream() << va_arg(args, int); | 
|  | break; | 
|  | case LogArgType::kLong: | 
|  | log_message.stream() << va_arg(args, long); | 
|  | break; | 
|  | case LogArgType::kLongLong: | 
|  | log_message.stream() << va_arg(args, long long); | 
|  | break; | 
|  | case LogArgType::kUInt: | 
|  | log_message.stream() << va_arg(args, unsigned); | 
|  | break; | 
|  | case LogArgType::kULong: | 
|  | log_message.stream() << va_arg(args, unsigned long); | 
|  | break; | 
|  | case LogArgType::kULongLong: | 
|  | log_message.stream() << va_arg(args, unsigned long long); | 
|  | break; | 
|  | case LogArgType::kDouble: | 
|  | log_message.stream() << va_arg(args, double); | 
|  | break; | 
|  | case LogArgType::kCharP: { | 
|  | const char* s = va_arg(args, const char*); | 
|  | log_message.stream() << (s ? s : "(null)"); | 
|  | break; | 
|  | } | 
|  | case LogArgType::kStdString: | 
|  | log_message.stream() << *va_arg(args, const std::string*); | 
|  | break; | 
|  | case LogArgType::kStringView: | 
|  | log_message.stream() << *va_arg(args, const absl::string_view*); | 
|  | break; | 
|  | case LogArgType::kVoidP: | 
|  | log_message.stream() | 
|  | << ToHex(reinterpret_cast<uintptr_t>(va_arg(args, const void*))); | 
|  | break; | 
|  | default: | 
|  | RTC_DCHECK_NOTREACHED(); | 
|  | va_end(args); | 
|  | return; | 
|  | } | 
|  | } | 
|  |  | 
|  | va_end(args); | 
|  | } | 
|  |  | 
|  | }  // namespace webrtc_logging_impl | 
|  | }  // namespace webrtc | 
|  | #endif | 
|  |  | 
|  | namespace webrtc { | 
|  | // Default implementation, override is recomended. | 
|  | void LogSink::OnLogMessage(const LogLineRef& log_line) { | 
|  | #if defined(WEBRTC_ANDROID) | 
|  | OnLogMessage(log_line.DefaultLogLine(), log_line.severity(), | 
|  | log_line.tag().data()); | 
|  | #else | 
|  | OnLogMessage(log_line.DefaultLogLine(), log_line.severity()); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | // Inefficient default implementation, override is recommended. | 
|  | void LogSink::OnLogMessage(const std::string& msg, | 
|  | LoggingSeverity severity, | 
|  | const char* tag) { | 
|  | OnLogMessage(tag + (": " + msg), severity); | 
|  | } | 
|  |  | 
|  | void LogSink::OnLogMessage(const std::string& msg, | 
|  | LoggingSeverity /* severity */) { | 
|  | OnLogMessage(msg); | 
|  | } | 
|  |  | 
|  | // Inefficient default implementation, override is recommended. | 
|  | void LogSink::OnLogMessage(absl::string_view msg, | 
|  | LoggingSeverity severity, | 
|  | const char* tag) { | 
|  | OnLogMessage(tag + (": " + std::string(msg)), severity); | 
|  | } | 
|  |  | 
|  | void LogSink::OnLogMessage(absl::string_view msg, | 
|  | LoggingSeverity /* severity */) { | 
|  | OnLogMessage(msg); | 
|  | } | 
|  |  | 
|  | void LogSink::OnLogMessage(absl::string_view msg) { | 
|  | OnLogMessage(std::string(msg)); | 
|  | } | 
|  | }  // namespace webrtc |