xref: /aosp_15_r20/system/vold/Benchmark.cpp (revision f40fafd4c6c2594924d919feffc1a1fd6e3b30f3)
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