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