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>
20 #include "environment.h"
21 #include "event_attr.h"
23 #include "event_type.h"
25 #include "record_file.h"
27 #include "record_equal_test.h"
29 using namespace PerfFileFormat;
31 class RecordFileTest : public ::testing::Test {
33 virtual void SetUp() {
34 filename = "temporary.record_file";
35 std::unique_ptr<EventTypeAndModifier> event_type_modifier = ParseEventType("cpu-cycles");
36 ASSERT_TRUE(event_type_modifier != nullptr);
37 event_attr = CreateDefaultPerfEventAttr(event_type_modifier->event_type);
38 event_attr.sample_id_all = 1;
39 event_attr.sample_type |= PERF_SAMPLE_TIME;
40 std::unique_ptr<EventFd> event_fd = EventFd::OpenEventFile(event_attr, getpid(), -1);
41 ASSERT_TRUE(event_fd != nullptr);
42 event_fds.push_back(std::move(event_fd));
46 perf_event_attr event_attr;
47 std::vector<std::unique_ptr<EventFd>> event_fds;
50 TEST_F(RecordFileTest, smoke) {
51 // Write to a record file.
52 std::unique_ptr<RecordFileWriter> writer =
53 RecordFileWriter::CreateInstance(filename, event_attr, event_fds);
54 ASSERT_TRUE(writer != nullptr);
56 // Write data section.
57 MmapRecord mmap_record =
58 CreateMmapRecord(event_attr, true, 1, 1, 0x1000, 0x2000, 0x3000, "mmap_record_example");
59 ASSERT_TRUE(writer->WriteData(mmap_record.BinaryFormat()));
61 // Write feature section.
62 ASSERT_TRUE(writer->WriteFeatureHeader(1));
63 char p[BuildId::Size()];
64 for (size_t i = 0; i < BuildId::Size(); ++i) {
68 BuildIdRecord build_id_record = CreateBuildIdRecord(false, getpid(), build_id, "init");
69 ASSERT_TRUE(writer->WriteBuildIdFeature({build_id_record}));
70 ASSERT_TRUE(writer->Close());
72 // Read from a record file.
73 std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(filename);
74 ASSERT_TRUE(reader != nullptr);
75 const FileHeader* file_header = reader->FileHeader();
76 ASSERT_TRUE(file_header != nullptr);
77 std::vector<const FileAttr*> attrs = reader->AttrSection();
78 ASSERT_EQ(1u, attrs.size());
79 ASSERT_EQ(0, memcmp(&attrs[0]->attr, &event_attr, sizeof(perf_event_attr)));
80 std::vector<uint64_t> ids = reader->IdsForAttr(attrs[0]);
81 ASSERT_EQ(1u, ids.size());
83 // Read and check data section.
84 std::vector<std::unique_ptr<const Record>> records = reader->DataSection();
85 ASSERT_EQ(1u, records.size());
86 CheckRecordEqual(mmap_record, *records[0]);
88 // Read and check feature section.
89 ASSERT_TRUE(file_header->features[FEAT_BUILD_ID / 8] & (1 << (FEAT_BUILD_ID % 8)));
90 std::map<int, SectionDesc> sections = reader->FeatureSectionDescriptors();
91 ASSERT_EQ(1u, sections.size());
92 ASSERT_TRUE(sections.find(FEAT_BUILD_ID) != sections.end());
93 const perf_event_header* header = reinterpret_cast<const perf_event_header*>(
94 reader->DataAtOffset(sections[FEAT_BUILD_ID].offset));
95 ASSERT_TRUE(header != nullptr);
96 ASSERT_EQ(sections[FEAT_BUILD_ID].size, header->size);
97 CheckRecordEqual(build_id_record, BuildIdRecord(header));
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 =
105 RecordFileWriter::CreateInstance(filename, event_attr, event_fds);
106 ASSERT_TRUE(writer != nullptr);
108 // Write data section.
109 MmapRecord r1 = CreateMmapRecord(event_attr, true, 1, 1, 0x100, 0x2000, 0x3000, "mmap_record1");
112 r1.sample_id.time_data.time = 2;
113 r2.sample_id.time_data.time = 1;
114 r3.sample_id.time_data.time = 3;
115 ASSERT_TRUE(writer->WriteData(r1.BinaryFormat()));
116 ASSERT_TRUE(writer->WriteData(r2.BinaryFormat()));
117 ASSERT_TRUE(writer->WriteData(r3.BinaryFormat()));
118 ASSERT_TRUE(writer->Close());
120 // Read from a record file.
121 std::unique_ptr<RecordFileReader> reader = RecordFileReader::CreateInstance(filename);
122 ASSERT_TRUE(reader != nullptr);
123 std::vector<std::unique_ptr<const Record>> records = reader->DataSection();
124 ASSERT_EQ(3u, records.size());
125 CheckRecordEqual(r2, *records[0]);
126 CheckRecordEqual(r1, *records[1]);
127 CheckRecordEqual(r3, *records[2]);
129 ASSERT_TRUE(reader->Close());