• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * Copyright (C) 2018 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "perfetto/public/consumer_api.h"
18 
19 #include <fcntl.h>
20 #include <inttypes.h>
21 #include <stdlib.h>
22 #include <sys/mman.h>
23 #include <sys/select.h>
24 #include <sys/stat.h>
25 #include <sys/time.h>
26 #include <sys/uio.h>
27 #include <unistd.h>
28 
29 #include <atomic>
30 #include <condition_variable>
31 #include <memory>
32 #include <mutex>
33 #include <thread>
34 
35 #include "perfetto/base/build_config.h"
36 #include "perfetto/ext/base/scoped_file.h"
37 #include "perfetto/ext/base/temp_file.h"
38 #include "perfetto/ext/base/thread_checker.h"
39 #include "perfetto/ext/base/unix_task_runner.h"
40 #include "perfetto/ext/base/utils.h"
41 #include "perfetto/ext/tracing/core/consumer.h"
42 #include "perfetto/ext/tracing/core/trace_packet.h"
43 #include "perfetto/ext/tracing/ipc/consumer_ipc_client.h"
44 #include "perfetto/ext/tracing/ipc/default_socket.h"
45 #include "perfetto/tracing/core/trace_config.h"
46 
47 #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
48 #include <linux/memfd.h>
49 #include <sys/syscall.h>
50 #endif
51 
52 #include "protos/perfetto/config/trace_config.gen.h"
53 
54 #define PERFETTO_EXPORTED_API __attribute__((visibility("default")))
55 
56 namespace perfetto {
57 namespace consumer {
58 
59 namespace {
60 
61 class TracingSession : public Consumer {
62  public:
63   TracingSession(base::TaskRunner*,
64                  Handle,
65                  OnStateChangedCb,
66                  void* callback_arg,
67                  const TraceConfig&);
68   ~TracingSession() override;
69 
70   // Note: if making this class moveable, the move-ctor/dtor must be updated
71   // to clear up mapped_buf_ on dtor.
72 
73   // These methods are called on a thread != |task_runner_|.
state() const74   State state() const { return state_; }
mapped_buf() const75   std::pair<char*, size_t> mapped_buf() const {
76     // The comparison operator will do an acquire-load on the atomic |state_|.
77     if (state_ == State::kTraceEnded)
78       return std::make_pair(mapped_buf_, mapped_buf_size_);
79     return std::make_pair(nullptr, 0);
80   }
81 
82   // All the methods below are called only on the |task_runner_| thread.
83 
84   bool Initialize();
85   void StartTracing();
86 
87   // perfetto::Consumer implementation.
88   void OnConnect() override;
89   void OnDisconnect() override;
90   void OnTracingDisabled() override;
91   void OnTraceData(std::vector<TracePacket>, bool has_more) override;
92   void OnDetach(bool) override;
93   void OnAttach(bool, const TraceConfig&) override;
94   void OnTraceStats(bool, const TraceStats&) override;
95   void OnObservableEvents(const ObservableEvents&) override;
96 
97  private:
98   TracingSession(const TracingSession&) = delete;
99   TracingSession& operator=(const TracingSession&) = delete;
100 
101   void DestroyConnection();
102   void NotifyCallback();
103 
104   base::TaskRunner* const task_runner_;
105   Handle const handle_;
106   OnStateChangedCb const callback_ = nullptr;
107   void* const callback_arg_ = nullptr;
108   TraceConfig trace_config_;
109   base::ScopedFile buf_fd_;
110   std::unique_ptr<TracingService::ConsumerEndpoint> consumer_endpoint_;
111 
112   // |mapped_buf_| and |mapped_buf_size_| are seq-consistent with |state_|.
113   std::atomic<State> state_{State::kIdle};
114   char* mapped_buf_ = nullptr;
115   size_t mapped_buf_size_ = 0;
116 
117   PERFETTO_THREAD_CHECKER(thread_checker_)
118 };
119 
TracingSession(base::TaskRunner * task_runner,Handle handle,OnStateChangedCb callback,void * callback_arg,const TraceConfig & trace_config_proto)120 TracingSession::TracingSession(base::TaskRunner* task_runner,
121                                Handle handle,
122                                OnStateChangedCb callback,
123                                void* callback_arg,
124                                const TraceConfig& trace_config_proto)
125     : task_runner_(task_runner),
126       handle_(handle),
127       callback_(callback),
128       callback_arg_(callback_arg) {
129   PERFETTO_DETACH_FROM_THREAD(thread_checker_);
130   trace_config_ = trace_config_proto;
131   trace_config_.set_write_into_file(true);
132 
133   // TODO(primiano): this really doesn't matter because the trace will be
134   // flushed into the file when stopping. We need a way to be able to say
135   // "disable periodic flushing and flush only when stopping".
136   trace_config_.set_file_write_period_ms(60000);
137 }
138 
~TracingSession()139 TracingSession::~TracingSession() {
140   PERFETTO_DCHECK_THREAD(thread_checker_);
141   if (mapped_buf_)
142     PERFETTO_CHECK(munmap(mapped_buf_, mapped_buf_size_) == 0);
143 }
144 
Initialize()145 bool TracingSession::Initialize() {
146   PERFETTO_DCHECK_THREAD(thread_checker_);
147 
148   if (state_ != State::kIdle)
149     return false;
150 
151 #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
152   char memfd_name[64];
153   snprintf(memfd_name, sizeof(memfd_name), "perfetto_trace_%" PRId64, handle_);
154   buf_fd_.reset(
155       static_cast<int>(syscall(__NR_memfd_create, memfd_name, MFD_CLOEXEC)));
156 #else
157   // Fallback for testing on Linux/mac.
158   buf_fd_ = base::TempFile::CreateUnlinked().ReleaseFD();
159 #endif
160 
161   if (!buf_fd_) {
162     PERFETTO_PLOG("Failed to allocate temporary tracing buffer");
163     return false;
164   }
165 
166   state_ = State::kConnecting;
167   consumer_endpoint_ =
168       ConsumerIPCClient::Connect(GetConsumerSocket(), this, task_runner_);
169 
170   return true;
171 }
172 
173 // Called after EnabledTracing, soon after the IPC connection is established.
OnConnect()174 void TracingSession::OnConnect() {
175   PERFETTO_DCHECK_THREAD(thread_checker_);
176 
177   PERFETTO_DLOG("OnConnect");
178   PERFETTO_DCHECK(state_ == State::kConnecting);
179   consumer_endpoint_->EnableTracing(trace_config_,
180                                     base::ScopedFile(dup(*buf_fd_)));
181   if (trace_config_.deferred_start())
182     state_ = State::kConfigured;
183   else
184     state_ = State::kTracing;
185   NotifyCallback();
186 }
187 
StartTracing()188 void TracingSession::StartTracing() {
189   PERFETTO_DCHECK_THREAD(thread_checker_);
190 
191   auto state = state_.load();
192   if (state != State::kConfigured) {
193     PERFETTO_ELOG("StartTracing(): invalid state (%d)",
194                   static_cast<int>(state));
195     return;
196   }
197   state_ = State::kTracing;
198   consumer_endpoint_->StartTracing();
199 }
200 
OnTracingDisabled()201 void TracingSession::OnTracingDisabled() {
202   PERFETTO_DCHECK_THREAD(thread_checker_);
203   PERFETTO_DLOG("OnTracingDisabled");
204 
205   struct stat stat_buf {};
206   int res = fstat(buf_fd_.get(), &stat_buf);
207   mapped_buf_size_ = res == 0 ? static_cast<size_t>(stat_buf.st_size) : 0;
208   mapped_buf_ =
209       static_cast<char*>(mmap(nullptr, mapped_buf_size_, PROT_READ | PROT_WRITE,
210                               MAP_SHARED, buf_fd_.get(), 0));
211   DestroyConnection();
212   if (mapped_buf_size_ == 0 || mapped_buf_ == MAP_FAILED) {
213     mapped_buf_ = nullptr;
214     mapped_buf_size_ = 0;
215     state_ = State::kTraceFailed;
216     PERFETTO_ELOG("Tracing session failed");
217   } else {
218     state_ = State::kTraceEnded;
219   }
220   NotifyCallback();
221 }
222 
OnDisconnect()223 void TracingSession::OnDisconnect() {
224   PERFETTO_DCHECK_THREAD(thread_checker_);
225   PERFETTO_DLOG("OnDisconnect");
226   DestroyConnection();
227   state_ = State::kConnectionError;
228   NotifyCallback();
229 }
230 
OnDetach(bool)231 void TracingSession::OnDetach(bool) {
232   PERFETTO_DCHECK(false);  // Should never be called, Detach() is not used here.
233 }
234 
OnAttach(bool,const TraceConfig &)235 void TracingSession::OnAttach(bool, const TraceConfig&) {
236   PERFETTO_DCHECK(false);  // Should never be called, Attach() is not used here.
237 }
238 
OnTraceStats(bool,const TraceStats &)239 void TracingSession::OnTraceStats(bool, const TraceStats&) {
240   // Should never be called, GetTraceStats() is not used here.
241   PERFETTO_DCHECK(false);
242 }
243 
OnObservableEvents(const ObservableEvents &)244 void TracingSession::OnObservableEvents(const ObservableEvents&) {
245   // Should never be called, ObserveEvents() is not used here.
246   PERFETTO_DCHECK(false);
247 }
248 
DestroyConnection()249 void TracingSession::DestroyConnection() {
250   // Destroys the connection in a separate task. This is to avoid destroying
251   // the IPC connection directly from within the IPC callback.
252   TracingService::ConsumerEndpoint* endpoint = consumer_endpoint_.release();
253   task_runner_->PostTask([endpoint] { delete endpoint; });
254 }
255 
OnTraceData(std::vector<TracePacket>,bool)256 void TracingSession::OnTraceData(std::vector<TracePacket>, bool) {
257   // This should be never called because we are using |write_into_file| and
258   // asking the traced service to directly write into the |buf_fd_|.
259   PERFETTO_DFATAL("Should be unreachable.");
260 }
261 
NotifyCallback()262 void TracingSession::NotifyCallback() {
263   if (!callback_)
264     return;
265   auto state = state_.load();
266   auto callback = callback_;
267   auto handle = handle_;
268   auto callback_arg = callback_arg_;
269   task_runner_->PostTask([callback, callback_arg, handle, state] {
270     callback(handle, state, callback_arg);
271   });
272 }
273 
274 class TracingController {
275  public:
276   static TracingController* GetInstance();
277   TracingController();
278 
279   // These methods are called from a thread != |task_runner_|.
280   Handle Create(const void*, size_t, OnStateChangedCb, void* callback_arg);
281   void StartTracing(Handle);
282   State PollState(Handle);
283   TraceBuffer ReadTrace(Handle);
284   void Destroy(Handle);
285 
286  private:
287   void ThreadMain();  // Called on |task_runner_| thread.
288 
289   std::mutex mutex_;
290   std::thread thread_;
291   std::unique_ptr<base::UnixTaskRunner> task_runner_;
292   std::condition_variable task_runner_initialized_;
293   Handle last_handle_ = 0;
294   std::map<Handle, std::unique_ptr<TracingSession>> sessions_;
295 };
296 
GetInstance()297 TracingController* TracingController::GetInstance() {
298   static TracingController* instance = new TracingController();
299   return instance;
300 }
301 
TracingController()302 TracingController::TracingController()
303     : thread_(&TracingController::ThreadMain, this) {
304   std::unique_lock<std::mutex> lock(mutex_);
305   task_runner_initialized_.wait(lock, [this] { return !!task_runner_; });
306 }
307 
ThreadMain()308 void TracingController::ThreadMain() {
309   {
310     std::unique_lock<std::mutex> lock(mutex_);
311     task_runner_.reset(new base::UnixTaskRunner());
312   }
313   task_runner_initialized_.notify_one();
314   task_runner_->Run();
315 }
316 
Create(const void * config_proto_buf,size_t config_len,OnStateChangedCb callback,void * callback_arg)317 Handle TracingController::Create(const void* config_proto_buf,
318                                  size_t config_len,
319                                  OnStateChangedCb callback,
320                                  void* callback_arg) {
321   TraceConfig config_proto;
322   bool parsed = config_proto.ParseFromArray(config_proto_buf, config_len);
323   if (!parsed) {
324     PERFETTO_ELOG("Failed to decode TraceConfig proto");
325     return kInvalidHandle;
326   }
327 
328   if (!config_proto.duration_ms()) {
329     PERFETTO_ELOG("The trace config must specify a duration");
330     return kInvalidHandle;
331   }
332 
333   std::unique_lock<std::mutex> lock(mutex_);
334   Handle handle = ++last_handle_;
335   auto* session = new TracingSession(task_runner_.get(), handle, callback,
336                                      callback_arg, config_proto);
337   sessions_.emplace(handle, std::unique_ptr<TracingSession>(session));
338 
339   // Enable the TracingSession on its own thread.
340   task_runner_->PostTask([session] { session->Initialize(); });
341 
342   return handle;
343 }
344 
StartTracing(Handle handle)345 void TracingController::StartTracing(Handle handle) {
346   std::unique_lock<std::mutex> lock(mutex_);
347   auto it = sessions_.find(handle);
348   if (it == sessions_.end()) {
349     PERFETTO_ELOG("StartTracing(): Invalid tracing session handle");
350     return;
351   }
352   TracingSession* session = it->second.get();
353   task_runner_->PostTask([session] { session->StartTracing(); });
354 }
355 
PollState(Handle handle)356 State TracingController::PollState(Handle handle) {
357   std::unique_lock<std::mutex> lock(mutex_);
358   auto it = sessions_.find(handle);
359   if (it == sessions_.end())
360     return State::kSessionNotFound;
361   return it->second->state();
362 }
363 
ReadTrace(Handle handle)364 TraceBuffer TracingController::ReadTrace(Handle handle) {
365   TraceBuffer buf{};
366 
367   std::unique_lock<std::mutex> lock(mutex_);
368   auto it = sessions_.find(handle);
369   if (it == sessions_.end()) {
370     PERFETTO_DLOG("Handle invalid");
371     return buf;
372   }
373 
374   TracingSession* session = it->second.get();
375   auto state = session->state();
376   if (state == State::kTraceEnded) {
377     std::tie(buf.begin, buf.size) = session->mapped_buf();
378     return buf;
379   }
380 
381   PERFETTO_DLOG("ReadTrace(): called in an unexpected state (%d)",
382                 static_cast<int>(state));
383   return buf;
384 }
385 
Destroy(Handle handle)386 void TracingController::Destroy(Handle handle) {
387   // Post an empty task on the task runner to delete the session on its own
388   // thread.
389   std::unique_lock<std::mutex> lock(mutex_);
390   auto it = sessions_.find(handle);
391   if (it == sessions_.end())
392     return;
393   TracingSession* session = it->second.release();
394   sessions_.erase(it);
395   task_runner_->PostTask([session] { delete session; });
396 }
397 
398 }  // namespace
399 
Create(const void * config_proto,size_t config_len,OnStateChangedCb callback,void * callback_arg)400 PERFETTO_EXPORTED_API Handle Create(const void* config_proto,
401                                     size_t config_len,
402                                     OnStateChangedCb callback,
403                                     void* callback_arg) {
404   return TracingController::GetInstance()->Create(config_proto, config_len,
405                                                   callback, callback_arg);
406 }
407 
408 PERFETTO_EXPORTED_API
StartTracing(Handle handle)409 void StartTracing(Handle handle) {
410   return TracingController::GetInstance()->StartTracing(handle);
411 }
412 
PollState(Handle handle)413 PERFETTO_EXPORTED_API State PollState(Handle handle) {
414   return TracingController::GetInstance()->PollState(handle);
415 }
416 
ReadTrace(Handle handle)417 PERFETTO_EXPORTED_API TraceBuffer ReadTrace(Handle handle) {
418   return TracingController::GetInstance()->ReadTrace(handle);
419 }
420 
Destroy(Handle handle)421 PERFETTO_EXPORTED_API void Destroy(Handle handle) {
422   TracingController::GetInstance()->Destroy(handle);
423 }
424 }  // namespace consumer
425 }  // namespace perfetto
426