xref: /aosp_15_r20/external/perfetto/src/shared_lib/test/utils.cc (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
1*6dbdd20aSAndroid Build Coastguard Worker /*
2*6dbdd20aSAndroid Build Coastguard Worker  * Copyright (C) 2023 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/shared_lib/test/utils.h"
18*6dbdd20aSAndroid Build Coastguard Worker 
19*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/public/abi/heap_buffer.h"
20*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/public/pb_msg.h"
21*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/public/pb_utils.h"
22*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/public/protos/config/data_source_config.pzc.h"
23*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/public/protos/config/trace_config.pzc.h"
24*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/public/protos/config/track_event/track_event_config.pzc.h"
25*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/public/tracing_session.h"
26*6dbdd20aSAndroid Build Coastguard Worker 
27*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
28*6dbdd20aSAndroid Build Coastguard Worker namespace shlib {
29*6dbdd20aSAndroid Build Coastguard Worker namespace test_utils {
30*6dbdd20aSAndroid Build Coastguard Worker namespace {
31*6dbdd20aSAndroid Build Coastguard Worker 
ToHexChars(uint8_t val)32*6dbdd20aSAndroid Build Coastguard Worker std::string ToHexChars(uint8_t val) {
33*6dbdd20aSAndroid Build Coastguard Worker   std::string ret;
34*6dbdd20aSAndroid Build Coastguard Worker   uint8_t high_nibble = (val & 0xF0) >> 4;
35*6dbdd20aSAndroid Build Coastguard Worker   uint8_t low_nibble = (val & 0xF);
36*6dbdd20aSAndroid Build Coastguard Worker   static const char hex_chars[] = "0123456789ABCDEF";
37*6dbdd20aSAndroid Build Coastguard Worker   ret.push_back(hex_chars[high_nibble]);
38*6dbdd20aSAndroid Build Coastguard Worker   ret.push_back(hex_chars[low_nibble]);
39*6dbdd20aSAndroid Build Coastguard Worker   return ret;
40*6dbdd20aSAndroid Build Coastguard Worker }
41*6dbdd20aSAndroid Build Coastguard Worker 
42*6dbdd20aSAndroid Build Coastguard Worker }  // namespace
43*6dbdd20aSAndroid Build Coastguard Worker 
Build()44*6dbdd20aSAndroid Build Coastguard Worker TracingSession TracingSession::Builder::Build() {
45*6dbdd20aSAndroid Build Coastguard Worker   struct PerfettoPbMsgWriter writer;
46*6dbdd20aSAndroid Build Coastguard Worker   struct PerfettoHeapBuffer* hb = PerfettoHeapBufferCreate(&writer.writer);
47*6dbdd20aSAndroid Build Coastguard Worker 
48*6dbdd20aSAndroid Build Coastguard Worker   struct perfetto_protos_TraceConfig cfg;
49*6dbdd20aSAndroid Build Coastguard Worker   PerfettoPbMsgInit(&cfg.msg, &writer);
50*6dbdd20aSAndroid Build Coastguard Worker 
51*6dbdd20aSAndroid Build Coastguard Worker   {
52*6dbdd20aSAndroid Build Coastguard Worker     struct perfetto_protos_TraceConfig_BufferConfig buffers;
53*6dbdd20aSAndroid Build Coastguard Worker     perfetto_protos_TraceConfig_begin_buffers(&cfg, &buffers);
54*6dbdd20aSAndroid Build Coastguard Worker 
55*6dbdd20aSAndroid Build Coastguard Worker     perfetto_protos_TraceConfig_BufferConfig_set_size_kb(&buffers, 1024);
56*6dbdd20aSAndroid Build Coastguard Worker 
57*6dbdd20aSAndroid Build Coastguard Worker     perfetto_protos_TraceConfig_end_buffers(&cfg, &buffers);
58*6dbdd20aSAndroid Build Coastguard Worker   }
59*6dbdd20aSAndroid Build Coastguard Worker 
60*6dbdd20aSAndroid Build Coastguard Worker   {
61*6dbdd20aSAndroid Build Coastguard Worker     struct perfetto_protos_TraceConfig_DataSource data_sources;
62*6dbdd20aSAndroid Build Coastguard Worker     perfetto_protos_TraceConfig_begin_data_sources(&cfg, &data_sources);
63*6dbdd20aSAndroid Build Coastguard Worker 
64*6dbdd20aSAndroid Build Coastguard Worker     {
65*6dbdd20aSAndroid Build Coastguard Worker       struct perfetto_protos_DataSourceConfig ds_cfg;
66*6dbdd20aSAndroid Build Coastguard Worker       perfetto_protos_TraceConfig_DataSource_begin_config(&data_sources,
67*6dbdd20aSAndroid Build Coastguard Worker                                                           &ds_cfg);
68*6dbdd20aSAndroid Build Coastguard Worker 
69*6dbdd20aSAndroid Build Coastguard Worker       perfetto_protos_DataSourceConfig_set_cstr_name(&ds_cfg,
70*6dbdd20aSAndroid Build Coastguard Worker                                                      data_source_name_.c_str());
71*6dbdd20aSAndroid Build Coastguard Worker       if (!enabled_categories_.empty() || !disabled_categories_.empty()) {
72*6dbdd20aSAndroid Build Coastguard Worker         perfetto_protos_TrackEventConfig te_cfg;
73*6dbdd20aSAndroid Build Coastguard Worker         perfetto_protos_DataSourceConfig_begin_track_event_config(&ds_cfg,
74*6dbdd20aSAndroid Build Coastguard Worker                                                                   &te_cfg);
75*6dbdd20aSAndroid Build Coastguard Worker         for (const std::string& cat : enabled_categories_) {
76*6dbdd20aSAndroid Build Coastguard Worker           perfetto_protos_TrackEventConfig_set_enabled_categories(
77*6dbdd20aSAndroid Build Coastguard Worker               &te_cfg, cat.data(), cat.size());
78*6dbdd20aSAndroid Build Coastguard Worker         }
79*6dbdd20aSAndroid Build Coastguard Worker         for (const std::string& cat : disabled_categories_) {
80*6dbdd20aSAndroid Build Coastguard Worker           perfetto_protos_TrackEventConfig_set_disabled_categories(
81*6dbdd20aSAndroid Build Coastguard Worker               &te_cfg, cat.data(), cat.size());
82*6dbdd20aSAndroid Build Coastguard Worker         }
83*6dbdd20aSAndroid Build Coastguard Worker         perfetto_protos_DataSourceConfig_end_track_event_config(&ds_cfg,
84*6dbdd20aSAndroid Build Coastguard Worker                                                                 &te_cfg);
85*6dbdd20aSAndroid Build Coastguard Worker       }
86*6dbdd20aSAndroid Build Coastguard Worker 
87*6dbdd20aSAndroid Build Coastguard Worker       perfetto_protos_TraceConfig_DataSource_end_config(&data_sources, &ds_cfg);
88*6dbdd20aSAndroid Build Coastguard Worker     }
89*6dbdd20aSAndroid Build Coastguard Worker 
90*6dbdd20aSAndroid Build Coastguard Worker     perfetto_protos_TraceConfig_end_data_sources(&cfg, &data_sources);
91*6dbdd20aSAndroid Build Coastguard Worker   }
92*6dbdd20aSAndroid Build Coastguard Worker   size_t cfg_size = PerfettoStreamWriterGetWrittenSize(&writer.writer);
93*6dbdd20aSAndroid Build Coastguard Worker   std::unique_ptr<uint8_t[]> ser(new uint8_t[cfg_size]);
94*6dbdd20aSAndroid Build Coastguard Worker   PerfettoHeapBufferCopyInto(hb, &writer.writer, ser.get(), cfg_size);
95*6dbdd20aSAndroid Build Coastguard Worker   PerfettoHeapBufferDestroy(hb, &writer.writer);
96*6dbdd20aSAndroid Build Coastguard Worker 
97*6dbdd20aSAndroid Build Coastguard Worker   struct PerfettoTracingSessionImpl* ts =
98*6dbdd20aSAndroid Build Coastguard Worker       PerfettoTracingSessionCreate(PERFETTO_BACKEND_IN_PROCESS);
99*6dbdd20aSAndroid Build Coastguard Worker 
100*6dbdd20aSAndroid Build Coastguard Worker   PerfettoTracingSessionSetup(ts, ser.get(), cfg_size);
101*6dbdd20aSAndroid Build Coastguard Worker 
102*6dbdd20aSAndroid Build Coastguard Worker   PerfettoTracingSessionStartBlocking(ts);
103*6dbdd20aSAndroid Build Coastguard Worker 
104*6dbdd20aSAndroid Build Coastguard Worker   return TracingSession::Adopt(ts);
105*6dbdd20aSAndroid Build Coastguard Worker }
106*6dbdd20aSAndroid Build Coastguard Worker 
Adopt(struct PerfettoTracingSessionImpl * session)107*6dbdd20aSAndroid Build Coastguard Worker TracingSession TracingSession::Adopt(
108*6dbdd20aSAndroid Build Coastguard Worker     struct PerfettoTracingSessionImpl* session) {
109*6dbdd20aSAndroid Build Coastguard Worker   TracingSession ret;
110*6dbdd20aSAndroid Build Coastguard Worker   ret.session_ = session;
111*6dbdd20aSAndroid Build Coastguard Worker   ret.stopped_ = std::make_unique<WaitableEvent>();
112*6dbdd20aSAndroid Build Coastguard Worker   PerfettoTracingSessionSetStopCb(
113*6dbdd20aSAndroid Build Coastguard Worker       ret.session_,
114*6dbdd20aSAndroid Build Coastguard Worker       [](struct PerfettoTracingSessionImpl*, void* arg) {
115*6dbdd20aSAndroid Build Coastguard Worker         static_cast<WaitableEvent*>(arg)->Notify();
116*6dbdd20aSAndroid Build Coastguard Worker       },
117*6dbdd20aSAndroid Build Coastguard Worker       ret.stopped_.get());
118*6dbdd20aSAndroid Build Coastguard Worker   return ret;
119*6dbdd20aSAndroid Build Coastguard Worker }
120*6dbdd20aSAndroid Build Coastguard Worker 
TracingSession(TracingSession && other)121*6dbdd20aSAndroid Build Coastguard Worker TracingSession::TracingSession(TracingSession&& other) noexcept {
122*6dbdd20aSAndroid Build Coastguard Worker   session_ = other.session_;
123*6dbdd20aSAndroid Build Coastguard Worker   other.session_ = nullptr;
124*6dbdd20aSAndroid Build Coastguard Worker   stopped_ = std::move(other.stopped_);
125*6dbdd20aSAndroid Build Coastguard Worker   other.stopped_ = nullptr;
126*6dbdd20aSAndroid Build Coastguard Worker }
127*6dbdd20aSAndroid Build Coastguard Worker 
~TracingSession()128*6dbdd20aSAndroid Build Coastguard Worker TracingSession::~TracingSession() {
129*6dbdd20aSAndroid Build Coastguard Worker   if (!session_) {
130*6dbdd20aSAndroid Build Coastguard Worker     return;
131*6dbdd20aSAndroid Build Coastguard Worker   }
132*6dbdd20aSAndroid Build Coastguard Worker   if (!stopped_->IsNotified()) {
133*6dbdd20aSAndroid Build Coastguard Worker     PerfettoTracingSessionStopBlocking(session_);
134*6dbdd20aSAndroid Build Coastguard Worker     stopped_->WaitForNotification();
135*6dbdd20aSAndroid Build Coastguard Worker   }
136*6dbdd20aSAndroid Build Coastguard Worker   PerfettoTracingSessionDestroy(session_);
137*6dbdd20aSAndroid Build Coastguard Worker }
138*6dbdd20aSAndroid Build Coastguard Worker 
FlushBlocking(uint32_t timeout_ms)139*6dbdd20aSAndroid Build Coastguard Worker bool TracingSession::FlushBlocking(uint32_t timeout_ms) {
140*6dbdd20aSAndroid Build Coastguard Worker   WaitableEvent notification;
141*6dbdd20aSAndroid Build Coastguard Worker   bool result;
142*6dbdd20aSAndroid Build Coastguard Worker   auto* cb = new std::function<void(bool)>([&](bool success) {
143*6dbdd20aSAndroid Build Coastguard Worker     result = success;
144*6dbdd20aSAndroid Build Coastguard Worker     notification.Notify();
145*6dbdd20aSAndroid Build Coastguard Worker   });
146*6dbdd20aSAndroid Build Coastguard Worker   PerfettoTracingSessionFlushAsync(
147*6dbdd20aSAndroid Build Coastguard Worker       session_, timeout_ms,
148*6dbdd20aSAndroid Build Coastguard Worker       [](PerfettoTracingSessionImpl*, bool success, void* user_arg) {
149*6dbdd20aSAndroid Build Coastguard Worker         auto* f = reinterpret_cast<std::function<void(bool)>*>(user_arg);
150*6dbdd20aSAndroid Build Coastguard Worker         (*f)(success);
151*6dbdd20aSAndroid Build Coastguard Worker         delete f;
152*6dbdd20aSAndroid Build Coastguard Worker       },
153*6dbdd20aSAndroid Build Coastguard Worker       cb);
154*6dbdd20aSAndroid Build Coastguard Worker   notification.WaitForNotification();
155*6dbdd20aSAndroid Build Coastguard Worker   return result;
156*6dbdd20aSAndroid Build Coastguard Worker }
157*6dbdd20aSAndroid Build Coastguard Worker 
WaitForStopped()158*6dbdd20aSAndroid Build Coastguard Worker void TracingSession::WaitForStopped() {
159*6dbdd20aSAndroid Build Coastguard Worker   stopped_->WaitForNotification();
160*6dbdd20aSAndroid Build Coastguard Worker }
161*6dbdd20aSAndroid Build Coastguard Worker 
StopAsync()162*6dbdd20aSAndroid Build Coastguard Worker void TracingSession::StopAsync() {
163*6dbdd20aSAndroid Build Coastguard Worker   PerfettoTracingSessionStopAsync(session_);
164*6dbdd20aSAndroid Build Coastguard Worker }
165*6dbdd20aSAndroid Build Coastguard Worker 
StopBlocking()166*6dbdd20aSAndroid Build Coastguard Worker void TracingSession::StopBlocking() {
167*6dbdd20aSAndroid Build Coastguard Worker   PerfettoTracingSessionStopBlocking(session_);
168*6dbdd20aSAndroid Build Coastguard Worker }
169*6dbdd20aSAndroid Build Coastguard Worker 
ReadBlocking()170*6dbdd20aSAndroid Build Coastguard Worker std::vector<uint8_t> TracingSession::ReadBlocking() {
171*6dbdd20aSAndroid Build Coastguard Worker   std::vector<uint8_t> data;
172*6dbdd20aSAndroid Build Coastguard Worker   PerfettoTracingSessionReadTraceBlocking(
173*6dbdd20aSAndroid Build Coastguard Worker       session_,
174*6dbdd20aSAndroid Build Coastguard Worker       [](struct PerfettoTracingSessionImpl*, const void* trace_data,
175*6dbdd20aSAndroid Build Coastguard Worker          size_t size, bool, void* user_arg) {
176*6dbdd20aSAndroid Build Coastguard Worker         auto& dst = *static_cast<std::vector<uint8_t>*>(user_arg);
177*6dbdd20aSAndroid Build Coastguard Worker         auto* src = static_cast<const uint8_t*>(trace_data);
178*6dbdd20aSAndroid Build Coastguard Worker         dst.insert(dst.end(), src, src + size);
179*6dbdd20aSAndroid Build Coastguard Worker       },
180*6dbdd20aSAndroid Build Coastguard Worker       &data);
181*6dbdd20aSAndroid Build Coastguard Worker   return data;
182*6dbdd20aSAndroid Build Coastguard Worker }
183*6dbdd20aSAndroid Build Coastguard Worker 
184*6dbdd20aSAndroid Build Coastguard Worker }  // namespace test_utils
185*6dbdd20aSAndroid Build Coastguard Worker }  // namespace shlib
186*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
187*6dbdd20aSAndroid Build Coastguard Worker 
PrintTo(const PerfettoPbDecoderField & field,std::ostream * pos)188*6dbdd20aSAndroid Build Coastguard Worker void PrintTo(const PerfettoPbDecoderField& field, std::ostream* pos) {
189*6dbdd20aSAndroid Build Coastguard Worker   std::ostream& os = *pos;
190*6dbdd20aSAndroid Build Coastguard Worker   PerfettoPbDecoderStatus status =
191*6dbdd20aSAndroid Build Coastguard Worker       static_cast<PerfettoPbDecoderStatus>(field.status);
192*6dbdd20aSAndroid Build Coastguard Worker   switch (status) {
193*6dbdd20aSAndroid Build Coastguard Worker     case PERFETTO_PB_DECODER_ERROR:
194*6dbdd20aSAndroid Build Coastguard Worker       os << "MALFORMED PROTOBUF";
195*6dbdd20aSAndroid Build Coastguard Worker       break;
196*6dbdd20aSAndroid Build Coastguard Worker     case PERFETTO_PB_DECODER_DONE:
197*6dbdd20aSAndroid Build Coastguard Worker       os << "DECODER DONE";
198*6dbdd20aSAndroid Build Coastguard Worker       break;
199*6dbdd20aSAndroid Build Coastguard Worker     case PERFETTO_PB_DECODER_OK:
200*6dbdd20aSAndroid Build Coastguard Worker       switch (field.wire_type) {
201*6dbdd20aSAndroid Build Coastguard Worker         case PERFETTO_PB_WIRE_TYPE_DELIMITED:
202*6dbdd20aSAndroid Build Coastguard Worker           os << "\"";
203*6dbdd20aSAndroid Build Coastguard Worker           for (size_t i = 0; i < field.value.delimited.len; i++) {
204*6dbdd20aSAndroid Build Coastguard Worker             os << perfetto::shlib::test_utils::ToHexChars(
205*6dbdd20aSAndroid Build Coastguard Worker                       field.value.delimited.start[i])
206*6dbdd20aSAndroid Build Coastguard Worker                << " ";
207*6dbdd20aSAndroid Build Coastguard Worker           }
208*6dbdd20aSAndroid Build Coastguard Worker           os << "\"";
209*6dbdd20aSAndroid Build Coastguard Worker           break;
210*6dbdd20aSAndroid Build Coastguard Worker         case PERFETTO_PB_WIRE_TYPE_VARINT:
211*6dbdd20aSAndroid Build Coastguard Worker           os << "varint: " << field.value.integer64;
212*6dbdd20aSAndroid Build Coastguard Worker           break;
213*6dbdd20aSAndroid Build Coastguard Worker         case PERFETTO_PB_WIRE_TYPE_FIXED32:
214*6dbdd20aSAndroid Build Coastguard Worker           os << "fixed32: " << field.value.integer32;
215*6dbdd20aSAndroid Build Coastguard Worker           break;
216*6dbdd20aSAndroid Build Coastguard Worker         case PERFETTO_PB_WIRE_TYPE_FIXED64:
217*6dbdd20aSAndroid Build Coastguard Worker           os << "fixed64: " << field.value.integer64;
218*6dbdd20aSAndroid Build Coastguard Worker           break;
219*6dbdd20aSAndroid Build Coastguard Worker       }
220*6dbdd20aSAndroid Build Coastguard Worker       break;
221*6dbdd20aSAndroid Build Coastguard Worker   }
222*6dbdd20aSAndroid Build Coastguard Worker }
223