xref: /aosp_15_r20/system/core/init/bootchart.cpp (revision 00c7fec1bb09f3284aad6a6f96d2f63dfc3650ad)
1*00c7fec1SAndroid Build Coastguard Worker /*
2*00c7fec1SAndroid Build Coastguard Worker  * Copyright (C) 2008 The Android Open Source Project
3*00c7fec1SAndroid Build Coastguard Worker  *
4*00c7fec1SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*00c7fec1SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*00c7fec1SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*00c7fec1SAndroid Build Coastguard Worker  *
8*00c7fec1SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*00c7fec1SAndroid Build Coastguard Worker  *
10*00c7fec1SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*00c7fec1SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*00c7fec1SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*00c7fec1SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*00c7fec1SAndroid Build Coastguard Worker  * limitations under the License.
15*00c7fec1SAndroid Build Coastguard Worker  */
16*00c7fec1SAndroid Build Coastguard Worker 
17*00c7fec1SAndroid Build Coastguard Worker #include "bootchart.h"
18*00c7fec1SAndroid Build Coastguard Worker 
19*00c7fec1SAndroid Build Coastguard Worker #include <dirent.h>
20*00c7fec1SAndroid Build Coastguard Worker #include <fcntl.h>
21*00c7fec1SAndroid Build Coastguard Worker #include <stdio.h>
22*00c7fec1SAndroid Build Coastguard Worker #include <stdlib.h>
23*00c7fec1SAndroid Build Coastguard Worker #include <string.h>
24*00c7fec1SAndroid Build Coastguard Worker #include <sys/stat.h>
25*00c7fec1SAndroid Build Coastguard Worker #include <sys/utsname.h>
26*00c7fec1SAndroid Build Coastguard Worker #include <time.h>
27*00c7fec1SAndroid Build Coastguard Worker #include <unistd.h>
28*00c7fec1SAndroid Build Coastguard Worker 
29*00c7fec1SAndroid Build Coastguard Worker #include <chrono>
30*00c7fec1SAndroid Build Coastguard Worker #include <condition_variable>
31*00c7fec1SAndroid Build Coastguard Worker #include <memory>
32*00c7fec1SAndroid Build Coastguard Worker #include <mutex>
33*00c7fec1SAndroid Build Coastguard Worker #include <thread>
34*00c7fec1SAndroid Build Coastguard Worker 
35*00c7fec1SAndroid Build Coastguard Worker #include <android-base/chrono_utils.h>
36*00c7fec1SAndroid Build Coastguard Worker #include <android-base/file.h>
37*00c7fec1SAndroid Build Coastguard Worker #include <android-base/logging.h>
38*00c7fec1SAndroid Build Coastguard Worker #include <android-base/properties.h>
39*00c7fec1SAndroid Build Coastguard Worker #include <android-base/stringprintf.h>
40*00c7fec1SAndroid Build Coastguard Worker 
41*00c7fec1SAndroid Build Coastguard Worker using android::base::StringPrintf;
42*00c7fec1SAndroid Build Coastguard Worker using android::base::boot_clock;
43*00c7fec1SAndroid Build Coastguard Worker using namespace std::chrono_literals;
44*00c7fec1SAndroid Build Coastguard Worker 
45*00c7fec1SAndroid Build Coastguard Worker namespace android {
46*00c7fec1SAndroid Build Coastguard Worker namespace init {
47*00c7fec1SAndroid Build Coastguard Worker 
48*00c7fec1SAndroid Build Coastguard Worker static std::thread* g_bootcharting_thread;
49*00c7fec1SAndroid Build Coastguard Worker 
50*00c7fec1SAndroid Build Coastguard Worker static std::mutex g_bootcharting_finished_mutex;
51*00c7fec1SAndroid Build Coastguard Worker static std::condition_variable g_bootcharting_finished_cv;
52*00c7fec1SAndroid Build Coastguard Worker static bool g_bootcharting_finished;
53*00c7fec1SAndroid Build Coastguard Worker 
get_uptime_jiffies()54*00c7fec1SAndroid Build Coastguard Worker static long long get_uptime_jiffies() {
55*00c7fec1SAndroid Build Coastguard Worker     constexpr int64_t kNanosecondsPerJiffy = 10000000;
56*00c7fec1SAndroid Build Coastguard Worker     boot_clock::time_point uptime = boot_clock::now();
57*00c7fec1SAndroid Build Coastguard Worker     return uptime.time_since_epoch().count() / kNanosecondsPerJiffy;
58*00c7fec1SAndroid Build Coastguard Worker }
59*00c7fec1SAndroid Build Coastguard Worker 
fopen_unique(const char * filename,const char * mode)60*00c7fec1SAndroid Build Coastguard Worker static std::unique_ptr<FILE, decltype(&fclose)> fopen_unique(const char* filename,
61*00c7fec1SAndroid Build Coastguard Worker                                                              const char* mode) {
62*00c7fec1SAndroid Build Coastguard Worker   std::unique_ptr<FILE, decltype(&fclose)> result(fopen(filename, mode), fclose);
63*00c7fec1SAndroid Build Coastguard Worker   if (!result) PLOG(ERROR) << "bootchart: failed to open " << filename;
64*00c7fec1SAndroid Build Coastguard Worker   return result;
65*00c7fec1SAndroid Build Coastguard Worker }
66*00c7fec1SAndroid Build Coastguard Worker 
log_header()67*00c7fec1SAndroid Build Coastguard Worker static void log_header() {
68*00c7fec1SAndroid Build Coastguard Worker   char date[32];
69*00c7fec1SAndroid Build Coastguard Worker   time_t now_t = time(NULL);
70*00c7fec1SAndroid Build Coastguard Worker   struct tm now = *localtime(&now_t);
71*00c7fec1SAndroid Build Coastguard Worker   strftime(date, sizeof(date), "%F %T", &now);
72*00c7fec1SAndroid Build Coastguard Worker 
73*00c7fec1SAndroid Build Coastguard Worker   utsname uts;
74*00c7fec1SAndroid Build Coastguard Worker   if (uname(&uts) == -1) return;
75*00c7fec1SAndroid Build Coastguard Worker 
76*00c7fec1SAndroid Build Coastguard Worker   std::string fingerprint = android::base::GetProperty("ro.build.fingerprint", "");
77*00c7fec1SAndroid Build Coastguard Worker   if (fingerprint.empty()) return;
78*00c7fec1SAndroid Build Coastguard Worker 
79*00c7fec1SAndroid Build Coastguard Worker   std::string kernel_cmdline;
80*00c7fec1SAndroid Build Coastguard Worker   android::base::ReadFileToString("/proc/cmdline", &kernel_cmdline);
81*00c7fec1SAndroid Build Coastguard Worker 
82*00c7fec1SAndroid Build Coastguard Worker   auto fp = fopen_unique("/data/bootchart/header", "we");
83*00c7fec1SAndroid Build Coastguard Worker   if (!fp) return;
84*00c7fec1SAndroid Build Coastguard Worker   fprintf(&*fp, "version = Android init 0.8\n");
85*00c7fec1SAndroid Build Coastguard Worker   fprintf(&*fp, "title = Boot chart for Android (%s)\n", date);
86*00c7fec1SAndroid Build Coastguard Worker   fprintf(&*fp, "system.uname = %s %s %s %s\n", uts.sysname, uts.release, uts.version, uts.machine);
87*00c7fec1SAndroid Build Coastguard Worker   fprintf(&*fp, "system.release = %s\n", fingerprint.c_str());
88*00c7fec1SAndroid Build Coastguard Worker   // TODO: use /proc/cpuinfo "model name" line for x86, "Processor" line for arm.
89*00c7fec1SAndroid Build Coastguard Worker   fprintf(&*fp, "system.cpu = %s\n", uts.machine);
90*00c7fec1SAndroid Build Coastguard Worker   fprintf(&*fp, "system.kernel.options = %s\n", kernel_cmdline.c_str());
91*00c7fec1SAndroid Build Coastguard Worker }
92*00c7fec1SAndroid Build Coastguard Worker 
log_uptime(FILE * log)93*00c7fec1SAndroid Build Coastguard Worker static void log_uptime(FILE* log) {
94*00c7fec1SAndroid Build Coastguard Worker   fprintf(log, "%lld\n", get_uptime_jiffies());
95*00c7fec1SAndroid Build Coastguard Worker }
96*00c7fec1SAndroid Build Coastguard Worker 
log_file(FILE * log,const char * procfile)97*00c7fec1SAndroid Build Coastguard Worker static void log_file(FILE* log, const char* procfile) {
98*00c7fec1SAndroid Build Coastguard Worker   log_uptime(log);
99*00c7fec1SAndroid Build Coastguard Worker 
100*00c7fec1SAndroid Build Coastguard Worker   std::string content;
101*00c7fec1SAndroid Build Coastguard Worker   if (android::base::ReadFileToString(procfile, &content)) {
102*00c7fec1SAndroid Build Coastguard Worker     fprintf(log, "%s\n", content.c_str());
103*00c7fec1SAndroid Build Coastguard Worker   }
104*00c7fec1SAndroid Build Coastguard Worker }
105*00c7fec1SAndroid Build Coastguard Worker 
log_processes(FILE * log)106*00c7fec1SAndroid Build Coastguard Worker static void log_processes(FILE* log) {
107*00c7fec1SAndroid Build Coastguard Worker   log_uptime(log);
108*00c7fec1SAndroid Build Coastguard Worker 
109*00c7fec1SAndroid Build Coastguard Worker   std::unique_ptr<DIR, int(*)(DIR*)> dir(opendir("/proc"), closedir);
110*00c7fec1SAndroid Build Coastguard Worker   struct dirent* entry;
111*00c7fec1SAndroid Build Coastguard Worker   while ((entry = readdir(dir.get())) != NULL) {
112*00c7fec1SAndroid Build Coastguard Worker     // Only match numeric values.
113*00c7fec1SAndroid Build Coastguard Worker     int pid = atoi(entry->d_name);
114*00c7fec1SAndroid Build Coastguard Worker     if (pid == 0) continue;
115*00c7fec1SAndroid Build Coastguard Worker 
116*00c7fec1SAndroid Build Coastguard Worker     // /proc/<pid>/stat only has truncated task names, so get the full
117*00c7fec1SAndroid Build Coastguard Worker     // name from /proc/<pid>/cmdline.
118*00c7fec1SAndroid Build Coastguard Worker     std::string cmdline;
119*00c7fec1SAndroid Build Coastguard Worker     android::base::ReadFileToString(StringPrintf("/proc/%d/cmdline", pid), &cmdline);
120*00c7fec1SAndroid Build Coastguard Worker     const char* full_name = cmdline.c_str(); // So we stop at the first NUL.
121*00c7fec1SAndroid Build Coastguard Worker 
122*00c7fec1SAndroid Build Coastguard Worker     // Read process stat line.
123*00c7fec1SAndroid Build Coastguard Worker     std::string stat;
124*00c7fec1SAndroid Build Coastguard Worker     if (android::base::ReadFileToString(StringPrintf("/proc/%d/stat", pid), &stat)) {
125*00c7fec1SAndroid Build Coastguard Worker       if (!cmdline.empty()) {
126*00c7fec1SAndroid Build Coastguard Worker         // Substitute the process name with its real name.
127*00c7fec1SAndroid Build Coastguard Worker         size_t open = stat.find('(');
128*00c7fec1SAndroid Build Coastguard Worker         size_t close = stat.find_last_of(')');
129*00c7fec1SAndroid Build Coastguard Worker         if (open != std::string::npos && close != std::string::npos) {
130*00c7fec1SAndroid Build Coastguard Worker           stat.replace(open + 1, close - open - 1, full_name);
131*00c7fec1SAndroid Build Coastguard Worker         }
132*00c7fec1SAndroid Build Coastguard Worker       }
133*00c7fec1SAndroid Build Coastguard Worker       fputs(stat.c_str(), log);
134*00c7fec1SAndroid Build Coastguard Worker     }
135*00c7fec1SAndroid Build Coastguard Worker   }
136*00c7fec1SAndroid Build Coastguard Worker 
137*00c7fec1SAndroid Build Coastguard Worker   fputc('\n', log);
138*00c7fec1SAndroid Build Coastguard Worker }
139*00c7fec1SAndroid Build Coastguard Worker 
bootchart_thread_main()140*00c7fec1SAndroid Build Coastguard Worker static void bootchart_thread_main() {
141*00c7fec1SAndroid Build Coastguard Worker   LOG(INFO) << "Bootcharting started";
142*00c7fec1SAndroid Build Coastguard Worker 
143*00c7fec1SAndroid Build Coastguard Worker   // Unshare the mount namespace of this thread so that the init process itself can switch
144*00c7fec1SAndroid Build Coastguard Worker   // the mount namespace later while this thread is still running.
145*00c7fec1SAndroid Build Coastguard Worker   // Otherwise, setns() call invoked as part of `enter_default_mount_ns` fails with EINVAL.
146*00c7fec1SAndroid Build Coastguard Worker   //
147*00c7fec1SAndroid Build Coastguard Worker   // Note that after unshare()'ing the mount namespace from the main thread, this thread won't
148*00c7fec1SAndroid Build Coastguard Worker   // receive mount/unmount events from the other mount namespace unless the events are happening
149*00c7fec1SAndroid Build Coastguard Worker   // from under a sharable mount.
150*00c7fec1SAndroid Build Coastguard Worker   //
151*00c7fec1SAndroid Build Coastguard Worker   // The bootchart thread is safe to unshare the mount namespace because it only reads from /proc
152*00c7fec1SAndroid Build Coastguard Worker   // and write to /data which are not private mounts.
153*00c7fec1SAndroid Build Coastguard Worker   if (unshare(CLONE_NEWNS) == -1) {
154*00c7fec1SAndroid Build Coastguard Worker       PLOG(ERROR) << "Cannot create mount namespace";
155*00c7fec1SAndroid Build Coastguard Worker       return;
156*00c7fec1SAndroid Build Coastguard Worker   }
157*00c7fec1SAndroid Build Coastguard Worker   // Open log files.
158*00c7fec1SAndroid Build Coastguard Worker   auto stat_log = fopen_unique("/data/bootchart/proc_stat.log", "we");
159*00c7fec1SAndroid Build Coastguard Worker   if (!stat_log) return;
160*00c7fec1SAndroid Build Coastguard Worker   auto proc_log = fopen_unique("/data/bootchart/proc_ps.log", "we");
161*00c7fec1SAndroid Build Coastguard Worker   if (!proc_log) return;
162*00c7fec1SAndroid Build Coastguard Worker   auto disk_log = fopen_unique("/data/bootchart/proc_diskstats.log", "we");
163*00c7fec1SAndroid Build Coastguard Worker   if (!disk_log) return;
164*00c7fec1SAndroid Build Coastguard Worker 
165*00c7fec1SAndroid Build Coastguard Worker   log_header();
166*00c7fec1SAndroid Build Coastguard Worker 
167*00c7fec1SAndroid Build Coastguard Worker   while (true) {
168*00c7fec1SAndroid Build Coastguard Worker     {
169*00c7fec1SAndroid Build Coastguard Worker       std::unique_lock<std::mutex> lock(g_bootcharting_finished_mutex);
170*00c7fec1SAndroid Build Coastguard Worker       g_bootcharting_finished_cv.wait_for(lock, 200ms);
171*00c7fec1SAndroid Build Coastguard Worker       if (g_bootcharting_finished) break;
172*00c7fec1SAndroid Build Coastguard Worker     }
173*00c7fec1SAndroid Build Coastguard Worker 
174*00c7fec1SAndroid Build Coastguard Worker     log_file(&*stat_log, "/proc/stat");
175*00c7fec1SAndroid Build Coastguard Worker     log_file(&*disk_log, "/proc/diskstats");
176*00c7fec1SAndroid Build Coastguard Worker     log_processes(&*proc_log);
177*00c7fec1SAndroid Build Coastguard Worker   }
178*00c7fec1SAndroid Build Coastguard Worker 
179*00c7fec1SAndroid Build Coastguard Worker   LOG(INFO) << "Bootcharting finished";
180*00c7fec1SAndroid Build Coastguard Worker }
181*00c7fec1SAndroid Build Coastguard Worker 
do_bootchart_start()182*00c7fec1SAndroid Build Coastguard Worker static Result<void> do_bootchart_start() {
183*00c7fec1SAndroid Build Coastguard Worker     // We don't care about the content, but we do care that /data/bootchart/enabled actually exists.
184*00c7fec1SAndroid Build Coastguard Worker     std::string start;
185*00c7fec1SAndroid Build Coastguard Worker     if (!android::base::ReadFileToString("/data/bootchart/enabled", &start)) {
186*00c7fec1SAndroid Build Coastguard Worker         LOG(VERBOSE) << "Not bootcharting";
187*00c7fec1SAndroid Build Coastguard Worker         return {};
188*00c7fec1SAndroid Build Coastguard Worker     }
189*00c7fec1SAndroid Build Coastguard Worker 
190*00c7fec1SAndroid Build Coastguard Worker     g_bootcharting_thread = new std::thread(bootchart_thread_main);
191*00c7fec1SAndroid Build Coastguard Worker     return {};
192*00c7fec1SAndroid Build Coastguard Worker }
193*00c7fec1SAndroid Build Coastguard Worker 
do_bootchart_stop()194*00c7fec1SAndroid Build Coastguard Worker static Result<void> do_bootchart_stop() {
195*00c7fec1SAndroid Build Coastguard Worker     if (!g_bootcharting_thread) return {};
196*00c7fec1SAndroid Build Coastguard Worker 
197*00c7fec1SAndroid Build Coastguard Worker     // Tell the worker thread it's time to quit.
198*00c7fec1SAndroid Build Coastguard Worker     {
199*00c7fec1SAndroid Build Coastguard Worker         std::lock_guard<std::mutex> lock(g_bootcharting_finished_mutex);
200*00c7fec1SAndroid Build Coastguard Worker         g_bootcharting_finished = true;
201*00c7fec1SAndroid Build Coastguard Worker         g_bootcharting_finished_cv.notify_one();
202*00c7fec1SAndroid Build Coastguard Worker     }
203*00c7fec1SAndroid Build Coastguard Worker 
204*00c7fec1SAndroid Build Coastguard Worker     g_bootcharting_thread->join();
205*00c7fec1SAndroid Build Coastguard Worker     delete g_bootcharting_thread;
206*00c7fec1SAndroid Build Coastguard Worker     g_bootcharting_thread = nullptr;
207*00c7fec1SAndroid Build Coastguard Worker     return {};
208*00c7fec1SAndroid Build Coastguard Worker }
209*00c7fec1SAndroid Build Coastguard Worker 
do_bootchart(const BuiltinArguments & args)210*00c7fec1SAndroid Build Coastguard Worker Result<void> do_bootchart(const BuiltinArguments& args) {
211*00c7fec1SAndroid Build Coastguard Worker     if (args[1] == "start") return do_bootchart_start();
212*00c7fec1SAndroid Build Coastguard Worker     return do_bootchart_stop();
213*00c7fec1SAndroid Build Coastguard Worker }
214*00c7fec1SAndroid Build Coastguard Worker 
215*00c7fec1SAndroid Build Coastguard Worker }  // namespace init
216*00c7fec1SAndroid Build Coastguard Worker }  // namespace android
217