Chromium Code Reviews| Index: src/debug.cc |
| diff --git a/src/debug.cc b/src/debug.cc |
| index 2d58ce1f54aefba1e93be618b9a1f76a69f7bcd3..6f60f9b5a65be1ee06105bfdfaeef0a0ab69ba4b 100644 |
| --- a/src/debug.cc |
| +++ b/src/debug.cc |
| @@ -40,6 +40,7 @@ |
| #include "global-handles.h" |
| #include "ic.h" |
| #include "ic-inl.h" |
| +#include "list.h" |
| #include "messages.h" |
| #include "natives.h" |
| #include "stub-cache.h" |
| @@ -1105,6 +1106,8 @@ void Debug::SetBreakPoint(Handle<SharedFunctionInfo> shared, |
| int* source_position) { |
| HandleScope scope(isolate_); |
| + PrepareForBreakPoints(); |
| + |
| if (!EnsureDebugInfo(shared)) { |
| // Return if retrieving debug info failed. |
| return; |
| @@ -1178,6 +1181,7 @@ void Debug::ClearAllBreakPoints() { |
| void Debug::FloodWithOneShot(Handle<SharedFunctionInfo> shared) { |
| + PrepareForBreakPoints(); |
| // Make sure the function has setup the debug info. |
| if (!EnsureDebugInfo(shared)) { |
| // Return if we failed to retrieve the debug info. |
| @@ -1234,6 +1238,9 @@ bool Debug::IsBreakOnException(ExceptionBreakType type) { |
| void Debug::PrepareStep(StepAction step_action, int step_count) { |
| HandleScope scope(isolate_); |
| + |
| + PrepareForBreakPoints(); |
| + |
| ASSERT(Debug::InDebugger()); |
| // Remember this step action and count. |
| @@ -1676,19 +1683,84 @@ void Debug::ClearStepNext() { |
| } |
| -// Ensures the debug information is present for shared. |
| -bool Debug::EnsureDebugInfo(Handle<SharedFunctionInfo> shared) { |
| - // Return if we already have the debug info for shared. |
| - if (HasDebugInfo(shared)) return true; |
| +bool SortedListContains(List<JSFunction*>* functions, JSFunction* function) { |
|
Sven Panne
2011/09/08 08:07:59
Perhaps this should be moved to the List template
Søren Thygesen Gjesse
2011/09/13 07:31:19
Added the following template functions to list.h.
|
| + int low = 0; |
| + int high = functions->length() - 1; |
| + while (low <= high) { |
| + int mid = (low + high) / 2; |
| + JSFunction* mid_function = (*functions)[mid]; |
| - // Ensure shared in compiled. Return false if this failed. |
| - if (!EnsureCompiled(shared, CLEAR_EXCEPTION)) return false; |
| + if (mid_function > function) { |
| + high = mid - 1; |
| + continue; |
| + } |
| + if (mid_function < function) { |
| + low = mid + 1; |
| + continue; |
| + } |
| + // Found the function. |
| + return true; |
| + } |
| + return false; |
| +} |
| + |
| +void Debug::PrepareForBreakPoints() { |
| // If preparing for the first break point make sure to deoptimize all |
| // functions as debugging does not work with optimized code. |
| if (!has_break_points_) { |
| Deoptimizer::DeoptimizeAll(); |
| + |
| + AssertNoAllocation no_allocation; |
| + Builtins* builtins = isolate_->builtins(); |
| + Code* lazy_compile = builtins->builtin(Builtins::kLazyCompile); |
| + |
| + // Find all non-optimized code functions with activation frames on |
| + // the stack. |
| + List<JSFunction*> functions(100); |
|
Sven Panne
2011/09/08 08:07:59
functions => active_functions
Søren Thygesen Gjesse
2011/09/13 07:31:19
Done.
|
| + for (JavaScriptFrameIterator it(isolate_); !it.done(); it.Advance()) { |
| + JavaScriptFrame* frame = it.frame(); |
| + if (frame->function()->IsJSFunction()) { |
| + JSFunction* function = JSFunction::cast(frame->function()); |
| + if (function->code()->kind() == Code::FUNCTION) |
| + functions.Add(function); |
| + } |
| + } |
| + functions.Sort(); |
| + |
| + // Scan the heap for all non-optimized functions which has no |
| + // debug break slots. |
| + HeapIterator iterator; |
| + HeapObject* obj = NULL; |
| + while (((obj = iterator.next()) != NULL)) { |
| + if (obj->IsJSFunction()) { |
| + JSFunction* function = JSFunction::cast(obj); |
| + if (function->shared()->allows_lazy_compilation() && |
| + function->shared()->script()->IsScript() && |
| + function->code()->kind() == Code::FUNCTION && |
| + !function->code()->has_debug_break_slots()) { |
| + bool has_activation = SortedListContains(&functions, function); |
| + if (!has_activation) { |
| + function->set_code(lazy_compile); |
| + function->shared()->set_code(lazy_compile); |
| + } |
| + } |
| + } |
| + } |
| } |
| +} |
| + |
| + |
| +// Ensures the debug information is present for shared. |
| +bool Debug::EnsureDebugInfo(Handle<SharedFunctionInfo> shared) { |
| + // Return if we already have the debug info for shared. |
| + if (HasDebugInfo(shared)) { |
| + ASSERT(shared->is_compiled()); |
| + return true; |
| + } |
| + |
| + // Ensure shared in compiled. Return false if this failed. |
| + if (!EnsureCompiled(shared, CLEAR_EXCEPTION)) return false; |
| // Create the debug info object. |
| Handle<DebugInfo> debug_info = FACTORY->NewDebugInfo(shared); |
| @@ -1739,6 +1811,8 @@ void Debug::RemoveDebugInfo(Handle<DebugInfo> debug_info) { |
| void Debug::SetAfterBreakTarget(JavaScriptFrame* frame) { |
| HandleScope scope(isolate_); |
| + PrepareForBreakPoints(); |
| + |
| // Get the executing function in which the debug break occurred. |
| Handle<SharedFunctionInfo> shared = |
| Handle<SharedFunctionInfo>(JSFunction::cast(frame->function())->shared()); |
| @@ -1829,6 +1903,8 @@ bool Debug::IsBreakAtReturn(JavaScriptFrame* frame) { |
| return false; |
| } |
| + PrepareForBreakPoints(); |
| + |
| // Get the executing function in which the debug break occurred. |
| Handle<SharedFunctionInfo> shared = |
| Handle<SharedFunctionInfo>(JSFunction::cast(frame->function())->shared()); |