// Copyright 2012 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.h" #include #include #include "src/api/api-inl.h" #include "src/api/api-natives.h" #include "src/base/platform/mutex.h" #include "src/builtins/builtins.h" #include "src/codegen/assembler-inl.h" #include "src/codegen/compilation-cache.h" #include "src/codegen/compiler.h" #include "src/common/assert-scope.h" #include "src/common/globals.h" #include "src/common/message-template.h" #include "src/debug/debug-evaluate.h" #include "src/debug/liveedit.h" #include "src/deoptimizer/deoptimizer.h" #include "src/execution/arguments.h" #include "src/execution/execution.h" #include "src/execution/frames-inl.h" #include "src/execution/isolate-inl.h" #include "src/execution/v8threads.h" #include "src/handles/global-handles-inl.h" #include "src/heap/heap-inl.h" // For NextDebuggingId. #include "src/init/bootstrapper.h" #include "src/interpreter/bytecode-array-iterator.h" #include "src/interpreter/interpreter.h" #include "src/logging/counters.h" #include "src/logging/runtime-call-stats-scope.h" #include "src/objects/api-callbacks-inl.h" #include "src/objects/debug-objects-inl.h" #include "src/objects/js-generator-inl.h" #include "src/objects/js-promise-inl.h" #include "src/objects/slots.h" #include "src/snapshot/embedded/embedded-data.h" #include "src/snapshot/snapshot.h" #if V8_ENABLE_WEBASSEMBLY #include "src/wasm/wasm-debug.h" #include "src/wasm/wasm-objects-inl.h" #endif // V8_ENABLE_WEBASSEMBLY namespace v8 { namespace internal { class Debug::TemporaryObjectsTracker : public HeapObjectAllocationTracker { public: TemporaryObjectsTracker() = default; ~TemporaryObjectsTracker() override = default; TemporaryObjectsTracker(const TemporaryObjectsTracker&) = delete; TemporaryObjectsTracker& operator=(const TemporaryObjectsTracker&) = delete; void AllocationEvent(Address addr, int) override { if (disabled) return; objects_.insert(addr); } void MoveEvent(Address from, Address to, int) override { if (from == to) return; base::MutexGuard guard(&mutex_); auto it = objects_.find(from); if (it == objects_.end()) { // If temporary object was collected we can get MoveEvent which moves // existing non temporary object to the address where we had temporary // object. So we should mark new address as non temporary. objects_.erase(to); return; } objects_.erase(it); objects_.insert(to); } bool HasObject(Handle obj) const { if (obj->IsJSObject() && Handle::cast(obj)->GetEmbedderFieldCount()) { // Embedder may store any pointers using embedder fields and implements // non trivial logic, e.g. create wrappers lazily and store pointer to // native object inside embedder field. We should consider all objects // with embedder fields as non temporary. return false; } return objects_.find(obj->address()) != objects_.end(); } bool disabled = false; private: std::unordered_set
objects_; base::Mutex mutex_; }; Debug::Debug(Isolate* isolate) : is_active_(false), hook_on_function_call_(false), is_suppressed_(false), break_disabled_(false), break_points_active_(true), break_on_exception_(false), break_on_uncaught_exception_(false), side_effect_check_failed_(false), debug_info_list_(nullptr), feature_tracker_(isolate), isolate_(isolate) { ThreadInit(); } Debug::~Debug() { DCHECK_NULL(debug_delegate_); } BreakLocation BreakLocation::FromFrame(Handle debug_info, JavaScriptFrame* frame) { if (debug_info->CanBreakAtEntry()) { return BreakLocation(Debug::kBreakAtEntryPosition, DEBUG_BREAK_AT_ENTRY); } auto summary = FrameSummary::GetTop(frame).AsJavaScript(); int offset = summary.code_offset(); Handle abstract_code = summary.abstract_code(); BreakIterator it(debug_info); it.SkipTo(BreakIndexFromCodeOffset(debug_info, abstract_code, offset)); return it.GetBreakLocation(); } MaybeHandle Debug::CheckBreakPointsForLocations( Handle debug_info, std::vector& break_locations, bool* has_break_points) { Handle break_points_hit = isolate_->factory()->NewFixedArray( debug_info->GetBreakPointCount(isolate_)); int break_points_hit_count = 0; bool has_break_points_at_all = false; for (size_t i = 0; i < break_locations.size(); i++) { bool location_has_break_points; MaybeHandle check_result = CheckBreakPoints( debug_info, &break_locations[i], &location_has_break_points); has_break_points_at_all |= location_has_break_points; if (!check_result.is_null()) { Handle break_points_current_hit = check_result.ToHandleChecked(); int num_objects = break_points_current_hit->length(); for (int j = 0; j < num_objects; ++j) { break_points_hit->set(break_points_hit_count++, break_points_current_hit->get(j)); } } } *has_break_points = has_break_points_at_all; if (break_points_hit_count == 0) return {}; break_points_hit->Shrink(isolate_, break_points_hit_count); return break_points_hit; } void BreakLocation::AllAtCurrentStatement( Handle debug_info, JavaScriptFrame* frame, std::vector* result_out) { DCHECK(!debug_info->CanBreakAtEntry()); auto summary = FrameSummary::GetTop(frame).AsJavaScript(); int offset = summary.code_offset(); Handle abstract_code = summary.abstract_code(); if (abstract_code->IsCode()) offset = offset - 1; int statement_position; { BreakIterator it(debug_info); it.SkipTo(BreakIndexFromCodeOffset(debug_info, abstract_code, offset)); statement_position = it.statement_position(); } for (BreakIterator it(debug_info); !it.Done(); it.Next()) { if (it.statement_position() == statement_position) { result_out->push_back(it.GetBreakLocation()); } } } JSGeneratorObject BreakLocation::GetGeneratorObjectForSuspendedFrame( JavaScriptFrame* frame) const { DCHECK(IsSuspend()); DCHECK_GE(generator_obj_reg_index_, 0); Object generator_obj = UnoptimizedFrame::cast(frame)->ReadInterpreterRegister( generator_obj_reg_index_); return JSGeneratorObject::cast(generator_obj); } int BreakLocation::BreakIndexFromCodeOffset(Handle debug_info, Handle abstract_code, int offset) { // Run through all break points to locate the one closest to the address. int closest_break = 0; int distance = kMaxInt; DCHECK(0 <= offset && offset < abstract_code->Size()); for (BreakIterator it(debug_info); !it.Done(); it.Next()) { // Check if this break point is closer that what was previously found. if (it.code_offset() <= offset && offset - it.code_offset() < distance) { closest_break = it.break_index(); distance = offset - it.code_offset(); // Check whether we can't get any closer. if (distance == 0) break; } } return closest_break; } bool BreakLocation::HasBreakPoint(Isolate* isolate, Handle debug_info) const { // First check whether there is a break point with the same source position. if (!debug_info->HasBreakPoint(isolate, position_)) return false; if (debug_info->CanBreakAtEntry()) { DCHECK_EQ(Debug::kBreakAtEntryPosition, position_); return debug_info->BreakAtEntry(); } else { // Then check whether a break point at that source position would have // the same code offset. Otherwise it's just a break location that we can // step to, but not actually a location where we can put a break point. DCHECK(abstract_code_->IsBytecodeArray()); BreakIterator it(debug_info); it.SkipToPosition(position_); return it.code_offset() == code_offset_; } } debug::BreakLocationType BreakLocation::type() const { switch (type_) { case DEBUGGER_STATEMENT: return debug::kDebuggerStatementBreakLocation; case DEBUG_BREAK_SLOT_AT_CALL: return debug::kCallBreakLocation; case DEBUG_BREAK_SLOT_AT_RETURN: return debug::kReturnBreakLocation; // Externally, suspend breaks should look like normal breaks. case DEBUG_BREAK_SLOT_AT_SUSPEND: default: return debug::kCommonBreakLocation; } } BreakIterator::BreakIterator(Handle debug_info) : debug_info_(debug_info), break_index_(-1), source_position_iterator_( debug_info->DebugBytecodeArray().SourcePositionTable()) { position_ = debug_info->shared().StartPosition(); statement_position_ = position_; // There is at least one break location. DCHECK(!Done()); Next(); } int BreakIterator::BreakIndexFromPosition(int source_position) { for (; !Done(); Next()) { if (source_position <= position()) { int first_break = break_index(); for (; !Done(); Next()) { if (source_position == position()) return break_index(); } return first_break; } } return break_index(); } void BreakIterator::Next() { DisallowGarbageCollection no_gc; DCHECK(!Done()); bool first = break_index_ == -1; while (!Done()) { if (!first) source_position_iterator_.Advance(); first = false; if (Done()) return; position_ = source_position_iterator_.source_position().ScriptOffset(); if (source_position_iterator_.is_statement()) { statement_position_ = position_; } DCHECK_LE(0, position_); DCHECK_LE(0, statement_position_); DebugBreakType type = GetDebugBreakType(); if (type != NOT_DEBUG_BREAK) break; } break_index_++; } DebugBreakType BreakIterator::GetDebugBreakType() { BytecodeArray bytecode_array = debug_info_->OriginalBytecodeArray(); interpreter::Bytecode bytecode = interpreter::Bytecodes::FromByte(bytecode_array.get(code_offset())); // Make sure we read the actual bytecode, not a prefix scaling bytecode. if (interpreter::Bytecodes::IsPrefixScalingBytecode(bytecode)) { bytecode = interpreter::Bytecodes::FromByte(bytecode_array.get(code_offset() + 1)); } if (bytecode == interpreter::Bytecode::kDebugger) { return DEBUGGER_STATEMENT; } else if (bytecode == interpreter::Bytecode::kReturn) { return DEBUG_BREAK_SLOT_AT_RETURN; } else if (bytecode == interpreter::Bytecode::kSuspendGenerator) { return DEBUG_BREAK_SLOT_AT_SUSPEND; } else if (interpreter::Bytecodes::IsCallOrConstruct(bytecode)) { return DEBUG_BREAK_SLOT_AT_CALL; } else if (source_position_iterator_.is_statement()) { return DEBUG_BREAK_SLOT; } else { return NOT_DEBUG_BREAK; } } void BreakIterator::SkipToPosition(int position) { BreakIterator it(debug_info_); SkipTo(it.BreakIndexFromPosition(position)); } void BreakIterator::SetDebugBreak() { DebugBreakType debug_break_type = GetDebugBreakType(); if (debug_break_type == DEBUGGER_STATEMENT) return; HandleScope scope(isolate()); DCHECK(debug_break_type >= DEBUG_BREAK_SLOT); Handle bytecode_array(debug_info_->DebugBytecodeArray(), isolate()); interpreter::BytecodeArrayIterator(bytecode_array, code_offset()) .ApplyDebugBreak(); } void BreakIterator::ClearDebugBreak() { DebugBreakType debug_break_type = GetDebugBreakType(); if (debug_break_type == DEBUGGER_STATEMENT) return; DCHECK(debug_break_type >= DEBUG_BREAK_SLOT); BytecodeArray bytecode_array = debug_info_->DebugBytecodeArray(); BytecodeArray original = debug_info_->OriginalBytecodeArray(); bytecode_array.set(code_offset(), original.get(code_offset())); } BreakLocation BreakIterator::GetBreakLocation() { Handle code( AbstractCode::cast(debug_info_->DebugBytecodeArray()), isolate()); DebugBreakType type = GetDebugBreakType(); int generator_object_reg_index = -1; int generator_suspend_id = -1; if (type == DEBUG_BREAK_SLOT_AT_SUSPEND) { // For suspend break, we'll need the generator object to be able to step // over the suspend as if it didn't return. We get the interpreter register // index that holds the generator object by reading it directly off the // bytecode array, and we'll read the actual generator object off the // interpreter stack frame in GetGeneratorObjectForSuspendedFrame. BytecodeArray bytecode_array = debug_info_->OriginalBytecodeArray(); interpreter::BytecodeArrayIterator iterator( handle(bytecode_array, isolate()), code_offset()); DCHECK_EQ(iterator.current_bytecode(), interpreter::Bytecode::kSuspendGenerator); interpreter::Register generator_obj_reg = iterator.GetRegisterOperand(0); generator_object_reg_index = generator_obj_reg.index(); // Also memorize the suspend ID, to be able to decide whether // we are paused on the implicit initial yield later. generator_suspend_id = iterator.GetUnsignedImmediateOperand(3); } return BreakLocation(code, type, code_offset(), position_, generator_object_reg_index, generator_suspend_id); } Isolate* BreakIterator::isolate() { return debug_info_->GetIsolate(); } void DebugFeatureTracker::Track(DebugFeatureTracker::Feature feature) { uint32_t mask = 1 << feature; // Only count one sample per feature and isolate. if (bitfield_ & mask) return; isolate_->counters()->debug_feature_usage()->AddSample(feature); bitfield_ |= mask; } // Threading support. void Debug::ThreadInit() { thread_local_.break_frame_id_ = StackFrameId::NO_ID; thread_local_.last_step_action_ = StepNone; thread_local_.last_statement_position_ = kNoSourcePosition; thread_local_.last_frame_count_ = -1; thread_local_.fast_forward_to_return_ = false; thread_local_.ignore_step_into_function_ = Smi::zero(); thread_local_.target_frame_count_ = -1; thread_local_.return_value_ = Smi::zero(); thread_local_.last_breakpoint_id_ = 0; clear_suspended_generator(); base::Relaxed_Store(&thread_local_.current_debug_scope_, static_cast(0)); thread_local_.break_on_next_function_call_ = false; UpdateHookOnFunctionCall(); thread_local_.promise_stack_ = Smi::zero(); } char* Debug::ArchiveDebug(char* storage) { MemCopy(storage, reinterpret_cast(&thread_local_), ArchiveSpacePerThread()); return storage + ArchiveSpacePerThread(); } char* Debug::RestoreDebug(char* storage) { MemCopy(reinterpret_cast(&thread_local_), storage, ArchiveSpacePerThread()); // Enter the debugger. DebugScope debug_scope(this); // Clear any one-shot breakpoints that may have been set by the other // thread, and reapply breakpoints for this thread. ClearOneShot(); if (thread_local_.last_step_action_ != StepNone) { int current_frame_count = CurrentFrameCount(); int target_frame_count = thread_local_.target_frame_count_; DCHECK(current_frame_count >= target_frame_count); StackTraceFrameIterator frames_it(isolate_); while (current_frame_count > target_frame_count) { current_frame_count -= frames_it.FrameFunctionCount(); frames_it.Advance(); } DCHECK(current_frame_count == target_frame_count); // Set frame to what it was at Step break thread_local_.break_frame_id_ = frames_it.frame()->id(); // Reset the previous step action for this thread. PrepareStep(thread_local_.last_step_action_); } return storage + ArchiveSpacePerThread(); } int Debug::ArchiveSpacePerThread() { return sizeof(ThreadLocal); } void Debug::Iterate(RootVisitor* v) { Iterate(v, &thread_local_); } char* Debug::Iterate(RootVisitor* v, char* thread_storage) { ThreadLocal* thread_local_data = reinterpret_cast(thread_storage); Iterate(v, thread_local_data); return thread_storage + ArchiveSpacePerThread(); } void Debug::Iterate(RootVisitor* v, ThreadLocal* thread_local_data) { v->VisitRootPointer(Root::kDebug, nullptr, FullObjectSlot(&thread_local_data->return_value_)); v->VisitRootPointer(Root::kDebug, nullptr, FullObjectSlot(&thread_local_data->suspended_generator_)); v->VisitRootPointer( Root::kDebug, nullptr, FullObjectSlot(&thread_local_data->ignore_step_into_function_)); v->VisitRootPointer(Root::kDebug, nullptr, FullObjectSlot(&thread_local_data->promise_stack_)); } DebugInfoListNode::DebugInfoListNode(Isolate* isolate, DebugInfo debug_info) : next_(nullptr) { // Globalize the request debug info object and make it weak. GlobalHandles* global_handles = isolate->global_handles(); debug_info_ = global_handles->Create(debug_info).location(); } DebugInfoListNode::~DebugInfoListNode() { if (debug_info_ == nullptr) return; GlobalHandles::Destroy(debug_info_); debug_info_ = nullptr; } void Debug::Unload() { RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); ClearAllBreakPoints(); ClearStepping(); RemoveAllCoverageInfos(); ClearAllDebuggerHints(); debug_delegate_ = nullptr; } void Debug::OnInstrumentationBreak() { RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); if (!debug_delegate_) return; DCHECK(in_debug_scope()); HandleScope scope(isolate_); DisableBreak no_recursive_break(this); Handle native_context(isolate_->native_context()); debug_delegate_->BreakOnInstrumentation(v8::Utils::ToLocal(native_context), kInstrumentationId); } void Debug::Break(JavaScriptFrame* frame, Handle break_target) { RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); // Just continue if breaks are disabled or debugger cannot be loaded. if (break_disabled()) return; // Enter the debugger. DebugScope debug_scope(this); DisableBreak no_recursive_break(this); // Return if we fail to retrieve debug info. Handle shared(break_target->shared(), isolate_); if (!EnsureBreakInfo(shared)) return; PrepareFunctionForDebugExecution(shared); Handle debug_info(shared->GetDebugInfo(), isolate_); // Find the break location where execution has stopped. BreakLocation location = BreakLocation::FromFrame(debug_info, frame); const bool hitInstrumentationBreak = IsBreakOnInstrumentation(debug_info, location); if (hitInstrumentationBreak) { OnInstrumentationBreak(); } // Find actual break points, if any, and trigger debug break event. bool has_break_points; MaybeHandle break_points_hit = CheckBreakPoints(debug_info, &location, &has_break_points); if (!break_points_hit.is_null() || break_on_next_function_call()) { StepAction lastStepAction = last_step_action(); // Clear all current stepping setup. ClearStepping(); // Notify the debug event listeners. OnDebugBreak(!break_points_hit.is_null() ? break_points_hit.ToHandleChecked() : isolate_->factory()->empty_fixed_array(), lastStepAction); return; } // Debug break at function entry, do not worry about stepping. if (location.IsDebugBreakAtEntry()) { DCHECK(debug_info->BreakAtEntry()); return; } DCHECK_NOT_NULL(frame); // No break point. Check for stepping. StepAction step_action = last_step_action(); int current_frame_count = CurrentFrameCount(); int target_frame_count = thread_local_.target_frame_count_; int last_frame_count = thread_local_.last_frame_count_; // StepOut at not return position was requested and return break locations // were flooded with one shots. if (thread_local_.fast_forward_to_return_) { // We might hit an instrumentation breakpoint before running into a // return/suspend location. DCHECK(location.IsReturnOrSuspend() || hitInstrumentationBreak); // We have to ignore recursive calls to function. if (current_frame_count > target_frame_count) return; ClearStepping(); PrepareStep(StepOut); return; } bool step_break = false; switch (step_action) { case StepNone: return; case StepOut: // StepOut should not break in a deeper frame than target frame. if (current_frame_count > target_frame_count) return; step_break = true; break; case StepOver: // StepOver should not break in a deeper frame than target frame. if (current_frame_count > target_frame_count) return; V8_FALLTHROUGH; case StepInto: { // Special case StepInto and StepOver for generators that are about to // suspend, in which case we go into "generator stepping" mode. The // exception here is the initial implicit yield in generators (which // always has a suspend ID of 0), where we return to the caller first, // instead of triggering "generator stepping" mode straight away. if (location.IsSuspend() && (!IsGeneratorFunction(shared->kind()) || location.generator_suspend_id() > 0)) { DCHECK(!has_suspended_generator()); thread_local_.suspended_generator_ = location.GetGeneratorObjectForSuspendedFrame(frame); ClearStepping(); return; } FrameSummary summary = FrameSummary::GetTop(frame); step_break = step_break || location.IsReturn() || current_frame_count != last_frame_count || thread_local_.last_statement_position_ != summary.SourceStatementPosition(); break; } } StepAction lastStepAction = last_step_action(); // Clear all current stepping setup. ClearStepping(); if (step_break) { // Notify the debug event listeners. OnDebugBreak(isolate_->factory()->empty_fixed_array(), lastStepAction); } else { // Re-prepare to continue. PrepareStep(step_action); } } bool Debug::IsBreakOnInstrumentation(Handle debug_info, const BreakLocation& location) { RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); bool has_break_points_to_check = break_points_active_ && location.HasBreakPoint(isolate_, debug_info); if (!has_break_points_to_check) return {}; Handle break_points = debug_info->GetBreakPoints(isolate_, location.position()); DCHECK(!break_points->IsUndefined(isolate_)); if (!break_points->IsFixedArray()) { const Handle break_point = Handle::cast(break_points); return break_point->id() == kInstrumentationId; } Handle array(FixedArray::cast(*break_points), isolate_); for (int i = 0; i < array->length(); ++i) { const Handle break_point = Handle::cast(handle(array->get(i), isolate_)); if (break_point->id() == kInstrumentationId) { return true; } } return false; } // Find break point objects for this location, if any, and evaluate them. // Return an array of break point objects that evaluated true, or an empty // handle if none evaluated true. // has_break_points will be true, if there is any (non-instrumentation) // breakpoint. MaybeHandle Debug::CheckBreakPoints(Handle debug_info, BreakLocation* location, bool* has_break_points) { RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); bool has_break_points_to_check = break_points_active_ && location->HasBreakPoint(isolate_, debug_info); if (!has_break_points_to_check) { *has_break_points = false; return {}; } return Debug::GetHitBreakPoints(debug_info, location->position(), has_break_points); } bool Debug::IsMutedAtCurrentLocation(JavaScriptFrame* frame) { // A break location is considered muted if break locations on the current // statement have at least one break point, and all of these break points // evaluate to false. Aside from not triggering a debug break event at the // break location, we also do not trigger one for debugger statements, nor // an exception event on exception at this location. RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); HandleScope scope(isolate_); bool has_break_points; MaybeHandle checked = GetHitBreakpointsAtCurrentStatement(frame, &has_break_points); return has_break_points && checked.is_null(); } MaybeHandle Debug::GetHitBreakpointsAtCurrentStatement( JavaScriptFrame* frame, bool* has_break_points) { RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); FrameSummary summary = FrameSummary::GetTop(frame); Handle function = summary.AsJavaScript().function(); if (!function->shared().HasBreakInfo()) { *has_break_points = false; return {}; } Handle debug_info(function->shared().GetDebugInfo(), isolate_); // Enter the debugger. DebugScope debug_scope(this); std::vector break_locations; BreakLocation::AllAtCurrentStatement(debug_info, frame, &break_locations); return CheckBreakPointsForLocations(debug_info, break_locations, has_break_points); } // Check whether a single break point object is triggered. bool Debug::CheckBreakPoint(Handle break_point, bool is_break_at_entry) { RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); HandleScope scope(isolate_); // Instrumentation breakpoints are handled separately. if (break_point->id() == kInstrumentationId) { return false; } if (!break_point->condition().length()) return true; Handle condition(break_point->condition(), isolate_); MaybeHandle maybe_result; Handle result; if (is_break_at_entry) { maybe_result = DebugEvaluate::WithTopmostArguments(isolate_, condition); } else { // Since we call CheckBreakpoint only for deoptimized frame on top of stack, // we can use 0 as index of inlined frame. const int inlined_jsframe_index = 0; const bool throw_on_side_effect = false; maybe_result = DebugEvaluate::Local(isolate_, break_frame_id(), inlined_jsframe_index, condition, throw_on_side_effect); } if (!maybe_result.ToHandle(&result)) { if (isolate_->has_pending_exception()) { isolate_->clear_pending_exception(); } return false; } return result->BooleanValue(isolate_); } bool Debug::SetBreakpoint(Handle shared, Handle break_point, int* source_position) { RCS_SCOPE(isolate_, RuntimeCallCounterId::kDebugger); HandleScope scope(isolate_); // Make sure the function is compiled and has set up the debug info. if (!EnsureBreakInfo(shared)) return false; PrepareFunctionForDebugExecution(shared); Handle debug_info(shared->GetDebugInfo(), isolate_); // Source positions starts with zero. DCHECK_LE(0, *source_position); // Find the break point and change it. *source_position = FindBreakablePosition(debug_info, *source_position); DebugInfo::SetBreakPoint(isolate_, debug_info, *source_position, break_point); // At least one active break point now. DCHECK_LT(0, debug_info->GetBreakPointCount(isolate_)); ClearBreakPoints(debug_info); ApplyBreakPoints(debug_info); feature_tracker()->Track(DebugFeatureTracker::kBreakPoint); return true; } bool Debug::SetBreakPointForScript(Handle