xref: /aosp_15_r20/external/perfetto/src/traced/probes/probes.cc (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
1*6dbdd20aSAndroid Build Coastguard Worker /*
2*6dbdd20aSAndroid Build Coastguard Worker  * Copyright (C) 2017 The Android Open Source Project
3*6dbdd20aSAndroid Build Coastguard Worker  *
4*6dbdd20aSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*6dbdd20aSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*6dbdd20aSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*6dbdd20aSAndroid Build Coastguard Worker  *
8*6dbdd20aSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*6dbdd20aSAndroid Build Coastguard Worker  *
10*6dbdd20aSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*6dbdd20aSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*6dbdd20aSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*6dbdd20aSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*6dbdd20aSAndroid Build Coastguard Worker  * limitations under the License.
15*6dbdd20aSAndroid Build Coastguard Worker  */
16*6dbdd20aSAndroid Build Coastguard Worker 
17*6dbdd20aSAndroid Build Coastguard Worker #include <fcntl.h>
18*6dbdd20aSAndroid Build Coastguard Worker #include <stdio.h>
19*6dbdd20aSAndroid Build Coastguard Worker #include <stdlib.h>
20*6dbdd20aSAndroid Build Coastguard Worker #include <unistd.h>
21*6dbdd20aSAndroid Build Coastguard Worker 
22*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/base/logging.h"
23*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/file_utils.h"
24*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/getopt.h"
25*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/unix_task_runner.h"
26*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/utils.h"
27*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/version.h"
28*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/traced/traced.h"
29*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/tracing/default_socket.h"
30*6dbdd20aSAndroid Build Coastguard Worker 
31*6dbdd20aSAndroid Build Coastguard Worker #include "src/traced/probes/ftrace/ftrace_procfs.h"
32*6dbdd20aSAndroid Build Coastguard Worker #include "src/traced/probes/kmem_activity_trigger.h"
33*6dbdd20aSAndroid Build Coastguard Worker #include "src/traced/probes/probes_producer.h"
34*6dbdd20aSAndroid Build Coastguard Worker 
35*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
36*6dbdd20aSAndroid Build Coastguard Worker 
ProbesMain(int argc,char ** argv)37*6dbdd20aSAndroid Build Coastguard Worker int PERFETTO_EXPORT_ENTRYPOINT ProbesMain(int argc, char** argv) {
38*6dbdd20aSAndroid Build Coastguard Worker   enum LongOption {
39*6dbdd20aSAndroid Build Coastguard Worker     OPT_CLEANUP_AFTER_CRASH = 1000,
40*6dbdd20aSAndroid Build Coastguard Worker     OPT_VERSION,
41*6dbdd20aSAndroid Build Coastguard Worker     OPT_BACKGROUND,
42*6dbdd20aSAndroid Build Coastguard Worker     OPT_RESET_FTRACE,
43*6dbdd20aSAndroid Build Coastguard Worker   };
44*6dbdd20aSAndroid Build Coastguard Worker 
45*6dbdd20aSAndroid Build Coastguard Worker   bool background = false;
46*6dbdd20aSAndroid Build Coastguard Worker   bool reset_ftrace = false;
47*6dbdd20aSAndroid Build Coastguard Worker 
48*6dbdd20aSAndroid Build Coastguard Worker   static const option long_options[] = {
49*6dbdd20aSAndroid Build Coastguard Worker       {"background", no_argument, nullptr, OPT_BACKGROUND},
50*6dbdd20aSAndroid Build Coastguard Worker       {"cleanup-after-crash", no_argument, nullptr, OPT_CLEANUP_AFTER_CRASH},
51*6dbdd20aSAndroid Build Coastguard Worker       {"reset-ftrace", no_argument, nullptr, OPT_RESET_FTRACE},
52*6dbdd20aSAndroid Build Coastguard Worker       {"version", no_argument, nullptr, OPT_VERSION},
53*6dbdd20aSAndroid Build Coastguard Worker       {nullptr, 0, nullptr, 0}};
54*6dbdd20aSAndroid Build Coastguard Worker 
55*6dbdd20aSAndroid Build Coastguard Worker   for (;;) {
56*6dbdd20aSAndroid Build Coastguard Worker     int option = getopt_long(argc, argv, "", long_options, nullptr);
57*6dbdd20aSAndroid Build Coastguard Worker     if (option == -1)
58*6dbdd20aSAndroid Build Coastguard Worker       break;
59*6dbdd20aSAndroid Build Coastguard Worker     switch (option) {
60*6dbdd20aSAndroid Build Coastguard Worker       case OPT_BACKGROUND:
61*6dbdd20aSAndroid Build Coastguard Worker         background = true;
62*6dbdd20aSAndroid Build Coastguard Worker         break;
63*6dbdd20aSAndroid Build Coastguard Worker       case OPT_CLEANUP_AFTER_CRASH:
64*6dbdd20aSAndroid Build Coastguard Worker         // Used by perfetto.rc in Android.
65*6dbdd20aSAndroid Build Coastguard Worker         PERFETTO_LOG("Hard resetting ftrace state.");
66*6dbdd20aSAndroid Build Coastguard Worker         HardResetFtraceState();
67*6dbdd20aSAndroid Build Coastguard Worker         return 0;
68*6dbdd20aSAndroid Build Coastguard Worker       case OPT_RESET_FTRACE:
69*6dbdd20aSAndroid Build Coastguard Worker         // This is like --cleanup-after-crash but doesn't quit.
70*6dbdd20aSAndroid Build Coastguard Worker         reset_ftrace = true;
71*6dbdd20aSAndroid Build Coastguard Worker         break;
72*6dbdd20aSAndroid Build Coastguard Worker       case OPT_VERSION:
73*6dbdd20aSAndroid Build Coastguard Worker         printf("%s\n", base::GetVersionString());
74*6dbdd20aSAndroid Build Coastguard Worker         return 0;
75*6dbdd20aSAndroid Build Coastguard Worker       default:
76*6dbdd20aSAndroid Build Coastguard Worker         fprintf(
77*6dbdd20aSAndroid Build Coastguard Worker             stderr,
78*6dbdd20aSAndroid Build Coastguard Worker             "Usage: %s [--background] [--reset-ftrace] [--cleanup-after-crash] "
79*6dbdd20aSAndroid Build Coastguard Worker             "[--version]\n",
80*6dbdd20aSAndroid Build Coastguard Worker             argv[0]);
81*6dbdd20aSAndroid Build Coastguard Worker         return 1;
82*6dbdd20aSAndroid Build Coastguard Worker     }
83*6dbdd20aSAndroid Build Coastguard Worker   }
84*6dbdd20aSAndroid Build Coastguard Worker 
85*6dbdd20aSAndroid Build Coastguard Worker   if (reset_ftrace && !HardResetFtraceState()) {
86*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_ELOG(
87*6dbdd20aSAndroid Build Coastguard Worker         "Failed to reset ftrace. Either run this as root or run "
88*6dbdd20aSAndroid Build Coastguard Worker         "`sudo chown -R $USER /sys/kernel/tracing`");
89*6dbdd20aSAndroid Build Coastguard Worker   }
90*6dbdd20aSAndroid Build Coastguard Worker 
91*6dbdd20aSAndroid Build Coastguard Worker   if (background) {
92*6dbdd20aSAndroid Build Coastguard Worker     base::Daemonize([] { return 0; });
93*6dbdd20aSAndroid Build Coastguard Worker   }
94*6dbdd20aSAndroid Build Coastguard Worker 
95*6dbdd20aSAndroid Build Coastguard Worker   base::Watchdog* watchdog = base::Watchdog::GetInstance();
96*6dbdd20aSAndroid Build Coastguard Worker   // The memory watchdog will be updated soon after connect, once the shmem
97*6dbdd20aSAndroid Build Coastguard Worker   // buffer size is known, in ProbesProducer::OnTracingSetup().
98*6dbdd20aSAndroid Build Coastguard Worker   watchdog->SetMemoryLimit(base::kWatchdogDefaultMemorySlack,
99*6dbdd20aSAndroid Build Coastguard Worker                            base::kWatchdogDefaultMemoryWindow);
100*6dbdd20aSAndroid Build Coastguard Worker   watchdog->SetCpuLimit(base::kWatchdogDefaultCpuLimit,
101*6dbdd20aSAndroid Build Coastguard Worker                         base::kWatchdogDefaultCpuWindow);
102*6dbdd20aSAndroid Build Coastguard Worker   watchdog->Start();
103*6dbdd20aSAndroid Build Coastguard Worker 
104*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_LOG("Starting %s service", argv[0]);
105*6dbdd20aSAndroid Build Coastguard Worker 
106*6dbdd20aSAndroid Build Coastguard Worker   // This environment variable is set by Android's init to a fd to /dev/kmsg
107*6dbdd20aSAndroid Build Coastguard Worker   // opened for writing (see perfetto.rc). We cannot open the file directly
108*6dbdd20aSAndroid Build Coastguard Worker   // due to permissions.
109*6dbdd20aSAndroid Build Coastguard Worker   const char* env = getenv("ANDROID_FILE__dev_kmsg");
110*6dbdd20aSAndroid Build Coastguard Worker   if (env) {
111*6dbdd20aSAndroid Build Coastguard Worker     FtraceProcfs::g_kmesg_fd = atoi(env);
112*6dbdd20aSAndroid Build Coastguard Worker     // The file descriptor passed by init doesn't have the FD_CLOEXEC bit set.
113*6dbdd20aSAndroid Build Coastguard Worker     // Set it so we don't leak this fd while invoking atrace.
114*6dbdd20aSAndroid Build Coastguard Worker     int res = fcntl(FtraceProcfs::g_kmesg_fd, F_SETFD, FD_CLOEXEC);
115*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DCHECK(res == 0);
116*6dbdd20aSAndroid Build Coastguard Worker   }
117*6dbdd20aSAndroid Build Coastguard Worker 
118*6dbdd20aSAndroid Build Coastguard Worker   base::UnixTaskRunner task_runner;
119*6dbdd20aSAndroid Build Coastguard Worker   ProbesProducer producer;
120*6dbdd20aSAndroid Build Coastguard Worker   // If the TRACED_PROBES_NOTIFY_FD env var is set, write 1 and close the FD,
121*6dbdd20aSAndroid Build Coastguard Worker   // when all data sources have been registered. This is used for //src/tracebox
122*6dbdd20aSAndroid Build Coastguard Worker   // --background-wait, to make sure that the data sources are registered before
123*6dbdd20aSAndroid Build Coastguard Worker   // waiting for them to be started.
124*6dbdd20aSAndroid Build Coastguard Worker   const char* env_notif = getenv("TRACED_PROBES_NOTIFY_FD");
125*6dbdd20aSAndroid Build Coastguard Worker   if (env_notif) {
126*6dbdd20aSAndroid Build Coastguard Worker     int notif_fd = atoi(env_notif);
127*6dbdd20aSAndroid Build Coastguard Worker     producer.SetAllDataSourcesRegisteredCb([notif_fd] {
128*6dbdd20aSAndroid Build Coastguard Worker       PERFETTO_CHECK(base::WriteAll(notif_fd, "1", 1) == 1);
129*6dbdd20aSAndroid Build Coastguard Worker       PERFETTO_CHECK(base::CloseFile(notif_fd) == 0);
130*6dbdd20aSAndroid Build Coastguard Worker     });
131*6dbdd20aSAndroid Build Coastguard Worker   }
132*6dbdd20aSAndroid Build Coastguard Worker   producer.ConnectWithRetries(GetProducerSocket(), &task_runner);
133*6dbdd20aSAndroid Build Coastguard Worker 
134*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
135*6dbdd20aSAndroid Build Coastguard Worker   // Start the thread that polls mm_event instance and triggers
136*6dbdd20aSAndroid Build Coastguard Worker   KmemActivityTrigger kmem_activity_trigger;
137*6dbdd20aSAndroid Build Coastguard Worker #endif
138*6dbdd20aSAndroid Build Coastguard Worker 
139*6dbdd20aSAndroid Build Coastguard Worker   task_runner.Run();
140*6dbdd20aSAndroid Build Coastguard Worker   return 0;
141*6dbdd20aSAndroid Build Coastguard Worker }
142*6dbdd20aSAndroid Build Coastguard Worker 
143*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
144