1*9190c2a8SAndroid Build Coastguard Worker /*
2*9190c2a8SAndroid Build Coastguard Worker * Copyright (C) 2020 The Android Open Source Project
3*9190c2a8SAndroid Build Coastguard Worker *
4*9190c2a8SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*9190c2a8SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*9190c2a8SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*9190c2a8SAndroid Build Coastguard Worker *
8*9190c2a8SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*9190c2a8SAndroid Build Coastguard Worker *
10*9190c2a8SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*9190c2a8SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*9190c2a8SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*9190c2a8SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*9190c2a8SAndroid Build Coastguard Worker * limitations under the License.
15*9190c2a8SAndroid Build Coastguard Worker */
16*9190c2a8SAndroid Build Coastguard Worker
17*9190c2a8SAndroid Build Coastguard Worker #include <android-base/file.h>
18*9190c2a8SAndroid Build Coastguard Worker #include <android-base/mapped_file.h>
19*9190c2a8SAndroid Build Coastguard Worker #include <benchmark/benchmark.h>
20*9190c2a8SAndroid Build Coastguard Worker #include <unistd.h>
21*9190c2a8SAndroid Build Coastguard Worker
22*9190c2a8SAndroid Build Coastguard Worker #include "incfs_support/access.h"
23*9190c2a8SAndroid Build Coastguard Worker #include "incfs_support/signal_handling.h"
24*9190c2a8SAndroid Build Coastguard Worker #include "util/map_ptr.h"
25*9190c2a8SAndroid Build Coastguard Worker
makeFile()26*9190c2a8SAndroid Build Coastguard Worker static std::unique_ptr<TemporaryFile> makeFile() {
27*9190c2a8SAndroid Build Coastguard Worker auto tmp = std::unique_ptr<TemporaryFile>(new TemporaryFile());
28*9190c2a8SAndroid Build Coastguard Worker char c = 1;
29*9190c2a8SAndroid Build Coastguard Worker write(tmp->fd, &c, sizeof(c));
30*9190c2a8SAndroid Build Coastguard Worker return tmp;
31*9190c2a8SAndroid Build Coastguard Worker }
32*9190c2a8SAndroid Build Coastguard Worker
33*9190c2a8SAndroid Build Coastguard Worker static std::pair<std::unique_ptr<TemporaryFile>, std::unique_ptr<android::base::MappedFile>>
makeEmptyFileMapping()34*9190c2a8SAndroid Build Coastguard Worker makeEmptyFileMapping() {
35*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
36*9190c2a8SAndroid Build Coastguard Worker // mmap() only works for non-empty files, but it's "ok" to resize it back to empty afterwards
37*9190c2a8SAndroid Build Coastguard Worker auto mapping = android::base::MappedFile::FromFd(tmp->fd, 0, 1, PROT_READ);
38*9190c2a8SAndroid Build Coastguard Worker ftruncate(tmp->fd, 0);
39*9190c2a8SAndroid Build Coastguard Worker return {std::move(tmp), std::move(mapping)};
40*9190c2a8SAndroid Build Coastguard Worker }
41*9190c2a8SAndroid Build Coastguard Worker
TestEmpty(benchmark::State & state)42*9190c2a8SAndroid Build Coastguard Worker static void TestEmpty(benchmark::State& state) {
43*9190c2a8SAndroid Build Coastguard Worker int val = 0;
44*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
45*9190c2a8SAndroid Build Coastguard Worker benchmark::DoNotOptimize(val += 1);
46*9190c2a8SAndroid Build Coastguard Worker }
47*9190c2a8SAndroid Build Coastguard Worker }
48*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestEmpty);
49*9190c2a8SAndroid Build Coastguard Worker
TestSignal(benchmark::State & state)50*9190c2a8SAndroid Build Coastguard Worker static void TestSignal(benchmark::State& state) {
51*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
52*9190c2a8SAndroid Build Coastguard Worker auto mapping = android::base::MappedFile::FromFd(tmp->fd, 0, 1, PROT_READ);
53*9190c2a8SAndroid Build Coastguard Worker
54*9190c2a8SAndroid Build Coastguard Worker int val = 0;
55*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
56*9190c2a8SAndroid Build Coastguard Worker SCOPED_SIGBUS_HANDLER({ break; });
57*9190c2a8SAndroid Build Coastguard Worker val += *mapping->data();
58*9190c2a8SAndroid Build Coastguard Worker }
59*9190c2a8SAndroid Build Coastguard Worker }
60*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestSignal);
61*9190c2a8SAndroid Build Coastguard Worker
TestRead(benchmark::State & state)62*9190c2a8SAndroid Build Coastguard Worker static void TestRead(benchmark::State& state) {
63*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
64*9190c2a8SAndroid Build Coastguard Worker int val = 0;
65*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
66*9190c2a8SAndroid Build Coastguard Worker char c;
67*9190c2a8SAndroid Build Coastguard Worker pread(tmp->fd, &c, sizeof(c), 0);
68*9190c2a8SAndroid Build Coastguard Worker val += c;
69*9190c2a8SAndroid Build Coastguard Worker }
70*9190c2a8SAndroid Build Coastguard Worker }
71*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestRead);
72*9190c2a8SAndroid Build Coastguard Worker
TestMapPtrRaw(benchmark::State & state)73*9190c2a8SAndroid Build Coastguard Worker static void TestMapPtrRaw(benchmark::State& state) {
74*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
75*9190c2a8SAndroid Build Coastguard Worker android::incfs::IncFsFileMap map;
76*9190c2a8SAndroid Build Coastguard Worker map.CreateForceVerification(tmp->fd, 0, 1, tmp->path, true);
77*9190c2a8SAndroid Build Coastguard Worker int val = 0;
78*9190c2a8SAndroid Build Coastguard Worker const uint8_t* prev_block = nullptr;
79*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
80*9190c2a8SAndroid Build Coastguard Worker auto start = static_cast<const uint8_t*>(map.unsafe_data());
81*9190c2a8SAndroid Build Coastguard Worker auto end = start + map.length();
82*9190c2a8SAndroid Build Coastguard Worker val += map.Verify(start, end, &prev_block);
83*9190c2a8SAndroid Build Coastguard Worker }
84*9190c2a8SAndroid Build Coastguard Worker }
85*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestMapPtrRaw);
86*9190c2a8SAndroid Build Coastguard Worker
TestMapPtr(benchmark::State & state)87*9190c2a8SAndroid Build Coastguard Worker static void TestMapPtr(benchmark::State& state) {
88*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
89*9190c2a8SAndroid Build Coastguard Worker android::incfs::IncFsFileMap map;
90*9190c2a8SAndroid Build Coastguard Worker map.CreateForceVerification(tmp->fd, 0, 1, tmp->path, true);
91*9190c2a8SAndroid Build Coastguard Worker int val = 0;
92*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
93*9190c2a8SAndroid Build Coastguard Worker val += map.data<char>().verify();
94*9190c2a8SAndroid Build Coastguard Worker }
95*9190c2a8SAndroid Build Coastguard Worker }
96*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestMapPtr);
97*9190c2a8SAndroid Build Coastguard Worker
TestAccess(benchmark::State & state)98*9190c2a8SAndroid Build Coastguard Worker static void TestAccess(benchmark::State& state) {
99*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
100*9190c2a8SAndroid Build Coastguard Worker auto mapping = android::base::MappedFile::FromFd(tmp->fd, 0, 1, PROT_READ);
101*9190c2a8SAndroid Build Coastguard Worker int val = 0;
102*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
103*9190c2a8SAndroid Build Coastguard Worker incfs::access(mapping->data(), [&](auto ptr) { val += *ptr; });
104*9190c2a8SAndroid Build Coastguard Worker }
105*9190c2a8SAndroid Build Coastguard Worker }
106*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestAccess);
107*9190c2a8SAndroid Build Coastguard Worker
TestAccessFast(benchmark::State & state)108*9190c2a8SAndroid Build Coastguard Worker static void TestAccessFast(benchmark::State& state) {
109*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
110*9190c2a8SAndroid Build Coastguard Worker auto mapping = android::base::MappedFile::FromFd(tmp->fd, 0, 1, PROT_READ);
111*9190c2a8SAndroid Build Coastguard Worker int val = 0;
112*9190c2a8SAndroid Build Coastguard Worker incfs::access(mapping->data(), [&](auto ptr) {
113*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
114*9190c2a8SAndroid Build Coastguard Worker val += *ptr;
115*9190c2a8SAndroid Build Coastguard Worker }
116*9190c2a8SAndroid Build Coastguard Worker });
117*9190c2a8SAndroid Build Coastguard Worker }
118*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestAccessFast);
119*9190c2a8SAndroid Build Coastguard Worker
TestAccessVal(benchmark::State & state)120*9190c2a8SAndroid Build Coastguard Worker static void TestAccessVal(benchmark::State& state) {
121*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
122*9190c2a8SAndroid Build Coastguard Worker auto mapping = android::base::MappedFile::FromFd(tmp->fd, 0, 1, PROT_READ);
123*9190c2a8SAndroid Build Coastguard Worker int val = 0;
124*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
125*9190c2a8SAndroid Build Coastguard Worker incfs::access(mapping->data(), [&](auto ptr) { return val += *ptr; });
126*9190c2a8SAndroid Build Coastguard Worker }
127*9190c2a8SAndroid Build Coastguard Worker }
128*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestAccessVal);
129*9190c2a8SAndroid Build Coastguard Worker
TestAccessNested(benchmark::State & state)130*9190c2a8SAndroid Build Coastguard Worker static void TestAccessNested(benchmark::State& state) {
131*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
132*9190c2a8SAndroid Build Coastguard Worker auto mapping = android::base::MappedFile::FromFd(tmp->fd, 0, 1, PROT_READ);
133*9190c2a8SAndroid Build Coastguard Worker int val = 0;
134*9190c2a8SAndroid Build Coastguard Worker incfs::access(nullptr, [&](auto) {
135*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
136*9190c2a8SAndroid Build Coastguard Worker incfs::access(mapping->data(), [&](auto ptr) { val += *ptr; });
137*9190c2a8SAndroid Build Coastguard Worker }
138*9190c2a8SAndroid Build Coastguard Worker });
139*9190c2a8SAndroid Build Coastguard Worker }
140*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestAccessNested);
141*9190c2a8SAndroid Build Coastguard Worker
TestAccessDoubleNested(benchmark::State & state)142*9190c2a8SAndroid Build Coastguard Worker static void TestAccessDoubleNested(benchmark::State& state) {
143*9190c2a8SAndroid Build Coastguard Worker auto tmp = makeFile();
144*9190c2a8SAndroid Build Coastguard Worker auto mapping = android::base::MappedFile::FromFd(tmp->fd, 0, 1, PROT_READ);
145*9190c2a8SAndroid Build Coastguard Worker int val = 0;
146*9190c2a8SAndroid Build Coastguard Worker incfs::access(nullptr, [&](auto) {
147*9190c2a8SAndroid Build Coastguard Worker incfs::access(nullptr, [&](auto) {
148*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
149*9190c2a8SAndroid Build Coastguard Worker incfs::access(mapping->data(), [&](auto ptr) { val += *ptr; });
150*9190c2a8SAndroid Build Coastguard Worker }
151*9190c2a8SAndroid Build Coastguard Worker });
152*9190c2a8SAndroid Build Coastguard Worker });
153*9190c2a8SAndroid Build Coastguard Worker }
154*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestAccessDoubleNested);
155*9190c2a8SAndroid Build Coastguard Worker
TestAccessError(benchmark::State & state)156*9190c2a8SAndroid Build Coastguard Worker static void TestAccessError(benchmark::State& state) {
157*9190c2a8SAndroid Build Coastguard Worker auto [tmp, mapping] = makeEmptyFileMapping();
158*9190c2a8SAndroid Build Coastguard Worker int val = 0;
159*9190c2a8SAndroid Build Coastguard Worker for (auto _ : state) {
160*9190c2a8SAndroid Build Coastguard Worker incfs::access(mapping->data(), [&](auto ptr) { val += *ptr; });
161*9190c2a8SAndroid Build Coastguard Worker }
162*9190c2a8SAndroid Build Coastguard Worker }
163*9190c2a8SAndroid Build Coastguard Worker BENCHMARK(TestAccessError);
164*9190c2a8SAndroid Build Coastguard Worker
165*9190c2a8SAndroid Build Coastguard Worker BENCHMARK_MAIN();
166