debug-frames.cc 4.42 KB
Newer Older
1 2 3 4 5 6
// Copyright 2015 the V8 project authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

#include "src/debug/debug-frames.h"

7
#include "src/accessors.h"
8
#include "src/frames-inl.h"
9
#include "src/wasm/wasm-interpreter.h"
10
#include "src/wasm/wasm-objects-inl.h"
11

12 13 14
namespace v8 {
namespace internal {

15
FrameInspector::FrameInspector(StandardFrame* frame, int inlined_frame_index,
16
                               Isolate* isolate)
17
    : frame_(frame),
18
      inlined_frame_index_(inlined_frame_index),
19
      isolate_(isolate) {
20 21
  // Extract the relevant information from the frame summary and discard it.
  FrameSummary summary = FrameSummary::Get(frame, inlined_frame_index);
22
  summary.EnsureSourcePositionsAvailable();
23 24 25 26 27 28 29 30 31 32 33

  is_constructor_ = summary.is_constructor();
  source_position_ = summary.SourcePosition();
  function_name_ = summary.FunctionName();
  script_ = Handle<Script>::cast(summary.script());
  receiver_ = summary.receiver();

  if (summary.IsJavaScript()) {
    function_ = summary.AsJavaScript().function();
  }

34 35 36 37
  JavaScriptFrame* js_frame =
      frame->is_java_script() ? javascript_frame() : nullptr;
  DCHECK(js_frame || frame->is_wasm());
  has_adapted_arguments_ = js_frame && js_frame->has_adapted_arguments();
38
  is_optimized_ = frame_->is_optimized();
39
  is_interpreted_ = frame_->is_interpreted();
40

41
  // Calculate the deoptimized frame.
42
  if (is_optimized_) {
43
    DCHECK_NOT_NULL(js_frame);
44 45 46 47
    deoptimized_frame_.reset(Deoptimizer::DebuggerInspectableFrame(
        js_frame, inlined_frame_index, isolate));
  } else if (frame_->is_wasm_interpreter_entry()) {
    wasm_interpreted_frame_ =
48 49 50
        WasmInterpreterEntryFrame::cast(frame_)
            ->debug_info()
            ->GetInterpretedFrame(frame_->fp(), inlined_frame_index);
51
    DCHECK(wasm_interpreted_frame_);
52 53 54
  }
}

55
// NOLINTNEXTLINE
56
FrameInspector::~FrameInspector() {
57 58
  // Destructor needs to be defined in the .cc file, because it instantiates
  // std::unique_ptr destructors but the types are not known in the header.
59 60 61
}

int FrameInspector::GetParametersCount() {
62 63 64 65
  if (is_optimized_) return deoptimized_frame_->parameters_count();
  if (wasm_interpreted_frame_)
    return wasm_interpreted_frame_->GetParameterCount();
  return frame_->ComputeParametersCount();
66 67
}

68
Handle<Object> FrameInspector::GetParameter(int index) {
69 70 71
  if (is_optimized_) return deoptimized_frame_->GetParameter(index);
  // TODO(clemensh): Handle wasm_interpreted_frame_.
  return handle(frame_->GetParameter(index), isolate_);
72 73
}

74
Handle<Object> FrameInspector::GetExpression(int index) {
75
  return is_optimized_ ? deoptimized_frame_->GetExpression(index)
76
                       : handle(frame_->GetExpression(index), isolate_);
77 78
}

79 80 81
Handle<Object> FrameInspector::GetContext() {
  return deoptimized_frame_ ? deoptimized_frame_->GetContext()
                            : handle(frame_->context(), isolate_);
82 83
}

84
bool FrameInspector::IsWasm() { return frame_->is_wasm(); }
85

86
bool FrameInspector::IsJavaScript() { return frame_->is_java_script(); }
87 88 89 90 91 92

bool FrameInspector::ParameterIsShadowedByContextLocal(
    Handle<ScopeInfo> info, Handle<String> parameter_name) {
  VariableMode mode;
  InitializationFlag init_flag;
  MaybeAssignedFlag maybe_assigned_flag;
93
  return ScopeInfo::ContextSlotIndex(*info, *parameter_name, &mode, &init_flag,
94
                                     &maybe_assigned_flag) != -1;
95
}
96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121

RedirectActiveFunctions::RedirectActiveFunctions(SharedFunctionInfo shared,
                                                 Mode mode)
    : shared_(shared), mode_(mode) {
  DCHECK(shared->HasBytecodeArray());
  if (mode == Mode::kUseDebugBytecode) {
    DCHECK(shared->HasDebugInfo());
  }
}

void RedirectActiveFunctions::VisitThread(Isolate* isolate,
                                          ThreadLocalTop* top) {
  for (JavaScriptFrameIterator it(isolate, top); !it.done(); it.Advance()) {
    JavaScriptFrame* frame = it.frame();
    JSFunction function = frame->function();
    if (!frame->is_interpreted()) continue;
    if (function->shared() != shared_) continue;
    InterpretedFrame* interpreted_frame =
        reinterpret_cast<InterpretedFrame*>(frame);
    BytecodeArray bytecode = mode_ == Mode::kUseDebugBytecode
                                 ? shared_->GetDebugInfo()->DebugBytecodeArray()
                                 : shared_->GetBytecodeArray();
    interpreted_frame->PatchBytecodeArray(bytecode);
  }
}

122 123
}  // namespace internal
}  // namespace v8