blob: 745fd6842cce8f34a6fc4b26ef681aa6969dcd3e [file] [log] [blame]
Florian Mayer824274d2018-09-17 11:33:45 +01001/*
2 * Copyright (C) 2018 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17#ifndef SRC_PROFILING_MEMORY_CLIENT_H_
18#define SRC_PROFILING_MEMORY_CLIENT_H_
19
20#include <stddef.h>
Florian Mayeraf266022018-09-17 15:18:15 +010021
Ryan Savitskia502bde2019-02-26 21:34:03 +000022#include <atomic>
Florian Mayereff98042018-12-10 17:44:44 +000023#include <condition_variable>
Florian Mayer824274d2018-09-17 11:33:45 +010024#include <mutex>
25#include <vector>
26
Primiano Tucci43ebf4d2019-01-02 20:04:34 +010027#include "perfetto/base/unix_socket.h"
Ryan Savitskia502bde2019-02-26 21:34:03 +000028#include "src/profiling/memory/sampler.h"
Florian Mayer51950592019-03-06 20:05:15 +000029#include "src/profiling/memory/shared_ring_buffer.h"
Ryan Savitski646d3662019-04-08 12:53:29 +010030#include "src/profiling/memory/unhooked_allocator.h"
Florian Mayerb85a9382018-09-27 13:59:01 +010031#include "src/profiling/memory/wire_protocol.h"
Florian Mayer824274d2018-09-17 11:33:45 +010032
33namespace perfetto {
Florian Mayer8c0d0482018-10-22 14:23:40 +010034namespace profiling {
Florian Mayer824274d2018-09-17 11:33:45 +010035
Florian Mayerb85a9382018-09-27 13:59:01 +010036const char* GetThreadStackBase();
37
Florian Mayerdac31272019-02-26 12:07:02 +000038constexpr uint32_t kClientSockTimeoutMs = 1000;
Ryan Savitski14c12d62019-01-08 22:38:53 +000039
Ryan Savitskia502bde2019-02-26 21:34:03 +000040// Profiling client, used to sample and record the malloc/free family of calls,
41// and communicate the necessary state to a separate profiling daemon process.
42//
43// Created and owned by the malloc hooks.
44//
45// Methods of this class are thread-safe unless otherwise stated, in which case
46// the caller needs to synchronize calls behind a mutex or similar.
Ryan Savitski646d3662019-04-08 12:53:29 +010047//
48// Implementation warning: this class should not use any heap, as otherwise its
49// destruction would enter the possibly-hooked |free|, which can reference the
50// Client itself. If avoiding the heap is not possible, then look at using
51// UnhookedAllocator.
Florian Mayerb85a9382018-09-27 13:59:01 +010052class Client {
53 public:
Ryan Savitski4a783172019-03-21 19:27:12 +000054 // Returns a client that is ready for sampling allocations, using the given
55 // socket (which should already be connected to heapprofd).
56 //
57 // Returns a shared_ptr since that is how the client will ultimately be used,
Ryan Savitski646d3662019-04-08 12:53:29 +010058 // and to take advantage of std::allocate_shared putting the object & the
59 // control block in one block of memory.
60 static std::shared_ptr<Client> CreateAndHandshake(
61 base::UnixSocketRaw sock,
62 UnhookedAllocator<Client> unhooked_allocator);
63
Ryan Savitski4a783172019-03-21 19:27:12 +000064 static base::Optional<base::UnixSocketRaw> ConnectToHeapprofd(
65 const std::string& sock_name);
66
Florian Mayer20c4b3c2019-01-24 18:25:35 +000067 bool RecordMalloc(uint64_t alloc_size,
Florian Mayer9a7fb2b2018-10-17 18:17:40 +010068 uint64_t total_size,
69 uint64_t alloc_address);
Ryan Savitski4a783172019-03-21 19:27:12 +000070
71 // Add address to buffer of deallocations. Flushes the buffer if necessary.
Florian Mayer20c4b3c2019-01-24 18:25:35 +000072 bool RecordFree(uint64_t alloc_address);
Florian Mayerb85a9382018-09-27 13:59:01 +010073
Ryan Savitskia502bde2019-02-26 21:34:03 +000074 // Returns the number of bytes to assign to an allocation with the given
75 // |alloc_size|, based on the current sampling rate. A return value of zero
76 // means that the allocation should not be recorded. Not idempotent, each
77 // invocation mutates the sampler state.
78 //
79 // Not thread-safe.
80 size_t GetSampleSizeLocked(size_t alloc_size) {
Ryan Savitskia502bde2019-02-26 21:34:03 +000081 return sampler_.SampleSize(alloc_size);
82 }
83
Ryan Savitski646d3662019-04-08 12:53:29 +010084 // Public for std::allocate_shared. Use CreateAndHandshake() to create
85 // instances instead.
Ryan Savitski4a783172019-03-21 19:27:12 +000086 Client(base::UnixSocketRaw sock,
87 ClientConfiguration client_config,
88 SharedRingBuffer shmem,
89 Sampler sampler,
90 const char* main_thread_stack_base);
91
Florian Mayer1fe77732018-10-02 17:25:53 +010092 ClientConfiguration client_config_for_testing() { return client_config_; }
Florian Mayer754b57f2019-04-15 13:17:55 +010093 bool IsConnected();
Florian Mayer1fe77732018-10-02 17:25:53 +010094
Florian Mayerb85a9382018-09-27 13:59:01 +010095 private:
96 const char* GetStackBase();
Ryan Savitski549753e2019-03-15 15:52:37 +000097 // Flush the contents of free_batch_. Must hold free_batch_lock_.
98 bool FlushFreesLocked();
Florian Mayer754b57f2019-04-15 13:17:55 +010099 bool SendControlSocketByte();
Ryan Savitski549753e2019-03-15 15:52:37 +0000100
Florian Mayer1fe77732018-10-02 17:25:53 +0100101 ClientConfiguration client_config_;
Ryan Savitskia502bde2019-02-26 21:34:03 +0000102 // sampler_ operations are not thread-safe.
103 Sampler sampler_;
Florian Mayer51950592019-03-06 20:05:15 +0000104 base::UnixSocketRaw sock_;
Ryan Savitski549753e2019-03-15 15:52:37 +0000105
106 // Protected by free_batch_lock_.
107 FreeBatch free_batch_;
108 std::timed_mutex free_batch_lock_;
109
Ryan Savitski4a783172019-03-21 19:27:12 +0000110 const char* main_thread_stack_base_{nullptr};
Florian Mayerb85a9382018-09-27 13:59:01 +0100111 std::atomic<uint64_t> sequence_number_{0};
Florian Mayer51950592019-03-06 20:05:15 +0000112 SharedRingBuffer shmem_;
Florian Mayerb85a9382018-09-27 13:59:01 +0100113};
114
Florian Mayer8c0d0482018-10-22 14:23:40 +0100115} // namespace profiling
Florian Mayer824274d2018-09-17 11:33:45 +0100116} // namespace perfetto
117
118#endif // SRC_PROFILING_MEMORY_CLIENT_H_