Commit 4d2bcd38 authored by Eric Seckler's avatar Eric Seckler Committed by Commit Bot

tracing: Add a system tracing integration test

The new test verifies that system tracing output on CrOS, Cast and Win
actually ends up in a trace - in preparation for porting the respective
tracing agents to the new perfetto service.

Bug: 900603
Change-Id: I8383a907d2aa843cb2f67f5300e82c7684f116fc
Reviewed-on: https://chromium-review.googlesource.com/c/1309835
Commit-Queue: Eric Seckler <eseckler@chromium.org>
Reviewed-by: default avataroysteine <oysteine@chromium.org>
Reviewed-by: default avatarMichael Spang <spang@chromium.org>
Reviewed-by: default avatarSteven Bennetts <stevenjb@chromium.org>
Reviewed-by: default avatarScott Violet <sky@chromium.org>
Cr-Commit-Position: refs/heads/master@{#604546}
parent 3ca0b0f3
...@@ -40,6 +40,7 @@ cast_source_set("system_tracer") { ...@@ -40,6 +40,7 @@ cast_source_set("system_tracer") {
public_deps = [ public_deps = [
":system_tracing_common", ":system_tracing_common",
"//base", "//base",
"//chromecast:chromecast_buildflags",
] ]
} }
......
...@@ -16,6 +16,7 @@ ...@@ -16,6 +16,7 @@
#include "base/posix/eintr_wrapper.h" #include "base/posix/eintr_wrapper.h"
#include "base/posix/unix_domain_socket.h" #include "base/posix/unix_domain_socket.h"
#include "base/trace_event/trace_config.h" #include "base/trace_event/trace_config.h"
#include "chromecast/chromecast_buildflags.h"
#include "chromecast/tracing/system_tracing_common.h" #include "chromecast/tracing/system_tracing_common.h"
namespace chromecast { namespace chromecast {
...@@ -43,16 +44,57 @@ base::ScopedFD CreateClientSocket() { ...@@ -43,16 +44,57 @@ base::ScopedFD CreateClientSocket() {
return socket_fd; return socket_fd;
} }
} // namespace class SystemTracerImpl : public SystemTracer {
public:
SystemTracer::SystemTracer() : buffer_(new char[kBufferSize]) {} SystemTracerImpl() : buffer_(new char[kBufferSize]) {}
~SystemTracerImpl() override { Cleanup(); }
SystemTracer::~SystemTracer() {
Cleanup(); void StartTracing(base::StringPiece categories,
} StartTracingCallback callback) override;
void SystemTracer::StartTracing(base::StringPiece categories, void StopTracing(const StopTracingCallback& callback) override;
StartTracingCallback callback) {
private:
enum class State {
INITIAL, // Not yet started.
STARTING, // Sent start message, waiting for ack.
TRACING, // Tracing, not yet requested stop.
READING, // Trace stopped, reading output.
FINISHED, // All done.
};
void ReceiveStartAckAndTracePipe();
void ReceiveTraceData();
void FailStartTracing();
void FailStopTracing();
void SendPartialTraceData();
void FinishTracing();
void Cleanup();
// Current state of tracing attempt.
State state_ = State::INITIAL;
// Unix socket connection to tracing daemon.
base::ScopedFD connection_fd_;
std::unique_ptr<base::FileDescriptorWatcher::Controller> connection_watcher_;
// Pipe for trace data.
base::ScopedFD trace_pipe_fd_;
std::unique_ptr<base::FileDescriptorWatcher::Controller> trace_pipe_watcher_;
// Read buffer (of size kBufferSize).
std::unique_ptr<char[]> buffer_;
// Callbacks for StartTracing() and StopTracing().
StartTracingCallback start_tracing_callback_;
StopTracingCallback stop_tracing_callback_;
// Trace data.
std::string trace_data_;
};
void SystemTracerImpl::StartTracing(base::StringPiece categories,
StartTracingCallback callback) {
start_tracing_callback_ = std::move(callback); start_tracing_callback_ = std::move(callback);
if (state_ != State::INITIAL) { if (state_ != State::INITIAL) {
FailStartTracing(); FailStartTracing();
...@@ -80,12 +122,12 @@ void SystemTracer::StartTracing(base::StringPiece categories, ...@@ -80,12 +122,12 @@ void SystemTracer::StartTracing(base::StringPiece categories,
connection_watcher_ = base::FileDescriptorWatcher::WatchReadable( connection_watcher_ = base::FileDescriptorWatcher::WatchReadable(
connection_fd_.get(), connection_fd_.get(),
base::BindRepeating(&SystemTracer::ReceiveStartAckAndTracePipe, base::BindRepeating(&SystemTracerImpl::ReceiveStartAckAndTracePipe,
base::Unretained(this))); base::Unretained(this)));
state_ = State::STARTING; state_ = State::STARTING;
} }
void SystemTracer::StopTracing(const StopTracingCallback& callback) { void SystemTracerImpl::StopTracing(const StopTracingCallback& callback) {
stop_tracing_callback_ = callback; stop_tracing_callback_ = callback;
if (state_ != State::TRACING) { if (state_ != State::TRACING) {
FailStopTracing(); FailStopTracing();
...@@ -102,12 +144,13 @@ void SystemTracer::StopTracing(const StopTracingCallback& callback) { ...@@ -102,12 +144,13 @@ void SystemTracer::StopTracing(const StopTracingCallback& callback) {
} }
trace_pipe_watcher_ = base::FileDescriptorWatcher::WatchReadable( trace_pipe_watcher_ = base::FileDescriptorWatcher::WatchReadable(
trace_pipe_fd_.get(), base::BindRepeating(&SystemTracer::ReceiveTraceData, trace_pipe_fd_.get(),
base::Unretained(this))); base::BindRepeating(&SystemTracerImpl::ReceiveTraceData,
base::Unretained(this)));
state_ = State::READING; state_ = State::READING;
} }
void SystemTracer::ReceiveStartAckAndTracePipe() { void SystemTracerImpl::ReceiveStartAckAndTracePipe() {
DCHECK_EQ(state_, State::STARTING); DCHECK_EQ(state_, State::STARTING);
std::vector<base::ScopedFD> fds; std::vector<base::ScopedFD> fds;
...@@ -135,7 +178,7 @@ void SystemTracer::ReceiveStartAckAndTracePipe() { ...@@ -135,7 +178,7 @@ void SystemTracer::ReceiveStartAckAndTracePipe() {
std::move(start_tracing_callback_).Run(Status::OK); std::move(start_tracing_callback_).Run(Status::OK);
} }
void SystemTracer::ReceiveTraceData() { void SystemTracerImpl::ReceiveTraceData() {
DCHECK_EQ(state_, State::READING); DCHECK_EQ(state_, State::READING);
for (;;) { for (;;) {
...@@ -164,29 +207,29 @@ void SystemTracer::ReceiveTraceData() { ...@@ -164,29 +207,29 @@ void SystemTracer::ReceiveTraceData() {
} }
} }
void SystemTracer::FailStartTracing() { void SystemTracerImpl::FailStartTracing() {
std::move(start_tracing_callback_).Run(Status::FAIL); std::move(start_tracing_callback_).Run(Status::FAIL);
Cleanup(); Cleanup();
} }
void SystemTracer::FailStopTracing() { void SystemTracerImpl::FailStopTracing() {
stop_tracing_callback_.Run(Status::FAIL, ""); stop_tracing_callback_.Run(Status::FAIL, "");
Cleanup(); Cleanup();
} }
void SystemTracer::SendPartialTraceData() { void SystemTracerImpl::SendPartialTraceData() {
DCHECK_EQ(state_, State::READING); DCHECK_EQ(state_, State::READING);
stop_tracing_callback_.Run(Status::KEEP_GOING, std::move(trace_data_)); stop_tracing_callback_.Run(Status::KEEP_GOING, std::move(trace_data_));
trace_data_ = ""; trace_data_ = "";
} }
void SystemTracer::FinishTracing() { void SystemTracerImpl::FinishTracing() {
DCHECK_EQ(state_, State::READING); DCHECK_EQ(state_, State::READING);
stop_tracing_callback_.Run(Status::OK, std::move(trace_data_)); stop_tracing_callback_.Run(Status::OK, std::move(trace_data_));
Cleanup(); Cleanup();
} }
void SystemTracer::Cleanup() { void SystemTracerImpl::Cleanup() {
connection_watcher_.reset(); connection_watcher_.reset();
connection_fd_.reset(); connection_fd_.reset();
trace_pipe_watcher_.reset(); trace_pipe_watcher_.reset();
...@@ -196,4 +239,31 @@ void SystemTracer::Cleanup() { ...@@ -196,4 +239,31 @@ void SystemTracer::Cleanup() {
state_ = State::FINISHED; state_ = State::FINISHED;
} }
class FakeSystemTracer : public SystemTracer {
public:
FakeSystemTracer() = default;
~FakeSystemTracer() override = default;
void StartTracing(base::StringPiece categories,
StartTracingCallback callback) override {
std::move(callback).Run(Status::OK);
}
void StopTracing(const StopTracingCallback& callback) override {
std::string trace_data = "# tracer: nop\n";
std::move(callback).Run(Status::OK, std::move(trace_data));
}
};
} // namespace
// static
std::unique_ptr<SystemTracer> SystemTracer::Create() {
#if BUILDFLAG(IS_CAST_DESKTOP_BUILD)
return std::make_unique<FakeSystemTracer>();
#else
return std::make_unique<SystemTracerImpl>();
#endif
}
} // namespace chromecast } // namespace chromecast
...@@ -18,8 +18,9 @@ namespace chromecast { ...@@ -18,8 +18,9 @@ namespace chromecast {
class SystemTracer { class SystemTracer {
public: public:
SystemTracer(); static std::unique_ptr<SystemTracer> Create();
~SystemTracer();
virtual ~SystemTracer() = default;
enum class Status { enum class Status {
OK, OK,
...@@ -32,54 +33,20 @@ class SystemTracer { ...@@ -32,54 +33,20 @@ class SystemTracer {
base::RepeatingCallback<void(Status status, std::string trace_data)>; base::RepeatingCallback<void(Status status, std::string trace_data)>;
// Start system tracing for categories in |categories| (comma separated). // Start system tracing for categories in |categories| (comma separated).
void StartTracing(base::StringPiece categories, virtual void StartTracing(base::StringPiece categories,
StartTracingCallback callback); StartTracingCallback callback) = 0;
// Stop system tracing. // Stop system tracing.
// //
// This will call |callback| on the current thread with the trace data. If // This will call |callback| on the current thread with the trace data. If
// |status| is Status::KEEP_GOING, another call will be made with additional // |status| is Status::KEEP_GOING, another call will be made with additional
// data. // data.
void StopTracing(const StopTracingCallback& callback); virtual void StopTracing(const StopTracingCallback& callback) = 0;
private:
enum class State {
INITIAL, // Not yet started.
STARTING, // Sent start message, waiting for ack.
TRACING, // Tracing, not yet requested stop.
READING, // Trace stopped, reading output.
FINISHED, // All done.
};
void ReceiveStartAckAndTracePipe();
void ReceiveTraceData();
void FailStartTracing();
void FailStopTracing();
void SendPartialTraceData();
void FinishTracing();
void Cleanup();
// Current state of tracing attempt.
State state_ = State::INITIAL;
// Unix socket connection to tracing daemon. protected:
base::ScopedFD connection_fd_; SystemTracer() = default;
std::unique_ptr<base::FileDescriptorWatcher::Controller> connection_watcher_;
// Pipe for trace data.
base::ScopedFD trace_pipe_fd_;
std::unique_ptr<base::FileDescriptorWatcher::Controller> trace_pipe_watcher_;
// Read buffer (of size kBufferSize).
std::unique_ptr<char[]> buffer_;
// Callbacks for StartTracing() and StopTracing().
StartTracingCallback start_tracing_callback_;
StopTracingCallback stop_tracing_callback_;
// Trace data.
std::string trace_data_;
private:
DISALLOW_COPY_AND_ASSIGN(SystemTracer); DISALLOW_COPY_AND_ASSIGN(SystemTracer);
}; };
......
...@@ -68,9 +68,9 @@ void FakeDebugDaemonClient::StartAgentTracing( ...@@ -68,9 +68,9 @@ void FakeDebugDaemonClient::StartAgentTracing(
void FakeDebugDaemonClient::StopAgentTracing( void FakeDebugDaemonClient::StopAgentTracing(
StopAgentTracingCallback callback) { StopAgentTracingCallback callback) {
std::string no_data; std::string trace_data = "# tracer: nop\n";
std::move(callback).Run(GetTracingAgentName(), GetTraceEventLabel(), std::move(callback).Run(GetTracingAgentName(), GetTraceEventLabel(),
base::RefCountedString::TakeString(&no_data)); base::RefCountedString::TakeString(&trace_data));
} }
void FakeDebugDaemonClient::SetStopAgentTracingTaskRunner( void FakeDebugDaemonClient::SetStopAgentTracingTaskRunner(
......
...@@ -92,7 +92,7 @@ void CastTracingAgent::GetCategories(Agent::GetCategoriesCallback callback) { ...@@ -92,7 +92,7 @@ void CastTracingAgent::GetCategories(Agent::GetCategoriesCallback callback) {
void CastTracingAgent::StartTracingOnIO( void CastTracingAgent::StartTracingOnIO(
scoped_refptr<base::TaskRunner> reply_task_runner, scoped_refptr<base::TaskRunner> reply_task_runner,
const std::string& categories) { const std::string& categories) {
system_tracer_ = std::make_unique<chromecast::SystemTracer>(); system_tracer_ = chromecast::SystemTracer::Create();
system_tracer_->StartTracing( system_tracer_->StartTracing(
categories, base::BindOnce(&CastTracingAgent::FinishStartOnIO, categories, base::BindOnce(&CastTracingAgent::FinishStartOnIO,
......
...@@ -27,6 +27,7 @@ ...@@ -27,6 +27,7 @@
#include "content/test/test_content_browser_client.h" #include "content/test/test_content_browser_client.h"
#include "services/network/public/cpp/shared_url_loader_factory.h" #include "services/network/public/cpp/shared_url_loader_factory.h"
#include "services/tracing/public/cpp/trace_event_agent.h" #include "services/tracing/public/cpp/trace_event_agent.h"
#include "services/tracing/public/cpp/tracing_features.h"
using base::trace_event::RECORD_CONTINUOUSLY; using base::trace_event::RECORD_CONTINUOUSLY;
using base::trace_event::RECORD_UNTIL_FULL; using base::trace_event::RECORD_UNTIL_FULL;
...@@ -198,7 +199,7 @@ class TracingControllerTest : public ContentBrowserTest { ...@@ -198,7 +199,7 @@ class TracingControllerTest : public ContentBrowserTest {
return last_data_; return last_data_;
} }
void TestStartAndStopTracingString() { void TestStartAndStopTracingString(bool enable_systrace = false) {
Navigate(shell()); Navigate(shell());
TracingController* controller = TracingController::GetInstance(); TracingController* controller = TracingController::GetInstance();
...@@ -208,8 +209,10 @@ class TracingControllerTest : public ContentBrowserTest { ...@@ -208,8 +209,10 @@ class TracingControllerTest : public ContentBrowserTest {
TracingController::StartTracingDoneCallback callback = TracingController::StartTracingDoneCallback callback =
base::BindOnce(&TracingControllerTest::StartTracingDoneCallbackTest, base::BindOnce(&TracingControllerTest::StartTracingDoneCallbackTest,
base::Unretained(this), run_loop.QuitClosure()); base::Unretained(this), run_loop.QuitClosure());
bool result = TraceConfig config;
controller->StartTracing(TraceConfig(), std::move(callback)); if (enable_systrace)
config.EnableSystrace();
bool result = controller->StartTracing(config, std::move(callback));
ASSERT_TRUE(result); ASSERT_TRUE(result);
run_loop.Run(); run_loop.Run();
EXPECT_EQ(enable_recording_done_callback_count(), 1); EXPECT_EQ(enable_recording_done_callback_count(), 1);
...@@ -537,4 +540,24 @@ IN_PROC_BROWSER_TEST_F(TracingControllerTest, MAYBE_DoubleStopTracing) { ...@@ -537,4 +540,24 @@ IN_PROC_BROWSER_TEST_F(TracingControllerTest, MAYBE_DoubleStopTracing) {
run_loop.Run(); run_loop.Run();
} }
// TODO(crbug.com/871770): Disabled for failing on ASAN.
#if defined(ADDRESS_SANITIZER)
#define MAYBE_SystemTraceEvents DISABLED_SystemTraceEvents
// Only CrOS, Cast, and Windows support system tracing.
#elif defined(OS_CHROMEOS) || (defined(IS_CHROMECAST) && defined(OS_LINUX)) || \
defined(OS_WIN)
#define MAYBE_SystemTraceEvents SystemTraceEvents
#else
#define MAYBE_SystemTraceEvents DISABLED_SystemTraceEvents
#endif
IN_PROC_BROWSER_TEST_F(TracingControllerTest, MAYBE_SystemTraceEvents) {
// TODO(crbug.com/900603): Enable this test for perfetto once passing.
if (tracing::TracingUsesPerfettoBackend())
return;
TestStartAndStopTracingString(true /* enable_systrace */);
EXPECT_TRUE(last_data().size() > 0);
EXPECT_TRUE(last_data().find("systemTraceEvents") != std::string::npos);
}
} // namespace content } // namespace content
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment