| /* |
| * Copyright (C) 2018 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 "src/profiling/memory/client.h" |
| |
| #include <inttypes.h> |
| #include <sys/syscall.h> |
| #include <unistd.h> |
| |
| #include <atomic> |
| #include <new> |
| |
| #include <unwindstack/MachineArm.h> |
| #include <unwindstack/MachineArm64.h> |
| #include <unwindstack/MachineMips.h> |
| #include <unwindstack/MachineMips64.h> |
| #include <unwindstack/MachineX86.h> |
| #include <unwindstack/MachineX86_64.h> |
| #include <unwindstack/Regs.h> |
| #include <unwindstack/RegsGetLocal.h> |
| |
| #include "perfetto/base/logging.h" |
| #include "perfetto/base/scoped_file.h" |
| #include "perfetto/base/thread_utils.h" |
| #include "perfetto/base/unix_socket.h" |
| #include "perfetto/base/utils.h" |
| #include "src/profiling/memory/sampler.h" |
| #include "src/profiling/memory/wire_protocol.h" |
| |
| namespace perfetto { |
| namespace profiling { |
| namespace { |
| |
| constexpr std::chrono::seconds kLockTimeout{1}; |
| |
| std::vector<base::UnixSocketRaw> ConnectPool(const std::string& sock_name, |
| size_t n) { |
| std::vector<base::UnixSocketRaw> res; |
| res.reserve(n); |
| for (size_t i = 0; i < n; ++i) { |
| auto sock = base::UnixSocketRaw::CreateMayFail(base::SockType::kStream); |
| if (!sock || !sock.Connect(sock_name)) { |
| PERFETTO_PLOG("Failed to connect to %s", sock_name.c_str()); |
| continue; |
| } |
| if (!sock.SetTxTimeout(kClientSockTxTimeoutMs)) { |
| PERFETTO_PLOG("Failed to set timeout for %s", sock_name.c_str()); |
| continue; |
| } |
| res.emplace_back(std::move(sock)); |
| } |
| return res; |
| } |
| |
| inline bool IsMainThread() { |
| return getpid() == base::GetThreadId(); |
| } |
| |
| // TODO(b/117203899): Remove this after making bionic implementation safe to |
| // use. |
| char* FindMainThreadStack() { |
| base::ScopedFstream maps(fopen("/proc/self/maps", "r")); |
| if (!maps) { |
| return nullptr; |
| } |
| while (!feof(*maps)) { |
| char line[1024]; |
| char* data = fgets(line, sizeof(line), *maps); |
| if (data != nullptr && strstr(data, "[stack]")) { |
| char* sep = strstr(data, "-"); |
| if (sep == nullptr) |
| continue; |
| sep++; |
| return reinterpret_cast<char*>(strtoll(sep, nullptr, 16)); |
| } |
| } |
| return nullptr; |
| } |
| |
| } // namespace |
| |
| bool FreePage::Add(const uint64_t addr, |
| const uint64_t sequence_number, |
| SocketPool* pool) { |
| std::unique_lock<std::timed_mutex> l(mutex_, kLockTimeout); |
| if (!l.owns_lock()) |
| return false; |
| if (offset_ == kFreePageSize) { |
| bool success = FlushLocked(pool); |
| // Now that we have flushed, reset to after the header. |
| offset_ = 0; |
| return success; |
| } |
| FreePageEntry& current_entry = free_page_.entries[offset_++]; |
| current_entry.sequence_number = sequence_number; |
| current_entry.addr = addr; |
| return true; |
| } |
| |
| bool FreePage::FlushLocked(SocketPool* pool) { |
| WireMessage msg = {}; |
| msg.record_type = RecordType::Free; |
| free_page_.num_entries = offset_; |
| msg.free_header = &free_page_; |
| BorrowedSocket sock(pool->Borrow()); |
| if (!sock || !SendWireMessage(sock.get(), msg)) { |
| PERFETTO_PLOG("Failed to send wire message"); |
| sock.Shutdown(); |
| return false; |
| } |
| return true; |
| } |
| |
| SocketPool::SocketPool(std::vector<base::UnixSocketRaw> sockets) |
| : sockets_(std::move(sockets)), available_sockets_(sockets_.size()) {} |
| |
| BorrowedSocket SocketPool::Borrow() { |
| std::unique_lock<std::timed_mutex> l(mutex_, kLockTimeout); |
| if (!l.owns_lock()) |
| return {base::UnixSocketRaw(), nullptr}; |
| cv_.wait(l, [this] { |
| return available_sockets_ > 0 || dead_sockets_ == sockets_.size() || |
| shutdown_; |
| }); |
| |
| if (dead_sockets_ == sockets_.size() || shutdown_) { |
| return {base::UnixSocketRaw(), nullptr}; |
| } |
| |
| PERFETTO_CHECK(available_sockets_ > 0); |
| return {std::move(sockets_[--available_sockets_]), this}; |
| } |
| |
| void SocketPool::Return(base::UnixSocketRaw sock) { |
| std::unique_lock<std::timed_mutex> l(mutex_, kLockTimeout); |
| if (!l.owns_lock()) |
| return; |
| PERFETTO_CHECK(dead_sockets_ + available_sockets_ < sockets_.size()); |
| if (sock && !shutdown_) { |
| PERFETTO_CHECK(available_sockets_ < sockets_.size()); |
| sockets_[available_sockets_++] = std::move(sock); |
| l.unlock(); |
| cv_.notify_one(); |
| } else { |
| dead_sockets_++; |
| if (dead_sockets_ == sockets_.size()) { |
| l.unlock(); |
| cv_.notify_all(); |
| } |
| } |
| } |
| |
| void SocketPool::Shutdown() { |
| { |
| std::unique_lock<std::timed_mutex> l(mutex_, kLockTimeout); |
| if (!l.owns_lock()) |
| return; |
| for (size_t i = 0; i < available_sockets_; ++i) |
| sockets_[i].Shutdown(); |
| dead_sockets_ += available_sockets_; |
| available_sockets_ = 0; |
| shutdown_ = true; |
| } |
| cv_.notify_all(); |
| } |
| |
| const char* GetThreadStackBase() { |
| pthread_attr_t attr; |
| if (pthread_getattr_np(pthread_self(), &attr) != 0) |
| return nullptr; |
| base::ScopedResource<pthread_attr_t*, pthread_attr_destroy, nullptr> cleanup( |
| &attr); |
| |
| char* stackaddr; |
| size_t stacksize; |
| if (pthread_attr_getstack(&attr, reinterpret_cast<void**>(&stackaddr), |
| &stacksize) != 0) |
| return nullptr; |
| return stackaddr + stacksize; |
| } |
| |
| Client::Client(std::vector<base::UnixSocketRaw> socks) |
| : pthread_key_(ThreadLocalSamplingData::KeyDestructor), |
| socket_pool_(std::move(socks)), |
| main_thread_stack_base_(FindMainThreadStack()) { |
| PERFETTO_DCHECK(pthread_key_.valid()); |
| |
| uint64_t size = 0; |
| base::ScopedFile maps(base::OpenFile("/proc/self/maps", O_RDONLY)); |
| base::ScopedFile mem(base::OpenFile("/proc/self/mem", O_RDONLY)); |
| if (!maps || !mem) { |
| PERFETTO_DFATAL("Failed to open /proc/self/{maps,mem}"); |
| return; |
| } |
| int fds[2]; |
| fds[0] = *maps; |
| fds[1] = *mem; |
| auto sock = socket_pool_.Borrow(); |
| if (!sock) |
| return; |
| // Send an empty record to transfer fds for /proc/self/maps and |
| // /proc/self/mem. |
| if (sock->Send(&size, sizeof(size), fds, 2) != sizeof(size)) { |
| PERFETTO_DFATAL("Failed to send file descriptors."); |
| return; |
| } |
| if (sock->Receive(&client_config_, sizeof(client_config_)) != |
| sizeof(client_config_)) { |
| PERFETTO_DFATAL("Failed to receive client config."); |
| return; |
| } |
| PERFETTO_DCHECK(client_config_.interval >= 1); |
| PERFETTO_DLOG("Initialized client."); |
| inited_.store(true, std::memory_order_release); |
| } |
| |
| Client::Client(const std::string& sock_name, size_t conns) |
| : Client(ConnectPool(sock_name, conns)) {} |
| |
| const char* Client::GetStackBase() { |
| if (IsMainThread()) { |
| if (!main_thread_stack_base_) |
| // Because pthread_attr_getstack reads and parses /proc/self/maps and |
| // /proc/self/stat, we have to cache the result here. |
| main_thread_stack_base_ = GetThreadStackBase(); |
| return main_thread_stack_base_; |
| } |
| return GetThreadStackBase(); |
| } |
| |
| // The stack grows towards numerically smaller addresses, so the stack layout |
| // of main calling malloc is as follows. |
| // |
| // +------------+ |
| // |SendWireMsg | |
| // stacktop +--> +------------+ 0x1000 |
| // |RecordMalloc| + |
| // +------------+ | |
| // | malloc | | |
| // +------------+ | |
| // | main | v |
| // stackbase +-> +------------+ 0xffff |
| void Client::RecordMalloc(uint64_t alloc_size, |
| uint64_t total_size, |
| uint64_t alloc_address) { |
| if (!inited_.load(std::memory_order_acquire)) |
| return; |
| AllocMetadata metadata; |
| const char* stackbase = GetStackBase(); |
| const char* stacktop = reinterpret_cast<char*>(__builtin_frame_address(0)); |
| unwindstack::AsmGetRegs(metadata.register_data); |
| |
| if (stackbase < stacktop) { |
| PERFETTO_DFATAL("Stackbase >= stacktop."); |
| return; |
| } |
| |
| uint64_t stack_size = static_cast<uint64_t>(stackbase - stacktop); |
| metadata.total_size = total_size; |
| metadata.alloc_size = alloc_size; |
| metadata.alloc_address = alloc_address; |
| metadata.stack_pointer = reinterpret_cast<uint64_t>(stacktop); |
| metadata.stack_pointer_offset = sizeof(AllocMetadata); |
| metadata.arch = unwindstack::Regs::CurrentArch(); |
| metadata.sequence_number = |
| 1 + sequence_number_.fetch_add(1, std::memory_order_acq_rel); |
| |
| WireMessage msg{}; |
| msg.record_type = RecordType::Malloc; |
| msg.alloc_header = &metadata; |
| msg.payload = const_cast<char*>(stacktop); |
| msg.payload_size = static_cast<size_t>(stack_size); |
| |
| BorrowedSocket sock = socket_pool_.Borrow(); |
| if (!sock || !SendWireMessage(sock.get(), msg)) { |
| PERFETTO_PLOG("Failed to send wire message."); |
| sock.Shutdown(); |
| Shutdown(); |
| } |
| } |
| |
| void Client::RecordFree(uint64_t alloc_address) { |
| if (!inited_.load(std::memory_order_acquire)) |
| return; |
| if (!free_page_.Add( |
| alloc_address, |
| 1 + sequence_number_.fetch_add(1, std::memory_order_acq_rel), |
| &socket_pool_)) |
| Shutdown(); |
| } |
| |
| size_t Client::ShouldSampleAlloc(uint64_t alloc_size, |
| void* (*unhooked_malloc)(size_t), |
| void (*unhooked_free)(void*)) { |
| if (!inited_.load(std::memory_order_acquire)) |
| return false; |
| return SampleSize(pthread_key_.get(), alloc_size, client_config_.interval, |
| unhooked_malloc, unhooked_free); |
| } |
| |
| void Client::MaybeSampleAlloc(uint64_t alloc_size, |
| uint64_t alloc_address, |
| void* (*unhooked_malloc)(size_t), |
| void (*unhooked_free)(void*)) { |
| size_t total_size = |
| ShouldSampleAlloc(alloc_size, unhooked_malloc, unhooked_free); |
| if (total_size > 0) |
| RecordMalloc(alloc_size, total_size, alloc_address); |
| } |
| |
| void Client::Shutdown() { |
| socket_pool_.Shutdown(); |
| inited_.store(false, std::memory_order_release); |
| } |
| |
| } // namespace profiling |
| } // namespace perfetto |