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 <stdio.h>
18*6dbdd20aSAndroid Build Coastguard Worker #include <algorithm>
19*6dbdd20aSAndroid Build Coastguard Worker
20*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/base/status.h"
21*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/file_utils.h"
22*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/getopt.h"
23*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/string_utils.h"
24*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/unix_socket.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/base/watchdog.h"
29*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/traced/traced.h"
30*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/tracing/core/tracing_service.h"
31*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/tracing/ipc/service_ipc_host.h"
32*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/tracing/default_socket.h"
33*6dbdd20aSAndroid Build Coastguard Worker #include "src/traced/service/builtin_producer.h"
34*6dbdd20aSAndroid Build Coastguard Worker
35*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
36*6dbdd20aSAndroid Build Coastguard Worker #include <sys/system_properties.h>
37*6dbdd20aSAndroid Build Coastguard Worker #endif
38*6dbdd20aSAndroid Build Coastguard Worker
39*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_ZLIB)
40*6dbdd20aSAndroid Build Coastguard Worker #include "src/tracing/service/zlib_compressor.h"
41*6dbdd20aSAndroid Build Coastguard Worker #endif
42*6dbdd20aSAndroid Build Coastguard Worker
43*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
44*6dbdd20aSAndroid Build Coastguard Worker namespace {
PrintUsage(const char * prog_name)45*6dbdd20aSAndroid Build Coastguard Worker void PrintUsage(const char* prog_name) {
46*6dbdd20aSAndroid Build Coastguard Worker fprintf(stderr, R"(
47*6dbdd20aSAndroid Build Coastguard Worker Usage: %s [option] ...
48*6dbdd20aSAndroid Build Coastguard Worker Options and arguments
49*6dbdd20aSAndroid Build Coastguard Worker --background : Exits immediately and continues running in the background
50*6dbdd20aSAndroid Build Coastguard Worker --version : print the version number and exit.
51*6dbdd20aSAndroid Build Coastguard Worker --set-socket-permissions <permissions> : sets group ownership and permission
52*6dbdd20aSAndroid Build Coastguard Worker mode bits of the producer and consumer sockets.
53*6dbdd20aSAndroid Build Coastguard Worker <permissions> format: <prod_group>:<prod_mode>:<cons_group>:<cons_mode>,
54*6dbdd20aSAndroid Build Coastguard Worker where <prod_group> is the group name for chgrp the producer socket,
55*6dbdd20aSAndroid Build Coastguard Worker <prod_mode> is the mode bits (e.g. 0660) for chmod the produce socket,
56*6dbdd20aSAndroid Build Coastguard Worker <cons_group> is the group name for chgrp the consumer socket, and
57*6dbdd20aSAndroid Build Coastguard Worker <cons_mode> is the mode bits (e.g. 0660) for chmod the consumer socket.
58*6dbdd20aSAndroid Build Coastguard Worker --enable-relay-endpoint : enables the relay endpoint on producer socket(s)
59*6dbdd20aSAndroid Build Coastguard Worker for traced_relay to communicate with traced in a multiple-machine
60*6dbdd20aSAndroid Build Coastguard Worker tracing session.
61*6dbdd20aSAndroid Build Coastguard Worker
62*6dbdd20aSAndroid Build Coastguard Worker Example:
63*6dbdd20aSAndroid Build Coastguard Worker %s --set-socket-permissions traced-producer:0660:traced-consumer:0660
64*6dbdd20aSAndroid Build Coastguard Worker starts the service and sets the group ownership of the producer and consumer
65*6dbdd20aSAndroid Build Coastguard Worker sockets to "traced-producer" and "traced-consumer", respectively. Both
66*6dbdd20aSAndroid Build Coastguard Worker producer and consumer sockets are chmod with 0660 (rw-rw----) mode bits.
67*6dbdd20aSAndroid Build Coastguard Worker )",
68*6dbdd20aSAndroid Build Coastguard Worker prog_name, prog_name);
69*6dbdd20aSAndroid Build Coastguard Worker }
70*6dbdd20aSAndroid Build Coastguard Worker } // namespace
71*6dbdd20aSAndroid Build Coastguard Worker
ServiceMain(int argc,char ** argv)72*6dbdd20aSAndroid Build Coastguard Worker int PERFETTO_EXPORT_ENTRYPOINT ServiceMain(int argc, char** argv) {
73*6dbdd20aSAndroid Build Coastguard Worker enum LongOption {
74*6dbdd20aSAndroid Build Coastguard Worker OPT_VERSION = 1000,
75*6dbdd20aSAndroid Build Coastguard Worker OPT_SET_SOCKET_PERMISSIONS = 1001,
76*6dbdd20aSAndroid Build Coastguard Worker OPT_BACKGROUND,
77*6dbdd20aSAndroid Build Coastguard Worker OPT_ENABLE_RELAY_ENDPOINT
78*6dbdd20aSAndroid Build Coastguard Worker };
79*6dbdd20aSAndroid Build Coastguard Worker
80*6dbdd20aSAndroid Build Coastguard Worker bool background = false;
81*6dbdd20aSAndroid Build Coastguard Worker bool enable_relay_endpoint = false;
82*6dbdd20aSAndroid Build Coastguard Worker
83*6dbdd20aSAndroid Build Coastguard Worker static const option long_options[] = {
84*6dbdd20aSAndroid Build Coastguard Worker {"background", no_argument, nullptr, OPT_BACKGROUND},
85*6dbdd20aSAndroid Build Coastguard Worker {"version", no_argument, nullptr, OPT_VERSION},
86*6dbdd20aSAndroid Build Coastguard Worker {"set-socket-permissions", required_argument, nullptr,
87*6dbdd20aSAndroid Build Coastguard Worker OPT_SET_SOCKET_PERMISSIONS},
88*6dbdd20aSAndroid Build Coastguard Worker {"enable-relay-endpoint", no_argument, nullptr,
89*6dbdd20aSAndroid Build Coastguard Worker OPT_ENABLE_RELAY_ENDPOINT},
90*6dbdd20aSAndroid Build Coastguard Worker {nullptr, 0, nullptr, 0}};
91*6dbdd20aSAndroid Build Coastguard Worker
92*6dbdd20aSAndroid Build Coastguard Worker std::string producer_socket_group, consumer_socket_group,
93*6dbdd20aSAndroid Build Coastguard Worker producer_socket_mode, consumer_socket_mode;
94*6dbdd20aSAndroid Build Coastguard Worker
95*6dbdd20aSAndroid Build Coastguard Worker for (;;) {
96*6dbdd20aSAndroid Build Coastguard Worker int option = getopt_long(argc, argv, "", long_options, nullptr);
97*6dbdd20aSAndroid Build Coastguard Worker if (option == -1)
98*6dbdd20aSAndroid Build Coastguard Worker break;
99*6dbdd20aSAndroid Build Coastguard Worker switch (option) {
100*6dbdd20aSAndroid Build Coastguard Worker case OPT_BACKGROUND:
101*6dbdd20aSAndroid Build Coastguard Worker background = true;
102*6dbdd20aSAndroid Build Coastguard Worker break;
103*6dbdd20aSAndroid Build Coastguard Worker case OPT_VERSION:
104*6dbdd20aSAndroid Build Coastguard Worker printf("%s\n", base::GetVersionString());
105*6dbdd20aSAndroid Build Coastguard Worker return 0;
106*6dbdd20aSAndroid Build Coastguard Worker case OPT_SET_SOCKET_PERMISSIONS: {
107*6dbdd20aSAndroid Build Coastguard Worker // Check that the socket permission argument is well formed.
108*6dbdd20aSAndroid Build Coastguard Worker auto parts = base::SplitString(std::string(optarg), ":");
109*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_CHECK(parts.size() == 4);
110*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_CHECK(
111*6dbdd20aSAndroid Build Coastguard Worker std::all_of(parts.cbegin(), parts.cend(),
112*6dbdd20aSAndroid Build Coastguard Worker [](const std::string& part) { return !part.empty(); }));
113*6dbdd20aSAndroid Build Coastguard Worker producer_socket_group = parts[0];
114*6dbdd20aSAndroid Build Coastguard Worker producer_socket_mode = parts[1];
115*6dbdd20aSAndroid Build Coastguard Worker consumer_socket_group = parts[2];
116*6dbdd20aSAndroid Build Coastguard Worker consumer_socket_mode = parts[3];
117*6dbdd20aSAndroid Build Coastguard Worker break;
118*6dbdd20aSAndroid Build Coastguard Worker }
119*6dbdd20aSAndroid Build Coastguard Worker case OPT_ENABLE_RELAY_ENDPOINT:
120*6dbdd20aSAndroid Build Coastguard Worker enable_relay_endpoint = true;
121*6dbdd20aSAndroid Build Coastguard Worker break;
122*6dbdd20aSAndroid Build Coastguard Worker default:
123*6dbdd20aSAndroid Build Coastguard Worker PrintUsage(argv[0]);
124*6dbdd20aSAndroid Build Coastguard Worker return 1;
125*6dbdd20aSAndroid Build Coastguard Worker }
126*6dbdd20aSAndroid Build Coastguard Worker }
127*6dbdd20aSAndroid Build Coastguard Worker
128*6dbdd20aSAndroid Build Coastguard Worker if (background) {
129*6dbdd20aSAndroid Build Coastguard Worker base::Daemonize([] { return 0; });
130*6dbdd20aSAndroid Build Coastguard Worker }
131*6dbdd20aSAndroid Build Coastguard Worker
132*6dbdd20aSAndroid Build Coastguard Worker base::UnixTaskRunner task_runner;
133*6dbdd20aSAndroid Build Coastguard Worker std::unique_ptr<ServiceIPCHost> svc;
134*6dbdd20aSAndroid Build Coastguard Worker TracingService::InitOpts init_opts = {};
135*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_ZLIB)
136*6dbdd20aSAndroid Build Coastguard Worker init_opts.compressor_fn = &ZlibCompressFn;
137*6dbdd20aSAndroid Build Coastguard Worker #endif
138*6dbdd20aSAndroid Build Coastguard Worker if (enable_relay_endpoint)
139*6dbdd20aSAndroid Build Coastguard Worker init_opts.enable_relay_endpoint = true;
140*6dbdd20aSAndroid Build Coastguard Worker svc = ServiceIPCHost::CreateInstance(&task_runner, init_opts);
141*6dbdd20aSAndroid Build Coastguard Worker
142*6dbdd20aSAndroid Build Coastguard Worker // When built as part of the Android tree, the two socket are created and
143*6dbdd20aSAndroid Build Coastguard Worker // bound by init and their fd number is passed in two env variables.
144*6dbdd20aSAndroid Build Coastguard Worker // See libcutils' android_get_control_socket().
145*6dbdd20aSAndroid Build Coastguard Worker const char* env_prod = getenv("ANDROID_SOCKET_traced_producer");
146*6dbdd20aSAndroid Build Coastguard Worker const char* env_cons = getenv("ANDROID_SOCKET_traced_consumer");
147*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_CHECK((!env_prod && !env_cons) || (env_prod && env_cons));
148*6dbdd20aSAndroid Build Coastguard Worker bool started;
149*6dbdd20aSAndroid Build Coastguard Worker if (env_prod) {
150*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_OS_WIN)
151*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_CHECK(false);
152*6dbdd20aSAndroid Build Coastguard Worker #else
153*6dbdd20aSAndroid Build Coastguard Worker base::ScopedFile producer_fd(atoi(env_prod));
154*6dbdd20aSAndroid Build Coastguard Worker base::ScopedFile consumer_fd(atoi(env_cons));
155*6dbdd20aSAndroid Build Coastguard Worker started = svc->Start(std::move(producer_fd), std::move(consumer_fd));
156*6dbdd20aSAndroid Build Coastguard Worker #endif
157*6dbdd20aSAndroid Build Coastguard Worker } else {
158*6dbdd20aSAndroid Build Coastguard Worker auto producer_sockets = TokenizeProducerSockets(GetProducerSocket());
159*6dbdd20aSAndroid Build Coastguard Worker for (const auto& producer_socket : producer_sockets) {
160*6dbdd20aSAndroid Build Coastguard Worker remove(producer_socket.c_str());
161*6dbdd20aSAndroid Build Coastguard Worker }
162*6dbdd20aSAndroid Build Coastguard Worker remove(GetConsumerSocket());
163*6dbdd20aSAndroid Build Coastguard Worker started = svc->Start(producer_sockets, GetConsumerSocket());
164*6dbdd20aSAndroid Build Coastguard Worker
165*6dbdd20aSAndroid Build Coastguard Worker if (!producer_socket_group.empty()) {
166*6dbdd20aSAndroid Build Coastguard Worker auto status = base::OkStatus();
167*6dbdd20aSAndroid Build Coastguard Worker for (const auto& producer_socket : producer_sockets) {
168*6dbdd20aSAndroid Build Coastguard Worker if (base::GetSockFamily(producer_socket.c_str()) !=
169*6dbdd20aSAndroid Build Coastguard Worker base::SockFamily::kUnix) {
170*6dbdd20aSAndroid Build Coastguard Worker // Socket permissions is only available to unix sockets.
171*6dbdd20aSAndroid Build Coastguard Worker continue;
172*6dbdd20aSAndroid Build Coastguard Worker }
173*6dbdd20aSAndroid Build Coastguard Worker status = base::SetFilePermissions(
174*6dbdd20aSAndroid Build Coastguard Worker producer_socket, producer_socket_group, producer_socket_mode);
175*6dbdd20aSAndroid Build Coastguard Worker if (!status.ok()) {
176*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_ELOG("%s", status.c_message());
177*6dbdd20aSAndroid Build Coastguard Worker return 1;
178*6dbdd20aSAndroid Build Coastguard Worker }
179*6dbdd20aSAndroid Build Coastguard Worker }
180*6dbdd20aSAndroid Build Coastguard Worker status = base::SetFilePermissions(
181*6dbdd20aSAndroid Build Coastguard Worker GetConsumerSocket(), consumer_socket_group, consumer_socket_mode);
182*6dbdd20aSAndroid Build Coastguard Worker if (!status.ok()) {
183*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_ELOG("%s", status.c_message());
184*6dbdd20aSAndroid Build Coastguard Worker return 1;
185*6dbdd20aSAndroid Build Coastguard Worker }
186*6dbdd20aSAndroid Build Coastguard Worker }
187*6dbdd20aSAndroid Build Coastguard Worker }
188*6dbdd20aSAndroid Build Coastguard Worker
189*6dbdd20aSAndroid Build Coastguard Worker if (!started) {
190*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_ELOG("Failed to start the traced service");
191*6dbdd20aSAndroid Build Coastguard Worker return 1;
192*6dbdd20aSAndroid Build Coastguard Worker }
193*6dbdd20aSAndroid Build Coastguard Worker
194*6dbdd20aSAndroid Build Coastguard Worker // Advertise builtin producers only on in-tree builds. These producers serve
195*6dbdd20aSAndroid Build Coastguard Worker // only to dynamically start heapprofd and other services via sysprops, but
196*6dbdd20aSAndroid Build Coastguard Worker // that can only ever happen in in-tree builds.
197*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_ANDROID_BUILD)
198*6dbdd20aSAndroid Build Coastguard Worker BuiltinProducer builtin_producer(&task_runner, /*lazy_stop_delay_ms=*/30000);
199*6dbdd20aSAndroid Build Coastguard Worker builtin_producer.ConnectInProcess(svc->service());
200*6dbdd20aSAndroid Build Coastguard Worker #endif
201*6dbdd20aSAndroid Build Coastguard Worker
202*6dbdd20aSAndroid Build Coastguard Worker // Set the CPU limit and start the watchdog running. The memory limit will
203*6dbdd20aSAndroid Build Coastguard Worker // be set inside the service code as it relies on the size of buffers.
204*6dbdd20aSAndroid Build Coastguard Worker // The CPU limit is the generic one defined in watchdog.h.
205*6dbdd20aSAndroid Build Coastguard Worker base::Watchdog* watchdog = base::Watchdog::GetInstance();
206*6dbdd20aSAndroid Build Coastguard Worker watchdog->SetCpuLimit(base::kWatchdogDefaultCpuLimit,
207*6dbdd20aSAndroid Build Coastguard Worker base::kWatchdogDefaultCpuWindow);
208*6dbdd20aSAndroid Build Coastguard Worker watchdog->Start();
209*6dbdd20aSAndroid Build Coastguard Worker
210*6dbdd20aSAndroid Build Coastguard Worker // If the TRACED_NOTIFY_FD env var is set, write 1 and close the FD. This is
211*6dbdd20aSAndroid Build Coastguard Worker // so tools can synchronize with the point where the IPC socket has been
212*6dbdd20aSAndroid Build Coastguard Worker // opened, without having to poll. This is used for //src/tracebox.
213*6dbdd20aSAndroid Build Coastguard Worker const char* env_notif = getenv("TRACED_NOTIFY_FD");
214*6dbdd20aSAndroid Build Coastguard Worker if (env_notif) {
215*6dbdd20aSAndroid Build Coastguard Worker int notif_fd = atoi(env_notif);
216*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_CHECK(base::WriteAll(notif_fd, "1", 1) == 1);
217*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_CHECK(base::CloseFile(notif_fd) == 0);
218*6dbdd20aSAndroid Build Coastguard Worker }
219*6dbdd20aSAndroid Build Coastguard Worker
220*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_ANDROID_BUILD) && \
221*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
222*6dbdd20aSAndroid Build Coastguard Worker // Notify init (perfetto.rc) that traced has been started. Used only by
223*6dbdd20aSAndroid Build Coastguard Worker // the perfetto_trace_on_boot init service.
224*6dbdd20aSAndroid Build Coastguard Worker // This property can be set only in in-tree builds. shell.te doesn't have
225*6dbdd20aSAndroid Build Coastguard Worker // SELinux permissions to set sys.trace.* properties.
226*6dbdd20aSAndroid Build Coastguard Worker if (__system_property_set("sys.trace.traced_started", "1") != 0) {
227*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_PLOG("Failed to set property sys.trace.traced_started");
228*6dbdd20aSAndroid Build Coastguard Worker }
229*6dbdd20aSAndroid Build Coastguard Worker #endif
230*6dbdd20aSAndroid Build Coastguard Worker
231*6dbdd20aSAndroid Build Coastguard Worker PERFETTO_ILOG("Started traced, listening on %s %s", GetProducerSocket(),
232*6dbdd20aSAndroid Build Coastguard Worker GetConsumerSocket());
233*6dbdd20aSAndroid Build Coastguard Worker task_runner.Run();
234*6dbdd20aSAndroid Build Coastguard Worker return 0;
235*6dbdd20aSAndroid Build Coastguard Worker }
236*6dbdd20aSAndroid Build Coastguard Worker
237*6dbdd20aSAndroid Build Coastguard Worker } // namespace perfetto
238