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