1*f40fafd4SAndroid Build Coastguard Worker /*
2*f40fafd4SAndroid Build Coastguard Worker * Copyright (C) 2015 The Android Open Source Project
3*f40fafd4SAndroid Build Coastguard Worker *
4*f40fafd4SAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*f40fafd4SAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*f40fafd4SAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*f40fafd4SAndroid Build Coastguard Worker *
8*f40fafd4SAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*f40fafd4SAndroid Build Coastguard Worker *
10*f40fafd4SAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*f40fafd4SAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*f40fafd4SAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*f40fafd4SAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*f40fafd4SAndroid Build Coastguard Worker * limitations under the License.
15*f40fafd4SAndroid Build Coastguard Worker */
16*f40fafd4SAndroid Build Coastguard Worker
17*f40fafd4SAndroid Build Coastguard Worker #include "Benchmark.h"
18*f40fafd4SAndroid Build Coastguard Worker #include "BenchmarkGen.h"
19*f40fafd4SAndroid Build Coastguard Worker #include "VolumeManager.h"
20*f40fafd4SAndroid Build Coastguard Worker
21*f40fafd4SAndroid Build Coastguard Worker #include <android-base/chrono_utils.h>
22*f40fafd4SAndroid Build Coastguard Worker #include <android-base/file.h>
23*f40fafd4SAndroid Build Coastguard Worker #include <android-base/logging.h>
24*f40fafd4SAndroid Build Coastguard Worker
25*f40fafd4SAndroid Build Coastguard Worker #include <cutils/iosched_policy.h>
26*f40fafd4SAndroid Build Coastguard Worker #include <private/android_filesystem_config.h>
27*f40fafd4SAndroid Build Coastguard Worker #include <wakelock/wakelock.h>
28*f40fafd4SAndroid Build Coastguard Worker
29*f40fafd4SAndroid Build Coastguard Worker #include <thread>
30*f40fafd4SAndroid Build Coastguard Worker
31*f40fafd4SAndroid Build Coastguard Worker #include <sys/resource.h>
32*f40fafd4SAndroid Build Coastguard Worker #include <sys/time.h>
33*f40fafd4SAndroid Build Coastguard Worker #include <unistd.h>
34*f40fafd4SAndroid Build Coastguard Worker
35*f40fafd4SAndroid Build Coastguard Worker using android::base::ReadFileToString;
36*f40fafd4SAndroid Build Coastguard Worker using android::base::WriteStringToFile;
37*f40fafd4SAndroid Build Coastguard Worker
38*f40fafd4SAndroid Build Coastguard Worker namespace android {
39*f40fafd4SAndroid Build Coastguard Worker namespace vold {
40*f40fafd4SAndroid Build Coastguard Worker
41*f40fafd4SAndroid Build Coastguard Worker // Benchmark currently uses chdir(), which means we can only
42*f40fafd4SAndroid Build Coastguard Worker // safely run one at a time.
43*f40fafd4SAndroid Build Coastguard Worker static std::mutex kBenchmarkLock;
44*f40fafd4SAndroid Build Coastguard Worker
45*f40fafd4SAndroid Build Coastguard Worker static const char* kWakeLock = "Benchmark";
46*f40fafd4SAndroid Build Coastguard Worker
47*f40fafd4SAndroid Build Coastguard Worker // Reasonable cards are able to complete the create/run stages
48*f40fafd4SAndroid Build Coastguard Worker // in under 20 seconds.
49*f40fafd4SAndroid Build Coastguard Worker constexpr auto kTimeout = 20s;
50*f40fafd4SAndroid Build Coastguard Worker
51*f40fafd4SAndroid Build Coastguard Worker // RAII class for boosting device performance during benchmarks.
52*f40fafd4SAndroid Build Coastguard Worker class PerformanceBoost {
53*f40fafd4SAndroid Build Coastguard Worker private:
54*f40fafd4SAndroid Build Coastguard Worker int orig_prio;
55*f40fafd4SAndroid Build Coastguard Worker int orig_ioprio;
56*f40fafd4SAndroid Build Coastguard Worker IoSchedClass orig_clazz;
57*f40fafd4SAndroid Build Coastguard Worker
58*f40fafd4SAndroid Build Coastguard Worker public:
PerformanceBoost()59*f40fafd4SAndroid Build Coastguard Worker PerformanceBoost() {
60*f40fafd4SAndroid Build Coastguard Worker errno = 0;
61*f40fafd4SAndroid Build Coastguard Worker orig_prio = getpriority(PRIO_PROCESS, 0);
62*f40fafd4SAndroid Build Coastguard Worker if (errno != 0) {
63*f40fafd4SAndroid Build Coastguard Worker PLOG(WARNING) << "Failed to getpriority";
64*f40fafd4SAndroid Build Coastguard Worker orig_prio = 0;
65*f40fafd4SAndroid Build Coastguard Worker }
66*f40fafd4SAndroid Build Coastguard Worker if (setpriority(PRIO_PROCESS, 0, -10) != 0) {
67*f40fafd4SAndroid Build Coastguard Worker PLOG(WARNING) << "Failed to setpriority";
68*f40fafd4SAndroid Build Coastguard Worker }
69*f40fafd4SAndroid Build Coastguard Worker if (android_get_ioprio(0, &orig_clazz, &orig_ioprio)) {
70*f40fafd4SAndroid Build Coastguard Worker PLOG(WARNING) << "Failed to android_get_ioprio";
71*f40fafd4SAndroid Build Coastguard Worker orig_ioprio = 0;
72*f40fafd4SAndroid Build Coastguard Worker orig_clazz = IoSchedClass_NONE;
73*f40fafd4SAndroid Build Coastguard Worker }
74*f40fafd4SAndroid Build Coastguard Worker if (android_set_ioprio(0, IoSchedClass_RT, 0)) {
75*f40fafd4SAndroid Build Coastguard Worker PLOG(WARNING) << "Failed to android_set_ioprio";
76*f40fafd4SAndroid Build Coastguard Worker }
77*f40fafd4SAndroid Build Coastguard Worker }
78*f40fafd4SAndroid Build Coastguard Worker
~PerformanceBoost()79*f40fafd4SAndroid Build Coastguard Worker ~PerformanceBoost() {
80*f40fafd4SAndroid Build Coastguard Worker if (android_set_ioprio(0, orig_clazz, orig_ioprio)) {
81*f40fafd4SAndroid Build Coastguard Worker PLOG(WARNING) << "Failed to android_set_ioprio";
82*f40fafd4SAndroid Build Coastguard Worker }
83*f40fafd4SAndroid Build Coastguard Worker if (setpriority(PRIO_PROCESS, 0, orig_prio) != 0) {
84*f40fafd4SAndroid Build Coastguard Worker PLOG(WARNING) << "Failed to setpriority";
85*f40fafd4SAndroid Build Coastguard Worker }
86*f40fafd4SAndroid Build Coastguard Worker }
87*f40fafd4SAndroid Build Coastguard Worker };
88*f40fafd4SAndroid Build Coastguard Worker
benchmarkInternal(const std::string & rootPath,const android::sp<android::os::IVoldTaskListener> & listener,android::os::PersistableBundle * extras)89*f40fafd4SAndroid Build Coastguard Worker static status_t benchmarkInternal(const std::string& rootPath,
90*f40fafd4SAndroid Build Coastguard Worker const android::sp<android::os::IVoldTaskListener>& listener,
91*f40fafd4SAndroid Build Coastguard Worker android::os::PersistableBundle* extras) {
92*f40fafd4SAndroid Build Coastguard Worker status_t res = 0;
93*f40fafd4SAndroid Build Coastguard Worker
94*f40fafd4SAndroid Build Coastguard Worker auto path = rootPath;
95*f40fafd4SAndroid Build Coastguard Worker path += "/misc";
96*f40fafd4SAndroid Build Coastguard Worker if (android::vold::PrepareDir(path, 01771, AID_SYSTEM, AID_MISC)) {
97*f40fafd4SAndroid Build Coastguard Worker return -1;
98*f40fafd4SAndroid Build Coastguard Worker }
99*f40fafd4SAndroid Build Coastguard Worker path += "/vold";
100*f40fafd4SAndroid Build Coastguard Worker if (android::vold::PrepareDir(path, 0700, AID_ROOT, AID_ROOT)) {
101*f40fafd4SAndroid Build Coastguard Worker return -1;
102*f40fafd4SAndroid Build Coastguard Worker }
103*f40fafd4SAndroid Build Coastguard Worker path += "/bench";
104*f40fafd4SAndroid Build Coastguard Worker if (android::vold::PrepareDir(path, 0700, AID_ROOT, AID_ROOT)) {
105*f40fafd4SAndroid Build Coastguard Worker return -1;
106*f40fafd4SAndroid Build Coastguard Worker }
107*f40fafd4SAndroid Build Coastguard Worker
108*f40fafd4SAndroid Build Coastguard Worker char orig_cwd[PATH_MAX];
109*f40fafd4SAndroid Build Coastguard Worker if (getcwd(orig_cwd, PATH_MAX) == NULL) {
110*f40fafd4SAndroid Build Coastguard Worker PLOG(ERROR) << "Failed getcwd";
111*f40fafd4SAndroid Build Coastguard Worker return -1;
112*f40fafd4SAndroid Build Coastguard Worker }
113*f40fafd4SAndroid Build Coastguard Worker if (chdir(path.c_str()) != 0) {
114*f40fafd4SAndroid Build Coastguard Worker PLOG(ERROR) << "Failed chdir";
115*f40fafd4SAndroid Build Coastguard Worker return -1;
116*f40fafd4SAndroid Build Coastguard Worker }
117*f40fafd4SAndroid Build Coastguard Worker
118*f40fafd4SAndroid Build Coastguard Worker sync();
119*f40fafd4SAndroid Build Coastguard Worker
120*f40fafd4SAndroid Build Coastguard Worker extras->putString(String16("path"), String16(path.c_str()));
121*f40fafd4SAndroid Build Coastguard Worker extras->putString(String16("ident"), String16(BenchmarkIdent().c_str()));
122*f40fafd4SAndroid Build Coastguard Worker
123*f40fafd4SAndroid Build Coastguard Worker // Always create
124*f40fafd4SAndroid Build Coastguard Worker {
125*f40fafd4SAndroid Build Coastguard Worker android::base::Timer timer;
126*f40fafd4SAndroid Build Coastguard Worker LOG(INFO) << "Creating " << path;
127*f40fafd4SAndroid Build Coastguard Worker res |= BenchmarkCreate([&](int progress) -> bool {
128*f40fafd4SAndroid Build Coastguard Worker if (listener) {
129*f40fafd4SAndroid Build Coastguard Worker listener->onStatus(progress, *extras);
130*f40fafd4SAndroid Build Coastguard Worker }
131*f40fafd4SAndroid Build Coastguard Worker return (timer.duration() < kTimeout);
132*f40fafd4SAndroid Build Coastguard Worker });
133*f40fafd4SAndroid Build Coastguard Worker sync();
134*f40fafd4SAndroid Build Coastguard Worker if (res == OK) extras->putLong(String16("create"), timer.duration().count());
135*f40fafd4SAndroid Build Coastguard Worker }
136*f40fafd4SAndroid Build Coastguard Worker
137*f40fafd4SAndroid Build Coastguard Worker // Only drop when we haven't aborted
138*f40fafd4SAndroid Build Coastguard Worker if (res == OK) {
139*f40fafd4SAndroid Build Coastguard Worker android::base::Timer timer;
140*f40fafd4SAndroid Build Coastguard Worker LOG(DEBUG) << "Before drop_caches";
141*f40fafd4SAndroid Build Coastguard Worker if (!WriteStringToFile("3", "/proc/sys/vm/drop_caches")) {
142*f40fafd4SAndroid Build Coastguard Worker PLOG(ERROR) << "Failed to drop_caches";
143*f40fafd4SAndroid Build Coastguard Worker res = -1;
144*f40fafd4SAndroid Build Coastguard Worker }
145*f40fafd4SAndroid Build Coastguard Worker LOG(DEBUG) << "After drop_caches";
146*f40fafd4SAndroid Build Coastguard Worker sync();
147*f40fafd4SAndroid Build Coastguard Worker if (res == OK) extras->putLong(String16("drop"), timer.duration().count());
148*f40fafd4SAndroid Build Coastguard Worker }
149*f40fafd4SAndroid Build Coastguard Worker
150*f40fafd4SAndroid Build Coastguard Worker // Only run when we haven't aborted
151*f40fafd4SAndroid Build Coastguard Worker if (res == OK) {
152*f40fafd4SAndroid Build Coastguard Worker android::base::Timer timer;
153*f40fafd4SAndroid Build Coastguard Worker LOG(INFO) << "Running " << path;
154*f40fafd4SAndroid Build Coastguard Worker res |= BenchmarkRun([&](int progress) -> bool {
155*f40fafd4SAndroid Build Coastguard Worker if (listener) {
156*f40fafd4SAndroid Build Coastguard Worker listener->onStatus(progress, *extras);
157*f40fafd4SAndroid Build Coastguard Worker }
158*f40fafd4SAndroid Build Coastguard Worker return (timer.duration() < kTimeout);
159*f40fafd4SAndroid Build Coastguard Worker });
160*f40fafd4SAndroid Build Coastguard Worker sync();
161*f40fafd4SAndroid Build Coastguard Worker if (res == OK) extras->putLong(String16("run"), timer.duration().count());
162*f40fafd4SAndroid Build Coastguard Worker }
163*f40fafd4SAndroid Build Coastguard Worker
164*f40fafd4SAndroid Build Coastguard Worker // Always destroy
165*f40fafd4SAndroid Build Coastguard Worker {
166*f40fafd4SAndroid Build Coastguard Worker android::base::Timer timer;
167*f40fafd4SAndroid Build Coastguard Worker LOG(INFO) << "Destroying " << path;
168*f40fafd4SAndroid Build Coastguard Worker res |= BenchmarkDestroy();
169*f40fafd4SAndroid Build Coastguard Worker sync();
170*f40fafd4SAndroid Build Coastguard Worker if (res == OK) extras->putLong(String16("destroy"), timer.duration().count());
171*f40fafd4SAndroid Build Coastguard Worker }
172*f40fafd4SAndroid Build Coastguard Worker
173*f40fafd4SAndroid Build Coastguard Worker if (chdir(orig_cwd) != 0) {
174*f40fafd4SAndroid Build Coastguard Worker PLOG(ERROR) << "Failed to chdir";
175*f40fafd4SAndroid Build Coastguard Worker return -1;
176*f40fafd4SAndroid Build Coastguard Worker }
177*f40fafd4SAndroid Build Coastguard Worker
178*f40fafd4SAndroid Build Coastguard Worker return res;
179*f40fafd4SAndroid Build Coastguard Worker }
180*f40fafd4SAndroid Build Coastguard Worker
Benchmark(const std::string & path,const android::sp<android::os::IVoldTaskListener> & listener)181*f40fafd4SAndroid Build Coastguard Worker void Benchmark(const std::string& path,
182*f40fafd4SAndroid Build Coastguard Worker const android::sp<android::os::IVoldTaskListener>& listener) {
183*f40fafd4SAndroid Build Coastguard Worker std::lock_guard<std::mutex> lock(kBenchmarkLock);
184*f40fafd4SAndroid Build Coastguard Worker auto wl = android::wakelock::WakeLock::tryGet(kWakeLock);
185*f40fafd4SAndroid Build Coastguard Worker if (!wl.has_value()) {
186*f40fafd4SAndroid Build Coastguard Worker return;
187*f40fafd4SAndroid Build Coastguard Worker }
188*f40fafd4SAndroid Build Coastguard Worker
189*f40fafd4SAndroid Build Coastguard Worker PerformanceBoost boost;
190*f40fafd4SAndroid Build Coastguard Worker android::os::PersistableBundle extras;
191*f40fafd4SAndroid Build Coastguard Worker
192*f40fafd4SAndroid Build Coastguard Worker status_t res = benchmarkInternal(path, listener, &extras);
193*f40fafd4SAndroid Build Coastguard Worker if (listener) {
194*f40fafd4SAndroid Build Coastguard Worker listener->onFinished(res, extras);
195*f40fafd4SAndroid Build Coastguard Worker }
196*f40fafd4SAndroid Build Coastguard Worker }
197*f40fafd4SAndroid Build Coastguard Worker
198*f40fafd4SAndroid Build Coastguard Worker } // namespace vold
199*f40fafd4SAndroid Build Coastguard Worker } // namespace android
200