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()); |