xref: /aosp_15_r20/external/perfetto/src/tracing/service/metatrace_writer.cc (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
1*6dbdd20aSAndroid Build Coastguard Worker /*
2*6dbdd20aSAndroid Build Coastguard Worker  * Copyright (C) 2019 The Android Open Source Project
3*6dbdd20aSAndroid Build Coastguard Worker  *
4*6dbdd20aSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*6dbdd20aSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*6dbdd20aSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*6dbdd20aSAndroid Build Coastguard Worker  *
8*6dbdd20aSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*6dbdd20aSAndroid Build Coastguard Worker  *
10*6dbdd20aSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*6dbdd20aSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*6dbdd20aSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*6dbdd20aSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*6dbdd20aSAndroid Build Coastguard Worker  * limitations under the License.
15*6dbdd20aSAndroid Build Coastguard Worker  */
16*6dbdd20aSAndroid Build Coastguard Worker 
17*6dbdd20aSAndroid Build Coastguard Worker #include "src/tracing/service/metatrace_writer.h"
18*6dbdd20aSAndroid Build Coastguard Worker 
19*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/base/logging.h"
20*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/base/task_runner.h"
21*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/tracing/core/trace_writer.h"
22*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/tracing/core/data_source_descriptor.h"
23*6dbdd20aSAndroid Build Coastguard Worker 
24*6dbdd20aSAndroid Build Coastguard Worker #include "protos/perfetto/trace/perfetto/perfetto_metatrace.pbzero.h"
25*6dbdd20aSAndroid Build Coastguard Worker #include "protos/perfetto/trace/trace_packet.pbzero.h"
26*6dbdd20aSAndroid Build Coastguard Worker 
27*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
28*6dbdd20aSAndroid Build Coastguard Worker 
MetatraceWriter()29*6dbdd20aSAndroid Build Coastguard Worker MetatraceWriter::MetatraceWriter() : weak_ptr_factory_(this) {}
30*6dbdd20aSAndroid Build Coastguard Worker 
~MetatraceWriter()31*6dbdd20aSAndroid Build Coastguard Worker MetatraceWriter::~MetatraceWriter() {
32*6dbdd20aSAndroid Build Coastguard Worker   Disable();
33*6dbdd20aSAndroid Build Coastguard Worker }
34*6dbdd20aSAndroid Build Coastguard Worker 
Enable(base::TaskRunner * task_runner,std::unique_ptr<TraceWriter> trace_writer,uint32_t tags)35*6dbdd20aSAndroid Build Coastguard Worker void MetatraceWriter::Enable(base::TaskRunner* task_runner,
36*6dbdd20aSAndroid Build Coastguard Worker                              std::unique_ptr<TraceWriter> trace_writer,
37*6dbdd20aSAndroid Build Coastguard Worker                              uint32_t tags) {
38*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_DCHECK_THREAD(thread_checker_);
39*6dbdd20aSAndroid Build Coastguard Worker   if (started_) {
40*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DFATAL_OR_ELOG("Metatrace already started from this instance");
41*6dbdd20aSAndroid Build Coastguard Worker     return;
42*6dbdd20aSAndroid Build Coastguard Worker   }
43*6dbdd20aSAndroid Build Coastguard Worker   task_runner_ = task_runner;
44*6dbdd20aSAndroid Build Coastguard Worker   trace_writer_ = std::move(trace_writer);
45*6dbdd20aSAndroid Build Coastguard Worker   auto weak_ptr = weak_ptr_factory_.GetWeakPtr();
46*6dbdd20aSAndroid Build Coastguard Worker   bool enabled = metatrace::Enable(
47*6dbdd20aSAndroid Build Coastguard Worker       [weak_ptr] {
48*6dbdd20aSAndroid Build Coastguard Worker         if (weak_ptr)
49*6dbdd20aSAndroid Build Coastguard Worker           weak_ptr->WriteAllAvailableEvents();
50*6dbdd20aSAndroid Build Coastguard Worker       },
51*6dbdd20aSAndroid Build Coastguard Worker       task_runner, tags);
52*6dbdd20aSAndroid Build Coastguard Worker   if (!enabled)
53*6dbdd20aSAndroid Build Coastguard Worker     return;
54*6dbdd20aSAndroid Build Coastguard Worker   started_ = true;
55*6dbdd20aSAndroid Build Coastguard Worker }
56*6dbdd20aSAndroid Build Coastguard Worker 
Disable()57*6dbdd20aSAndroid Build Coastguard Worker void MetatraceWriter::Disable() {
58*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_DCHECK_THREAD(thread_checker_);
59*6dbdd20aSAndroid Build Coastguard Worker   if (!started_)
60*6dbdd20aSAndroid Build Coastguard Worker     return;
61*6dbdd20aSAndroid Build Coastguard Worker   metatrace::Disable();
62*6dbdd20aSAndroid Build Coastguard Worker   started_ = false;
63*6dbdd20aSAndroid Build Coastguard Worker   trace_writer_.reset();
64*6dbdd20aSAndroid Build Coastguard Worker }
65*6dbdd20aSAndroid Build Coastguard Worker 
WriteAllAvailableEvents()66*6dbdd20aSAndroid Build Coastguard Worker void MetatraceWriter::WriteAllAvailableEvents() {
67*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_DCHECK_THREAD(thread_checker_);
68*6dbdd20aSAndroid Build Coastguard Worker   if (!started_)
69*6dbdd20aSAndroid Build Coastguard Worker     return;
70*6dbdd20aSAndroid Build Coastguard Worker   for (auto it = metatrace::RingBuffer::GetReadIterator(); it; ++it) {
71*6dbdd20aSAndroid Build Coastguard Worker     auto type_and_id = it->type_and_id.load(std::memory_order_acquire);
72*6dbdd20aSAndroid Build Coastguard Worker     if (type_and_id == 0)
73*6dbdd20aSAndroid Build Coastguard Worker       break;  // Stop at the first incomplete event.
74*6dbdd20aSAndroid Build Coastguard Worker 
75*6dbdd20aSAndroid Build Coastguard Worker     auto packet = trace_writer_->NewTracePacket();
76*6dbdd20aSAndroid Build Coastguard Worker     packet->set_timestamp(it->timestamp_ns());
77*6dbdd20aSAndroid Build Coastguard Worker     auto* evt = packet->set_perfetto_metatrace();
78*6dbdd20aSAndroid Build Coastguard Worker     uint16_t type = type_and_id & metatrace::Record::kTypeMask;
79*6dbdd20aSAndroid Build Coastguard Worker     uint16_t id = type_and_id & ~metatrace::Record::kTypeMask;
80*6dbdd20aSAndroid Build Coastguard Worker     if (type == metatrace::Record::kTypeCounter) {
81*6dbdd20aSAndroid Build Coastguard Worker       evt->set_counter_id(id);
82*6dbdd20aSAndroid Build Coastguard Worker       evt->set_counter_value(it->counter_value);
83*6dbdd20aSAndroid Build Coastguard Worker     } else {
84*6dbdd20aSAndroid Build Coastguard Worker       evt->set_event_id(id);
85*6dbdd20aSAndroid Build Coastguard Worker       evt->set_event_duration_ns(it->duration_ns);
86*6dbdd20aSAndroid Build Coastguard Worker     }
87*6dbdd20aSAndroid Build Coastguard Worker 
88*6dbdd20aSAndroid Build Coastguard Worker     evt->set_thread_id(static_cast<uint32_t>(it->thread_id));
89*6dbdd20aSAndroid Build Coastguard Worker 
90*6dbdd20aSAndroid Build Coastguard Worker     if (metatrace::RingBuffer::has_overruns())
91*6dbdd20aSAndroid Build Coastguard Worker       evt->set_has_overruns(true);
92*6dbdd20aSAndroid Build Coastguard Worker   }
93*6dbdd20aSAndroid Build Coastguard Worker   // The |it| destructor will automatically update the read index position in
94*6dbdd20aSAndroid Build Coastguard Worker   // the meta-trace ring buffer.
95*6dbdd20aSAndroid Build Coastguard Worker }
96*6dbdd20aSAndroid Build Coastguard Worker 
WriteAllAndFlushTraceWriter(std::function<void ()> callback)97*6dbdd20aSAndroid Build Coastguard Worker void MetatraceWriter::WriteAllAndFlushTraceWriter(
98*6dbdd20aSAndroid Build Coastguard Worker     std::function<void()> callback) {
99*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_DCHECK_THREAD(thread_checker_);
100*6dbdd20aSAndroid Build Coastguard Worker   if (!started_)
101*6dbdd20aSAndroid Build Coastguard Worker     return;
102*6dbdd20aSAndroid Build Coastguard Worker   WriteAllAvailableEvents();
103*6dbdd20aSAndroid Build Coastguard Worker   trace_writer_->Flush(std::move(callback));
104*6dbdd20aSAndroid Build Coastguard Worker }
105*6dbdd20aSAndroid Build Coastguard Worker 
106*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
107