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 "src/traced/probes/ftrace/ftrace_config_muxer.h"
18
19 #include <memory>
20
21 #include "gmock/gmock.h"
22 #include "gtest/gtest.h"
23 #include "src/traced/probes/ftrace/atrace_wrapper.h"
24 #include "src/traced/probes/ftrace/ftrace_procfs.h"
25 #include "src/traced/probes/ftrace/proto_translation_table.h"
26
27 using testing::_;
28 using testing::AnyNumber;
29 using testing::MatchesRegex;
30 using testing::Contains;
31 using testing::ElementsAreArray;
32 using testing::Eq;
33 using testing::IsEmpty;
34 using testing::NiceMock;
35 using testing::Not;
36 using testing::Return;
37 using testing::UnorderedElementsAre;
38
39 namespace perfetto {
40 namespace {
41
42 class MockFtraceProcfs : public FtraceProcfs {
43 public:
MockFtraceProcfs()44 MockFtraceProcfs() : FtraceProcfs("/root/") {
45 ON_CALL(*this, NumberOfCpus()).WillByDefault(Return(1));
46 ON_CALL(*this, WriteToFile(_, _)).WillByDefault(Return(true));
47 ON_CALL(*this, ClearFile(_)).WillByDefault(Return(true));
48 EXPECT_CALL(*this, NumberOfCpus()).Times(AnyNumber());
49 }
50
51 MOCK_METHOD2(WriteToFile,
52 bool(const std::string& path, const std::string& str));
53 MOCK_METHOD2(AppendToFile,
54 bool(const std::string& path, const std::string& str));
55 MOCK_METHOD1(ReadOneCharFromFile, char(const std::string& path));
56 MOCK_METHOD1(ClearFile, bool(const std::string& path));
57 MOCK_CONST_METHOD1(ReadFileIntoString, std::string(const std::string& path));
58 MOCK_CONST_METHOD0(NumberOfCpus, size_t());
59 MOCK_CONST_METHOD1(GetEventNamesForGroup,
60 const std::set<std::string>(const std::string& path));
61 };
62
63 struct MockRunAtrace {
MockRunAtraceperfetto::__anon32c4cc6d0111::MockRunAtrace64 MockRunAtrace() {
65 static MockRunAtrace* instance;
66 instance = this;
67 SetRunAtraceForTesting([](const std::vector<std::string>& args) {
68 return instance->RunAtrace(args);
69 });
70 }
71
~MockRunAtraceperfetto::__anon32c4cc6d0111::MockRunAtrace72 ~MockRunAtrace() { SetRunAtraceForTesting(nullptr); }
73
74 MOCK_METHOD1(RunAtrace, bool(const std::vector<std::string>&));
75 };
76
77 class MockProtoTranslationTable : public ProtoTranslationTable {
78 public:
MockProtoTranslationTable(NiceMock<MockFtraceProcfs> * ftrace_procfs,const std::vector<Event> & events,std::vector<Field> common_fields,FtracePageHeaderSpec ftrace_page_header_spec)79 MockProtoTranslationTable(NiceMock<MockFtraceProcfs>* ftrace_procfs,
80 const std::vector<Event>& events,
81 std::vector<Field> common_fields,
82 FtracePageHeaderSpec ftrace_page_header_spec)
83 : ProtoTranslationTable(ftrace_procfs,
84 events,
85 common_fields,
86 ftrace_page_header_spec) {}
87 MOCK_METHOD1(GetOrCreateEvent, Event*(const GroupAndName& group_and_name));
88 MOCK_CONST_METHOD1(GetEvent,
89 const Event*(const GroupAndName& group_and_name));
90 };
91
92 class FtraceConfigMuxerTest : public ::testing::Test {
93 protected:
GetMockTable()94 std::unique_ptr<MockProtoTranslationTable> GetMockTable() {
95 std::vector<Field> common_fields;
96 std::vector<Event> events;
97 return std::unique_ptr<MockProtoTranslationTable>(
98 new MockProtoTranslationTable(
99 &table_procfs_, events, std::move(common_fields),
100 ProtoTranslationTable::DefaultPageHeaderSpecForTesting()));
101 }
CreateFakeTable()102 std::unique_ptr<ProtoTranslationTable> CreateFakeTable() {
103 std::vector<Field> common_fields;
104 std::vector<Event> events;
105 {
106 Event event;
107 event.name = "sched_switch";
108 event.group = "sched";
109 event.ftrace_event_id = 1;
110 events.push_back(event);
111 }
112
113 {
114 Event event;
115 event.name = "sched_wakeup";
116 event.group = "sched";
117 event.ftrace_event_id = 10;
118 events.push_back(event);
119 }
120
121 {
122 Event event;
123 event.name = "sched_new";
124 event.group = "sched";
125 event.ftrace_event_id = 11;
126 events.push_back(event);
127 }
128
129 {
130 Event event;
131 event.name = "cgroup_mkdir";
132 event.group = "cgroup";
133 event.ftrace_event_id = 12;
134 events.push_back(event);
135 }
136
137 {
138 Event event;
139 event.name = "mm_vmscan_direct_reclaim_begin";
140 event.group = "vmscan";
141 event.ftrace_event_id = 13;
142 events.push_back(event);
143 }
144
145 {
146 Event event;
147 event.name = "lowmemory_kill";
148 event.group = "lowmemorykiller";
149 event.ftrace_event_id = 14;
150 events.push_back(event);
151 }
152
153 {
154 Event event;
155 event.name = "print";
156 event.group = "ftrace";
157 event.ftrace_event_id = 20;
158 events.push_back(event);
159 }
160
161 return std::unique_ptr<ProtoTranslationTable>(new ProtoTranslationTable(
162 &table_procfs_, events, std::move(common_fields),
163 ProtoTranslationTable::DefaultPageHeaderSpecForTesting()));
164 }
165
166 NiceMock<MockFtraceProcfs> table_procfs_;
167 std::unique_ptr<ProtoTranslationTable> table_ = CreateFakeTable();
168 };
169
TEST_F(FtraceConfigMuxerTest,ComputeCpuBufferSizeInPages)170 TEST_F(FtraceConfigMuxerTest, ComputeCpuBufferSizeInPages) {
171 static constexpr size_t kMaxBufSizeInPages = 16 * 1024u;
172 // No buffer size given: good default (128 pages = 2mb).
173 EXPECT_EQ(ComputeCpuBufferSizeInPages(0), 512u);
174 // Buffer size given way too big: good default.
175 EXPECT_EQ(ComputeCpuBufferSizeInPages(10 * 1024 * 1024), kMaxBufSizeInPages);
176 // The limit is 64mb per CPU, 512mb is too much.
177 EXPECT_EQ(ComputeCpuBufferSizeInPages(512 * 1024), kMaxBufSizeInPages);
178 // Your size ends up with less than 1 page per cpu -> 1 page.
179 EXPECT_EQ(ComputeCpuBufferSizeInPages(3), 1u);
180 // You picked a good size -> your size rounded to nearest page.
181 EXPECT_EQ(ComputeCpuBufferSizeInPages(42), 10u);
182 }
183
TEST_F(FtraceConfigMuxerTest,AddGenericEvent)184 TEST_F(FtraceConfigMuxerTest, AddGenericEvent) {
185 auto mock_table = GetMockTable();
186 MockFtraceProcfs ftrace;
187
188 FtraceConfig config = CreateFtraceConfig({"power/cpu_frequency"});
189
190 FtraceConfigMuxer model(&ftrace, mock_table.get());
191
192 ON_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
193 .WillByDefault(Return("[local] global boot"));
194 EXPECT_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
195 .Times(AnyNumber());
196
197 EXPECT_CALL(ftrace, ReadOneCharFromFile("/root/tracing_on"))
198 .Times(2)
199 .WillRepeatedly(Return('0'));
200 EXPECT_CALL(ftrace, WriteToFile("/root/buffer_size_kb", _));
201 EXPECT_CALL(ftrace, WriteToFile("/root/trace_clock", "boot"));
202 EXPECT_CALL(ftrace, WriteToFile("/root/tracing_on", "1"));
203 EXPECT_CALL(ftrace,
204 WriteToFile("/root/events/power/cpu_frequency/enable", "1"));
205 EXPECT_CALL(*mock_table, GetEvent(GroupAndName("power", "cpu_frequency")))
206 .Times(AnyNumber());
207
208 Event event_to_return;
209 event_to_return.name = "cpu_frequency";
210 event_to_return.group = "power";
211 event_to_return.ftrace_event_id = 1;
212 ON_CALL(*mock_table, GetOrCreateEvent(GroupAndName("power", "cpu_frequency")))
213 .WillByDefault(Return(&event_to_return));
214 EXPECT_CALL(*mock_table,
215 GetOrCreateEvent(GroupAndName("power", "cpu_frequency")));
216
217 FtraceConfigId id = model.SetupConfig(config);
218 ASSERT_TRUE(model.ActivateConfig(id));
219 const FtraceConfig* actual_config = model.GetConfigForTesting(id);
220 EXPECT_TRUE(actual_config);
221 EXPECT_THAT(actual_config->ftrace_events(), Contains("power/cpu_frequency"));
222 }
223
TEST_F(FtraceConfigMuxerTest,AddSameNameEvents)224 TEST_F(FtraceConfigMuxerTest, AddSameNameEvents) {
225 auto mock_table = GetMockTable();
226 NiceMock<MockFtraceProcfs> ftrace;
227
228 FtraceConfig config = CreateFtraceConfig({"group_one/foo", "group_two/foo"});
229
230 FtraceConfigMuxer model(&ftrace, mock_table.get());
231
232 Event event1;
233 event1.name = "foo";
234 event1.group = "group_one";
235 event1.ftrace_event_id = 1;
236 ON_CALL(*mock_table, GetOrCreateEvent(GroupAndName("group_one", "foo")))
237 .WillByDefault(Return(&event1));
238 EXPECT_CALL(*mock_table, GetOrCreateEvent(GroupAndName("group_one", "foo")));
239
240 Event event2;
241 event2.name = "foo";
242 event2.group = "group_two";
243 event2.ftrace_event_id = 2;
244 ON_CALL(*mock_table, GetOrCreateEvent(GroupAndName("group_two", "foo")))
245 .WillByDefault(Return(&event2));
246 EXPECT_CALL(*mock_table, GetOrCreateEvent(GroupAndName("group_two", "foo")));
247
248 FtraceConfigId id = model.SetupConfig(config);
249 ASSERT_TRUE(model.ActivateConfig(id));
250 const FtraceConfig* actual_config = model.GetConfigForTesting(id);
251 EXPECT_TRUE(actual_config);
252 EXPECT_THAT(actual_config->ftrace_events(), Contains("group_one/foo"));
253 EXPECT_THAT(actual_config->ftrace_events(), Contains("group_two/foo"));
254 }
255
TEST_F(FtraceConfigMuxerTest,AddAllEvents)256 TEST_F(FtraceConfigMuxerTest, AddAllEvents) {
257 auto mock_table = GetMockTable();
258 MockFtraceProcfs ftrace;
259
260 FtraceConfig config = CreateFtraceConfig({"sched/*"});
261
262 ON_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
263 .WillByDefault(Return("[local] global boot"));
264 EXPECT_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
265 .Times(AnyNumber());
266
267 EXPECT_CALL(ftrace, ReadOneCharFromFile("/root/tracing_on"))
268 .Times(2)
269 .WillRepeatedly(Return('0'));
270 EXPECT_CALL(ftrace, WriteToFile("/root/buffer_size_kb", _));
271 EXPECT_CALL(ftrace, WriteToFile("/root/trace_clock", "boot"));
272 EXPECT_CALL(ftrace, WriteToFile("/root/tracing_on", "1"));
273 EXPECT_CALL(ftrace,
274 WriteToFile("/root/events/sched/sched_switch/enable", "1"));
275 EXPECT_CALL(ftrace,
276 WriteToFile("/root/events/sched/sched_new_event/enable", "1"));
277
278 FtraceConfigMuxer model(&ftrace, mock_table.get());
279 std::set<std::string> n = {"sched_switch", "sched_new_event"};
280 ON_CALL(ftrace, GetEventNamesForGroup("events/sched"))
281 .WillByDefault(Return(n));
282 EXPECT_CALL(ftrace, GetEventNamesForGroup("events/sched")).Times(1);
283
284 // Non-generic event.
285 std::map<std::string, const Event*> events;
286 Event sched_switch = {"sched_switch", "sched"};
287 sched_switch.ftrace_event_id = 1;
288 ON_CALL(*mock_table, GetOrCreateEvent(GroupAndName("sched", "sched_switch")))
289 .WillByDefault(Return(&sched_switch));
290 EXPECT_CALL(*mock_table,
291 GetOrCreateEvent(GroupAndName("sched", "sched_switch")))
292 .Times(AnyNumber());
293
294 // Generic event.
295 Event event_to_return;
296 event_to_return.name = "sched_new_event";
297 event_to_return.group = "sched";
298 event_to_return.ftrace_event_id = 2;
299 ON_CALL(*mock_table,
300 GetOrCreateEvent(GroupAndName("sched", "sched_new_event")))
301 .WillByDefault(Return(&event_to_return));
302 EXPECT_CALL(*mock_table,
303 GetOrCreateEvent(GroupAndName("sched", "sched_new_event")));
304
305 FtraceConfigId id = model.SetupConfig(config);
306 ASSERT_TRUE(id);
307 ASSERT_TRUE(model.ActivateConfig(id));
308
309 const FtraceConfig* actual_config = model.GetConfigForTesting(id);
310 EXPECT_THAT(actual_config->ftrace_events(), Contains("sched/sched_switch"));
311 EXPECT_THAT(actual_config->ftrace_events(),
312 Contains("sched/sched_new_event"));
313 }
314
TEST_F(FtraceConfigMuxerTest,TwoWildcardGroups)315 TEST_F(FtraceConfigMuxerTest, TwoWildcardGroups) {
316 auto mock_table = GetMockTable();
317 NiceMock<MockFtraceProcfs> ftrace;
318
319 FtraceConfig config = CreateFtraceConfig({"group_one/*", "group_two/*"});
320
321 FtraceConfigMuxer model(&ftrace, mock_table.get());
322
323 std::set<std::string> event_names = {"foo"};
324 ON_CALL(ftrace, GetEventNamesForGroup("events/group_one"))
325 .WillByDefault(Return(event_names));
326 EXPECT_CALL(ftrace, GetEventNamesForGroup("events/group_one"))
327 .Times(AnyNumber());
328
329 ON_CALL(ftrace, GetEventNamesForGroup("events/group_two"))
330 .WillByDefault(Return(event_names));
331 EXPECT_CALL(ftrace, GetEventNamesForGroup("events/group_two"))
332 .Times(AnyNumber());
333
334 Event event1;
335 event1.name = "foo";
336 event1.group = "group_one";
337 event1.ftrace_event_id = 1;
338 ON_CALL(*mock_table, GetOrCreateEvent(GroupAndName("group_one", "foo")))
339 .WillByDefault(Return(&event1));
340 EXPECT_CALL(*mock_table, GetOrCreateEvent(GroupAndName("group_one", "foo")));
341
342 Event event2;
343 event2.name = "foo";
344 event2.group = "group_two";
345 event2.ftrace_event_id = 2;
346 ON_CALL(*mock_table, GetOrCreateEvent(GroupAndName("group_two", "foo")))
347 .WillByDefault(Return(&event2));
348 EXPECT_CALL(*mock_table, GetOrCreateEvent(GroupAndName("group_two", "foo")));
349
350 FtraceConfigId id = model.SetupConfig(config);
351 ASSERT_TRUE(model.ActivateConfig(id));
352 const FtraceConfig* actual_config = model.GetConfigForTesting(id);
353 EXPECT_TRUE(actual_config);
354 EXPECT_THAT(actual_config->ftrace_events(), Contains("group_one/foo"));
355 EXPECT_THAT(actual_config->ftrace_events(), Contains("group_two/foo"));
356 }
357
TEST_F(FtraceConfigMuxerTest,TurnFtraceOnOff)358 TEST_F(FtraceConfigMuxerTest, TurnFtraceOnOff) {
359 MockFtraceProcfs ftrace;
360
361 FtraceConfig config = CreateFtraceConfig({"sched_switch", "foo"});
362
363 FtraceConfigMuxer model(&ftrace, table_.get());
364
365 ON_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
366 .WillByDefault(Return("[local] global boot"));
367 EXPECT_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
368 .Times(AnyNumber());
369
370 EXPECT_CALL(ftrace, ReadOneCharFromFile("/root/tracing_on"))
371 .Times(2)
372 .WillRepeatedly(Return('0'));
373 EXPECT_CALL(ftrace, WriteToFile("/root/buffer_size_kb", _));
374 EXPECT_CALL(ftrace, WriteToFile("/root/trace_clock", "boot"));
375 EXPECT_CALL(ftrace, WriteToFile("/root/tracing_on", "1"));
376 EXPECT_CALL(ftrace,
377 WriteToFile("/root/events/sched/sched_switch/enable", "1"));
378 FtraceConfigId id = model.SetupConfig(config);
379 ASSERT_TRUE(id);
380 ASSERT_TRUE(model.ActivateConfig(id));
381
382 const FtraceConfig* actual_config = model.GetConfigForTesting(id);
383 EXPECT_TRUE(actual_config);
384 EXPECT_THAT(actual_config->ftrace_events(), Contains("sched/sched_switch"));
385 EXPECT_THAT(actual_config->ftrace_events(), Not(Contains("foo")));
386
387 EXPECT_CALL(ftrace, WriteToFile("/root/tracing_on", "0"));
388 EXPECT_CALL(ftrace, WriteToFile("/root/buffer_size_kb", "0"));
389 EXPECT_CALL(ftrace, WriteToFile("/root/events/enable", "0"));
390 EXPECT_CALL(ftrace,
391 WriteToFile("/root/events/sched/sched_switch/enable", "0"));
392 EXPECT_CALL(ftrace, ClearFile("/root/trace"));
393 EXPECT_CALL(ftrace, ClearFile(MatchesRegex("/root/per_cpu/cpu[0-9]/trace")));
394 ASSERT_TRUE(model.RemoveConfig(id));
395 }
396
TEST_F(FtraceConfigMuxerTest,FtraceIsAlreadyOn)397 TEST_F(FtraceConfigMuxerTest, FtraceIsAlreadyOn) {
398 MockFtraceProcfs ftrace;
399
400 FtraceConfig config = CreateFtraceConfig({"sched/sched_switch"});
401
402 FtraceConfigMuxer model(&ftrace, table_.get());
403
404 // If someone is using ftrace already don't stomp on what they are doing.
405 EXPECT_CALL(ftrace, ReadOneCharFromFile("/root/tracing_on"))
406 .WillOnce(Return('1'));
407 FtraceConfigId id = model.SetupConfig(config);
408 ASSERT_FALSE(id);
409 }
410
TEST_F(FtraceConfigMuxerTest,Atrace)411 TEST_F(FtraceConfigMuxerTest, Atrace) {
412 NiceMock<MockFtraceProcfs> ftrace;
413 MockRunAtrace atrace;
414
415 FtraceConfig config = CreateFtraceConfig({"sched/sched_switch"});
416 *config.add_atrace_categories() = "sched";
417
418 FtraceConfigMuxer model(&ftrace, table_.get());
419
420 EXPECT_CALL(ftrace, ReadOneCharFromFile("/root/tracing_on"))
421 .WillOnce(Return('0'));
422 EXPECT_CALL(atrace,
423 RunAtrace(ElementsAreArray(
424 {"atrace", "--async_start", "--only_userspace", "sched"})))
425 .WillOnce(Return(true));
426
427 FtraceConfigId id = model.SetupConfig(config);
428 ASSERT_TRUE(id);
429
430 const FtraceConfig* actual_config = model.GetConfigForTesting(id);
431 EXPECT_TRUE(actual_config);
432 EXPECT_THAT(actual_config->ftrace_events(), Contains("sched/sched_switch"));
433 EXPECT_THAT(actual_config->ftrace_events(), Contains("ftrace/print"));
434
435 EXPECT_CALL(atrace, RunAtrace(ElementsAreArray(
436 {"atrace", "--async_stop", "--only_userspace"})))
437 .WillOnce(Return(true));
438 ASSERT_TRUE(model.RemoveConfig(id));
439 }
440
TEST_F(FtraceConfigMuxerTest,AtraceTwoApps)441 TEST_F(FtraceConfigMuxerTest, AtraceTwoApps) {
442 NiceMock<MockFtraceProcfs> ftrace;
443 MockRunAtrace atrace;
444
445 FtraceConfig config = CreateFtraceConfig({});
446 *config.add_atrace_apps() = "com.google.android.gms.persistent";
447 *config.add_atrace_apps() = "com.google.android.gms";
448
449 FtraceConfigMuxer model(&ftrace, table_.get());
450
451 EXPECT_CALL(ftrace, ReadOneCharFromFile("/root/tracing_on"))
452 .WillOnce(Return('0'));
453 EXPECT_CALL(
454 atrace,
455 RunAtrace(ElementsAreArray(
456 {"atrace", "--async_start", "--only_userspace", "-a",
457 "com.google.android.gms.persistent,com.google.android.gms"})))
458 .WillOnce(Return(true));
459
460 FtraceConfigId id = model.SetupConfig(config);
461 ASSERT_TRUE(id);
462
463 const FtraceConfig* actual_config = model.GetConfigForTesting(id);
464 EXPECT_TRUE(actual_config);
465 EXPECT_THAT(actual_config->ftrace_events(), Contains("ftrace/print"));
466
467 EXPECT_CALL(atrace, RunAtrace(ElementsAreArray(
468 {"atrace", "--async_stop", "--only_userspace"})))
469 .WillOnce(Return(true));
470 ASSERT_TRUE(model.RemoveConfig(id));
471 }
472
TEST_F(FtraceConfigMuxerTest,SetupClockForTesting)473 TEST_F(FtraceConfigMuxerTest, SetupClockForTesting) {
474 MockFtraceProcfs ftrace;
475 FtraceConfig config;
476
477 FtraceConfigMuxer model(&ftrace, table_.get());
478
479 EXPECT_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
480 .Times(AnyNumber());
481
482 ON_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
483 .WillByDefault(Return("[local] global boot"));
484 EXPECT_CALL(ftrace, WriteToFile("/root/trace_clock", "boot"));
485 model.SetupClockForTesting(config);
486
487 ON_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
488 .WillByDefault(Return("[local] global"));
489 EXPECT_CALL(ftrace, WriteToFile("/root/trace_clock", "global"));
490 model.SetupClockForTesting(config);
491
492 ON_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
493 .WillByDefault(Return(""));
494 model.SetupClockForTesting(config);
495
496 ON_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
497 .WillByDefault(Return("local [global]"));
498 model.SetupClockForTesting(config);
499 }
500
TEST_F(FtraceConfigMuxerTest,GetFtraceEvents)501 TEST_F(FtraceConfigMuxerTest, GetFtraceEvents) {
502 MockFtraceProcfs ftrace;
503 FtraceConfigMuxer model(&ftrace, table_.get());
504
505 FtraceConfig config = CreateFtraceConfig({"sched/sched_switch"});
506 std::set<GroupAndName> events =
507 model.GetFtraceEventsForTesting(config, table_.get());
508
509 EXPECT_THAT(events, Contains(GroupAndName("sched", "sched_switch")));
510 EXPECT_THAT(events, Not(Contains(GroupAndName("ftrace", "print"))));
511 }
512
TEST_F(FtraceConfigMuxerTest,GetFtraceEventsAtrace)513 TEST_F(FtraceConfigMuxerTest, GetFtraceEventsAtrace) {
514 MockFtraceProcfs ftrace;
515 FtraceConfigMuxer model(&ftrace, table_.get());
516
517 FtraceConfig config = CreateFtraceConfig({});
518 *config.add_atrace_categories() = "sched";
519 std::set<GroupAndName> events =
520 model.GetFtraceEventsForTesting(config, table_.get());
521
522 EXPECT_THAT(events, Contains(GroupAndName("sched", "sched_switch")));
523 EXPECT_THAT(events, Contains(GroupAndName("sched", "sched_cpu_hotplug")));
524 EXPECT_THAT(events, Contains(GroupAndName("ftrace", "print")));
525 }
526
TEST_F(FtraceConfigMuxerTest,GetFtraceEventsAtraceCategories)527 TEST_F(FtraceConfigMuxerTest, GetFtraceEventsAtraceCategories) {
528 MockFtraceProcfs ftrace;
529 FtraceConfigMuxer model(&ftrace, table_.get());
530
531 FtraceConfig config = CreateFtraceConfig({});
532 *config.add_atrace_categories() = "sched";
533 *config.add_atrace_categories() = "memreclaim";
534 std::set<GroupAndName> events =
535 model.GetFtraceEventsForTesting(config, table_.get());
536
537 EXPECT_THAT(events, Contains(GroupAndName("sched", "sched_switch")));
538 EXPECT_THAT(events, Contains(GroupAndName("sched", "sched_cpu_hotplug")));
539 EXPECT_THAT(events, Contains(GroupAndName("cgroup", "cgroup_mkdir")));
540 EXPECT_THAT(events, Contains(GroupAndName("vmscan",
541 "mm_vmscan_direct_reclaim_begin")));
542 EXPECT_THAT(events,
543 Contains(GroupAndName("lowmemorykiller", "lowmemory_kill")));
544 EXPECT_THAT(events, Contains(GroupAndName("ftrace", "print")));
545 }
546
547 // Tests the enabling fallback logic that tries to use the "set_event" interface
548 // if writing the individual xxx/enable file fails.
TEST_F(FtraceConfigMuxerTest,FallbackOnSetEvent)549 TEST_F(FtraceConfigMuxerTest, FallbackOnSetEvent) {
550 MockFtraceProcfs ftrace;
551 FtraceConfig config =
552 CreateFtraceConfig({"sched/sched_switch", "cgroup/cgroup_mkdir"});
553 FtraceConfigMuxer model(&ftrace, table_.get());
554
555 ON_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
556 .WillByDefault(Return("[local] global boot"));
557 EXPECT_CALL(ftrace, ReadFileIntoString("/root/trace_clock"))
558 .Times(AnyNumber());
559
560 EXPECT_CALL(ftrace, ReadOneCharFromFile("/root/tracing_on"))
561 .Times(2)
562 .WillRepeatedly(Return('0'));
563 EXPECT_CALL(ftrace, WriteToFile("/root/buffer_size_kb", _));
564 EXPECT_CALL(ftrace, WriteToFile("/root/trace_clock", "boot"));
565 EXPECT_CALL(ftrace, WriteToFile("/root/tracing_on", "1"));
566 EXPECT_CALL(ftrace,
567 WriteToFile("/root/events/sched/sched_switch/enable", "1"));
568 EXPECT_CALL(ftrace,
569 WriteToFile("/root/events/cgroup/cgroup_mkdir/enable", "1"))
570 .WillOnce(Return(false));
571 EXPECT_CALL(ftrace, AppendToFile("/root/set_event", "cgroup:cgroup_mkdir"))
572 .WillOnce(Return(true));
573 FtraceConfigId id = model.SetupConfig(config);
574 ASSERT_TRUE(id);
575 ASSERT_TRUE(model.ActivateConfig(id));
576
577 const FtraceConfig* actual_config = model.GetConfigForTesting(id);
578 EXPECT_TRUE(actual_config);
579 EXPECT_THAT(actual_config->ftrace_events(), Contains("sched/sched_switch"));
580 EXPECT_THAT(actual_config->ftrace_events(), Contains("cgroup/cgroup_mkdir"));
581
582 EXPECT_CALL(ftrace, WriteToFile("/root/tracing_on", "0"));
583 EXPECT_CALL(ftrace, WriteToFile("/root/buffer_size_kb", "0"));
584 EXPECT_CALL(ftrace, WriteToFile("/root/events/enable", "0"));
585 EXPECT_CALL(ftrace,
586 WriteToFile("/root/events/sched/sched_switch/enable", "0"));
587 EXPECT_CALL(ftrace,
588 WriteToFile("/root/events/cgroup/cgroup_mkdir/enable", "0"))
589 .WillOnce(Return(false));
590 EXPECT_CALL(ftrace, AppendToFile("/root/set_event", "!cgroup:cgroup_mkdir"))
591 .WillOnce(Return(true));
592 EXPECT_CALL(ftrace, ClearFile("/root/trace"));
593 EXPECT_CALL(ftrace, ClearFile(MatchesRegex("/root/per_cpu/cpu[0-9]/trace")));
594 ASSERT_TRUE(model.RemoveConfig(id));
595 }
596
597 } // namespace
598 } // namespace perfetto
599