/* * 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 "inplace_sampler_lib.h" #include <inttypes.h> #include <pthread.h> #include <signal.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <sys/syscall.h> #include <sys/ucontext.h> #include <unistd.h> #include <map> #include <memory> #include <queue> #include <set> #include <string> #include <unordered_map> #include <android-base/logging.h> #include <android-base/macros.h> #include <backtrace/Backtrace.h> #define LOG_TAG "InplaceSampler" #include <log/log.h> #include "environment.h" #include "UnixSocket.h" #include "utils.h" #define DEFAULT_SIGNO SIGRTMAX static constexpr int DEFAULT_SAMPLE_FREQ = 4000; static constexpr int CHECK_THREADS_INTERVAL_IN_MS = 200; namespace { struct ThreadInfo { std::string name; }; // SampleManager controls the whole sampling process: // Read commands from simpleperf // Set up timers to send signals for each profiled thread regularly. // Send thread info and map info to simpleperf. class SampleManager { public: SampleManager(std::unique_ptr<UnixSocketConnection> conn) : conn_(std::move(conn)), tid_(gettid()), signo_(DEFAULT_SIGNO), sample_freq_(DEFAULT_SAMPLE_FREQ), sample_period_in_ns_(0), dump_callchain_(false), monitor_all_threads_(true) { } void Run(); private: bool HandleMessage(const UnixSocketMessage& msg); bool ParseStartProfilingMessage(const UnixSocketMessage& msg); bool SendStartProfilingReplyMessage(bool ok); bool StartProfiling(); bool InstallSignalHandler(); bool CheckThreads(); bool CheckThreadNameChange(uint64_t timestamp); bool CheckMapChange(uint64_t timestamp); void SendThreadMapInfo(); void SendFakeSampleRecord(); std::unique_ptr<UnixSocketConnection> conn_; int tid_; int signo_; uint32_t sample_freq_; uint32_t sample_period_in_ns_; bool dump_callchain_; bool monitor_all_threads_; std::set<int> monitor_tid_filter_; std::map<int, ThreadInfo> threads_; std::map<uint64_t, ThreadMmap> maps_; std::queue<std::unique_ptr<char[]>> thread_map_info_q_; IOEventLoop loop_; }; void SampleManager::Run() { auto read_callback = [&](const UnixSocketMessage& msg) { return HandleMessage(msg); }; auto close_callback = [&]() { return loop_.ExitLoop(); }; if (!conn_->PrepareForIO(loop_, read_callback, close_callback)) { return; } loop_.RunLoop(); } bool SampleManager::HandleMessage(const UnixSocketMessage& msg) { if (msg.type == START_PROFILING) { if (!ParseStartProfilingMessage(msg)) { if (!SendStartProfilingReplyMessage(false)) { return false; } return conn_->NoMoreMessage(); } if (!SendStartProfilingReplyMessage(true)) { return false; } return StartProfiling(); } if (msg.type == END_PROFILING) { // Close connection after clearing send buffer. return conn_->NoMoreMessage(); } LOG(ERROR) << "Unexpected msg type: " << msg.type; return false; } bool SampleManager::ParseStartProfilingMessage(const UnixSocketMessage& msg) { char* option = const_cast<char*>(msg.data); while (option != nullptr && *option != '\0') { char* next_option = strchr(option, ' '); if (next_option != nullptr) { *next_option++ = '\0'; } char* equal_op = strchr(option, '='); if (equal_op != nullptr) { char* key = option; *equal_op = '\0'; char* value = equal_op + 1; if (strcmp(key, "freq") == 0) { sample_freq_ = atoi(value); } else if (strcmp(key, "signal") == 0) { signo_ = atoi(value); } else if (strcmp(key, "tids") == 0) { monitor_all_threads_ = false; while (*value != '\0') { int tid = static_cast<int>(strtol(value, &value, 10)); monitor_tid_filter_.insert(tid); if (*value == ',') { ++value; } } } else if (strcmp(key, "dump_callchain") == 0) { dump_callchain_ = (strcmp(value, "1") == 0); } } option = next_option; } if (sample_freq_ == 0 || sample_freq_ > 1000000000) { LOG(ERROR) << "Unexpected sample_freq: " << sample_freq_; return false; } if (sample_freq_ == 1) { sample_period_in_ns_ = 999999999; } else { sample_period_in_ns_ = 1000000000 / sample_freq_; } return true; } bool SampleManager::SendStartProfilingReplyMessage(bool ok) { const char* s = ok ? "ok" : "error"; size_t size = sizeof(UnixSocketMessage) + strlen(s) + 1; std::unique_ptr<char[]> data(new char[size]); UnixSocketMessage* msg = reinterpret_cast<UnixSocketMessage*>(data.get()); msg->len = size; msg->type = START_PROFILING_REPLY; strcpy(msg->data, s); return conn_->SendMessage(*msg, true); } bool SampleManager::StartProfiling() { if (!InstallSignalHandler()) { return false; } if (!CheckThreads()) { return false; } timeval tv; tv.tv_sec = CHECK_THREADS_INTERVAL_IN_MS / 1000; tv.tv_usec = CHECK_THREADS_INTERVAL_IN_MS % 1000 * 1000; return loop_.AddPeriodicEvent(tv, [&]() { return CheckThreads(); }); } bool SampleManager::InstallSignalHandler() { return true; } bool SampleManager::CheckThreads() { uint64_t timestamp = GetSystemClock(); if (!CheckMapChange(timestamp)) { return false; } if (!CheckThreadNameChange(timestamp)) { return false; } SendThreadMapInfo(); // For testing. SendFakeSampleRecord(); return true; } bool SampleManager::CheckThreadNameChange(uint64_t timestamp) { std::vector<pid_t> tids = GetThreadsInProcess(getpid()); std::map<pid_t, std::string> current; for (auto& tid : tids) { if (tid == tid_) { // Skip sample thread. continue; } if (monitor_all_threads_ || monitor_tid_filter_.find(tid) != monitor_tid_filter_.end()) { std::string name; if (GetThreadName(tid, &name)) { current[tid] = name; } } } // Check new threads or threads with new names. for (auto& pair : current) { pid_t tid = pair.first; auto it = threads_.find(tid); if (it == threads_.end() || it->second.name != pair.second) { threads_[tid].name = pair.second; size_t size = sizeof(UnixSocketMessage) + sizeof(uint64_t) + sizeof(uint32_t) + pair.second.size() + 1; std::unique_ptr<char[]> data(new char[size]); UnixSocketMessage* msg = reinterpret_cast<UnixSocketMessage*>(data.get()); msg->len = size; msg->type = THREAD_INFO; char* p = msg->data; MoveToBinaryFormat(timestamp, p); MoveToBinaryFormat(static_cast<uint32_t>(tid), p); MoveToBinaryFormat(pair.second.c_str(), pair.second.size() + 1, p); thread_map_info_q_.push(std::move(data)); } } // Check deleted threads. for (auto it = threads_.begin(); it != threads_.end();) { int tid = it->first; if (current.find(tid) == current.end()) { it = threads_.erase(it); } else { ++it; } } return true; } bool SampleManager::CheckMapChange(uint64_t timestamp) { std::vector<ThreadMmap> maps; if (!GetThreadMmapsInProcess(getpid(), &maps)) { return false; } // Check new maps or changed maps. for (auto& map : maps) { if (!(map.prot & PROT_EXEC)) { continue; } auto it = maps_.find(map.start_addr); if (it == maps_.end() || it->second.len != map.len || it->second.pgoff != map.pgoff || it->second.name != map.name) { maps_[map.start_addr] = map; size_t size = sizeof(UnixSocketMessage) + sizeof(uint64_t) * 4 + map.name.size() + 1; std::unique_ptr<char[]> data(new char[size]); UnixSocketMessage* msg = reinterpret_cast<UnixSocketMessage*>(data.get()); msg->len = size; msg->type = MAP_INFO; char* p = msg->data; MoveToBinaryFormat(timestamp, p); MoveToBinaryFormat(map.start_addr, p); MoveToBinaryFormat(map.len, p); MoveToBinaryFormat(map.pgoff, p); MoveToBinaryFormat(map.name.c_str(), map.name.size() + 1, p); thread_map_info_q_.push(std::move(data)); } } return true; } void SampleManager::SendThreadMapInfo() { while (!thread_map_info_q_.empty()) { auto& data = thread_map_info_q_.front(); UnixSocketMessage* msg = reinterpret_cast<UnixSocketMessage*>(data.get()); if (!conn_->SendMessage(*msg, false)) { break; } thread_map_info_q_.pop(); } } static void FakeFunction() { } void SampleManager::SendFakeSampleRecord() { size_t size = sizeof(UnixSocketMessage) + sizeof(uint64_t) * 2 + sizeof(uint32_t) * 3; std::unique_ptr<char[]> data(new char[size]); UnixSocketMessage* msg = reinterpret_cast<UnixSocketMessage*>(data.get()); uint64_t ip = static_cast<uint64_t>(reinterpret_cast<uintptr_t>(&FakeFunction)); msg->len = size; msg->type = SAMPLE_INFO; char* p = msg->data; MoveToBinaryFormat(GetSystemClock(), p); MoveToBinaryFormat(static_cast<uint32_t>(tid_), p); MoveToBinaryFormat(1u, p); MoveToBinaryFormat(1u, p); MoveToBinaryFormat(ip, p); conn_->SendMessage(*msg, false); } static void* CommunicationThread(void*) { pthread_setname_np(pthread_self(), "inplace_sampler"); std::string server_path = "inplace_sampler_server_" + std::to_string(getpid()); std::unique_ptr<UnixSocketServer> server = UnixSocketServer::Create(server_path, true); if (server == nullptr) { LOG(ERROR) << "failed to create server at path " << server_path; return nullptr; } LOG(INFO) << "Create inplace_sampler_server at " << server_path; while (true) { std::unique_ptr<UnixSocketConnection> conn = server->AcceptConnection(); if (conn == nullptr) { break; } SampleManager manager(std::move(conn)); manager.Run(); } return nullptr; } __attribute__((constructor)) void InitSampler() { pthread_attr_t attr; if (pthread_attr_init(&attr) != 0) { LOG(ERROR) << "pthread_attr_init failed"; return; } if (pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED) != 0) { LOG(ERROR) << "pthread_attr_setdetachstate failed"; return; } pthread_t thread; if (pthread_create(&thread, &attr, CommunicationThread, nullptr) != 0) { LOG(ERROR) << "pthread_create failed"; return; } pthread_attr_destroy(&attr); } } // namespace