| // Copyright 2014 The Chromium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "base/bind.h" |
| #include "base/memory/scoped_vector.h" |
| #include "mojo/application/application_runner_chromium.h" |
| #include "mojo/common/weak_binding_set.h" |
| #include "mojo/common/weak_interface_ptr_set.h" |
| #include "mojo/public/c/system/main.h" |
| #include "mojo/public/cpp/application/application_delegate.h" |
| #include "mojo/public/cpp/application/application_impl.h" |
| #include "mojo/public/cpp/bindings/strong_binding.h" |
| #include "services/tracing/trace_data_sink.h" |
| #include "services/tracing/tracing.mojom.h" |
| |
| namespace tracing { |
| |
| namespace { |
| |
| class CollectorImpl : public TraceDataCollector { |
| public: |
| CollectorImpl(mojo::InterfaceRequest<TraceDataCollector> request, |
| TraceDataSink* sink) |
| : sink_(sink), binding_(this, request.Pass()) {} |
| |
| ~CollectorImpl() override {} |
| |
| // tracing::TraceDataCollector implementation. |
| void DataCollected(const mojo::String& json) override { |
| sink_->AddChunk(json.To<std::string>()); |
| } |
| |
| private: |
| TraceDataSink* sink_; |
| mojo::Binding<TraceDataCollector> binding_; |
| |
| DISALLOW_COPY_AND_ASSIGN(CollectorImpl); |
| }; |
| |
| } // namespace |
| |
| class TracingApp : public mojo::ApplicationDelegate, |
| public mojo::InterfaceFactory<TraceCoordinator>, |
| public TraceCoordinator { |
| public: |
| TracingApp() {} |
| ~TracingApp() override {} |
| |
| private: |
| // mojo::ApplicationDelegate implementation. |
| bool ConfigureIncomingConnection( |
| mojo::ApplicationConnection* connection) override { |
| connection->AddService<TraceCoordinator>(this); |
| |
| // If someone connects to us they may want to use the TraceCoordinator |
| // interface and/or they may want to expose themselves to be traced. Attempt |
| // to connect to the TraceController interface to see if the application |
| // connecting to us wants to be traced. They can refuse the connection or |
| // close the pipe if not. |
| TraceControllerPtr controller_ptr; |
| connection->ConnectToService(&controller_ptr); |
| controller_ptrs_.AddInterfacePtr(controller_ptr.Pass()); |
| return true; |
| } |
| |
| // mojo::InterfaceFactory<TraceCoordinator> implementation. |
| void Create(mojo::ApplicationConnection* connection, |
| mojo::InterfaceRequest<TraceCoordinator> request) override { |
| coordinator_bindings_.AddBinding(this, request.Pass()); |
| } |
| |
| // tracing::TraceCoordinator implementation. |
| void Start(mojo::ScopedDataPipeProducerHandle stream, |
| const mojo::String& categories) override { |
| sink_.reset(new TraceDataSink(stream.Pass())); |
| controller_ptrs_.ForAllPtrs( |
| [categories, this](TraceController* controller) { |
| TraceDataCollectorPtr ptr; |
| collector_impls_.push_back( |
| new CollectorImpl(GetProxy(&ptr), sink_.get())); |
| controller->StartTracing(categories, ptr.Pass()); |
| }); |
| } |
| void StopAndFlush() override { |
| controller_ptrs_.ForAllPtrs( |
| [](TraceController* controller) { controller->StopTracing(); }); |
| |
| // TODO: We really should keep track of how many connections we have here |
| // and flush + reset the sink after we receive a EndTracing or a detect a |
| // pipe closure on all pipes. |
| base::MessageLoop::current()->PostDelayedTask( |
| FROM_HERE, |
| base::Bind(&TracingApp::AllDataCollected, base::Unretained(this)), |
| base::TimeDelta::FromSeconds(1)); |
| } |
| |
| void AllDataCollected() { |
| collector_impls_.clear(); |
| sink_->Flush(); |
| } |
| |
| scoped_ptr<TraceDataSink> sink_; |
| ScopedVector<CollectorImpl> collector_impls_; |
| mojo::WeakInterfacePtrSet<TraceController> controller_ptrs_; |
| mojo::WeakBindingSet<TraceCoordinator> coordinator_bindings_; |
| |
| DISALLOW_COPY_AND_ASSIGN(TracingApp); |
| }; |
| |
| } // namespace tracing |
| |
| MojoResult MojoMain(MojoHandle shell_handle) { |
| mojo::ApplicationRunnerChromium runner(new tracing::TracingApp); |
| return runner.Run(shell_handle); |
| } |