xref: /aosp_15_r20/external/perfetto/src/profiling/memory/unwinding.cc (revision 6dbdd20afdafa5e3ca9b8809fa73465d530080dc)
1*6dbdd20aSAndroid Build Coastguard Worker /*
2*6dbdd20aSAndroid Build Coastguard Worker  * Copyright (C) 2018 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 "src/profiling/memory/unwinding.h"
18*6dbdd20aSAndroid Build Coastguard Worker 
19*6dbdd20aSAndroid Build Coastguard Worker #include <sys/types.h>
20*6dbdd20aSAndroid Build Coastguard Worker #include <unistd.h>
21*6dbdd20aSAndroid Build Coastguard Worker 
22*6dbdd20aSAndroid Build Coastguard Worker #include <condition_variable>
23*6dbdd20aSAndroid Build Coastguard Worker #include <mutex>
24*6dbdd20aSAndroid Build Coastguard Worker 
25*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/MachineArm.h>
26*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/MachineArm64.h>
27*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/MachineRiscv64.h>
28*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/MachineX86.h>
29*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/MachineX86_64.h>
30*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/Maps.h>
31*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/Memory.h>
32*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/Regs.h>
33*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/RegsArm.h>
34*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/RegsArm64.h>
35*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/RegsRiscv64.h>
36*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/RegsX86.h>
37*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/RegsX86_64.h>
38*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/Unwinder.h>
39*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/UserArm.h>
40*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/UserArm64.h>
41*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/UserRiscv64.h>
42*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/UserX86.h>
43*6dbdd20aSAndroid Build Coastguard Worker #include <unwindstack/UserX86_64.h>
44*6dbdd20aSAndroid Build Coastguard Worker 
45*6dbdd20aSAndroid Build Coastguard Worker #include <procinfo/process_map.h>
46*6dbdd20aSAndroid Build Coastguard Worker 
47*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/base/logging.h"
48*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/base/task_runner.h"
49*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/file_utils.h"
50*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/scoped_file.h"
51*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/string_utils.h"
52*6dbdd20aSAndroid Build Coastguard Worker #include "perfetto/ext/base/thread_task_runner.h"
53*6dbdd20aSAndroid Build Coastguard Worker 
54*6dbdd20aSAndroid Build Coastguard Worker #include "src/profiling/memory/unwound_messages.h"
55*6dbdd20aSAndroid Build Coastguard Worker #include "src/profiling/memory/wire_protocol.h"
56*6dbdd20aSAndroid Build Coastguard Worker 
57*6dbdd20aSAndroid Build Coastguard Worker namespace perfetto {
58*6dbdd20aSAndroid Build Coastguard Worker namespace profiling {
59*6dbdd20aSAndroid Build Coastguard Worker namespace {
60*6dbdd20aSAndroid Build Coastguard Worker 
61*6dbdd20aSAndroid Build Coastguard Worker constexpr base::TimeMillis kMapsReparseInterval{500};
62*6dbdd20aSAndroid Build Coastguard Worker constexpr uint32_t kRetryDelayMs = 100;
63*6dbdd20aSAndroid Build Coastguard Worker 
64*6dbdd20aSAndroid Build Coastguard Worker constexpr size_t kMaxFrames = 500;
65*6dbdd20aSAndroid Build Coastguard Worker 
66*6dbdd20aSAndroid Build Coastguard Worker // We assume average ~300us per unwind. If we handle up to 1000 unwinds, this
67*6dbdd20aSAndroid Build Coastguard Worker // makes sure other tasks get to be run at least every 300ms if the unwinding
68*6dbdd20aSAndroid Build Coastguard Worker // saturates this thread.
69*6dbdd20aSAndroid Build Coastguard Worker constexpr size_t kUnwindBatchSize = 1000;
70*6dbdd20aSAndroid Build Coastguard Worker constexpr size_t kRecordBatchSize = 1024;
71*6dbdd20aSAndroid Build Coastguard Worker constexpr size_t kMaxAllocRecordArenaSize = 2 * kRecordBatchSize;
72*6dbdd20aSAndroid Build Coastguard Worker 
73*6dbdd20aSAndroid Build Coastguard Worker #pragma GCC diagnostic push
74*6dbdd20aSAndroid Build Coastguard Worker // We do not care about deterministic destructor order.
75*6dbdd20aSAndroid Build Coastguard Worker #pragma GCC diagnostic ignored "-Wglobal-constructors"
76*6dbdd20aSAndroid Build Coastguard Worker #pragma GCC diagnostic ignored "-Wexit-time-destructors"
77*6dbdd20aSAndroid Build Coastguard Worker static std::vector<std::string> kSkipMaps{"heapprofd_client.so",
78*6dbdd20aSAndroid Build Coastguard Worker                                           "heapprofd_client_api.so"};
79*6dbdd20aSAndroid Build Coastguard Worker #pragma GCC diagnostic pop
80*6dbdd20aSAndroid Build Coastguard Worker 
GetRegsSize(unwindstack::Regs * regs)81*6dbdd20aSAndroid Build Coastguard Worker size_t GetRegsSize(unwindstack::Regs* regs) {
82*6dbdd20aSAndroid Build Coastguard Worker   if (regs->Is32Bit())
83*6dbdd20aSAndroid Build Coastguard Worker     return sizeof(uint32_t) * regs->total_regs();
84*6dbdd20aSAndroid Build Coastguard Worker   return sizeof(uint64_t) * regs->total_regs();
85*6dbdd20aSAndroid Build Coastguard Worker }
86*6dbdd20aSAndroid Build Coastguard Worker 
ReadFromRawData(unwindstack::Regs * regs,void * raw_data)87*6dbdd20aSAndroid Build Coastguard Worker void ReadFromRawData(unwindstack::Regs* regs, void* raw_data) {
88*6dbdd20aSAndroid Build Coastguard Worker   memcpy(regs->RawData(), raw_data, GetRegsSize(regs));
89*6dbdd20aSAndroid Build Coastguard Worker }
90*6dbdd20aSAndroid Build Coastguard Worker 
91*6dbdd20aSAndroid Build Coastguard Worker }  // namespace
92*6dbdd20aSAndroid Build Coastguard Worker 
CreateRegsFromRawData(unwindstack::ArchEnum arch,void * raw_data)93*6dbdd20aSAndroid Build Coastguard Worker std::unique_ptr<unwindstack::Regs> CreateRegsFromRawData(
94*6dbdd20aSAndroid Build Coastguard Worker     unwindstack::ArchEnum arch,
95*6dbdd20aSAndroid Build Coastguard Worker     void* raw_data) {
96*6dbdd20aSAndroid Build Coastguard Worker   std::unique_ptr<unwindstack::Regs> ret;
97*6dbdd20aSAndroid Build Coastguard Worker   switch (arch) {
98*6dbdd20aSAndroid Build Coastguard Worker     case unwindstack::ARCH_X86:
99*6dbdd20aSAndroid Build Coastguard Worker       ret.reset(new unwindstack::RegsX86());
100*6dbdd20aSAndroid Build Coastguard Worker       break;
101*6dbdd20aSAndroid Build Coastguard Worker     case unwindstack::ARCH_X86_64:
102*6dbdd20aSAndroid Build Coastguard Worker       ret.reset(new unwindstack::RegsX86_64());
103*6dbdd20aSAndroid Build Coastguard Worker       break;
104*6dbdd20aSAndroid Build Coastguard Worker     case unwindstack::ARCH_ARM:
105*6dbdd20aSAndroid Build Coastguard Worker       ret.reset(new unwindstack::RegsArm());
106*6dbdd20aSAndroid Build Coastguard Worker       break;
107*6dbdd20aSAndroid Build Coastguard Worker     case unwindstack::ARCH_ARM64:
108*6dbdd20aSAndroid Build Coastguard Worker       ret.reset(new unwindstack::RegsArm64());
109*6dbdd20aSAndroid Build Coastguard Worker       break;
110*6dbdd20aSAndroid Build Coastguard Worker     case unwindstack::ARCH_RISCV64:
111*6dbdd20aSAndroid Build Coastguard Worker       ret.reset(new unwindstack::RegsRiscv64());
112*6dbdd20aSAndroid Build Coastguard Worker       break;
113*6dbdd20aSAndroid Build Coastguard Worker     case unwindstack::ARCH_UNKNOWN:
114*6dbdd20aSAndroid Build Coastguard Worker       break;
115*6dbdd20aSAndroid Build Coastguard Worker   }
116*6dbdd20aSAndroid Build Coastguard Worker   if (ret)
117*6dbdd20aSAndroid Build Coastguard Worker     ReadFromRawData(ret.get(), raw_data);
118*6dbdd20aSAndroid Build Coastguard Worker   return ret;
119*6dbdd20aSAndroid Build Coastguard Worker }
120*6dbdd20aSAndroid Build Coastguard Worker 
DoUnwind(WireMessage * msg,UnwindingMetadata * metadata,AllocRecord * out)121*6dbdd20aSAndroid Build Coastguard Worker bool DoUnwind(WireMessage* msg, UnwindingMetadata* metadata, AllocRecord* out) {
122*6dbdd20aSAndroid Build Coastguard Worker   AllocMetadata* alloc_metadata = msg->alloc_header;
123*6dbdd20aSAndroid Build Coastguard Worker   std::unique_ptr<unwindstack::Regs> regs(CreateRegsFromRawData(
124*6dbdd20aSAndroid Build Coastguard Worker       alloc_metadata->arch, alloc_metadata->register_data));
125*6dbdd20aSAndroid Build Coastguard Worker   if (regs == nullptr) {
126*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DLOG("Unable to construct unwindstack::Regs");
127*6dbdd20aSAndroid Build Coastguard Worker     unwindstack::FrameData frame_data{};
128*6dbdd20aSAndroid Build Coastguard Worker     frame_data.function_name = "ERROR READING REGISTERS";
129*6dbdd20aSAndroid Build Coastguard Worker 
130*6dbdd20aSAndroid Build Coastguard Worker     out->frames.clear();
131*6dbdd20aSAndroid Build Coastguard Worker     out->build_ids.clear();
132*6dbdd20aSAndroid Build Coastguard Worker     out->frames.emplace_back(std::move(frame_data));
133*6dbdd20aSAndroid Build Coastguard Worker     out->build_ids.emplace_back("");
134*6dbdd20aSAndroid Build Coastguard Worker     out->error = true;
135*6dbdd20aSAndroid Build Coastguard Worker     return false;
136*6dbdd20aSAndroid Build Coastguard Worker   }
137*6dbdd20aSAndroid Build Coastguard Worker   uint8_t* stack = reinterpret_cast<uint8_t*>(msg->payload);
138*6dbdd20aSAndroid Build Coastguard Worker   std::shared_ptr<unwindstack::Memory> mems =
139*6dbdd20aSAndroid Build Coastguard Worker       std::make_shared<StackOverlayMemory>(metadata->fd_mem,
140*6dbdd20aSAndroid Build Coastguard Worker                                            alloc_metadata->stack_pointer, stack,
141*6dbdd20aSAndroid Build Coastguard Worker                                            msg->payload_size);
142*6dbdd20aSAndroid Build Coastguard Worker 
143*6dbdd20aSAndroid Build Coastguard Worker   unwindstack::Unwinder unwinder(kMaxFrames, &metadata->fd_maps, regs.get(),
144*6dbdd20aSAndroid Build Coastguard Worker                                  mems);
145*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_ANDROID_BUILD)
146*6dbdd20aSAndroid Build Coastguard Worker   unwinder.SetJitDebug(metadata->GetJitDebug(regs->Arch()));
147*6dbdd20aSAndroid Build Coastguard Worker   unwinder.SetDexFiles(metadata->GetDexFiles(regs->Arch()));
148*6dbdd20aSAndroid Build Coastguard Worker #endif
149*6dbdd20aSAndroid Build Coastguard Worker   // Suppress incorrect "variable may be uninitialized" error for if condition
150*6dbdd20aSAndroid Build Coastguard Worker   // after this loop. error_code = LastErrorCode gets run at least once.
151*6dbdd20aSAndroid Build Coastguard Worker   unwindstack::ErrorCode error_code = unwindstack::ERROR_NONE;
152*6dbdd20aSAndroid Build Coastguard Worker   for (int attempt = 0; attempt < 2; ++attempt) {
153*6dbdd20aSAndroid Build Coastguard Worker     if (attempt > 0) {
154*6dbdd20aSAndroid Build Coastguard Worker       if (metadata->last_maps_reparse_time + kMapsReparseInterval >
155*6dbdd20aSAndroid Build Coastguard Worker           base::GetWallTimeMs()) {
156*6dbdd20aSAndroid Build Coastguard Worker         PERFETTO_DLOG("Skipping reparse due to rate limit.");
157*6dbdd20aSAndroid Build Coastguard Worker         break;
158*6dbdd20aSAndroid Build Coastguard Worker       }
159*6dbdd20aSAndroid Build Coastguard Worker       PERFETTO_DLOG("Reparsing maps");
160*6dbdd20aSAndroid Build Coastguard Worker       metadata->ReparseMaps();
161*6dbdd20aSAndroid Build Coastguard Worker       metadata->last_maps_reparse_time = base::GetWallTimeMs();
162*6dbdd20aSAndroid Build Coastguard Worker       // Regs got invalidated by libuwindstack's speculative jump.
163*6dbdd20aSAndroid Build Coastguard Worker       // Reset.
164*6dbdd20aSAndroid Build Coastguard Worker       ReadFromRawData(regs.get(), alloc_metadata->register_data);
165*6dbdd20aSAndroid Build Coastguard Worker       out->reparsed_map = true;
166*6dbdd20aSAndroid Build Coastguard Worker #if PERFETTO_BUILDFLAG(PERFETTO_ANDROID_BUILD)
167*6dbdd20aSAndroid Build Coastguard Worker       unwinder.SetJitDebug(metadata->GetJitDebug(regs->Arch()));
168*6dbdd20aSAndroid Build Coastguard Worker       unwinder.SetDexFiles(metadata->GetDexFiles(regs->Arch()));
169*6dbdd20aSAndroid Build Coastguard Worker #endif
170*6dbdd20aSAndroid Build Coastguard Worker     }
171*6dbdd20aSAndroid Build Coastguard Worker     out->frames.swap(unwinder.frames());  // Provide the unwinder buffer to use.
172*6dbdd20aSAndroid Build Coastguard Worker     unwinder.Unwind(&kSkipMaps, /*map_suffixes_to_ignore=*/nullptr);
173*6dbdd20aSAndroid Build Coastguard Worker     out->frames.swap(unwinder.frames());  // Take the buffer back.
174*6dbdd20aSAndroid Build Coastguard Worker     error_code = unwinder.LastErrorCode();
175*6dbdd20aSAndroid Build Coastguard Worker     if (error_code != unwindstack::ERROR_INVALID_MAP &&
176*6dbdd20aSAndroid Build Coastguard Worker         (unwinder.warnings() & unwindstack::WARNING_DEX_PC_NOT_IN_MAP) == 0) {
177*6dbdd20aSAndroid Build Coastguard Worker       break;
178*6dbdd20aSAndroid Build Coastguard Worker     }
179*6dbdd20aSAndroid Build Coastguard Worker   }
180*6dbdd20aSAndroid Build Coastguard Worker   out->build_ids.resize(out->frames.size());
181*6dbdd20aSAndroid Build Coastguard Worker   for (size_t i = 0; i < out->frames.size(); ++i) {
182*6dbdd20aSAndroid Build Coastguard Worker     out->build_ids[i] = metadata->GetBuildId(out->frames[i]);
183*6dbdd20aSAndroid Build Coastguard Worker   }
184*6dbdd20aSAndroid Build Coastguard Worker 
185*6dbdd20aSAndroid Build Coastguard Worker   if (error_code != unwindstack::ERROR_NONE) {
186*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DLOG("Unwinding error %" PRIu8, error_code);
187*6dbdd20aSAndroid Build Coastguard Worker     unwindstack::FrameData frame_data{};
188*6dbdd20aSAndroid Build Coastguard Worker     frame_data.function_name =
189*6dbdd20aSAndroid Build Coastguard Worker         "ERROR " + StringifyLibUnwindstackError(error_code);
190*6dbdd20aSAndroid Build Coastguard Worker 
191*6dbdd20aSAndroid Build Coastguard Worker     out->frames.emplace_back(std::move(frame_data));
192*6dbdd20aSAndroid Build Coastguard Worker     out->build_ids.emplace_back("");
193*6dbdd20aSAndroid Build Coastguard Worker     out->error = true;
194*6dbdd20aSAndroid Build Coastguard Worker   }
195*6dbdd20aSAndroid Build Coastguard Worker   return true;
196*6dbdd20aSAndroid Build Coastguard Worker }
197*6dbdd20aSAndroid Build Coastguard Worker 
~UnwindingWorker()198*6dbdd20aSAndroid Build Coastguard Worker UnwindingWorker::~UnwindingWorker() {
199*6dbdd20aSAndroid Build Coastguard Worker   if (thread_task_runner_.get() == nullptr) {
200*6dbdd20aSAndroid Build Coastguard Worker     return;
201*6dbdd20aSAndroid Build Coastguard Worker   }
202*6dbdd20aSAndroid Build Coastguard Worker   std::mutex mutex;
203*6dbdd20aSAndroid Build Coastguard Worker   std::condition_variable cv;
204*6dbdd20aSAndroid Build Coastguard Worker 
205*6dbdd20aSAndroid Build Coastguard Worker   std::unique_lock<std::mutex> lock(mutex);
206*6dbdd20aSAndroid Build Coastguard Worker   bool done = false;
207*6dbdd20aSAndroid Build Coastguard Worker   thread_task_runner_.PostTask([&mutex, &cv, &done, this] {
208*6dbdd20aSAndroid Build Coastguard Worker     for (auto& it : client_data_) {
209*6dbdd20aSAndroid Build Coastguard Worker       auto& client_data = it.second;
210*6dbdd20aSAndroid Build Coastguard Worker       client_data.sock->Shutdown(false);
211*6dbdd20aSAndroid Build Coastguard Worker     }
212*6dbdd20aSAndroid Build Coastguard Worker     client_data_.clear();
213*6dbdd20aSAndroid Build Coastguard Worker 
214*6dbdd20aSAndroid Build Coastguard Worker     std::lock_guard<std::mutex> inner_lock(mutex);
215*6dbdd20aSAndroid Build Coastguard Worker     done = true;
216*6dbdd20aSAndroid Build Coastguard Worker     cv.notify_one();
217*6dbdd20aSAndroid Build Coastguard Worker   });
218*6dbdd20aSAndroid Build Coastguard Worker   cv.wait(lock, [&done] { return done; });
219*6dbdd20aSAndroid Build Coastguard Worker }
220*6dbdd20aSAndroid Build Coastguard Worker 
OnDisconnect(base::UnixSocket * self)221*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::OnDisconnect(base::UnixSocket* self) {
222*6dbdd20aSAndroid Build Coastguard Worker   pid_t peer_pid = self->peer_pid_linux();
223*6dbdd20aSAndroid Build Coastguard Worker   auto it = client_data_.find(peer_pid);
224*6dbdd20aSAndroid Build Coastguard Worker   if (it == client_data_.end()) {
225*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DFATAL_OR_ELOG("Disconnected unexpected socket.");
226*6dbdd20aSAndroid Build Coastguard Worker     return;
227*6dbdd20aSAndroid Build Coastguard Worker   }
228*6dbdd20aSAndroid Build Coastguard Worker 
229*6dbdd20aSAndroid Build Coastguard Worker   ClientData& client_data = it->second;
230*6dbdd20aSAndroid Build Coastguard Worker   SharedRingBuffer& shmem = client_data.shmem;
231*6dbdd20aSAndroid Build Coastguard Worker   client_data.drain_bytes = shmem.read_avail();
232*6dbdd20aSAndroid Build Coastguard Worker 
233*6dbdd20aSAndroid Build Coastguard Worker   if (client_data.drain_bytes != 0) {
234*6dbdd20aSAndroid Build Coastguard Worker     DrainJob(peer_pid);
235*6dbdd20aSAndroid Build Coastguard Worker   } else {
236*6dbdd20aSAndroid Build Coastguard Worker     FinishDisconnect(it);
237*6dbdd20aSAndroid Build Coastguard Worker   }
238*6dbdd20aSAndroid Build Coastguard Worker }
239*6dbdd20aSAndroid Build Coastguard Worker 
RemoveClientData(std::map<pid_t,ClientData>::iterator client_data_iterator)240*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::RemoveClientData(
241*6dbdd20aSAndroid Build Coastguard Worker     std::map<pid_t, ClientData>::iterator client_data_iterator) {
242*6dbdd20aSAndroid Build Coastguard Worker   client_data_.erase(client_data_iterator);
243*6dbdd20aSAndroid Build Coastguard Worker   if (client_data_.empty()) {
244*6dbdd20aSAndroid Build Coastguard Worker     // We got rid of the last client. Flush and destruct AllocRecords in
245*6dbdd20aSAndroid Build Coastguard Worker     // arena. Disable the arena (will not accept returning borrowed records)
246*6dbdd20aSAndroid Build Coastguard Worker     // in case there are pending AllocRecords on the main thread.
247*6dbdd20aSAndroid Build Coastguard Worker     alloc_record_arena_.Disable();
248*6dbdd20aSAndroid Build Coastguard Worker   }
249*6dbdd20aSAndroid Build Coastguard Worker }
250*6dbdd20aSAndroid Build Coastguard Worker 
FinishDisconnect(std::map<pid_t,ClientData>::iterator client_data_iterator)251*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::FinishDisconnect(
252*6dbdd20aSAndroid Build Coastguard Worker     std::map<pid_t, ClientData>::iterator client_data_iterator) {
253*6dbdd20aSAndroid Build Coastguard Worker   pid_t peer_pid = client_data_iterator->first;
254*6dbdd20aSAndroid Build Coastguard Worker   ClientData& client_data = client_data_iterator->second;
255*6dbdd20aSAndroid Build Coastguard Worker   SharedRingBuffer& shmem = client_data.shmem;
256*6dbdd20aSAndroid Build Coastguard Worker 
257*6dbdd20aSAndroid Build Coastguard Worker   if (!client_data.free_records.empty()) {
258*6dbdd20aSAndroid Build Coastguard Worker     delegate_->PostFreeRecord(this, std::move(client_data.free_records));
259*6dbdd20aSAndroid Build Coastguard Worker   }
260*6dbdd20aSAndroid Build Coastguard Worker 
261*6dbdd20aSAndroid Build Coastguard Worker   SharedRingBuffer::Stats stats = {};
262*6dbdd20aSAndroid Build Coastguard Worker   {
263*6dbdd20aSAndroid Build Coastguard Worker     auto lock = shmem.AcquireLock(ScopedSpinlock::Mode::Try);
264*6dbdd20aSAndroid Build Coastguard Worker     if (lock.locked())
265*6dbdd20aSAndroid Build Coastguard Worker       stats = shmem.GetStats(lock);
266*6dbdd20aSAndroid Build Coastguard Worker     else
267*6dbdd20aSAndroid Build Coastguard Worker       PERFETTO_ELOG("Failed to log shmem to get stats.");
268*6dbdd20aSAndroid Build Coastguard Worker   }
269*6dbdd20aSAndroid Build Coastguard Worker   DataSourceInstanceID ds_id = client_data.data_source_instance_id;
270*6dbdd20aSAndroid Build Coastguard Worker 
271*6dbdd20aSAndroid Build Coastguard Worker   RemoveClientData(client_data_iterator);
272*6dbdd20aSAndroid Build Coastguard Worker   delegate_->PostSocketDisconnected(this, ds_id, peer_pid, stats);
273*6dbdd20aSAndroid Build Coastguard Worker }
274*6dbdd20aSAndroid Build Coastguard Worker 
OnDataAvailable(base::UnixSocket * self)275*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::OnDataAvailable(base::UnixSocket* self) {
276*6dbdd20aSAndroid Build Coastguard Worker   // Drain buffer to clear the notification.
277*6dbdd20aSAndroid Build Coastguard Worker   char recv_buf[kUnwindBatchSize];
278*6dbdd20aSAndroid Build Coastguard Worker   self->Receive(recv_buf, sizeof(recv_buf));
279*6dbdd20aSAndroid Build Coastguard Worker   BatchUnwindJob(self->peer_pid_linux());
280*6dbdd20aSAndroid Build Coastguard Worker }
281*6dbdd20aSAndroid Build Coastguard Worker 
ReadAndUnwindBatch(ClientData * client_data)282*6dbdd20aSAndroid Build Coastguard Worker UnwindingWorker::ReadAndUnwindBatchResult UnwindingWorker::ReadAndUnwindBatch(
283*6dbdd20aSAndroid Build Coastguard Worker     ClientData* client_data) {
284*6dbdd20aSAndroid Build Coastguard Worker   SharedRingBuffer& shmem = client_data->shmem;
285*6dbdd20aSAndroid Build Coastguard Worker   SharedRingBuffer::Buffer buf;
286*6dbdd20aSAndroid Build Coastguard Worker   ReadAndUnwindBatchResult res;
287*6dbdd20aSAndroid Build Coastguard Worker 
288*6dbdd20aSAndroid Build Coastguard Worker   size_t i;
289*6dbdd20aSAndroid Build Coastguard Worker   for (i = 0; i < kUnwindBatchSize; ++i) {
290*6dbdd20aSAndroid Build Coastguard Worker     uint64_t reparses_before = client_data->metadata.reparses;
291*6dbdd20aSAndroid Build Coastguard Worker     buf = shmem.BeginRead();
292*6dbdd20aSAndroid Build Coastguard Worker     if (!buf)
293*6dbdd20aSAndroid Build Coastguard Worker       break;
294*6dbdd20aSAndroid Build Coastguard Worker     HandleBuffer(this, &alloc_record_arena_, buf, client_data,
295*6dbdd20aSAndroid Build Coastguard Worker                  client_data->sock->peer_pid_linux(), delegate_);
296*6dbdd20aSAndroid Build Coastguard Worker     res.bytes_read += shmem.EndRead(std::move(buf));
297*6dbdd20aSAndroid Build Coastguard Worker     // Reparsing takes time, so process the rest in a new batch to avoid timing
298*6dbdd20aSAndroid Build Coastguard Worker     // out.
299*6dbdd20aSAndroid Build Coastguard Worker     if (reparses_before < client_data->metadata.reparses) {
300*6dbdd20aSAndroid Build Coastguard Worker       res.status = ReadAndUnwindBatchResult::Status::kHasMore;
301*6dbdd20aSAndroid Build Coastguard Worker       return res;
302*6dbdd20aSAndroid Build Coastguard Worker     }
303*6dbdd20aSAndroid Build Coastguard Worker   }
304*6dbdd20aSAndroid Build Coastguard Worker 
305*6dbdd20aSAndroid Build Coastguard Worker   if (i == kUnwindBatchSize) {
306*6dbdd20aSAndroid Build Coastguard Worker     res.status = ReadAndUnwindBatchResult::Status::kHasMore;
307*6dbdd20aSAndroid Build Coastguard Worker   } else if (i > 0) {
308*6dbdd20aSAndroid Build Coastguard Worker     res.status = ReadAndUnwindBatchResult::Status::kReadSome;
309*6dbdd20aSAndroid Build Coastguard Worker   } else {
310*6dbdd20aSAndroid Build Coastguard Worker     res.status = ReadAndUnwindBatchResult::Status::kReadNone;
311*6dbdd20aSAndroid Build Coastguard Worker   }
312*6dbdd20aSAndroid Build Coastguard Worker   return res;
313*6dbdd20aSAndroid Build Coastguard Worker }
314*6dbdd20aSAndroid Build Coastguard Worker 
BatchUnwindJob(pid_t peer_pid)315*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::BatchUnwindJob(pid_t peer_pid) {
316*6dbdd20aSAndroid Build Coastguard Worker   auto it = client_data_.find(peer_pid);
317*6dbdd20aSAndroid Build Coastguard Worker   if (it == client_data_.end()) {
318*6dbdd20aSAndroid Build Coastguard Worker     // This can happen if the client disconnected before the buffer was fully
319*6dbdd20aSAndroid Build Coastguard Worker     // handled.
320*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DLOG("Unexpected data.");
321*6dbdd20aSAndroid Build Coastguard Worker     return;
322*6dbdd20aSAndroid Build Coastguard Worker   }
323*6dbdd20aSAndroid Build Coastguard Worker   ClientData& client_data = it->second;
324*6dbdd20aSAndroid Build Coastguard Worker   if (client_data.drain_bytes != 0) {
325*6dbdd20aSAndroid Build Coastguard Worker     // This process disconnected and we're reading out the remainder of its
326*6dbdd20aSAndroid Build Coastguard Worker     // buffered data in a dedicated recurring task (DrainJob), so this task has
327*6dbdd20aSAndroid Build Coastguard Worker     // nothing to do.
328*6dbdd20aSAndroid Build Coastguard Worker     return;
329*6dbdd20aSAndroid Build Coastguard Worker   }
330*6dbdd20aSAndroid Build Coastguard Worker 
331*6dbdd20aSAndroid Build Coastguard Worker   bool job_reposted = false;
332*6dbdd20aSAndroid Build Coastguard Worker   bool reader_paused = false;
333*6dbdd20aSAndroid Build Coastguard Worker   switch (ReadAndUnwindBatch(&client_data).status) {
334*6dbdd20aSAndroid Build Coastguard Worker     case ReadAndUnwindBatchResult::Status::kHasMore:
335*6dbdd20aSAndroid Build Coastguard Worker       thread_task_runner_.get()->PostTask(
336*6dbdd20aSAndroid Build Coastguard Worker           [this, peer_pid] { BatchUnwindJob(peer_pid); });
337*6dbdd20aSAndroid Build Coastguard Worker       job_reposted = true;
338*6dbdd20aSAndroid Build Coastguard Worker       break;
339*6dbdd20aSAndroid Build Coastguard Worker     case ReadAndUnwindBatchResult::Status::kReadSome:
340*6dbdd20aSAndroid Build Coastguard Worker       thread_task_runner_.get()->PostDelayedTask(
341*6dbdd20aSAndroid Build Coastguard Worker           [this, peer_pid] { BatchUnwindJob(peer_pid); }, kRetryDelayMs);
342*6dbdd20aSAndroid Build Coastguard Worker       job_reposted = true;
343*6dbdd20aSAndroid Build Coastguard Worker       break;
344*6dbdd20aSAndroid Build Coastguard Worker     case ReadAndUnwindBatchResult::Status::kReadNone:
345*6dbdd20aSAndroid Build Coastguard Worker       client_data.shmem.SetReaderPaused();
346*6dbdd20aSAndroid Build Coastguard Worker       reader_paused = true;
347*6dbdd20aSAndroid Build Coastguard Worker       break;
348*6dbdd20aSAndroid Build Coastguard Worker   }
349*6dbdd20aSAndroid Build Coastguard Worker 
350*6dbdd20aSAndroid Build Coastguard Worker   // We need to either repost the job, or set the reader paused bit. By
351*6dbdd20aSAndroid Build Coastguard Worker   // setting that bit, we inform the client that we want to be notified when
352*6dbdd20aSAndroid Build Coastguard Worker   // new data is written to the shared memory buffer.
353*6dbdd20aSAndroid Build Coastguard Worker   // If we do neither of these things, we will not read from the shared memory
354*6dbdd20aSAndroid Build Coastguard Worker   // buffer again.
355*6dbdd20aSAndroid Build Coastguard Worker   PERFETTO_CHECK(job_reposted || reader_paused);
356*6dbdd20aSAndroid Build Coastguard Worker }
357*6dbdd20aSAndroid Build Coastguard Worker 
DrainJob(pid_t peer_pid)358*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::DrainJob(pid_t peer_pid) {
359*6dbdd20aSAndroid Build Coastguard Worker   auto it = client_data_.find(peer_pid);
360*6dbdd20aSAndroid Build Coastguard Worker   if (it == client_data_.end()) {
361*6dbdd20aSAndroid Build Coastguard Worker     return;
362*6dbdd20aSAndroid Build Coastguard Worker   }
363*6dbdd20aSAndroid Build Coastguard Worker   ClientData& client_data = it->second;
364*6dbdd20aSAndroid Build Coastguard Worker   auto res = ReadAndUnwindBatch(&client_data);
365*6dbdd20aSAndroid Build Coastguard Worker   switch (res.status) {
366*6dbdd20aSAndroid Build Coastguard Worker     case ReadAndUnwindBatchResult::Status::kHasMore:
367*6dbdd20aSAndroid Build Coastguard Worker       if (res.bytes_read < client_data.drain_bytes) {
368*6dbdd20aSAndroid Build Coastguard Worker         client_data.drain_bytes -= res.bytes_read;
369*6dbdd20aSAndroid Build Coastguard Worker         thread_task_runner_.get()->PostTask(
370*6dbdd20aSAndroid Build Coastguard Worker             [this, peer_pid] { DrainJob(peer_pid); });
371*6dbdd20aSAndroid Build Coastguard Worker         return;
372*6dbdd20aSAndroid Build Coastguard Worker       }
373*6dbdd20aSAndroid Build Coastguard Worker       // ReadAndUnwindBatch read more than client_data.drain_bytes.
374*6dbdd20aSAndroid Build Coastguard Worker       break;
375*6dbdd20aSAndroid Build Coastguard Worker     case ReadAndUnwindBatchResult::Status::kReadSome:
376*6dbdd20aSAndroid Build Coastguard Worker       // ReadAndUnwindBatch read all the available data (for now) in the shared
377*6dbdd20aSAndroid Build Coastguard Worker       // memory buffer.
378*6dbdd20aSAndroid Build Coastguard Worker     case ReadAndUnwindBatchResult::Status::kReadNone:
379*6dbdd20aSAndroid Build Coastguard Worker       // There was no data in the shared memory buffer.
380*6dbdd20aSAndroid Build Coastguard Worker       break;
381*6dbdd20aSAndroid Build Coastguard Worker   }
382*6dbdd20aSAndroid Build Coastguard Worker   // No further drain task has been scheduled. Drain is finished. Finish the
383*6dbdd20aSAndroid Build Coastguard Worker   // disconnect operation as well.
384*6dbdd20aSAndroid Build Coastguard Worker 
385*6dbdd20aSAndroid Build Coastguard Worker   FinishDisconnect(it);
386*6dbdd20aSAndroid Build Coastguard Worker }
387*6dbdd20aSAndroid Build Coastguard Worker 
388*6dbdd20aSAndroid Build Coastguard Worker // static
HandleBuffer(UnwindingWorker * self,AllocRecordArena * alloc_record_arena,const SharedRingBuffer::Buffer & buf,ClientData * client_data,pid_t peer_pid,Delegate * delegate)389*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::HandleBuffer(UnwindingWorker* self,
390*6dbdd20aSAndroid Build Coastguard Worker                                    AllocRecordArena* alloc_record_arena,
391*6dbdd20aSAndroid Build Coastguard Worker                                    const SharedRingBuffer::Buffer& buf,
392*6dbdd20aSAndroid Build Coastguard Worker                                    ClientData* client_data,
393*6dbdd20aSAndroid Build Coastguard Worker                                    pid_t peer_pid,
394*6dbdd20aSAndroid Build Coastguard Worker                                    Delegate* delegate) {
395*6dbdd20aSAndroid Build Coastguard Worker   UnwindingMetadata* unwinding_metadata = &client_data->metadata;
396*6dbdd20aSAndroid Build Coastguard Worker   DataSourceInstanceID data_source_instance_id =
397*6dbdd20aSAndroid Build Coastguard Worker       client_data->data_source_instance_id;
398*6dbdd20aSAndroid Build Coastguard Worker   WireMessage msg;
399*6dbdd20aSAndroid Build Coastguard Worker   // TODO(fmayer): standardise on char* or uint8_t*.
400*6dbdd20aSAndroid Build Coastguard Worker   // char* has stronger guarantees regarding aliasing.
401*6dbdd20aSAndroid Build Coastguard Worker   // see https://timsong-cpp.github.io/cppwp/n3337/basic.lval#10.8
402*6dbdd20aSAndroid Build Coastguard Worker   if (!ReceiveWireMessage(reinterpret_cast<char*>(buf.data), buf.size, &msg)) {
403*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DFATAL_OR_ELOG("Failed to receive wire message.");
404*6dbdd20aSAndroid Build Coastguard Worker     return;
405*6dbdd20aSAndroid Build Coastguard Worker   }
406*6dbdd20aSAndroid Build Coastguard Worker 
407*6dbdd20aSAndroid Build Coastguard Worker   if (msg.record_type == RecordType::Malloc) {
408*6dbdd20aSAndroid Build Coastguard Worker     std::unique_ptr<AllocRecord> rec = alloc_record_arena->BorrowAllocRecord();
409*6dbdd20aSAndroid Build Coastguard Worker     rec->alloc_metadata = *msg.alloc_header;
410*6dbdd20aSAndroid Build Coastguard Worker     rec->pid = peer_pid;
411*6dbdd20aSAndroid Build Coastguard Worker     rec->data_source_instance_id = data_source_instance_id;
412*6dbdd20aSAndroid Build Coastguard Worker     auto start_time_us = base::GetWallTimeNs() / 1000;
413*6dbdd20aSAndroid Build Coastguard Worker     if (!client_data->stream_allocations)
414*6dbdd20aSAndroid Build Coastguard Worker       DoUnwind(&msg, unwinding_metadata, rec.get());
415*6dbdd20aSAndroid Build Coastguard Worker     rec->unwinding_time_us = static_cast<uint64_t>(
416*6dbdd20aSAndroid Build Coastguard Worker         ((base::GetWallTimeNs() / 1000) - start_time_us).count());
417*6dbdd20aSAndroid Build Coastguard Worker     delegate->PostAllocRecord(self, std::move(rec));
418*6dbdd20aSAndroid Build Coastguard Worker   } else if (msg.record_type == RecordType::Free) {
419*6dbdd20aSAndroid Build Coastguard Worker     FreeRecord rec;
420*6dbdd20aSAndroid Build Coastguard Worker     rec.pid = peer_pid;
421*6dbdd20aSAndroid Build Coastguard Worker     rec.data_source_instance_id = data_source_instance_id;
422*6dbdd20aSAndroid Build Coastguard Worker     // We need to copy this, so we can return the memory to the shmem buffer.
423*6dbdd20aSAndroid Build Coastguard Worker     memcpy(&rec.entry, msg.free_header, sizeof(*msg.free_header));
424*6dbdd20aSAndroid Build Coastguard Worker     client_data->free_records.emplace_back(std::move(rec));
425*6dbdd20aSAndroid Build Coastguard Worker     if (client_data->free_records.size() == kRecordBatchSize) {
426*6dbdd20aSAndroid Build Coastguard Worker       delegate->PostFreeRecord(self, std::move(client_data->free_records));
427*6dbdd20aSAndroid Build Coastguard Worker       client_data->free_records.clear();
428*6dbdd20aSAndroid Build Coastguard Worker       client_data->free_records.reserve(kRecordBatchSize);
429*6dbdd20aSAndroid Build Coastguard Worker     }
430*6dbdd20aSAndroid Build Coastguard Worker   } else if (msg.record_type == RecordType::HeapName) {
431*6dbdd20aSAndroid Build Coastguard Worker     HeapNameRecord rec;
432*6dbdd20aSAndroid Build Coastguard Worker     rec.pid = peer_pid;
433*6dbdd20aSAndroid Build Coastguard Worker     rec.data_source_instance_id = data_source_instance_id;
434*6dbdd20aSAndroid Build Coastguard Worker     memcpy(&rec.entry, msg.heap_name_header, sizeof(*msg.heap_name_header));
435*6dbdd20aSAndroid Build Coastguard Worker     rec.entry.heap_name[sizeof(rec.entry.heap_name) - 1] = '\0';
436*6dbdd20aSAndroid Build Coastguard Worker     delegate->PostHeapNameRecord(self, std::move(rec));
437*6dbdd20aSAndroid Build Coastguard Worker   } else {
438*6dbdd20aSAndroid Build Coastguard Worker     PERFETTO_DFATAL_OR_ELOG("Invalid record type.");
439*6dbdd20aSAndroid Build Coastguard Worker   }
440*6dbdd20aSAndroid Build Coastguard Worker }
441*6dbdd20aSAndroid Build Coastguard Worker 
PostHandoffSocket(HandoffData handoff_data)442*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::PostHandoffSocket(HandoffData handoff_data) {
443*6dbdd20aSAndroid Build Coastguard Worker   // Even with C++14, this cannot be moved, as std::function has to be
444*6dbdd20aSAndroid Build Coastguard Worker   // copyable, which HandoffData is not.
445*6dbdd20aSAndroid Build Coastguard Worker   HandoffData* raw_data = new HandoffData(std::move(handoff_data));
446*6dbdd20aSAndroid Build Coastguard Worker   // We do not need to use a WeakPtr here because the task runner will not
447*6dbdd20aSAndroid Build Coastguard Worker   // outlive its UnwindingWorker.
448*6dbdd20aSAndroid Build Coastguard Worker   thread_task_runner_.get()->PostTask([this, raw_data] {
449*6dbdd20aSAndroid Build Coastguard Worker     HandoffData data = std::move(*raw_data);
450*6dbdd20aSAndroid Build Coastguard Worker     delete raw_data;
451*6dbdd20aSAndroid Build Coastguard Worker     HandleHandoffSocket(std::move(data));
452*6dbdd20aSAndroid Build Coastguard Worker   });
453*6dbdd20aSAndroid Build Coastguard Worker }
454*6dbdd20aSAndroid Build Coastguard Worker 
HandleHandoffSocket(HandoffData handoff_data)455*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::HandleHandoffSocket(HandoffData handoff_data) {
456*6dbdd20aSAndroid Build Coastguard Worker   auto sock = base::UnixSocket::AdoptConnected(
457*6dbdd20aSAndroid Build Coastguard Worker       handoff_data.sock.ReleaseFd(), this, this->thread_task_runner_.get(),
458*6dbdd20aSAndroid Build Coastguard Worker       base::SockFamily::kUnix, base::SockType::kStream);
459*6dbdd20aSAndroid Build Coastguard Worker   pid_t peer_pid = sock->peer_pid_linux();
460*6dbdd20aSAndroid Build Coastguard Worker 
461*6dbdd20aSAndroid Build Coastguard Worker   UnwindingMetadata metadata(std::move(handoff_data.maps_fd),
462*6dbdd20aSAndroid Build Coastguard Worker                              std::move(handoff_data.mem_fd));
463*6dbdd20aSAndroid Build Coastguard Worker   ClientData client_data{
464*6dbdd20aSAndroid Build Coastguard Worker       handoff_data.data_source_instance_id,
465*6dbdd20aSAndroid Build Coastguard Worker       std::move(sock),
466*6dbdd20aSAndroid Build Coastguard Worker       std::move(metadata),
467*6dbdd20aSAndroid Build Coastguard Worker       std::move(handoff_data.shmem),
468*6dbdd20aSAndroid Build Coastguard Worker       std::move(handoff_data.client_config),
469*6dbdd20aSAndroid Build Coastguard Worker       handoff_data.stream_allocations,
470*6dbdd20aSAndroid Build Coastguard Worker       /*drain_bytes=*/0,
471*6dbdd20aSAndroid Build Coastguard Worker       /*free_records=*/{},
472*6dbdd20aSAndroid Build Coastguard Worker   };
473*6dbdd20aSAndroid Build Coastguard Worker   client_data.free_records.reserve(kRecordBatchSize);
474*6dbdd20aSAndroid Build Coastguard Worker   client_data.shmem.SetReaderPaused();
475*6dbdd20aSAndroid Build Coastguard Worker   client_data_.emplace(peer_pid, std::move(client_data));
476*6dbdd20aSAndroid Build Coastguard Worker   alloc_record_arena_.Enable();
477*6dbdd20aSAndroid Build Coastguard Worker }
478*6dbdd20aSAndroid Build Coastguard Worker 
HandleDrainFree(DataSourceInstanceID ds_id,pid_t pid)479*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::HandleDrainFree(DataSourceInstanceID ds_id, pid_t pid) {
480*6dbdd20aSAndroid Build Coastguard Worker   auto it = client_data_.find(pid);
481*6dbdd20aSAndroid Build Coastguard Worker   if (it != client_data_.end()) {
482*6dbdd20aSAndroid Build Coastguard Worker     ClientData& client_data = it->second;
483*6dbdd20aSAndroid Build Coastguard Worker 
484*6dbdd20aSAndroid Build Coastguard Worker     if (!client_data.free_records.empty()) {
485*6dbdd20aSAndroid Build Coastguard Worker       delegate_->PostFreeRecord(this, std::move(client_data.free_records));
486*6dbdd20aSAndroid Build Coastguard Worker       client_data.free_records.clear();
487*6dbdd20aSAndroid Build Coastguard Worker       client_data.free_records.reserve(kRecordBatchSize);
488*6dbdd20aSAndroid Build Coastguard Worker     }
489*6dbdd20aSAndroid Build Coastguard Worker   }
490*6dbdd20aSAndroid Build Coastguard Worker   delegate_->PostDrainDone(this, ds_id);
491*6dbdd20aSAndroid Build Coastguard Worker }
492*6dbdd20aSAndroid Build Coastguard Worker 
PostDisconnectSocket(pid_t pid)493*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::PostDisconnectSocket(pid_t pid) {
494*6dbdd20aSAndroid Build Coastguard Worker   // We do not need to use a WeakPtr here because the task runner will not
495*6dbdd20aSAndroid Build Coastguard Worker   // outlive its UnwindingWorker.
496*6dbdd20aSAndroid Build Coastguard Worker   thread_task_runner_.get()->PostTask(
497*6dbdd20aSAndroid Build Coastguard Worker       [this, pid] { HandleDisconnectSocket(pid); });
498*6dbdd20aSAndroid Build Coastguard Worker }
499*6dbdd20aSAndroid Build Coastguard Worker 
PostPurgeProcess(pid_t pid)500*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::PostPurgeProcess(pid_t pid) {
501*6dbdd20aSAndroid Build Coastguard Worker   // We do not need to use a WeakPtr here because the task runner will not
502*6dbdd20aSAndroid Build Coastguard Worker   // outlive its UnwindingWorker.
503*6dbdd20aSAndroid Build Coastguard Worker   thread_task_runner_.get()->PostTask([this, pid] {
504*6dbdd20aSAndroid Build Coastguard Worker     auto it = client_data_.find(pid);
505*6dbdd20aSAndroid Build Coastguard Worker     if (it == client_data_.end()) {
506*6dbdd20aSAndroid Build Coastguard Worker       return;
507*6dbdd20aSAndroid Build Coastguard Worker     }
508*6dbdd20aSAndroid Build Coastguard Worker     RemoveClientData(it);
509*6dbdd20aSAndroid Build Coastguard Worker   });
510*6dbdd20aSAndroid Build Coastguard Worker }
511*6dbdd20aSAndroid Build Coastguard Worker 
PostDrainFree(DataSourceInstanceID ds_id,pid_t pid)512*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::PostDrainFree(DataSourceInstanceID ds_id, pid_t pid) {
513*6dbdd20aSAndroid Build Coastguard Worker   // We do not need to use a WeakPtr here because the task runner will not
514*6dbdd20aSAndroid Build Coastguard Worker   // outlive its UnwindingWorker.
515*6dbdd20aSAndroid Build Coastguard Worker   thread_task_runner_.get()->PostTask(
516*6dbdd20aSAndroid Build Coastguard Worker       [this, ds_id, pid] { HandleDrainFree(ds_id, pid); });
517*6dbdd20aSAndroid Build Coastguard Worker }
518*6dbdd20aSAndroid Build Coastguard Worker 
HandleDisconnectSocket(pid_t pid)519*6dbdd20aSAndroid Build Coastguard Worker void UnwindingWorker::HandleDisconnectSocket(pid_t pid) {
520*6dbdd20aSAndroid Build Coastguard Worker   auto it = client_data_.find(pid);
521*6dbdd20aSAndroid Build Coastguard Worker   if (it == client_data_.end()) {
522*6dbdd20aSAndroid Build Coastguard Worker     // This is expected if the client voluntarily disconnects before the
523*6dbdd20aSAndroid Build Coastguard Worker     // profiling session ended. In that case, there is a race between the main
524*6dbdd20aSAndroid Build Coastguard Worker     // thread learning about the disconnect and it calling back here.
525*6dbdd20aSAndroid Build Coastguard Worker     return;
526*6dbdd20aSAndroid Build Coastguard Worker   }
527*6dbdd20aSAndroid Build Coastguard Worker   ClientData& client_data = it->second;
528*6dbdd20aSAndroid Build Coastguard Worker   // Shutdown and call OnDisconnect handler.
529*6dbdd20aSAndroid Build Coastguard Worker   client_data.shmem.SetShuttingDown();
530*6dbdd20aSAndroid Build Coastguard Worker   client_data.sock->Shutdown(/* notify= */ true);
531*6dbdd20aSAndroid Build Coastguard Worker }
532*6dbdd20aSAndroid Build Coastguard Worker 
BorrowAllocRecord()533*6dbdd20aSAndroid Build Coastguard Worker std::unique_ptr<AllocRecord> AllocRecordArena::BorrowAllocRecord() {
534*6dbdd20aSAndroid Build Coastguard Worker   std::lock_guard<std::mutex> l(*alloc_records_mutex_);
535*6dbdd20aSAndroid Build Coastguard Worker   if (!alloc_records_.empty()) {
536*6dbdd20aSAndroid Build Coastguard Worker     std::unique_ptr<AllocRecord> result = std::move(alloc_records_.back());
537*6dbdd20aSAndroid Build Coastguard Worker     alloc_records_.pop_back();
538*6dbdd20aSAndroid Build Coastguard Worker     return result;
539*6dbdd20aSAndroid Build Coastguard Worker   }
540*6dbdd20aSAndroid Build Coastguard Worker   return std::unique_ptr<AllocRecord>(new AllocRecord());
541*6dbdd20aSAndroid Build Coastguard Worker }
542*6dbdd20aSAndroid Build Coastguard Worker 
ReturnAllocRecord(std::unique_ptr<AllocRecord> record)543*6dbdd20aSAndroid Build Coastguard Worker void AllocRecordArena::ReturnAllocRecord(std::unique_ptr<AllocRecord> record) {
544*6dbdd20aSAndroid Build Coastguard Worker   std::lock_guard<std::mutex> l(*alloc_records_mutex_);
545*6dbdd20aSAndroid Build Coastguard Worker   if (enabled_ && record && alloc_records_.size() < kMaxAllocRecordArenaSize)
546*6dbdd20aSAndroid Build Coastguard Worker     alloc_records_.emplace_back(std::move(record));
547*6dbdd20aSAndroid Build Coastguard Worker }
548*6dbdd20aSAndroid Build Coastguard Worker 
Disable()549*6dbdd20aSAndroid Build Coastguard Worker void AllocRecordArena::Disable() {
550*6dbdd20aSAndroid Build Coastguard Worker   std::lock_guard<std::mutex> l(*alloc_records_mutex_);
551*6dbdd20aSAndroid Build Coastguard Worker   alloc_records_.clear();
552*6dbdd20aSAndroid Build Coastguard Worker   enabled_ = false;
553*6dbdd20aSAndroid Build Coastguard Worker }
554*6dbdd20aSAndroid Build Coastguard Worker 
Enable()555*6dbdd20aSAndroid Build Coastguard Worker void AllocRecordArena::Enable() {
556*6dbdd20aSAndroid Build Coastguard Worker   std::lock_guard<std::mutex> l(*alloc_records_mutex_);
557*6dbdd20aSAndroid Build Coastguard Worker   enabled_ = true;
558*6dbdd20aSAndroid Build Coastguard Worker }
559*6dbdd20aSAndroid Build Coastguard Worker 
560*6dbdd20aSAndroid Build Coastguard Worker UnwindingWorker::Delegate::~Delegate() = default;
561*6dbdd20aSAndroid Build Coastguard Worker 
562*6dbdd20aSAndroid Build Coastguard Worker }  // namespace profiling
563*6dbdd20aSAndroid Build Coastguard Worker }  // namespace perfetto
564