blob: 79c054a045c930dbaa40dff5fe58daff23c960d4 [file] [log] [blame]
Lalit Magantic4c3ceb2018-03-29 20:38:13 +01001/*
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 "test/test_helper.h"
18
Florian Mayer67b26852021-02-04 19:13:01 +000019#include "perfetto/base/compiler.h"
Primiano Tucci2c5488f2019-06-01 03:27:28 +010020#include "perfetto/ext/tracing/core/trace_packet.h"
Stephen Nuskoac0c1972019-06-25 13:57:13 +010021#include "perfetto/ext/tracing/ipc/default_socket.h"
Primiano Tucci9c41ceb2020-04-14 13:23:01 +010022#include "perfetto/tracing/core/tracing_service_state.h"
Florian Mayerc29e0d32018-04-04 15:55:46 +010023
Primiano Tucci355b8c82019-08-29 08:37:51 +020024#include "protos/perfetto/trace/trace_packet.pbzero.h"
Primiano Tucci07e104d2018-04-03 20:45:35 +020025
Lalit Magantic4c3ceb2018-03-29 20:38:13 +010026namespace perfetto {
27
Florian Mayer67b26852021-02-04 19:13:01 +000028namespace {
29const char* ProducerSocketForMode(TestHelper::Mode mode) {
30#if !PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
31 base::ignore_result(mode);
32 return ::perfetto::GetProducerSocket();
Lalit Magantic4c3ceb2018-03-29 20:38:13 +010033#else
Florian Mayer67b26852021-02-04 19:13:01 +000034 switch (mode) {
35 case TestHelper::Mode::kStartDaemons:
36 return "/data/local/tmp/traced_producer";
37 case TestHelper::Mode::kUseSystemService:
38 return ::perfetto::GetProducerSocket();
39 }
40#endif
41}
42
43const char* ConsumerSocketForMode(TestHelper::Mode mode) {
44#if !PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
45 base::ignore_result(mode);
46 return ::perfetto::GetConsumerSocket();
47#else
48 switch (mode) {
49 case TestHelper::Mode::kStartDaemons:
50 return "/data/local/tmp/traced_consumer";
51 case TestHelper::Mode::kUseSystemService:
52 return ::perfetto::GetConsumerSocket();
53 }
54#endif
55}
56} // namespace
57
58uint64_t TestHelper::next_instance_num_ = 0;
59#if PERFETTO_BUILDFLAG(PERFETTO_START_DAEMONS)
60TestHelper::Mode TestHelper::kDefaultMode = Mode::kStartDaemons;
61#else
62TestHelper::Mode TestHelper::kDefaultMode = Mode::kUseSystemService;
Lalit Magantic4c3ceb2018-03-29 20:38:13 +010063#endif
64
Florian Mayer67b26852021-02-04 19:13:01 +000065TestHelper::TestHelper(base::TestTaskRunner* task_runner, Mode mode)
Florian Mayer05a87c92019-01-30 13:17:51 +000066 : instance_num_(next_instance_num_++),
67 task_runner_(task_runner),
Florian Mayer67b26852021-02-04 19:13:01 +000068 mode_(mode),
69 producer_socket_(ProducerSocketForMode(mode)),
70 consumer_socket_(ConsumerSocketForMode(mode)),
71 service_thread_(producer_socket_, consumer_socket_),
72 fake_producer_thread_(producer_socket_,
Primiano Tuccibbe68be2020-04-16 22:17:12 +010073 WrapTask(CreateCheckpoint("producer.connect")),
Lalit Maganti9782f492020-01-10 18:13:13 +000074 WrapTask(CreateCheckpoint("producer.setup")),
75 WrapTask(CreateCheckpoint("producer.enabled"))) {}
Lalit Magantic4c3ceb2018-03-29 20:38:13 +010076
77void TestHelper::OnConnect() {
Lalit Maganti36557d82018-04-11 14:36:17 +010078 std::move(on_connect_callback_)();
Lalit Magantic4c3ceb2018-03-29 20:38:13 +010079}
80
81void TestHelper::OnDisconnect() {
Primiano Tucci008cdb92019-07-19 19:52:41 +010082 PERFETTO_FATAL("Consumer unexpectedly disconnected from the service");
Lalit Magantic4c3ceb2018-03-29 20:38:13 +010083}
84
Primiano Tuccidf440312020-10-30 16:43:58 +010085void TestHelper::OnTracingDisabled(const std::string& /*error*/) {
Lalit Maganti36557d82018-04-11 14:36:17 +010086 std::move(on_stop_tracing_callback_)();
Stephen Nusko1af720e2020-11-18 14:04:16 -050087 on_stop_tracing_callback_ = nullptr;
Lalit Maganti36557d82018-04-11 14:36:17 +010088}
Lalit Magantic4c3ceb2018-03-29 20:38:13 +010089
Florian Mayer709de7f2021-01-28 21:41:14 +000090void TestHelper::ReadTraceData(std::vector<TracePacket> packets) {
Primiano Tucci07e104d2018-04-03 20:45:35 +020091 for (auto& encoded_packet : packets) {
Primiano Tuccife502c42019-12-11 01:00:27 +000092 protos::gen::TracePacket packet;
Primiano Tuccif7851ee2019-09-09 06:20:30 -070093 PERFETTO_CHECK(
94 packet.ParseFromString(encoded_packet.GetRawBytesForTesting()));
Florian Mayer51016d32020-09-17 11:34:44 +010095 full_trace_.push_back(packet);
Primiano Tucci5e33cad2018-04-30 14:41:25 +010096 if (packet.has_clock_snapshot() || packet.has_trace_config() ||
Hector Dearman685f7522019-03-12 14:28:56 +000097 packet.has_trace_stats() || !packet.synchronization_marker().empty() ||
Primiano Tucci79e4dcb2020-04-08 09:51:02 +010098 packet.has_system_info() || packet.has_service_event()) {
Lalit Magantic4c3ceb2018-03-29 20:38:13 +010099 continue;
Primiano Tucci5e33cad2018-04-30 14:41:25 +0100100 }
Primiano Tuccife502c42019-12-11 01:00:27 +0000101 PERFETTO_CHECK(packet.has_trusted_uid());
Lalit Maganti36557d82018-04-11 14:36:17 +0100102 trace_.push_back(std::move(packet));
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100103 }
Florian Mayer709de7f2021-01-28 21:41:14 +0000104}
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100105
Florian Mayer709de7f2021-01-28 21:41:14 +0000106void TestHelper::OnTraceData(std::vector<TracePacket> packets, bool has_more) {
107 ReadTraceData(std::move(packets));
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100108 if (!has_more) {
Lalit Maganti36557d82018-04-11 14:36:17 +0100109 std::move(on_packets_finished_callback_)();
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100110 }
111}
112
113void TestHelper::StartServiceIfRequired() {
Florian Mayer67b26852021-02-04 19:13:01 +0000114 if (mode_ == Mode::kStartDaemons)
115 service_thread_.Start();
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100116}
117
118FakeProducer* TestHelper::ConnectFakeProducer() {
Lalit Maganti9782f492020-01-10 18:13:13 +0000119 fake_producer_thread_.Connect();
Primiano Tuccibbe68be2020-04-16 22:17:12 +0100120 // This will wait until the service has seen the RegisterDataSource() call
121 // (because of the Sync() in FakeProducer::OnConnect()).
122 RunUntilCheckpoint("producer.connect");
Lalit Maganti9782f492020-01-10 18:13:13 +0000123 return fake_producer_thread_.producer();
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100124}
125
126void TestHelper::ConnectConsumer() {
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100127 cur_consumer_num_++;
Florian Mayer05a87c92019-01-30 13:17:51 +0000128 on_connect_callback_ = CreateCheckpoint("consumer.connected." +
129 std::to_string(cur_consumer_num_));
Florian Mayer67b26852021-02-04 19:13:01 +0000130 endpoint_ = ConsumerIPCClient::Connect(consumer_socket_, this, task_runner_);
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100131}
132
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100133void TestHelper::DetachConsumer(const std::string& key) {
Florian Mayer05a87c92019-01-30 13:17:51 +0000134 on_detach_callback_ = CreateCheckpoint("detach." + key);
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100135 endpoint_->Detach(key);
Florian Mayer05a87c92019-01-30 13:17:51 +0000136 RunUntilCheckpoint("detach." + key);
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100137 endpoint_.reset();
138}
139
140bool TestHelper::AttachConsumer(const std::string& key) {
141 bool success = false;
Florian Mayer05a87c92019-01-30 13:17:51 +0000142 auto checkpoint = CreateCheckpoint("attach." + key);
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100143 on_attach_callback_ = [&success, checkpoint](bool s) {
144 success = s;
145 checkpoint();
146 };
147 endpoint_->Attach(key);
Florian Mayer05a87c92019-01-30 13:17:51 +0000148 RunUntilCheckpoint("attach." + key);
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100149 return success;
150}
151
Primiano Tucci40207fe2020-12-03 18:27:55 +0100152bool TestHelper::SaveTraceForBugreportAndWait() {
153 bool success = false;
154 auto checkpoint = CreateCheckpoint("bugreport");
155 auto callback = [&success, checkpoint](bool s, const std::string&) {
156 success = s;
157 checkpoint();
158 };
159 endpoint_->SaveTraceForBugreport(callback);
160 RunUntilCheckpoint("bugreport");
161 return success;
162}
163
Eric Seckler326a3d32020-02-04 11:24:56 +0000164void TestHelper::CreateProducerProvidedSmb() {
165 fake_producer_thread_.CreateProducerProvidedSmb();
166}
167
168bool TestHelper::IsShmemProvidedByProducer() {
169 return fake_producer_thread_.producer()->IsShmemProvidedByProducer();
170}
171
Eric Seckler526921b2020-02-18 11:44:30 +0000172void TestHelper::ProduceStartupEventBatch(
173 const protos::gen::TestConfig& config) {
174 auto on_data_written = CreateCheckpoint("startup_data_written");
175 fake_producer_thread_.ProduceStartupEventBatch(config,
176 WrapTask(on_data_written));
177 RunUntilCheckpoint("startup_data_written");
178}
179
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100180void TestHelper::StartTracing(const TraceConfig& config,
181 base::ScopedFile file) {
Stephen Nusko1af720e2020-11-18 14:04:16 -0500182 PERFETTO_CHECK(!on_stop_tracing_callback_);
Florian Mayer05a87c92019-01-30 13:17:51 +0000183 trace_.clear();
Stephen Nusko1af720e2020-11-18 14:04:16 -0500184 on_stop_tracing_callback_ =
185 CreateCheckpoint("stop.tracing" + std::to_string(++trace_count_));
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100186 endpoint_->EnableTracing(config, std::move(file));
Lalit Maganti36557d82018-04-11 14:36:17 +0100187}
188
Primiano Tuccic1855302018-12-06 10:36:55 +0000189void TestHelper::DisableTracing() {
190 endpoint_->DisableTracing();
191}
192
193void TestHelper::FlushAndWait(uint32_t timeout_ms) {
194 static int flush_num = 0;
195 std::string checkpoint_name = "flush." + std::to_string(flush_num++);
Florian Mayer05a87c92019-01-30 13:17:51 +0000196 auto checkpoint = CreateCheckpoint(checkpoint_name);
Primiano Tuccic1855302018-12-06 10:36:55 +0000197 endpoint_->Flush(timeout_ms, [checkpoint](bool) { checkpoint(); });
Florian Mayer05a87c92019-01-30 13:17:51 +0000198 RunUntilCheckpoint(checkpoint_name, timeout_ms + 1000);
Primiano Tuccic1855302018-12-06 10:36:55 +0000199}
200
Lalit Maganti3f0b7c62018-04-18 19:10:09 +0100201void TestHelper::ReadData(uint32_t read_count) {
Florian Mayer05a87c92019-01-30 13:17:51 +0000202 on_packets_finished_callback_ =
203 CreateCheckpoint("readback.complete." + std::to_string(read_count));
Lalit Maganti36557d82018-04-11 14:36:17 +0100204 endpoint_->ReadBuffers();
205}
206
Stephen Nusko1af720e2020-11-18 14:04:16 -0500207void TestHelper::FreeBuffers() {
208 endpoint_->FreeBuffers();
209}
210
Lalit Maganti36557d82018-04-11 14:36:17 +0100211void TestHelper::WaitForConsumerConnect() {
Florian Mayer05a87c92019-01-30 13:17:51 +0000212 RunUntilCheckpoint("consumer.connected." + std::to_string(cur_consumer_num_));
Lalit Maganti36557d82018-04-11 14:36:17 +0100213}
214
Stephen Nuskoe8238112019-04-09 18:37:00 +0100215void TestHelper::WaitForProducerSetup() {
216 RunUntilCheckpoint("producer.setup");
217}
218
Lalit Maganti36557d82018-04-11 14:36:17 +0100219void TestHelper::WaitForProducerEnabled() {
Florian Mayer05a87c92019-01-30 13:17:51 +0000220 RunUntilCheckpoint("producer.enabled");
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100221}
222
Primiano Tuccic1855302018-12-06 10:36:55 +0000223void TestHelper::WaitForTracingDisabled(uint32_t timeout_ms) {
Stephen Nusko1af720e2020-11-18 14:04:16 -0500224 RunUntilCheckpoint(std::string("stop.tracing") + std::to_string(trace_count_),
225 timeout_ms);
Lalit Maganti36557d82018-04-11 14:36:17 +0100226}
227
Florian Mayer3e6e4b42019-06-05 10:17:36 +0100228void TestHelper::WaitForReadData(uint32_t read_count, uint32_t timeout_ms) {
229 RunUntilCheckpoint("readback.complete." + std::to_string(read_count),
230 timeout_ms);
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100231}
232
Primiano Tuccif26cb9c2022-01-18 23:19:36 +0000233void TestHelper::WaitFor(std::function<bool()> predicate,
234 const std::string& error_msg,
235 uint32_t timeout_ms) {
236 int64_t deadline_ms = base::GetWallTimeMs().count() + timeout_ms;
237 while (base::GetWallTimeMs().count() < deadline_ms) {
238 if (predicate())
239 return;
240 base::SleepMicroseconds(500 * 1000); // 0.5 s.
241 }
242 PERFETTO_FATAL("Test timed out waiting for: %s", error_msg.c_str());
243}
244
245void TestHelper::WaitForDataSourceConnected(const std::string& ds_name) {
246 auto predicate = [&] {
247 auto dss = QueryServiceStateAndWait().data_sources();
248 return std::any_of(dss.begin(), dss.end(),
249 [&](const TracingServiceState::DataSource& ds) {
250 return ds.ds_descriptor().name() == ds_name;
251 });
252 };
253 WaitFor(predicate, "connection of data source " + ds_name);
254}
255
Primiano Tuccibbe68be2020-04-16 22:17:12 +0100256void TestHelper::SyncAndWaitProducer() {
257 static int sync_id = 0;
258 std::string checkpoint_name = "producer_sync_" + std::to_string(++sync_id);
259 auto checkpoint = CreateCheckpoint(checkpoint_name);
260 fake_producer_thread_.producer()->Sync(
261 [this, &checkpoint] { task_runner_->PostTask(checkpoint); });
262 RunUntilCheckpoint(checkpoint_name);
263}
264
Primiano Tucci9c41ceb2020-04-14 13:23:01 +0100265TracingServiceState TestHelper::QueryServiceStateAndWait() {
266 TracingServiceState res;
Primiano Tuccif26cb9c2022-01-18 23:19:36 +0000267 static int n = 0;
268 std::string checkpoint_name = "query_svc_state_" + std::to_string(n++);
269 auto checkpoint = CreateCheckpoint(checkpoint_name);
Primiano Tucci9c41ceb2020-04-14 13:23:01 +0100270 auto callback = [&checkpoint, &res](bool, const TracingServiceState& tss) {
271 res = tss;
272 checkpoint();
273 };
274 endpoint_->QueryServiceState(callback);
Primiano Tuccif26cb9c2022-01-18 23:19:36 +0000275 RunUntilCheckpoint(checkpoint_name);
Primiano Tucci9c41ceb2020-04-14 13:23:01 +0100276 return res;
277}
278
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100279std::function<void()> TestHelper::WrapTask(
280 const std::function<void()>& function) {
281 return [this, function] { task_runner_->PostTask(function); };
282}
283
Primiano Tucci9ba1d842018-12-20 17:31:04 +0100284void TestHelper::OnDetach(bool) {
285 if (on_detach_callback_)
286 std::move(on_detach_callback_)();
287}
288
289void TestHelper::OnAttach(bool success, const TraceConfig&) {
290 if (on_attach_callback_)
291 std::move(on_attach_callback_)(success);
292}
293
Eric Secklereaf29ed2019-01-23 09:53:55 +0000294void TestHelper::OnTraceStats(bool, const TraceStats&) {}
295
Eric Seckler7b0c9452019-03-18 13:14:36 +0000296void TestHelper::OnObservableEvents(const ObservableEvents&) {}
297
Primiano Tucci106605c2019-01-08 21:12:58 +0000298// static
Florian Mayer67b26852021-02-04 19:13:01 +0000299const char* TestHelper::GetDefaultModeConsumerSocketName() {
300 return ConsumerSocketForMode(TestHelper::kDefaultMode);
Primiano Tucci106605c2019-01-08 21:12:58 +0000301}
302
Stephen Nuskoe8238112019-04-09 18:37:00 +0100303// static
Florian Mayer67b26852021-02-04 19:13:01 +0000304const char* TestHelper::GetDefaultModeProducerSocketName() {
305 return ProducerSocketForMode(TestHelper::kDefaultMode);
Stephen Nuskoe8238112019-04-09 18:37:00 +0100306}
307
Lalit Magantic4c3ceb2018-03-29 20:38:13 +0100308} // namespace perfetto