2 * Copyright (C) 2015 The Android Open Source Project
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
8 * http://www.apache.org/licenses/LICENSE-2.0
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.
17 #include <gtest/gtest.h>
23 #include <android-base/test_utils.h>
25 #include "environment.h"
26 #include "event_attr.h"
27 #include "event_type.h"
29 #include "record_file.h"
31 #include "record_equal_test.h"
33 using namespace PerfFileFormat;
35 class RecordFileTest : public ::testing::Test {
37 void AddEventType(const std::string& event_type_str) {
38 std::unique_ptr<EventTypeAndModifier> event_type_modifier = ParseEventType(event_type_str);
39 ASSERT_TRUE(event_type_modifier != nullptr);
40 perf_event_attr attr = CreateDefaultPerfEventAttr(event_type_modifier->event_type);
41 attr.sample_id_all = 1;
42 attrs_.push_back(std::unique_ptr<perf_event_attr>(new perf_event_attr(attr)));
43 EventAttrWithId attr_id;
44 attr_id.attr = attrs_.back().get();
45 attr_id.ids.push_back(attrs_.size()); // Fake id.
46 attr_ids_.push_back(attr_id);
49 TemporaryFile tmpfile_;
50 std::vector<std::unique_ptr<perf_event_attr>> attrs_;
51 std::vector<EventAttrWithId> attr_ids_;
54 TEST_F(RecordFileTest, smoke) {
55 // Write to a record file.
56 std::unique_ptr<RecordFileWriter> writer = RecordFileWriter::CreateInstance(tmpfile_.path);
57 ASSERT_TRUE(writer != nullptr);
59 // Write attr section.
60 AddEventType("cpu-cycles");
61 ASSERT_TRUE(writer->WriteAttrSection(attr_ids_));
63 // Write data section.
64 MmapRecord mmap_record(*(attr_ids_[0].attr), true, 1, 1, 0x1000, 0x2000,
65 0x3000, "mmap_record_example", attr_ids_[0].ids[0]);
66 ASSERT_TRUE(writer->WriteRecord(mmap_record));
68 // Write feature section.
69 ASSERT_TRUE(writer->BeginWriteFeatures(1));
70 char p[BuildId::Size()];
71 for (size_t i = 0; i < BuildId::Size(); ++i) {
75 std::vector<BuildIdRecord> build_id_records;
76 build_id_records.push_back(BuildIdRecord(false, getpid(), build_id, "init"));
77 ASSERT_TRUE(writer->WriteBuildIdFeature(build_id_records));
78 ASSERT_TRUE(writer->EndWriteFeatures());
79 ASSERT_TRUE(writer->Close());
81 // Read from a record file.
82 std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(tmpfile_.path);
83 ASSERT_TRUE(reader != nullptr);
84 std::vector<EventAttrWithId> attrs = reader->AttrSection();
85 ASSERT_EQ(1u, attrs.size());
86 ASSERT_EQ(0, memcmp(attrs[0].attr, attr_ids_[0].attr, sizeof(perf_event_attr)));
87 ASSERT_EQ(attrs[0].ids, attr_ids_[0].ids);
89 // Read and check data section.
90 std::vector<std::unique_ptr<Record>> records = reader->DataSection();
91 ASSERT_EQ(1u, records.size());
92 CheckRecordEqual(mmap_record, *records[0]);
94 // Read and check feature section.
95 std::vector<BuildIdRecord> read_build_id_records = reader->ReadBuildIdFeature();
96 ASSERT_EQ(1u, read_build_id_records.size());
97 CheckRecordEqual(read_build_id_records[0], build_id_records[0]);
99 ASSERT_TRUE(reader->Close());
102 TEST_F(RecordFileTest, records_sorted_by_time) {
103 // Write to a record file.
104 std::unique_ptr<RecordFileWriter> writer = RecordFileWriter::CreateInstance(tmpfile_.path);
105 ASSERT_TRUE(writer != nullptr);
107 // Write attr section.
108 AddEventType("cpu-cycles");
109 attrs_[0]->sample_id_all = 1;
110 attrs_[0]->sample_type |= PERF_SAMPLE_TIME;
111 ASSERT_TRUE(writer->WriteAttrSection(attr_ids_));
113 // Write data section.
114 MmapRecord r1(*(attr_ids_[0].attr), true, 1, 1, 0x100, 0x2000, 0x3000, "mmap_record1",
115 attr_ids_[0].ids[0], 2);
116 MmapRecord r2(*(attr_ids_[0].attr), true, 1, 1, 0x100, 0x2000, 0x3000, "mmap_record1",
117 attr_ids_[0].ids[0], 1);
118 MmapRecord r3(*(attr_ids_[0].attr), true, 1, 1, 0x100, 0x2000, 0x3000, "mmap_record1",
119 attr_ids_[0].ids[0], 3);
120 ASSERT_TRUE(writer->WriteRecord(r1));
121 ASSERT_TRUE(writer->WriteRecord(r2));
122 ASSERT_TRUE(writer->WriteRecord(r3));
123 ASSERT_TRUE(writer->Close());
125 // Read from a record file.
126 std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(tmpfile_.path);
127 ASSERT_TRUE(reader != nullptr);
128 std::vector<std::unique_ptr<Record>> records = reader->DataSection();
129 ASSERT_EQ(3u, records.size());
130 CheckRecordEqual(r2, *records[0]);
131 CheckRecordEqual(r1, *records[1]);
132 CheckRecordEqual(r3, *records[2]);
134 ASSERT_TRUE(reader->Close());
137 TEST_F(RecordFileTest, record_more_than_one_attr) {
138 // Write to a record file.
139 std::unique_ptr<RecordFileWriter> writer = RecordFileWriter::CreateInstance(tmpfile_.path);
140 ASSERT_TRUE(writer != nullptr);
142 // Write attr section.
143 AddEventType("cpu-cycles");
144 AddEventType("cpu-clock");
145 AddEventType("task-clock");
146 ASSERT_TRUE(writer->WriteAttrSection(attr_ids_));
148 ASSERT_TRUE(writer->Close());
150 // Read from a record file.
151 std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(tmpfile_.path);
152 ASSERT_TRUE(reader != nullptr);
153 std::vector<EventAttrWithId> attrs = reader->AttrSection();
154 ASSERT_EQ(3u, attrs.size());
155 for (size_t i = 0; i < attrs.size(); ++i) {
156 ASSERT_EQ(0, memcmp(attrs[i].attr, attr_ids_[i].attr, sizeof(perf_event_attr)));
157 ASSERT_EQ(attrs[i].ids, attr_ids_[i].ids);
161 TEST_F(RecordFileTest, write_meta_info_feature_section) {
162 // Write to a record file.
163 std::unique_ptr<RecordFileWriter> writer = RecordFileWriter::CreateInstance(tmpfile_.path);
164 ASSERT_TRUE(writer != nullptr);
165 AddEventType("cpu-cycles");
166 ASSERT_TRUE(writer->WriteAttrSection(attr_ids_));
168 // Write meta_info feature section.
169 ASSERT_TRUE(writer->BeginWriteFeatures(1));
170 std::unordered_map<std::string, std::string> info_map;
171 for (int i = 0; i < 100; ++i) {
172 std::string s = std::to_string(i);
175 ASSERT_TRUE(writer->WriteMetaInfoFeature(info_map));
176 ASSERT_TRUE(writer->EndWriteFeatures());
177 ASSERT_TRUE(writer->Close());
179 // Read from a record file.
180 std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(tmpfile_.path);
181 ASSERT_TRUE(reader != nullptr);
182 std::unordered_map<std::string, std::string> read_info_map;
183 ASSERT_TRUE(reader->ReadMetaInfoFeature(&read_info_map));
184 ASSERT_EQ(read_info_map, info_map);