diff options
author | 2015-05-15 13:20:41 +0100 | |
---|---|---|
committer | 2015-05-18 14:58:41 +0100 | |
commit | 57f61612a6101a2d246727640676acf8d8a6b51c (patch) | |
tree | 03f63405d5bf4fe38ed580721d05ca5b270c3b66 | |
parent | 17f1bc531ea2f8c1a6fac3def13dee1b901949dd (diff) |
Stack walking of inlined frames.
Be "true" to the stack visitor design, which seems to be:
1) Carry minimal state, so this CL just adds an inlining depth to it.
2) Compute needed information for each method, which is what GetDexPc,
GetMethod, and GetVRegFromOptimized code do.
Change-Id: I3d14126766f12216a7b448ce856259f1e7f37395
-rw-r--r-- | runtime/stack.cc | 77 | ||||
-rw-r--r-- | runtime/stack.h | 20 |
2 files changed, 77 insertions, 20 deletions
diff --git a/runtime/stack.cc b/runtime/stack.cc index 800acaa320..6795516c83 100644 --- a/runtime/stack.cc +++ b/runtime/stack.cc @@ -99,15 +99,45 @@ StackVisitor::StackVisitor(Thread* thread, cur_quick_frame_pc_(0), num_frames_(num_frames), cur_depth_(0), + current_inlining_depth_(0), context_(context) { DCHECK(thread == Thread::Current() || thread->IsSuspended()) << *thread; } +InlineInfo StackVisitor::GetCurrentInlineInfo() const SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) { + mirror::ArtMethod* outer_method = GetCurrentQuickFrame()->AsMirrorPtr(); + uint32_t native_pc_offset = outer_method->NativeQuickPcOffset(cur_quick_frame_pc_); + CodeInfo code_info = outer_method->GetOptimizedCodeInfo(); + StackMap stack_map = code_info.GetStackMapForNativePcOffset(native_pc_offset); + return code_info.GetInlineInfoOf(stack_map); +} + +mirror::ArtMethod* StackVisitor::GetMethod() const SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) { + if (cur_shadow_frame_ != nullptr) { + return cur_shadow_frame_->GetMethod(); + } else if (cur_quick_frame_ != nullptr) { + if (IsInInlinedFrame()) { + size_t depth_in_stack_map = current_inlining_depth_ - 1; + return GetCurrentQuickFrame()->AsMirrorPtr()->GetDexCacheResolvedMethod( + GetCurrentInlineInfo().GetMethodIndexAtDepth(depth_in_stack_map)); + } else { + return cur_quick_frame_->AsMirrorPtr(); + } + } else { + return nullptr; + } +} + uint32_t StackVisitor::GetDexPc(bool abort_on_failure) const { if (cur_shadow_frame_ != nullptr) { return cur_shadow_frame_->GetDexPC(); } else if (cur_quick_frame_ != nullptr) { - return GetMethod()->ToDexPc(cur_quick_frame_pc_, abort_on_failure); + if (IsInInlinedFrame()) { + size_t depth_in_stack_map = current_inlining_depth_ - 1; + return GetCurrentInlineInfo().GetDexPcAtDepth(depth_in_stack_map); + } else { + return GetMethod()->ToDexPc(cur_quick_frame_pc_, abort_on_failure); + } } else { return 0; } @@ -225,18 +255,27 @@ bool StackVisitor::GetVRegFromQuickCode(mirror::ArtMethod* m, uint16_t vreg, VRe bool StackVisitor::GetVRegFromOptimizedCode(mirror::ArtMethod* m, uint16_t vreg, VRegKind kind, uint32_t* val) const { - const void* code_pointer = m->GetQuickOatCodePointer(sizeof(void*)); - DCHECK(code_pointer != nullptr); - uint32_t native_pc_offset = m->NativeQuickPcOffset(cur_quick_frame_pc_); - CodeInfo code_info = m->GetOptimizedCodeInfo(); - StackMap stack_map = code_info.GetStackMapForNativePcOffset(native_pc_offset); + DCHECK_EQ(m, GetMethod()); const DexFile::CodeItem* code_item = m->GetCodeItem(); DCHECK(code_item != nullptr) << PrettyMethod(m); // Can't be null or how would we compile // its instructions? - DCHECK_LT(vreg, code_item->registers_size_); uint16_t number_of_dex_registers = code_item->registers_size_; - DexRegisterMap dex_register_map = - code_info.GetDexRegisterMapOf(stack_map, number_of_dex_registers); + DCHECK_LT(vreg, code_item->registers_size_); + + mirror::ArtMethod* outer_method = GetCurrentQuickFrame()->AsMirrorPtr(); + const void* code_pointer = outer_method->GetQuickOatCodePointer(sizeof(void*)); + DCHECK(code_pointer != nullptr); + CodeInfo code_info = outer_method->GetOptimizedCodeInfo(); + + uint32_t native_pc_offset = outer_method->NativeQuickPcOffset(cur_quick_frame_pc_); + StackMap stack_map = code_info.GetStackMapForNativePcOffset(native_pc_offset); + size_t depth_in_stack_map = current_inlining_depth_ - 1; + + DexRegisterMap dex_register_map = IsInInlinedFrame() + ? code_info.GetDexRegisterMapAtDepth( + depth_in_stack_map, code_info.GetInlineInfoOf(stack_map), number_of_dex_registers) + : code_info.GetDexRegisterMapOf(stack_map, number_of_dex_registers); + DexRegisterLocation::Kind location_kind = dex_register_map.GetLocationKind(vreg, number_of_dex_registers, code_info); switch (location_kind) { @@ -704,6 +743,26 @@ void StackVisitor::WalkStack(bool include_transitions) { mirror::ArtMethod* method = cur_quick_frame_->AsMirrorPtr(); while (method != nullptr) { SanityCheckFrame(); + + if ((walk_kind_ == StackWalkKind::kIncludeInlinedFrames) + && method->IsOptimized(sizeof(void*))) { + CodeInfo code_info = method->GetOptimizedCodeInfo(); + uint32_t native_pc_offset = method->NativeQuickPcOffset(cur_quick_frame_pc_); + StackMap stack_map = code_info.GetStackMapForNativePcOffset(native_pc_offset); + if (stack_map.HasInlineInfo(code_info)) { + InlineInfo inline_info = code_info.GetInlineInfoOf(stack_map); + DCHECK_EQ(current_inlining_depth_, 0u); + for (current_inlining_depth_ = inline_info.GetDepth(); + current_inlining_depth_ != 0; + --current_inlining_depth_) { + bool should_continue = VisitFrame(); + if (UNLIKELY(!should_continue)) { + return; + } + } + } + } + bool should_continue = VisitFrame(); if (UNLIKELY(!should_continue)) { return; diff --git a/runtime/stack.h b/runtime/stack.h index bf6101619d..5b43848de0 100644 --- a/runtime/stack.h +++ b/runtime/stack.h @@ -36,9 +36,10 @@ namespace mirror { } // namespace mirror class Context; -class ShadowFrame; class HandleScope; +class InlineInfo; class ScopedObjectAccess; +class ShadowFrame; class StackVisitor; class Thread; @@ -430,15 +431,7 @@ class StackVisitor { void WalkStack(bool include_transitions = false) SHARED_LOCKS_REQUIRED(Locks::mutator_lock_); - mirror::ArtMethod* GetMethod() const SHARED_LOCKS_REQUIRED(Locks::mutator_lock_) { - if (cur_shadow_frame_ != nullptr) { - return cur_shadow_frame_->GetMethod(); - } else if (cur_quick_frame_ != nullptr) { - return cur_quick_frame_->AsMirrorPtr(); - } else { - return nullptr; - } - } + mirror::ArtMethod* GetMethod() const SHARED_LOCKS_REQUIRED(Locks::mutator_lock_); bool IsShadowFrame() const { return cur_shadow_frame_ != nullptr; @@ -611,7 +604,7 @@ class StackVisitor { } bool IsInInlinedFrame() const { - return false; + return current_inlining_depth_ != 0; } uintptr_t GetCurrentQuickFramePc() const { @@ -703,6 +696,8 @@ class StackVisitor { void SanityCheckFrame() const SHARED_LOCKS_REQUIRED(Locks::mutator_lock_); + InlineInfo GetCurrentInlineInfo() const SHARED_LOCKS_REQUIRED(Locks::mutator_lock_); + Thread* const thread_; const StackWalkKind walk_kind_; ShadowFrame* cur_shadow_frame_; @@ -712,6 +707,9 @@ class StackVisitor { size_t num_frames_; // Depth of the frame we're currently at. size_t cur_depth_; + // Current inlining depth of the method we are currently at. + // 0 if there is no inlined frame. + size_t current_inlining_depth_; protected: Context* const context_; |