| // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| |
| #include "vm/dart_api_impl.h" |
| #include "vm/dart_api_message.h" |
| #include "vm/debugger.h" |
| #include "vm/message.h" |
| #include "vm/unit_test.h" |
| |
| namespace dart { |
| |
| #ifndef PRODUCT |
| |
| DECLARE_FLAG(bool, background_compilation); |
| DECLARE_FLAG(bool, enable_inlining_annotations); |
| DECLARE_FLAG(bool, prune_dead_locals); |
| DECLARE_FLAG(bool, remove_script_timestamps_for_test); |
| DECLARE_FLAG(bool, trace_rewind); |
| DECLARE_FLAG(int, optimization_counter_threshold); |
| |
| // Search for the formatted string in buffer. |
| // |
| // TODO(turnidge): This function obscures the line number of failing |
| // EXPECTs. Rework this. |
| static void ExpectSubstringF(const char* buff, const char* fmt, ...) { |
| va_list args; |
| va_start(args, fmt); |
| intptr_t len = OS::VSNPrint(NULL, 0, fmt, args); |
| va_end(args); |
| |
| char* buffer = Thread::Current()->zone()->Alloc<char>(len + 1); |
| va_list args2; |
| va_start(args2, fmt); |
| OS::VSNPrint(buffer, (len + 1), fmt, args2); |
| va_end(args2); |
| |
| EXPECT_SUBSTRING(buffer, buff); |
| } |
| |
| |
| TEST_CASE(Debugger_PrintBreakpointsToJSONArray) { |
| const char* kScriptChars = |
| "main() {\n" |
| " var x = new StringBuffer();\n" |
| " x.add('won');\n" |
| " x.add('too');\n" |
| " return x.toString();\n" |
| "}\n"; |
| SetFlagScope<bool> sfs(&FLAG_remove_script_timestamps_for_test, true); |
| Dart_Handle lib = TestCase::LoadTestScript(kScriptChars, NULL); |
| EXPECT_VALID(lib); |
| Library& vmlib = Library::Handle(); |
| vmlib ^= Api::UnwrapHandle(lib); |
| EXPECT(!vmlib.IsNull()); |
| const String& private_key = String::Handle(vmlib.private_key()); |
| |
| Isolate* isolate = Isolate::Current(); |
| Debugger* debugger = isolate->debugger(); |
| |
| // Empty case. |
| { |
| JSONStream js; |
| { |
| JSONArray jsarr(&js); |
| debugger->PrintBreakpointsToJSONArray(&jsarr); |
| } |
| EXPECT_STREQ("[]", js.ToCString()); |
| } |
| |
| // Test with a couple of breakpoints. |
| Dart_Handle url = NewString(TestCase::url()); |
| EXPECT_VALID(Dart_SetBreakpoint(url, 2)); |
| EXPECT_VALID(Dart_SetBreakpoint(url, 3)); |
| { |
| JSONStream js; |
| { |
| JSONArray jsarr(&js); |
| debugger->PrintBreakpointsToJSONArray(&jsarr); |
| } |
| ExpectSubstringF( |
| js.ToCString(), |
| "[{\"type\":\"Breakpoint\",\"fixedId\":true,\"id\":\"breakpoints\\/2\"," |
| "\"breakpointNumber\":2,\"resolved\":false," |
| "\"location\":{\"type\":\"UnresolvedSourceLocation\"," |
| "\"script\":{\"type\":\"@Script\",\"fixedId\":true," |
| "\"id\":\"libraries\\/%s" |
| "\\/scripts\\/test-lib\\/0\"," |
| "\"uri\":\"test-lib\"," |
| "\"_kind\":\"script\"},\"line\":3}}," |
| "{\"type\":\"Breakpoint\",\"fixedId\":true,\"id\":\"breakpoints\\/1\"," |
| "\"breakpointNumber\":1,\"resolved\":false," |
| "\"location\":{\"type\":\"UnresolvedSourceLocation\"," |
| "\"script\":{\"type\":\"@Script\",\"fixedId\":true," |
| "\"id\":\"libraries\\/%s" |
| "\\/scripts\\/test-lib\\/0\"," |
| "\"uri\":\"test-lib\"," |
| "\"_kind\":\"script\"},\"line\":2}}]", |
| private_key.ToCString(), private_key.ToCString()); |
| } |
| } |
| |
| |
| static bool saw_paused_event = false; |
| |
| static void InspectPausedEvent(Dart_IsolateId isolate_id, |
| intptr_t bp_id, |
| const Dart_CodeLocation& loc) { |
| Isolate* isolate = Isolate::Current(); |
| Debugger* debugger = isolate->debugger(); |
| |
| // The debugger knows that it is paused, and why. |
| EXPECT(debugger->IsPaused()); |
| const ServiceEvent* event = debugger->PauseEvent(); |
| EXPECT(event != NULL); |
| EXPECT(event->kind() == ServiceEvent::kPauseBreakpoint); |
| saw_paused_event = true; |
| } |
| |
| |
| TEST_CASE(Debugger_PauseEvent) { |
| const char* kScriptChars = |
| "main() {\n" |
| " var x = new StringBuffer();\n" |
| " x.write('won');\n" |
| " x.write('too');\n" |
| " return x.toString();\n" |
| "}\n"; |
| Dart_Handle lib = TestCase::LoadTestScript(kScriptChars, NULL); |
| EXPECT_VALID(lib); |
| |
| Isolate* isolate = Isolate::Current(); |
| Debugger* debugger = isolate->debugger(); |
| |
| // No pause event. |
| EXPECT(!debugger->IsPaused()); |
| EXPECT(debugger->PauseEvent() == NULL); |
| |
| saw_paused_event = false; |
| Dart_SetPausedEventHandler(InspectPausedEvent); |
| |
| // Set a breakpoint and run. |
| EXPECT_VALID(Dart_SetBreakpoint(NewString(TestCase::url()), 2)); |
| Dart_Handle result = Dart_Invoke(lib, NewString("main"), 0, NULL); |
| EXPECT_VALID(result); |
| EXPECT(Dart_IsString(result)); |
| |
| // We ran the code in InspectPausedEvent. |
| EXPECT(saw_paused_event); |
| } |
| |
| |
| static uint8_t* malloc_allocator(uint8_t* ptr, |
| intptr_t old_size, |
| intptr_t new_size) { |
| void* new_ptr = realloc(reinterpret_cast<void*>(ptr), new_size); |
| return reinterpret_cast<uint8_t*>(new_ptr); |
| } |
| |
| |
| const char* rewind_frame_index = "-1"; |
| |
| |
| // Build and send a fake resume OOB message for testing purposes. |
| void SendResumeMessage(Isolate* isolate) { |
| // Format is: [ oob_type, port, seq, method_name, [keys], [values] ] |
| Dart_CObject msg; |
| Dart_CObject* list_values[6]; |
| msg.type = Dart_CObject_kArray; |
| msg.value.as_array.length = 6; |
| msg.value.as_array.values = list_values; |
| |
| Dart_CObject oob; |
| oob.type = Dart_CObject_kInt32; |
| oob.value.as_int32 = Message::kServiceOOBMsg; |
| list_values[0] = &oob; |
| |
| Dart_CObject reply_port; |
| reply_port.type = Dart_CObject_kNull; |
| list_values[1] = &reply_port; |
| |
| Dart_CObject seq; |
| seq.type = Dart_CObject_kNull; |
| list_values[2] = &seq; |
| |
| Dart_CObject method_name; |
| method_name.type = Dart_CObject_kString; |
| method_name.value.as_string = const_cast<char*>("resume"); |
| list_values[3] = &method_name; |
| |
| const int kParamCount = 3; |
| Dart_CObject param_keys; |
| Dart_CObject* param_keys_list[kParamCount]; |
| param_keys.type = Dart_CObject_kArray; |
| param_keys.value.as_array.values = param_keys_list; |
| param_keys.value.as_array.length = kParamCount; |
| list_values[4] = ¶m_keys; |
| |
| Dart_CObject param_values; |
| Dart_CObject* param_values_list[kParamCount]; |
| param_values.type = Dart_CObject_kArray; |
| param_values.value.as_array.values = param_values_list; |
| param_values.value.as_array.length = kParamCount; |
| list_values[5] = ¶m_values; |
| |
| Dart_CObject param0_name; |
| param0_name.type = Dart_CObject_kString; |
| param0_name.value.as_string = const_cast<char*>("isolateId"); |
| param_keys_list[0] = ¶m0_name; |
| |
| Dart_CObject param0_value; |
| param0_value.type = Dart_CObject_kString; |
| const char* isolate_id = Thread::Current()->zone()->PrintToString( |
| ISOLATE_SERVICE_ID_FORMAT_STRING, |
| static_cast<int64_t>(isolate->main_port())); |
| param0_value.value.as_string = const_cast<char*>(isolate_id); |
| param_values_list[0] = ¶m0_value; |
| |
| Dart_CObject param1_name; |
| param1_name.type = Dart_CObject_kString; |
| param1_name.value.as_string = const_cast<char*>("step"); |
| param_keys_list[1] = ¶m1_name; |
| |
| Dart_CObject param1_value; |
| param1_value.type = Dart_CObject_kString; |
| param1_value.value.as_string = const_cast<char*>("Rewind"); |
| param_values_list[1] = ¶m1_value; |
| |
| Dart_CObject param2_name; |
| param2_name.type = Dart_CObject_kString; |
| param2_name.value.as_string = const_cast<char*>("frameIndex"); |
| param_keys_list[2] = ¶m2_name; |
| |
| Dart_CObject param2_value; |
| param2_value.type = Dart_CObject_kString; |
| param2_value.value.as_string = const_cast<char*>(rewind_frame_index); |
| param_values_list[2] = ¶m2_value; |
| |
| { |
| uint8_t* buffer = NULL; |
| ApiMessageWriter writer(&buffer, &malloc_allocator); |
| bool success = writer.WriteCMessage(&msg); |
| ASSERT(success); |
| |
| // Post the message at the given port. |
| success = PortMap::PostMessage(new Message(isolate->main_port(), buffer, |
| writer.BytesWritten(), |
| Message::kOOBPriority)); |
| ASSERT(success); |
| } |
| } |
| |
| |
| static void RewindOnce(Dart_IsolateId isolate_id, |
| intptr_t bp_id, |
| const Dart_CodeLocation& loc) { |
| bool first_time = !saw_paused_event; |
| saw_paused_event = true; |
| if (first_time) { |
| Thread* T = Thread::Current(); |
| Isolate* I = T->isolate(); |
| // TODO(turnidge): It is weird that the isolate can get to this |
| // point in our tests without being marked runnable. Clear this up |
| // at some point. |
| I->set_is_runnable(true); |
| SendResumeMessage(I); |
| I->PauseEventHandler(); |
| } |
| } |
| |
| |
| TEST_CASE(Debugger_RewindOneFrame_Unoptimized) { |
| SetFlagScope<bool> sfs(&FLAG_trace_rewind, true); |
| |
| // These variables are global state used by RewindOnce. |
| saw_paused_event = false; |
| rewind_frame_index = "1"; |
| |
| const char* kScriptChars = |
| "import 'dart:developer';\n" |
| "\n" |
| "var msg = new StringBuffer();\n" |
| "\n" |
| "foo() {\n" |
| " msg.write('enter(foo) ');\n" |
| " debugger();\n" |
| " msg.write('exit(foo) ');\n" |
| "}\n" |
| "\n" |
| "main() {\n" |
| " msg.write('enter(main) ');\n" |
| " foo();\n" |
| " msg.write('exit(main) ');\n" |
| " return msg.toString();\n" |
| "}\n"; |
| Dart_Handle lib = TestCase::LoadTestScript(kScriptChars, NULL); |
| EXPECT_VALID(lib); |
| |
| Dart_SetPausedEventHandler(RewindOnce); |
| Dart_Handle result = Dart_Invoke(lib, NewString("main"), 0, NULL); |
| const char* result_cstr; |
| EXPECT_VALID(result); |
| EXPECT(Dart_IsString(result)); |
| EXPECT_VALID(Dart_StringToCString(result, &result_cstr)); |
| EXPECT_STREQ("enter(main) enter(foo) enter(foo) exit(foo) exit(main) ", |
| result_cstr); |
| EXPECT(saw_paused_event); |
| } |
| |
| |
| TEST_CASE(Debugger_RewindTwoFrames_Unoptimized) { |
| SetFlagScope<bool> sfs(&FLAG_trace_rewind, true); |
| |
| // These variables are global state used by RewindOnce. |
| saw_paused_event = false; |
| rewind_frame_index = "2"; |
| |
| const char* kScriptChars = |
| "import 'dart:developer';\n" |
| "\n" |
| "var msg = new StringBuffer();\n" |
| "\n" |
| "foo() {\n" |
| " msg.write('enter(foo) ');\n" |
| " debugger();\n" |
| " msg.write('exit(foo) ');\n" |
| "}\n" |
| "\n" |
| "bar() {\n" |
| " msg.write('enter(bar) ');\n" |
| " foo();\n" |
| " msg.write('exit(bar) ');\n" |
| "}\n" |
| "\n" |
| "main() {\n" |
| " msg.write('enter(main) ');\n" |
| " bar();\n" |
| " msg.write('exit(main) ');\n" |
| " return msg.toString();\n" |
| "}\n"; |
| Dart_Handle lib = TestCase::LoadTestScript(kScriptChars, NULL); |
| EXPECT_VALID(lib); |
| |
| Dart_SetPausedEventHandler(RewindOnce); |
| Dart_Handle result = Dart_Invoke(lib, NewString("main"), 0, NULL); |
| const char* result_cstr; |
| EXPECT_VALID(result); |
| EXPECT(Dart_IsString(result)); |
| EXPECT_VALID(Dart_StringToCString(result, &result_cstr)); |
| EXPECT_STREQ( |
| "enter(main) enter(bar) enter(foo) enter(bar) enter(foo) " |
| "exit(foo) exit(bar) exit(main) ", |
| result_cstr); |
| EXPECT(saw_paused_event); |
| } |
| |
| |
| TEST_CASE(Debugger_Rewind_Optimized) { |
| SetFlagScope<bool> sfs1(&FLAG_trace_rewind, true); |
| SetFlagScope<bool> sfs2(&FLAG_prune_dead_locals, false); |
| SetFlagScope<bool> sfs3(&FLAG_enable_inlining_annotations, true); |
| SetFlagScope<bool> sfs4(&FLAG_background_compilation, false); |
| SetFlagScope<int> sfs5(&FLAG_optimization_counter_threshold, 10); |
| |
| // These variables are global state used by RewindOnce. |
| saw_paused_event = false; |
| rewind_frame_index = "2"; |
| |
| const char* kScriptChars = |
| "import 'dart:developer';\n" |
| "\n" |
| "const alwaysInline = \"AlwaysInline\";\n" |
| "const noInline = \"NeverInline\";\n" |
| "\n" |
| "var msg = new StringBuffer();\n" |
| "int i;\n" |
| "\n" |
| "@noInline\n" |
| "foo() {\n" |
| " msg.write('enter(foo) ');\n" |
| " if (i > 15) {\n" |
| " debugger();\n" |
| " msg.write('exit(foo) ');\n" |
| " return true;\n" |
| " } else {\n" |
| " msg.write('exit(foo) ');\n" |
| " return false;\n" |
| " }\n" |
| "}\n" |
| "\n" |
| "@alwaysInline\n" |
| "bar3() {\n" |
| " msg.write('enter(bar3) ');\n" |
| " var result = foo();\n" |
| " msg.write('exit(bar3) ');\n" |
| " return result;\n" |
| "}\n" |
| "\n" |
| "@alwaysInline\n" |
| "bar2() {\n" |
| " msg.write('enter(bar2) ');\n" |
| " var result = bar3();\n" |
| " msg.write('exit(bar2) ');\n" |
| " return result;\n" |
| "}\n" |
| "\n" |
| "@alwaysInline\n" |
| "bar1() {\n" |
| " msg.write('enter(bar1) ');\n" |
| " var result = bar2();\n" |
| " msg.write('exit(bar1) ');\n" |
| " return result;\n" |
| "}\n" |
| "\n" |
| "main() {\n" |
| " for (i = 0; i < 20; i++) {\n" |
| " msg.clear();\n" |
| " if (bar1()) break;\n;" |
| " }\n" |
| " return msg.toString();\n" |
| "}\n"; |
| Dart_Handle lib = TestCase::LoadTestScript(kScriptChars, NULL); |
| EXPECT_VALID(lib); |
| |
| Dart_SetPausedEventHandler(RewindOnce); |
| Dart_Handle result = Dart_Invoke(lib, NewString("main"), 0, NULL); |
| const char* result_cstr; |
| EXPECT_VALID(result); |
| EXPECT(Dart_IsString(result)); |
| EXPECT_VALID(Dart_StringToCString(result, &result_cstr)); |
| EXPECT_STREQ( |
| "enter(bar1) enter(bar2) enter(bar3) enter(foo) " |
| "enter(bar3) enter(foo) " |
| "exit(foo) exit(bar3) exit(bar2) exit(bar1) ", |
| result_cstr); |
| EXPECT(saw_paused_event); |
| } |
| |
| #endif // !PRODUCT |
| |
| } // namespace dart |