| /* | 
 |  *  Copyright (c) 2012 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 "webrtc/base/event_tracer.h" | 
 |  | 
 | #include <inttypes.h> | 
 |  | 
 | #include <vector> | 
 |  | 
 | #include "webrtc/base/checks.h" | 
 | #include "webrtc/base/criticalsection.h" | 
 | #include "webrtc/base/event.h" | 
 | #include "webrtc/base/logging.h" | 
 | #include "webrtc/base/platform_thread.h" | 
 | #include "webrtc/base/timeutils.h" | 
 | #include "webrtc/base/trace_event.h" | 
 |  | 
 | namespace webrtc { | 
 |  | 
 | namespace { | 
 |  | 
 | GetCategoryEnabledPtr g_get_category_enabled_ptr = nullptr; | 
 | AddTraceEventPtr g_add_trace_event_ptr = nullptr; | 
 |  | 
 | }  // namespace | 
 |  | 
 | void SetupEventTracer(GetCategoryEnabledPtr get_category_enabled_ptr, | 
 |                       AddTraceEventPtr add_trace_event_ptr) { | 
 |   g_get_category_enabled_ptr = get_category_enabled_ptr; | 
 |   g_add_trace_event_ptr = add_trace_event_ptr; | 
 | } | 
 |  | 
 | const unsigned char* EventTracer::GetCategoryEnabled(const char* name) { | 
 |   if (g_get_category_enabled_ptr) | 
 |     return g_get_category_enabled_ptr(name); | 
 |  | 
 |   // A string with null terminator means category is disabled. | 
 |   return reinterpret_cast<const unsigned char*>("\0"); | 
 | } | 
 |  | 
 | // Arguments to this function (phase, etc.) are as defined in | 
 | // webrtc/base/trace_event.h. | 
 | void EventTracer::AddTraceEvent(char phase, | 
 |                                 const unsigned char* category_enabled, | 
 |                                 const char* name, | 
 |                                 unsigned long long id, | 
 |                                 int num_args, | 
 |                                 const char** arg_names, | 
 |                                 const unsigned char* arg_types, | 
 |                                 const unsigned long long* arg_values, | 
 |                                 unsigned char flags) { | 
 |   if (g_add_trace_event_ptr) { | 
 |     g_add_trace_event_ptr(phase, | 
 |                           category_enabled, | 
 |                           name, | 
 |                           id, | 
 |                           num_args, | 
 |                           arg_names, | 
 |                           arg_types, | 
 |                           arg_values, | 
 |                           flags); | 
 |   } | 
 | } | 
 |  | 
 | }  // namespace webrtc | 
 |  | 
 | namespace rtc { | 
 | namespace tracing { | 
 | namespace { | 
 |  | 
 | static bool EventTracingThreadFunc(void* params); | 
 |  | 
 | // Atomic-int fast path for avoiding logging when disabled. | 
 | static volatile int g_event_logging_active = 0; | 
 |  | 
 | // TODO(pbos): Log metadata for all threads, etc. | 
 | class EventLogger final { | 
 |  public: | 
 |   EventLogger() | 
 |       : logging_thread_(EventTracingThreadFunc, this, "EventTracingThread"), | 
 |         shutdown_event_(false, false) {} | 
 |   ~EventLogger() { RTC_DCHECK(thread_checker_.CalledOnValidThread()); } | 
 |  | 
 |   void AddTraceEvent(const char* name, | 
 |                      const unsigned char* category_enabled, | 
 |                      char phase, | 
 |                      uint64_t timestamp, | 
 |                      int pid, | 
 |                      rtc::PlatformThreadId thread_id) { | 
 |     rtc::CritScope lock(&crit_); | 
 |     trace_events_.push_back( | 
 |         {name, category_enabled, phase, timestamp, 1, thread_id}); | 
 |   } | 
 |  | 
 | // The TraceEvent format is documented here: | 
 | // https://docs.google.com/document/d/1CvAClvFfyA5R-PhYUmn5OOQtYMH4h6I0nSsKchNAySU/preview | 
 |   void Log() { | 
 |     RTC_DCHECK(output_file_); | 
 |     static const int kLoggingIntervalMs = 100; | 
 |     fprintf(output_file_, "{ \"traceEvents\": [\n"); | 
 |     bool has_logged_event = false; | 
 |     while (true) { | 
 |       bool shutting_down = shutdown_event_.Wait(kLoggingIntervalMs); | 
 |       std::vector<TraceEvent> events; | 
 |       { | 
 |         rtc::CritScope lock(&crit_); | 
 |         trace_events_.swap(events); | 
 |       } | 
 |       for (const TraceEvent& e : events) { | 
 |         fprintf(output_file_, | 
 |                 "%s{ \"name\": \"%s\"" | 
 |                 ", \"cat\": \"%s\"" | 
 |                 ", \"ph\": \"%c\"" | 
 |                 ", \"ts\": %" PRIu64 | 
 |                 ", \"pid\": %d" | 
 | #if defined(WEBRTC_WIN) | 
 |                 ", \"tid\": %lu" | 
 | #else | 
 |                 ", \"tid\": %d" | 
 | #endif  // defined(WEBRTC_WIN) | 
 |                 "}\n", | 
 |                 has_logged_event ? "," : " ", e.name, e.category_enabled, | 
 |                 e.phase, e.timestamp, e.pid, e.tid); | 
 |         has_logged_event = true; | 
 |       } | 
 |       if (shutting_down) | 
 |         break; | 
 |     } | 
 |     fprintf(output_file_, "]}\n"); | 
 |     if (output_file_owned_) | 
 |       fclose(output_file_); | 
 |     output_file_ = nullptr; | 
 |   } | 
 |  | 
 |   void Start(FILE* file, bool owned) { | 
 |     RTC_DCHECK(thread_checker_.CalledOnValidThread()); | 
 |     RTC_DCHECK(file); | 
 |     RTC_DCHECK(!output_file_); | 
 |     output_file_ = file; | 
 |     output_file_owned_ = owned; | 
 |     { | 
 |       rtc::CritScope lock(&crit_); | 
 |       // Since the atomic fast-path for adding events to the queue can be | 
 |       // bypassed while the logging thread is shutting down there may be some | 
 |       // stale events in the queue, hence the vector needs to be cleared to not | 
 |       // log events from a previous logging session (which may be days old). | 
 |       trace_events_.clear(); | 
 |     } | 
 |     // Enable event logging (fast-path). This should be disabled since starting | 
 |     // shouldn't be done twice. | 
 |     RTC_CHECK_EQ(0, | 
 |                  rtc::AtomicOps::CompareAndSwap(&g_event_logging_active, 0, 1)); | 
 |  | 
 |     // Finally start, everything should be set up now. | 
 |     logging_thread_.Start(); | 
 |     TRACE_EVENT_INSTANT0("webrtc", "EventLogger::Start"); | 
 |     logging_thread_.SetPriority(kLowPriority); | 
 |   } | 
 |  | 
 |   void Stop() { | 
 |     RTC_DCHECK(thread_checker_.CalledOnValidThread()); | 
 |     TRACE_EVENT_INSTANT0("webrtc", "EventLogger::Stop"); | 
 |     // Try to stop. Abort if we're not currently logging. | 
 |     if (rtc::AtomicOps::CompareAndSwap(&g_event_logging_active, 1, 0) == 0) | 
 |       return; | 
 |  | 
 |     // Wake up logging thread to finish writing. | 
 |     shutdown_event_.Set(); | 
 |     // Join the logging thread. | 
 |     logging_thread_.Stop(); | 
 |   } | 
 |  | 
 |  private: | 
 |   struct TraceEvent { | 
 |     const char* name; | 
 |     const unsigned char* category_enabled; | 
 |     char phase; | 
 |     uint64_t timestamp; | 
 |     int pid; | 
 |     rtc::PlatformThreadId tid; | 
 |   }; | 
 |  | 
 |   rtc::CriticalSection crit_; | 
 |   std::vector<TraceEvent> trace_events_ GUARDED_BY(crit_); | 
 |   rtc::PlatformThread logging_thread_; | 
 |   rtc::Event shutdown_event_; | 
 |   rtc::ThreadChecker thread_checker_; | 
 |   FILE* output_file_ = nullptr; | 
 |   bool output_file_owned_ = false; | 
 | }; | 
 |  | 
 | static bool EventTracingThreadFunc(void* params) { | 
 |   static_cast<EventLogger*>(params)->Log(); | 
 |   return true; | 
 | } | 
 |  | 
 | static EventLogger* volatile g_event_logger = nullptr; | 
 | static const char* const kDisabledTracePrefix = TRACE_DISABLED_BY_DEFAULT(""); | 
 | const unsigned char* InternalGetCategoryEnabled(const char* name) { | 
 |   const char* prefix_ptr = &kDisabledTracePrefix[0]; | 
 |   const char* name_ptr = name; | 
 |   // Check whether name contains the default-disabled prefix. | 
 |   while (*prefix_ptr == *name_ptr && *prefix_ptr != '\0') { | 
 |     ++prefix_ptr; | 
 |     ++name_ptr; | 
 |   } | 
 |   return reinterpret_cast<const unsigned char*>(*prefix_ptr == '\0' ? "" | 
 |                                                                     : name); | 
 | } | 
 |  | 
 | void InternalAddTraceEvent(char phase, | 
 |                            const unsigned char* category_enabled, | 
 |                            const char* name, | 
 |                            unsigned long long id, | 
 |                            int num_args, | 
 |                            const char** arg_names, | 
 |                            const unsigned char* arg_types, | 
 |                            const unsigned long long* arg_values, | 
 |                            unsigned char flags) { | 
 |   // Fast path for when event tracing is inactive. | 
 |   if (rtc::AtomicOps::AcquireLoad(&g_event_logging_active) == 0) | 
 |     return; | 
 |  | 
 |   g_event_logger->AddTraceEvent(name, category_enabled, phase, | 
 |                                 rtc::TimeMicros(), 1, rtc::CurrentThreadId()); | 
 | } | 
 |  | 
 | }  // namespace | 
 |  | 
 | void SetupInternalTracer() { | 
 |   RTC_CHECK(rtc::AtomicOps::CompareAndSwapPtr( | 
 |                 &g_event_logger, static_cast<EventLogger*>(nullptr), | 
 |                 new EventLogger()) == nullptr); | 
 |   g_event_logger = new EventLogger(); | 
 |   webrtc::SetupEventTracer(InternalGetCategoryEnabled, InternalAddTraceEvent); | 
 | } | 
 |  | 
 | void StartInternalCaptureToFile(FILE* file) { | 
 |   g_event_logger->Start(file, false); | 
 | } | 
 |  | 
 | bool StartInternalCapture(const char* filename) { | 
 |   FILE* file = fopen(filename, "w"); | 
 |   if (!file) { | 
 |     LOG(LS_ERROR) << "Failed to open trace file '" << filename | 
 |                   << "' for writing."; | 
 |     return false; | 
 |   } | 
 |   g_event_logger->Start(file, true); | 
 |   return true; | 
 | } | 
 |  | 
 | void StopInternalCapture() { | 
 |   g_event_logger->Stop(); | 
 | } | 
 |  | 
 | void ShutdownInternalTracer() { | 
 |   StopInternalCapture(); | 
 |   EventLogger* old_logger = rtc::AtomicOps::AcquireLoadPtr(&g_event_logger); | 
 |   RTC_DCHECK(old_logger); | 
 |   RTC_CHECK(rtc::AtomicOps::CompareAndSwapPtr( | 
 |                 &g_event_logger, old_logger, | 
 |                 static_cast<EventLogger*>(nullptr)) == old_logger); | 
 |   delete old_logger; | 
 |   webrtc::SetupEventTracer(nullptr, nullptr); | 
 | } | 
 |  | 
 | }  // namespace tracing | 
 | }  // namespace rtc |