xref: /aosp_15_r20/external/libchrome/base/test/trace_to_file.cc (revision 635a864187cb8b6c713ff48b7e790a6b21769273)
1*635a8641SAndroid Build Coastguard Worker // Copyright (c) 2014 The Chromium Authors. All rights reserved.
2*635a8641SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*635a8641SAndroid Build Coastguard Worker // found in the LICENSE file.
4*635a8641SAndroid Build Coastguard Worker 
5*635a8641SAndroid Build Coastguard Worker #include "base/test/trace_to_file.h"
6*635a8641SAndroid Build Coastguard Worker 
7*635a8641SAndroid Build Coastguard Worker #include "base/base_switches.h"
8*635a8641SAndroid Build Coastguard Worker #include "base/bind.h"
9*635a8641SAndroid Build Coastguard Worker #include "base/command_line.h"
10*635a8641SAndroid Build Coastguard Worker #include "base/files/file_util.h"
11*635a8641SAndroid Build Coastguard Worker #include "base/memory/ref_counted_memory.h"
12*635a8641SAndroid Build Coastguard Worker #include "base/run_loop.h"
13*635a8641SAndroid Build Coastguard Worker #include "base/trace_event/trace_buffer.h"
14*635a8641SAndroid Build Coastguard Worker #include "base/trace_event/trace_log.h"
15*635a8641SAndroid Build Coastguard Worker 
16*635a8641SAndroid Build Coastguard Worker namespace base {
17*635a8641SAndroid Build Coastguard Worker namespace test {
18*635a8641SAndroid Build Coastguard Worker 
TraceToFile()19*635a8641SAndroid Build Coastguard Worker TraceToFile::TraceToFile() : started_(false) {
20*635a8641SAndroid Build Coastguard Worker }
21*635a8641SAndroid Build Coastguard Worker 
~TraceToFile()22*635a8641SAndroid Build Coastguard Worker TraceToFile::~TraceToFile() {
23*635a8641SAndroid Build Coastguard Worker   EndTracingIfNeeded();
24*635a8641SAndroid Build Coastguard Worker }
25*635a8641SAndroid Build Coastguard Worker 
BeginTracingFromCommandLineOptions()26*635a8641SAndroid Build Coastguard Worker void TraceToFile::BeginTracingFromCommandLineOptions() {
27*635a8641SAndroid Build Coastguard Worker   DCHECK(CommandLine::InitializedForCurrentProcess());
28*635a8641SAndroid Build Coastguard Worker   DCHECK(!started_);
29*635a8641SAndroid Build Coastguard Worker 
30*635a8641SAndroid Build Coastguard Worker   if (!CommandLine::ForCurrentProcess()->HasSwitch(switches::kTraceToFile))
31*635a8641SAndroid Build Coastguard Worker     return;
32*635a8641SAndroid Build Coastguard Worker 
33*635a8641SAndroid Build Coastguard Worker   // Empty filter (i.e. just --trace-to-file) turns into default categories in
34*635a8641SAndroid Build Coastguard Worker   // TraceEventImpl
35*635a8641SAndroid Build Coastguard Worker   std::string filter = CommandLine::ForCurrentProcess()->GetSwitchValueASCII(
36*635a8641SAndroid Build Coastguard Worker       switches::kTraceToFile);
37*635a8641SAndroid Build Coastguard Worker 
38*635a8641SAndroid Build Coastguard Worker   FilePath path;
39*635a8641SAndroid Build Coastguard Worker   if (CommandLine::ForCurrentProcess()->HasSwitch(switches::kTraceToFileName)) {
40*635a8641SAndroid Build Coastguard Worker     path = FilePath(CommandLine::ForCurrentProcess()
41*635a8641SAndroid Build Coastguard Worker                         ->GetSwitchValuePath(switches::kTraceToFileName));
42*635a8641SAndroid Build Coastguard Worker   } else {
43*635a8641SAndroid Build Coastguard Worker     path = FilePath(FILE_PATH_LITERAL("trace.json"));
44*635a8641SAndroid Build Coastguard Worker   }
45*635a8641SAndroid Build Coastguard Worker 
46*635a8641SAndroid Build Coastguard Worker   BeginTracing(path, filter);
47*635a8641SAndroid Build Coastguard Worker }
48*635a8641SAndroid Build Coastguard Worker 
BeginTracing(const FilePath & path,const std::string & categories)49*635a8641SAndroid Build Coastguard Worker void TraceToFile::BeginTracing(const FilePath& path,
50*635a8641SAndroid Build Coastguard Worker                                const std::string& categories) {
51*635a8641SAndroid Build Coastguard Worker   DCHECK(!started_);
52*635a8641SAndroid Build Coastguard Worker   started_ = true;
53*635a8641SAndroid Build Coastguard Worker   path_ = path;
54*635a8641SAndroid Build Coastguard Worker   WriteFileHeader();
55*635a8641SAndroid Build Coastguard Worker 
56*635a8641SAndroid Build Coastguard Worker   trace_event::TraceLog::GetInstance()->SetEnabled(
57*635a8641SAndroid Build Coastguard Worker       trace_event::TraceConfig(categories, trace_event::RECORD_UNTIL_FULL),
58*635a8641SAndroid Build Coastguard Worker       trace_event::TraceLog::RECORDING_MODE);
59*635a8641SAndroid Build Coastguard Worker }
60*635a8641SAndroid Build Coastguard Worker 
WriteFileHeader()61*635a8641SAndroid Build Coastguard Worker void TraceToFile::WriteFileHeader() {
62*635a8641SAndroid Build Coastguard Worker   const char str[] = "{\"traceEvents\": [";
63*635a8641SAndroid Build Coastguard Worker   WriteFile(path_, str, static_cast<int>(strlen(str)));
64*635a8641SAndroid Build Coastguard Worker }
65*635a8641SAndroid Build Coastguard Worker 
AppendFileFooter()66*635a8641SAndroid Build Coastguard Worker void TraceToFile::AppendFileFooter() {
67*635a8641SAndroid Build Coastguard Worker   const char str[] = "]}";
68*635a8641SAndroid Build Coastguard Worker   AppendToFile(path_, str, static_cast<int>(strlen(str)));
69*635a8641SAndroid Build Coastguard Worker }
70*635a8641SAndroid Build Coastguard Worker 
TraceOutputCallback(const std::string & data)71*635a8641SAndroid Build Coastguard Worker void TraceToFile::TraceOutputCallback(const std::string& data) {
72*635a8641SAndroid Build Coastguard Worker   bool ret = AppendToFile(path_, data.c_str(), static_cast<int>(data.size()));
73*635a8641SAndroid Build Coastguard Worker   DCHECK(ret);
74*635a8641SAndroid Build Coastguard Worker }
75*635a8641SAndroid Build Coastguard Worker 
OnTraceDataCollected(Closure quit_closure,trace_event::TraceResultBuffer * buffer,const scoped_refptr<RefCountedString> & json_events_str,bool has_more_events)76*635a8641SAndroid Build Coastguard Worker static void OnTraceDataCollected(
77*635a8641SAndroid Build Coastguard Worker     Closure quit_closure,
78*635a8641SAndroid Build Coastguard Worker     trace_event::TraceResultBuffer* buffer,
79*635a8641SAndroid Build Coastguard Worker     const scoped_refptr<RefCountedString>& json_events_str,
80*635a8641SAndroid Build Coastguard Worker     bool has_more_events) {
81*635a8641SAndroid Build Coastguard Worker   buffer->AddFragment(json_events_str->data());
82*635a8641SAndroid Build Coastguard Worker   if (!has_more_events)
83*635a8641SAndroid Build Coastguard Worker     quit_closure.Run();
84*635a8641SAndroid Build Coastguard Worker }
85*635a8641SAndroid Build Coastguard Worker 
EndTracingIfNeeded()86*635a8641SAndroid Build Coastguard Worker void TraceToFile::EndTracingIfNeeded() {
87*635a8641SAndroid Build Coastguard Worker   if (!started_)
88*635a8641SAndroid Build Coastguard Worker     return;
89*635a8641SAndroid Build Coastguard Worker   started_ = false;
90*635a8641SAndroid Build Coastguard Worker 
91*635a8641SAndroid Build Coastguard Worker   trace_event::TraceLog::GetInstance()->SetDisabled();
92*635a8641SAndroid Build Coastguard Worker 
93*635a8641SAndroid Build Coastguard Worker   trace_event::TraceResultBuffer buffer;
94*635a8641SAndroid Build Coastguard Worker   buffer.SetOutputCallback(
95*635a8641SAndroid Build Coastguard Worker       Bind(&TraceToFile::TraceOutputCallback, Unretained(this)));
96*635a8641SAndroid Build Coastguard Worker 
97*635a8641SAndroid Build Coastguard Worker   RunLoop run_loop;
98*635a8641SAndroid Build Coastguard Worker   trace_event::TraceLog::GetInstance()->Flush(
99*635a8641SAndroid Build Coastguard Worker       Bind(&OnTraceDataCollected, run_loop.QuitClosure(), Unretained(&buffer)));
100*635a8641SAndroid Build Coastguard Worker   run_loop.Run();
101*635a8641SAndroid Build Coastguard Worker 
102*635a8641SAndroid Build Coastguard Worker   AppendFileFooter();
103*635a8641SAndroid Build Coastguard Worker }
104*635a8641SAndroid Build Coastguard Worker 
105*635a8641SAndroid Build Coastguard Worker }  // namespace test
106*635a8641SAndroid Build Coastguard Worker }  // namespace base
107