1 /*
2  * Copyright (C) 2015 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 <gtest/gtest.h>
18 
19 #include <string.h>
20 
21 #include <memory>
22 
23 #include <android-base/file.h>
24 
25 #include "environment.h"
26 #include "event_attr.h"
27 #include "event_type.h"
28 #include "record.h"
29 #include "record_file.h"
30 #include "utils.h"
31 
32 #include "record_equal_test.h"
33 
34 using namespace simpleperf;
35 using namespace simpleperf::PerfFileFormat;
36 
37 class RecordFileTest : public ::testing::Test {
38  protected:
SetUp()39   void SetUp() override { close(tmpfile_.release()); }
40 
AddEventType(const std::string & event_type_str)41   void AddEventType(const std::string& event_type_str) {
42     std::unique_ptr<EventTypeAndModifier> event_type_modifier = ParseEventType(event_type_str);
43     ASSERT_TRUE(event_type_modifier != nullptr);
44     perf_event_attr attr = CreateDefaultPerfEventAttr(event_type_modifier->event_type);
45     attr.sample_id_all = 1;
46     attrs_.push_back(std::unique_ptr<perf_event_attr>(new perf_event_attr(attr)));
47     EventAttrWithId attr_id;
48     attr_id.attr = attrs_.back().get();
49     attr_id.ids.push_back(attrs_.size());  // Fake id.
50     attr_ids_.push_back(attr_id);
51   }
52 
53   TemporaryFile tmpfile_;
54   std::vector<std::unique_ptr<perf_event_attr>> attrs_;
55   std::vector<EventAttrWithId> attr_ids_;
56 };
57 
TEST_F(RecordFileTest,smoke)58 TEST_F(RecordFileTest, smoke) {
59   // Write to a record file.
60   std::unique_ptr<RecordFileWriter> writer = RecordFileWriter::CreateInstance(tmpfile_.path);
61   ASSERT_TRUE(writer != nullptr);
62 
63   // Write attr section.
64   AddEventType("cpu-cycles");
65   ASSERT_TRUE(writer->WriteAttrSection(attr_ids_));
66 
67   // Write data section.
68   MmapRecord mmap_record(*(attr_ids_[0].attr), true, 1, 1, 0x1000, 0x2000, 0x3000,
69                          "mmap_record_example", attr_ids_[0].ids[0]);
70   ASSERT_TRUE(writer->WriteRecord(mmap_record));
71 
72   // Write feature section.
73   ASSERT_TRUE(writer->BeginWriteFeatures(1));
74   char p[BuildId::Size()];
75   for (size_t i = 0; i < BuildId::Size(); ++i) {
76     p[i] = i;
77   }
78   BuildId build_id(p);
79   std::vector<BuildIdRecord> build_id_records;
80   build_id_records.push_back(BuildIdRecord(false, getpid(), build_id, "init"));
81   ASSERT_TRUE(writer->WriteBuildIdFeature(build_id_records));
82   ASSERT_TRUE(writer->EndWriteFeatures());
83   ASSERT_TRUE(writer->Close());
84 
85   // Read from a record file.
86   std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(tmpfile_.path);
87   ASSERT_TRUE(reader != nullptr);
88   std::vector<EventAttrWithId> attrs = reader->AttrSection();
89   ASSERT_EQ(1u, attrs.size());
90   ASSERT_EQ(0, memcmp(attrs[0].attr, attr_ids_[0].attr, sizeof(perf_event_attr)));
91   ASSERT_EQ(attrs[0].ids, attr_ids_[0].ids);
92 
93   // Read and check data section.
94   std::vector<std::unique_ptr<Record>> records = reader->DataSection();
95   ASSERT_EQ(1u, records.size());
96   CheckRecordEqual(mmap_record, *records[0]);
97 
98   // Read and check feature section.
99   std::vector<BuildIdRecord> read_build_id_records = reader->ReadBuildIdFeature();
100   ASSERT_EQ(1u, read_build_id_records.size());
101   CheckRecordEqual(read_build_id_records[0], build_id_records[0]);
102 
103   ASSERT_TRUE(reader->Close());
104 }
105 
TEST_F(RecordFileTest,record_more_than_one_attr)106 TEST_F(RecordFileTest, record_more_than_one_attr) {
107   // Write to a record file.
108   std::unique_ptr<RecordFileWriter> writer = RecordFileWriter::CreateInstance(tmpfile_.path);
109   ASSERT_TRUE(writer != nullptr);
110 
111   // Write attr section.
112   AddEventType("cpu-cycles");
113   AddEventType("cpu-clock");
114   AddEventType("task-clock");
115   ASSERT_TRUE(writer->WriteAttrSection(attr_ids_));
116 
117   ASSERT_TRUE(writer->Close());
118 
119   // Read from a record file.
120   std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(tmpfile_.path);
121   ASSERT_TRUE(reader != nullptr);
122   std::vector<EventAttrWithId> attrs = reader->AttrSection();
123   ASSERT_EQ(3u, attrs.size());
124   for (size_t i = 0; i < attrs.size(); ++i) {
125     ASSERT_EQ(0, memcmp(attrs[i].attr, attr_ids_[i].attr, sizeof(perf_event_attr)));
126     ASSERT_EQ(attrs[i].ids, attr_ids_[i].ids);
127   }
128 }
129 
TEST_F(RecordFileTest,write_meta_info_feature_section)130 TEST_F(RecordFileTest, write_meta_info_feature_section) {
131   // Write to a record file.
132   std::unique_ptr<RecordFileWriter> writer = RecordFileWriter::CreateInstance(tmpfile_.path);
133   ASSERT_TRUE(writer != nullptr);
134   AddEventType("cpu-cycles");
135   ASSERT_TRUE(writer->WriteAttrSection(attr_ids_));
136 
137   // Write meta_info feature section.
138   ASSERT_TRUE(writer->BeginWriteFeatures(1));
139   std::unordered_map<std::string, std::string> info_map;
140   for (int i = 0; i < 100; ++i) {
141     std::string s = std::to_string(i);
142     info_map[s] = s + s;
143   }
144   ASSERT_TRUE(writer->WriteMetaInfoFeature(info_map));
145   ASSERT_TRUE(writer->EndWriteFeatures());
146   ASSERT_TRUE(writer->Close());
147 
148   // Read from a record file.
149   std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(tmpfile_.path);
150   ASSERT_TRUE(reader != nullptr);
151   ASSERT_EQ(reader->GetMetaInfoFeature(), info_map);
152 }
153 
TEST_F(RecordFileTest,write_debug_unwind_feature_section)154 TEST_F(RecordFileTest, write_debug_unwind_feature_section) {
155   // Write to a record file.
156   std::unique_ptr<RecordFileWriter> writer = RecordFileWriter::CreateInstance(tmpfile_.path);
157   ASSERT_TRUE(writer != nullptr);
158   AddEventType("cpu-cycles");
159   ASSERT_TRUE(writer->WriteAttrSection(attr_ids_));
160 
161   // Write debug_unwind feature section.
162   ASSERT_TRUE(writer->BeginWriteFeatures(1));
163   DebugUnwindFeature debug_unwind(2);
164   debug_unwind[0].path = "file1";
165   debug_unwind[0].size = 1000;
166   debug_unwind[1].path = "file2";
167   debug_unwind[1].size = 2000;
168   ASSERT_TRUE(writer->WriteDebugUnwindFeature(debug_unwind));
169   ASSERT_TRUE(writer->EndWriteFeatures());
170   ASSERT_TRUE(writer->Close());
171 
172   // Read from a record file.
173   std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(tmpfile_.path);
174   ASSERT_TRUE(reader != nullptr);
175   std::optional<DebugUnwindFeature> opt_debug_unwind = reader->ReadDebugUnwindFeature();
176   ASSERT_TRUE(opt_debug_unwind.has_value());
177   ASSERT_EQ(opt_debug_unwind.value().size(), debug_unwind.size());
178   for (size_t i = 0; i < debug_unwind.size(); i++) {
179     ASSERT_EQ(opt_debug_unwind.value()[i].path, debug_unwind[i].path);
180     ASSERT_EQ(opt_debug_unwind.value()[i].size, debug_unwind[i].size);
181   }
182 }
183