| Index: test/cctest/test-api.cc
 | 
| ===================================================================
 | 
| --- test/cctest/test-api.cc	(revision 4596)
 | 
| +++ test/cctest/test-api.cc	(working copy)
 | 
| @@ -9584,6 +9584,115 @@
 | 
|  }
 | 
|  
 | 
|  
 | 
| +// Checks that a StackFrame has certain expected values.
 | 
| +void checkStackFrame(const char* expected_script_name,
 | 
| +    const char* expected_func_name, int expected_line_number,
 | 
| +    int expected_column, bool is_eval, bool is_constructor,
 | 
| +    v8::Handle<v8::StackFrame> frame) {
 | 
| +  v8::HandleScope scope;
 | 
| +  v8::String::Utf8Value func_name(frame->GetFunctionName());
 | 
| +  v8::String::Utf8Value script_name(frame->GetScriptName());
 | 
| +  if (*script_name == NULL) {
 | 
| +    // The situation where there is no associated script, like for evals.
 | 
| +    CHECK(expected_script_name == NULL);
 | 
| +  } else {
 | 
| +    CHECK(strstr(*script_name, expected_script_name) != NULL);
 | 
| +  }
 | 
| +  CHECK(strstr(*func_name, expected_func_name) != NULL);
 | 
| +  CHECK_EQ(expected_line_number, frame->GetLineNumber());
 | 
| +  CHECK_EQ(expected_column, frame->GetColumn());
 | 
| +  CHECK_EQ(is_eval, frame->IsEval());
 | 
| +  CHECK_EQ(is_constructor, frame->IsConstructor());
 | 
| +}
 | 
| +
 | 
| +
 | 
| +v8::Handle<Value> AnalyzeStackInNativeCode(const v8::Arguments& args) {
 | 
| +  v8::HandleScope scope;
 | 
| +  const char* origin = "capture-stack-trace-test";
 | 
| +  const int kOverviewTest = 1;
 | 
| +  const int kDetailedTest = 2;
 | 
| +
 | 
| +  ASSERT(args.Length() == 1);
 | 
| +
 | 
| +  int testGroup = args[0]->ToNumber()->Value();
 | 
| +  if (testGroup == kOverviewTest) {
 | 
| +    v8::Handle<v8::StackTrace> stackTrace =
 | 
| +        v8::StackTrace::CurrentStackTrace(10, v8::StackTrace::kOverview);
 | 
| +    CHECK_EQ(4, stackTrace->GetFrameCount());
 | 
| +    checkStackFrame(origin, "bar", 2, 10, false, false,
 | 
| +                    stackTrace->GetFrame(0));
 | 
| +    checkStackFrame(origin, "foo", 6, 3, false, false,
 | 
| +                    stackTrace->GetFrame(1));
 | 
| +    checkStackFrame(NULL, "", 1, 1, false, false,
 | 
| +                    stackTrace->GetFrame(2));
 | 
| +    // The last frame is an anonymous function that has the initial call.
 | 
| +    checkStackFrame(origin, "", 8, 7, false, false,
 | 
| +                    stackTrace->GetFrame(3));
 | 
| +
 | 
| +    CHECK(stackTrace->AsArray()->IsArray());
 | 
| +  } else if (testGroup == kDetailedTest) {
 | 
| +    v8::Handle<v8::StackTrace> stackTrace =
 | 
| +        v8::StackTrace::CurrentStackTrace(10, v8::StackTrace::kDetailed);
 | 
| +    CHECK_EQ(4, stackTrace->GetFrameCount());
 | 
| +    checkStackFrame(origin, "bat", 2, 1, false, false,
 | 
| +                    stackTrace->GetFrame(0));
 | 
| +    checkStackFrame(origin, "baz", 5, 3, false, true,
 | 
| +                    stackTrace->GetFrame(1));
 | 
| +    checkStackFrame(NULL, "", 1, 1, true, false,
 | 
| +                    stackTrace->GetFrame(2));
 | 
| +    // The last frame is an anonymous function that has the initial call to foo.
 | 
| +    checkStackFrame(origin, "", 7, 1, false, false,
 | 
| +                    stackTrace->GetFrame(3));
 | 
| +
 | 
| +    CHECK(stackTrace->AsArray()->IsArray());
 | 
| +  }
 | 
| +  return v8::Undefined();
 | 
| +}
 | 
| +
 | 
| +
 | 
| +// Tests the C++ StackTrace API.
 | 
| +THREADED_TEST(CaptureStackTrace) {
 | 
| +  v8::HandleScope scope;
 | 
| +  v8::Handle<v8::String> origin = v8::String::New("capture-stack-trace-test");
 | 
| +  Local<ObjectTemplate> templ = ObjectTemplate::New();
 | 
| +  templ->Set(v8_str("AnalyzeStackInNativeCode"),
 | 
| +             v8::FunctionTemplate::New(AnalyzeStackInNativeCode));
 | 
| +  LocalContext context(0, templ);
 | 
| +
 | 
| +  // Test getting OVERVIEW information. Should ignore information that is not
 | 
| +  // script name, function name, line number, and column offset.
 | 
| +  const char *overview_source =
 | 
| +    "function bar() {\n"
 | 
| +    "  var y; AnalyzeStackInNativeCode(1);\n"
 | 
| +    "}\n"
 | 
| +    "function foo() {\n"
 | 
| +    "\n"
 | 
| +    "  bar();\n"
 | 
| +    "}\n"
 | 
| +    "var x;eval('new foo();');";
 | 
| +  v8::Handle<v8::String> overview_src = v8::String::New(overview_source);
 | 
| +  v8::Handle<Value> overview_result =
 | 
| +      v8::Script::New(overview_src, origin)->Run();
 | 
| +  ASSERT(!overview_result.IsEmpty());
 | 
| +  ASSERT(overview_result->IsObject());
 | 
| +
 | 
| +  // Test getting DETAILED information.
 | 
| +  const char *detailed_source =
 | 
| +    "function bat() {\n"
 | 
| +    "AnalyzeStackInNativeCode(2);\n"
 | 
| +    "}\n"
 | 
| +    "function baz() {\n"
 | 
| +    "  bat();\n"
 | 
| +    "}\n"
 | 
| +    "eval('new baz();');";
 | 
| +  v8::Handle<v8::String> detailed_src = v8::String::New(detailed_source);
 | 
| +  v8::Handle<Value> detailed_result =
 | 
| +      v8::Script::New(detailed_src, origin)->Run();
 | 
| +  ASSERT(!detailed_result.IsEmpty());
 | 
| +  ASSERT(detailed_result->IsObject());
 | 
| +}
 | 
| +
 | 
| +
 | 
|  // Test that idle notification can be handled and eventually returns true.
 | 
|  THREADED_TEST(IdleNotification) {
 | 
|    bool rv = false;
 | 
| 
 |