|  | /* | 
|  | * Copyright (C) 2019 The Android Open Source Project | 
|  | * | 
|  | * Licensed under the Apache License, Version 2.0 (the "License"); | 
|  | * you may not use this file except in compliance with the License. | 
|  | * You may obtain a copy of the License at | 
|  | * | 
|  | *      http://www.apache.org/licenses/LICENSE-2.0 | 
|  | * | 
|  | * Unless required by applicable law or agreed to in writing, software | 
|  | * distributed under the License is distributed on an "AS IS" BASIS, | 
|  | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
|  | * See the License for the specific language governing permissions and | 
|  | * limitations under the License. | 
|  | */ | 
|  |  | 
|  | #include "perfetto/base/logging.h" | 
|  |  | 
|  | #include <stdarg.h> | 
|  | #include <stdio.h> | 
|  |  | 
|  | #if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) | 
|  | #include <unistd.h>  // For isatty() | 
|  | #endif | 
|  |  | 
|  | #include <atomic> | 
|  | #include <memory> | 
|  |  | 
|  | #include "perfetto/base/build_config.h" | 
|  | #include "perfetto/base/time.h" | 
|  | #include "perfetto/ext/base/crash_keys.h" | 
|  | #include "perfetto/ext/base/string_utils.h" | 
|  | #include "perfetto/ext/base/string_view.h" | 
|  | #include "src/base/log_ring_buffer.h" | 
|  |  | 
|  | #if PERFETTO_ENABLE_LOG_RING_BUFFER() && PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) | 
|  | #include <android/set_abort_message.h> | 
|  | #endif | 
|  |  | 
|  | namespace perfetto { | 
|  | namespace base { | 
|  |  | 
|  | namespace { | 
|  | const char kReset[] = "\x1b[0m"; | 
|  | const char kDefault[] = "\x1b[39m"; | 
|  | const char kDim[] = "\x1b[2m"; | 
|  | const char kRed[] = "\x1b[31m"; | 
|  | const char kBoldGreen[] = "\x1b[1m\x1b[32m"; | 
|  | const char kLightGray[] = "\x1b[90m"; | 
|  |  | 
|  | std::atomic<LogMessageCallback> g_log_callback{}; | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_STDERR_CRASH_DUMP) | 
|  | // __attribute__((constructor)) causes a static initializer that automagically | 
|  | // early runs this function before the main(). | 
|  | void PERFETTO_EXPORT_COMPONENT __attribute__((constructor)) | 
|  | InitDebugCrashReporter() { | 
|  | // This function is defined in debug_crash_stack_trace.cc. | 
|  | // The dynamic initializer is in logging.cc because logging.cc is included | 
|  | // in virtually any target that depends on base. Having it in | 
|  | // debug_crash_stack_trace.cc would require figuring out -Wl,whole-archive | 
|  | // which is not worth it. | 
|  | EnableStacktraceOnCrashForDebug(); | 
|  | } | 
|  | #endif | 
|  |  | 
|  | #if PERFETTO_ENABLE_LOG_RING_BUFFER() | 
|  | LogRingBuffer g_log_ring_buffer{}; | 
|  |  | 
|  | // This is global to avoid allocating memory or growing too much the stack | 
|  | // in MaybeSerializeLastLogsForCrashReporting(), which is called from | 
|  | // arbitrary code paths hitting PERFETTO_CHECK()/FATAL(). | 
|  | char g_crash_buf[kLogRingBufEntries * kLogRingBufMsgLen]; | 
|  | #endif | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | void SetLogMessageCallback(LogMessageCallback callback) { | 
|  | g_log_callback.store(callback, std::memory_order_relaxed); | 
|  | } | 
|  |  | 
|  | void LogMessage(LogLev level, | 
|  | const char* fname, | 
|  | int line, | 
|  | const char* fmt, | 
|  | ...) { | 
|  | char stack_buf[512]; | 
|  | std::unique_ptr<char[]> large_buf; | 
|  | char* log_msg = &stack_buf[0]; | 
|  | size_t log_msg_len = 0; | 
|  |  | 
|  | // By default use a stack allocated buffer because most log messages are quite | 
|  | // short. In rare cases they can be larger (e.g. --help). In those cases we | 
|  | // pay the cost of allocating the buffer on the heap. | 
|  | for (size_t max_len = sizeof(stack_buf);;) { | 
|  | va_list args; | 
|  | va_start(args, fmt); | 
|  | int res = vsnprintf(log_msg, max_len, fmt, args); | 
|  | va_end(args); | 
|  |  | 
|  | // If for any reason the print fails, overwrite the message but still print | 
|  | // it. The code below will attach the filename and line, which is still | 
|  | // useful. | 
|  | if (res < 0) { | 
|  | snprintf(log_msg, max_len, "%s", "[printf format error]"); | 
|  | break; | 
|  | } | 
|  |  | 
|  | // if res == max_len, vsnprintf saturated the input buffer. Retry with a | 
|  | // larger buffer in that case (within reasonable limits). | 
|  | if (res < static_cast<int>(max_len) || max_len >= 128 * 1024) { | 
|  | // In case of truncation vsnprintf returns the len that "would have been | 
|  | // written if the string was longer", not the actual chars written. | 
|  | log_msg_len = std::min(static_cast<size_t>(res), max_len - 1); | 
|  | break; | 
|  | } | 
|  | max_len *= 4; | 
|  | large_buf.reset(new char[max_len]); | 
|  | log_msg = &large_buf[0]; | 
|  | } | 
|  |  | 
|  | LogMessageCallback cb = g_log_callback.load(std::memory_order_relaxed); | 
|  | if (cb) { | 
|  | cb({level, line, fname, log_msg}); | 
|  | return; | 
|  | } | 
|  |  | 
|  | const char* color = kDefault; | 
|  | switch (level) { | 
|  | case kLogDebug: | 
|  | color = kDim; | 
|  | break; | 
|  | case kLogInfo: | 
|  | color = kDefault; | 
|  | break; | 
|  | case kLogImportant: | 
|  | color = kBoldGreen; | 
|  | break; | 
|  | case kLogError: | 
|  | color = kRed; | 
|  | break; | 
|  | } | 
|  |  | 
|  | #if !PERFETTO_BUILDFLAG(PERFETTO_OS_WIN) &&  \ | 
|  | !PERFETTO_BUILDFLAG(PERFETTO_OS_WASM) && \ | 
|  | !PERFETTO_BUILDFLAG(PERFETTO_CHROMIUM_BUILD) | 
|  | static const bool use_colors = isatty(STDERR_FILENO); | 
|  | #else | 
|  | static const bool use_colors = false; | 
|  | #endif | 
|  |  | 
|  | // Formats file.cc:line as a space-padded fixed width string. If the file name | 
|  | // |fname| is too long, truncate it on the left-hand side. | 
|  | StackString<10> line_str("%d", line); | 
|  |  | 
|  | // 24 will be the width of the file.cc:line column in the log event. | 
|  | static constexpr size_t kMaxNameAndLine = 24; | 
|  | size_t fname_len = strlen(fname); | 
|  | size_t fname_max = kMaxNameAndLine - line_str.len() - 2;  // 2 = ':' + '\0'. | 
|  | size_t fname_offset = fname_len <= fname_max ? 0 : fname_len - fname_max; | 
|  | StackString<kMaxNameAndLine> file_and_line( | 
|  | "%*s:%s", static_cast<int>(fname_max), &fname[fname_offset], | 
|  | line_str.c_str()); | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) | 
|  | // Logcat has already timestamping, don't re-emit it. | 
|  | __android_log_print(int{ANDROID_LOG_DEBUG} + level, "perfetto", "%s %s", | 
|  | file_and_line.c_str(), log_msg); | 
|  | #endif | 
|  |  | 
|  | // When printing on stderr, print also the timestamp. We don't really care | 
|  | // about the actual time. We just need some reference clock that can be used | 
|  | // to correlated events across differrent processses (e.g. traced and | 
|  | // traced_probes). The wall time % 1000 is good enough. | 
|  | uint32_t t_ms = static_cast<uint32_t>(GetWallTimeMs().count()); | 
|  | uint32_t t_sec = t_ms / 1000; | 
|  | t_ms -= t_sec * 1000; | 
|  | t_sec = t_sec % 1000; | 
|  | StackString<32> timestamp("[%03u.%03u] ", t_sec, t_ms); | 
|  |  | 
|  | if (use_colors) { | 
|  | fprintf(stderr, "%s%s%s%s %s%s%s\n", kLightGray, timestamp.c_str(), | 
|  | file_and_line.c_str(), kReset, color, log_msg, kReset); | 
|  | } else { | 
|  | fprintf(stderr, "%s%s %s\n", timestamp.c_str(), file_and_line.c_str(), | 
|  | log_msg); | 
|  | } | 
|  |  | 
|  | #if PERFETTO_ENABLE_LOG_RING_BUFFER() | 
|  | // Append the message to the ring buffer for crash reporting postmortems. | 
|  | StringView timestamp_sv = timestamp.string_view(); | 
|  | StringView file_and_line_sv = file_and_line.string_view(); | 
|  | StringView log_msg_sv(log_msg, static_cast<size_t>(log_msg_len)); | 
|  | g_log_ring_buffer.Append(timestamp_sv, file_and_line_sv, log_msg_sv); | 
|  | #else | 
|  | ignore_result(log_msg_len); | 
|  | #endif | 
|  | } | 
|  |  | 
|  | #if PERFETTO_ENABLE_LOG_RING_BUFFER() | 
|  | void MaybeSerializeLastLogsForCrashReporting() { | 
|  | // Keep this function minimal. This is called from the watchdog thread, often | 
|  | // when the system is thrashing. | 
|  |  | 
|  | // This is racy because two threads could hit a CHECK/FATAL at the same time. | 
|  | // But if that happens we have bigger problems, not worth designing around it. | 
|  | // The behaviour is still defined in the race case (the string attached to | 
|  | // the crash report will contain a mixture of log strings). | 
|  | size_t wr = 0; | 
|  | wr += SerializeCrashKeys(&g_crash_buf[wr], sizeof(g_crash_buf) - wr); | 
|  | wr += g_log_ring_buffer.Read(&g_crash_buf[wr], sizeof(g_crash_buf) - wr); | 
|  |  | 
|  | // Read() null-terminates the string properly. This is just to avoid UB when | 
|  | // two threads race on each other (T1 writes a shorter string, T2 | 
|  | // overwrites the \0 writing a longer string. T1 continues here before T2 | 
|  | // finishes writing the longer string with the \0 -> boom. | 
|  | g_crash_buf[sizeof(g_crash_buf) - 1] = '\0'; | 
|  |  | 
|  | #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID) | 
|  | // android_set_abort_message() will cause debuggerd to report the message | 
|  | // in the tombstone and in the crash log in logcat. | 
|  | // NOTE: android_set_abort_message() can be called only once. This should | 
|  | // be called only when we are sure we are about to crash. | 
|  | android_set_abort_message(g_crash_buf); | 
|  | #else | 
|  | // Print out the message on stderr on Linux/Mac/Win. | 
|  | fputs("\n-----BEGIN PERFETTO PRE-CRASH LOG-----\n", stderr); | 
|  | fputs(g_crash_buf, stderr); | 
|  | fputs("\n-----END PERFETTO PRE-CRASH LOG-----\n", stderr); | 
|  | #endif | 
|  | } | 
|  | #endif  // PERFETTO_ENABLE_LOG_RING_BUFFER | 
|  |  | 
|  | }  // namespace base | 
|  | }  // namespace perfetto |