blob: b02b6d5cd94470897c85bbef2abdd45b3e7089c7 [file] [log] [blame]
/*
* Copyright (C) 2017 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include "perfetto/base/logging.h"
#include "perfetto/ext/base/getopt.h"
#include "perfetto/ext/base/unix_task_runner.h"
#include "perfetto/ext/base/utils.h"
#include "perfetto/ext/base/version.h"
#include "perfetto/ext/traced/traced.h"
#include "perfetto/ext/tracing/ipc/default_socket.h"
#include "src/traced/probes/ftrace/ftrace_procfs.h"
#include "src/traced/probes/kmem_activity_trigger.h"
#include "src/traced/probes/probes_producer.h"
namespace perfetto {
int PERFETTO_EXPORT_ENTRYPOINT ProbesMain(int argc, char** argv) {
enum LongOption {
OPT_CLEANUP_AFTER_CRASH = 1000,
OPT_VERSION,
OPT_BACKGROUND,
};
bool background = false;
static const option long_options[] = {
{"background", no_argument, nullptr, OPT_BACKGROUND},
{"cleanup-after-crash", no_argument, nullptr, OPT_CLEANUP_AFTER_CRASH},
{"version", no_argument, nullptr, OPT_VERSION},
{nullptr, 0, nullptr, 0}};
for (;;) {
int option = getopt_long(argc, argv, "", long_options, nullptr);
if (option == -1)
break;
switch (option) {
case OPT_BACKGROUND:
background = true;
break;
case OPT_CLEANUP_AFTER_CRASH:
HardResetFtraceState();
return 0;
case OPT_VERSION:
printf("%s\n", base::GetVersionString());
return 0;
default:
PERFETTO_ELOG("Usage: %s [--background|--cleanup-after-crash|--version]", argv[0]);
return 1;
}
}
if (background) {
base::Daemonize();
}
base::Watchdog* watchdog = base::Watchdog::GetInstance();
// The memory watchdog will be updated soon after connect, once the shmem
// buffer size is known, in ProbesProducer::OnTracingSetup().
watchdog->SetMemoryLimit(base::kWatchdogDefaultMemorySlack,
base::kWatchdogDefaultMemoryWindow);
watchdog->SetCpuLimit(base::kWatchdogDefaultCpuLimit,
base::kWatchdogDefaultCpuWindow);
watchdog->Start();
PERFETTO_LOG("Starting %s service", argv[0]);
// This environment variable is set by Android's init to a fd to /dev/kmsg
// opened for writing (see perfetto.rc). We cannot open the file directly
// due to permissions.
const char* env = getenv("ANDROID_FILE__dev_kmsg");
if (env) {
FtraceProcfs::g_kmesg_fd = atoi(env);
// The file descriptor passed by init doesn't have the FD_CLOEXEC bit set.
// Set it so we don't leak this fd while invoking atrace.
int res = fcntl(FtraceProcfs::g_kmesg_fd, F_SETFD, FD_CLOEXEC);
PERFETTO_DCHECK(res == 0);
}
base::UnixTaskRunner task_runner;
ProbesProducer producer;
producer.ConnectWithRetries(GetProducerSocket(), &task_runner);
#if PERFETTO_BUILDFLAG(PERFETTO_OS_ANDROID)
// Start the thread that polls mm_event instance and triggers
KmemActivityTrigger kmem_activity_trigger;
#endif
task_runner.Run();
return 0;
}
} // namespace perfetto