xref: /aosp_15_r20/external/cronet/base/profiler/stack_copier_suspend.cc (revision 6777b5387eb2ff775bb5750e3f5d96f37fb7352b)
1*6777b538SAndroid Build Coastguard Worker // Copyright 2019 The Chromium Authors
2*6777b538SAndroid Build Coastguard Worker // Use of this source code is governed by a BSD-style license that can be
3*6777b538SAndroid Build Coastguard Worker // found in the LICENSE file.
4*6777b538SAndroid Build Coastguard Worker 
5*6777b538SAndroid Build Coastguard Worker #include "base/profiler/stack_copier_suspend.h"
6*6777b538SAndroid Build Coastguard Worker 
7*6777b538SAndroid Build Coastguard Worker #include "base/profiler/stack_buffer.h"
8*6777b538SAndroid Build Coastguard Worker #include "base/profiler/suspendable_thread_delegate.h"
9*6777b538SAndroid Build Coastguard Worker 
10*6777b538SAndroid Build Coastguard Worker namespace base {
11*6777b538SAndroid Build Coastguard Worker 
StackCopierSuspend(std::unique_ptr<SuspendableThreadDelegate> thread_delegate)12*6777b538SAndroid Build Coastguard Worker StackCopierSuspend::StackCopierSuspend(
13*6777b538SAndroid Build Coastguard Worker     std::unique_ptr<SuspendableThreadDelegate> thread_delegate)
14*6777b538SAndroid Build Coastguard Worker     : thread_delegate_(std::move(thread_delegate)) {}
15*6777b538SAndroid Build Coastguard Worker 
16*6777b538SAndroid Build Coastguard Worker StackCopierSuspend::~StackCopierSuspend() = default;
17*6777b538SAndroid Build Coastguard Worker 
18*6777b538SAndroid Build Coastguard Worker // Suspends the thread, copies the stack state, and resumes the thread. The
19*6777b538SAndroid Build Coastguard Worker // copied stack state includes the stack itself, the top address of the stack
20*6777b538SAndroid Build Coastguard Worker // copy, and the register context. Returns true on success, and returns the
21*6777b538SAndroid Build Coastguard Worker // copied state via the params.
22*6777b538SAndroid Build Coastguard Worker //
23*6777b538SAndroid Build Coastguard Worker // NO HEAP ALLOCATIONS within the ScopedSuspendThread scope.
CopyStack(StackBuffer * stack_buffer,uintptr_t * stack_top,TimeTicks * timestamp,RegisterContext * thread_context,Delegate * delegate)24*6777b538SAndroid Build Coastguard Worker bool StackCopierSuspend::CopyStack(StackBuffer* stack_buffer,
25*6777b538SAndroid Build Coastguard Worker                                    uintptr_t* stack_top,
26*6777b538SAndroid Build Coastguard Worker                                    TimeTicks* timestamp,
27*6777b538SAndroid Build Coastguard Worker                                    RegisterContext* thread_context,
28*6777b538SAndroid Build Coastguard Worker                                    Delegate* delegate) {
29*6777b538SAndroid Build Coastguard Worker   const uintptr_t top = thread_delegate_->GetStackBaseAddress();
30*6777b538SAndroid Build Coastguard Worker   uintptr_t bottom = 0;
31*6777b538SAndroid Build Coastguard Worker   const uint8_t* stack_copy_bottom = nullptr;
32*6777b538SAndroid Build Coastguard Worker   {
33*6777b538SAndroid Build Coastguard Worker     // Allocation of the ScopedSuspendThread object itself is OK since it
34*6777b538SAndroid Build Coastguard Worker     // necessarily occurs before the thread is suspended by the object.
35*6777b538SAndroid Build Coastguard Worker     std::unique_ptr<SuspendableThreadDelegate::ScopedSuspendThread>
36*6777b538SAndroid Build Coastguard Worker         suspend_thread = thread_delegate_->CreateScopedSuspendThread();
37*6777b538SAndroid Build Coastguard Worker 
38*6777b538SAndroid Build Coastguard Worker     // TimeTicks::Now() is implemented in terms of reads to the timer tick
39*6777b538SAndroid Build Coastguard Worker     // counter or TSC register on x86/x86_64 so is reentrant.
40*6777b538SAndroid Build Coastguard Worker     *timestamp = TimeTicks::Now();
41*6777b538SAndroid Build Coastguard Worker 
42*6777b538SAndroid Build Coastguard Worker     if (!suspend_thread->WasSuccessful())
43*6777b538SAndroid Build Coastguard Worker       return false;
44*6777b538SAndroid Build Coastguard Worker 
45*6777b538SAndroid Build Coastguard Worker     if (!thread_delegate_->GetThreadContext(thread_context))
46*6777b538SAndroid Build Coastguard Worker       return false;
47*6777b538SAndroid Build Coastguard Worker 
48*6777b538SAndroid Build Coastguard Worker     bottom = RegisterContextStackPointer(thread_context);
49*6777b538SAndroid Build Coastguard Worker 
50*6777b538SAndroid Build Coastguard Worker     // The StackBuffer allocation is expected to be at least as large as the
51*6777b538SAndroid Build Coastguard Worker     // largest stack region allocation on the platform, but check just in case
52*6777b538SAndroid Build Coastguard Worker     // it isn't *and* the actual stack itself exceeds the buffer allocation
53*6777b538SAndroid Build Coastguard Worker     // size.
54*6777b538SAndroid Build Coastguard Worker     if ((top - bottom) > stack_buffer->size())
55*6777b538SAndroid Build Coastguard Worker       return false;
56*6777b538SAndroid Build Coastguard Worker 
57*6777b538SAndroid Build Coastguard Worker     if (!thread_delegate_->CanCopyStack(bottom))
58*6777b538SAndroid Build Coastguard Worker       return false;
59*6777b538SAndroid Build Coastguard Worker 
60*6777b538SAndroid Build Coastguard Worker     delegate->OnStackCopy();
61*6777b538SAndroid Build Coastguard Worker 
62*6777b538SAndroid Build Coastguard Worker     stack_copy_bottom = CopyStackContentsAndRewritePointers(
63*6777b538SAndroid Build Coastguard Worker         reinterpret_cast<uint8_t*>(bottom), reinterpret_cast<uintptr_t*>(top),
64*6777b538SAndroid Build Coastguard Worker         StackBuffer::kPlatformStackAlignment, stack_buffer->buffer());
65*6777b538SAndroid Build Coastguard Worker   }
66*6777b538SAndroid Build Coastguard Worker 
67*6777b538SAndroid Build Coastguard Worker   *stack_top = reinterpret_cast<uintptr_t>(stack_copy_bottom) + (top - bottom);
68*6777b538SAndroid Build Coastguard Worker 
69*6777b538SAndroid Build Coastguard Worker   for (uintptr_t* reg :
70*6777b538SAndroid Build Coastguard Worker        thread_delegate_->GetRegistersToRewrite(thread_context)) {
71*6777b538SAndroid Build Coastguard Worker     *reg = RewritePointerIfInOriginalStack(reinterpret_cast<uint8_t*>(bottom),
72*6777b538SAndroid Build Coastguard Worker                                            reinterpret_cast<uintptr_t*>(top),
73*6777b538SAndroid Build Coastguard Worker                                            stack_copy_bottom, *reg);
74*6777b538SAndroid Build Coastguard Worker   }
75*6777b538SAndroid Build Coastguard Worker 
76*6777b538SAndroid Build Coastguard Worker   return true;
77*6777b538SAndroid Build Coastguard Worker }
78*6777b538SAndroid Build Coastguard Worker 
79*6777b538SAndroid Build Coastguard Worker }  // namespace base
80