| // Copyright (c) 2012, 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 "platform/assert.h" |
| #include "vm/class_finalizer.h" |
| #include "vm/code_patcher.h" |
| #include "vm/compiler.h" |
| #include "vm/dart_api_impl.h" |
| #include "vm/object.h" |
| #include "vm/symbols.h" |
| #include "vm/thread_pool.h" |
| #include "vm/unit_test.h" |
| |
| namespace dart { |
| |
| TEST_CASE(CompileScript) { |
| const char* kScriptChars = |
| "class A {\n" |
| " static foo() { return 42; }\n" |
| "}\n"; |
| String& url = String::Handle(String::New("dart-test:CompileScript")); |
| String& source = String::Handle(String::New(kScriptChars)); |
| Script& script = Script::Handle(Script::New(url, |
| source, |
| RawScript::kScriptTag)); |
| Library& lib = Library::Handle(Library::CoreLibrary()); |
| EXPECT(CompilerTest::TestCompileScript(lib, script)); |
| } |
| |
| |
| TEST_CASE(CompileFunction) { |
| const char* kScriptChars = |
| "class A {\n" |
| " static foo() { return 42; }\n" |
| " static moo() {\n" |
| " // A.foo();\n" |
| " }\n" |
| "}\n"; |
| String& url = String::Handle(String::New("dart-test:CompileFunction")); |
| String& source = String::Handle(String::New(kScriptChars)); |
| Script& script = Script::Handle(Script::New(url, |
| source, |
| RawScript::kScriptTag)); |
| Library& lib = Library::Handle(Library::CoreLibrary()); |
| EXPECT(CompilerTest::TestCompileScript(lib, script)); |
| EXPECT(ClassFinalizer::ProcessPendingClasses()); |
| Class& cls = Class::Handle( |
| lib.LookupClass(String::Handle(Symbols::New("A")))); |
| EXPECT(!cls.IsNull()); |
| String& function_foo_name = String::Handle(String::New("foo")); |
| Function& function_foo = |
| Function::Handle(cls.LookupStaticFunction(function_foo_name)); |
| EXPECT(!function_foo.IsNull()); |
| String& function_source = String::Handle(function_foo.GetSource()); |
| EXPECT_STREQ("static foo() { return 42; }", function_source.ToCString()); |
| EXPECT(CompilerTest::TestCompileFunction(function_foo)); |
| EXPECT(function_foo.HasCode()); |
| |
| String& function_moo_name = String::Handle(String::New("moo")); |
| Function& function_moo = |
| Function::Handle(cls.LookupStaticFunction(function_moo_name)); |
| EXPECT(!function_moo.IsNull()); |
| |
| EXPECT(CompilerTest::TestCompileFunction(function_moo)); |
| EXPECT(function_moo.HasCode()); |
| function_source = function_moo.GetSource(); |
| EXPECT_STREQ("static moo() {\n // A.foo();\n }", |
| function_source.ToCString()); |
| } |
| |
| |
| class CompileFunctionTask : public ThreadPool::Task { |
| public: |
| CompileFunctionTask(Isolate* isolate, |
| const Function& func, |
| Monitor* done_monitor, |
| bool* done) |
| : isolate_(isolate), |
| func_(func), |
| done_monitor_(done_monitor), |
| done_(done) { |
| } |
| |
| virtual void Run() { |
| Thread::EnterIsolateAsHelper(isolate_); |
| { |
| Thread* thread = Thread::Current(); |
| StackZone stack_zone(thread); |
| HANDLESCOPE(thread); |
| EXPECT(func_.HasCode()); |
| EXPECT(!func_.HasOptimizedCode()); |
| const Error& err = |
| Error::Handle(Compiler::CompileOptimizedFunction(thread, func_)); |
| EXPECT(err.IsNull()); |
| EXPECT(func_.HasOptimizedCode()); |
| } |
| Thread::ExitIsolateAsHelper(); |
| // Tell main thread that we are done. |
| { |
| MonitorLocker ml(done_monitor_); |
| ASSERT(!*done_); |
| *done_ = true; |
| ml.Notify(); |
| } |
| } |
| |
| private: |
| Isolate* isolate_; |
| const Function& func_; |
| Monitor* done_monitor_; |
| bool* done_; |
| }; |
| |
| |
| TEST_CASE(CompileFunctionOnHelperThread) { |
| Monitor done_monitor; |
| bool done = false; |
| Isolate* isolate = Thread::Current()->isolate(); |
| // Flush store buffers, etc. |
| // TODO(koda): Currently, the GC only does this for the current thread, (i.e, |
| // the helper, in this test), but it should be done for all *threads* |
| // after/at safepointing. |
| Thread::PrepareForGC(); |
| |
| // Create a simple function and compile it without optimization. |
| const char* kScriptChars = |
| "class A {\n" |
| " static foo() { return 42; }\n" |
| "}\n"; |
| String& url = |
| String::Handle(String::New("dart-test:CompileFunctionOnHelperThread")); |
| String& source = String::Handle(String::New(kScriptChars)); |
| Script& script = Script::Handle(Script::New(url, |
| source, |
| RawScript::kScriptTag)); |
| Library& lib = Library::Handle(Library::CoreLibrary()); |
| EXPECT(CompilerTest::TestCompileScript(lib, script)); |
| EXPECT(ClassFinalizer::ProcessPendingClasses()); |
| Class& cls = Class::Handle( |
| lib.LookupClass(String::Handle(Symbols::New("A")))); |
| EXPECT(!cls.IsNull()); |
| String& function_foo_name = String::Handle(String::New("foo")); |
| Function& func = |
| Function::Handle(cls.LookupStaticFunction(function_foo_name)); |
| EXPECT(!func.HasCode()); |
| CompilerTest::TestCompileFunction(func); |
| EXPECT(func.HasCode()); |
| EXPECT(!func.HasOptimizedCode()); |
| |
| // Now optimize it on a helper thread. |
| Dart::thread_pool()->Run( |
| new CompileFunctionTask(isolate, func, &done_monitor, &done)); |
| { |
| // Manually wait. |
| // TODO(koda): Replace with execution of Dart and/or VM code when GC |
| // actually safepoints everything. |
| MonitorLocker ml(&done_monitor); |
| while (!done) { |
| ml.Wait(); |
| } |
| } |
| } |
| |
| |
| TEST_CASE(RegenerateAllocStubs) { |
| const char* kScriptChars = |
| "class A {\n" |
| "}\n" |
| "unOpt() => new A(); \n" |
| "optIt() => new A(); \n" |
| "A main() {\n" |
| " return unOpt();\n" |
| "}\n"; |
| |
| // Isolate::Current()->flags().set_checked(true); |
| Dart_Handle lib = TestCase::LoadTestScript(kScriptChars, NULL); |
| Dart_Handle result = Dart_Invoke(lib, NewString("main"), 0, NULL); |
| EXPECT_VALID(result); |
| RawLibrary* raw_library = Library::RawCast(Api::UnwrapHandle(lib)); |
| Library& lib_handle = Library::ZoneHandle(raw_library); |
| Class& cls = Class::Handle( |
| lib_handle.LookupClass(String::Handle(Symbols::New("A")))); |
| EXPECT(!cls.IsNull()); |
| |
| Zone* zone = thread->zone(); |
| const Code& stub = Code::Handle(zone, |
| StubCode::GetAllocationStubForClass(cls)); |
| Class& owner = Class::Handle(); |
| owner ^= stub.owner(); |
| owner.DisableAllocationStub(); |
| result = Dart_Invoke(lib, NewString("main"), 0, NULL); |
| EXPECT_VALID(result); |
| |
| owner.DisableAllocationStub(); |
| result = Dart_Invoke(lib, NewString("main"), 0, NULL); |
| EXPECT_VALID(result); |
| |
| owner.DisableAllocationStub(); |
| result = Dart_Invoke(lib, NewString("main"), 0, NULL); |
| EXPECT_VALID(result); |
| } |
| |
| |
| TEST_CASE(EvalExpression) { |
| const char* kScriptChars = |
| "int ten = 2 * 5; \n" |
| "get dot => '.'; \n" |
| "class A { \n" |
| " var apa = 'Herr Nilsson'; \n" |
| " calc(x) => '${x*ten}'; \n" |
| "} \n" |
| "makeObj() => new A(); \n"; |
| |
| Dart_Handle lib = TestCase::LoadTestScript(kScriptChars, NULL); |
| Dart_Handle obj_handle = |
| Dart_Invoke(lib, Dart_NewStringFromCString("makeObj"), 0, NULL); |
| EXPECT(!Dart_IsNull(obj_handle)); |
| EXPECT(!Dart_IsError(obj_handle)); |
| const Object& obj = Object::Handle(Api::UnwrapHandle(obj_handle)); |
| EXPECT(!obj.IsNull()); |
| EXPECT(obj.IsInstance()); |
| |
| String& expr_text = String::Handle(); |
| expr_text = String::New("apa + ' ${calc(10)}' + dot"); |
| Object& val = Object::Handle(); |
| val = Instance::Cast(obj).Evaluate(expr_text, |
| Array::empty_array(), |
| Array::empty_array()); |
| EXPECT(!val.IsNull()); |
| EXPECT(!val.IsError()); |
| EXPECT(val.IsString()); |
| EXPECT_STREQ("Herr Nilsson 100.", val.ToCString()); |
| } |
| |
| |
| TEST_CASE(EvalExpressionWithLazyCompile) { |
| Library& lib = Library::Handle(Library::CoreLibrary()); |
| |
| const String& expression = String::Handle(String::New( |
| "(){ return (){ return (){ return 3 + 4; }(); }(); }()")); |
| Object& val = Object::Handle(); |
| val = lib.Evaluate(expression, Array::empty_array(), Array::empty_array()); |
| |
| EXPECT(!val.IsNull()); |
| EXPECT(!val.IsError()); |
| EXPECT(val.IsInteger()); |
| EXPECT_EQ(7, Integer::Cast(val).AsInt64Value()); |
| } |
| |
| |
| TEST_CASE(EvalExpressionExhaustCIDs) { |
| Library& lib = Library::Handle(Library::CoreLibrary()); |
| |
| const String& expression = String::Handle(String::New("3 + 4")); |
| Object& val = Object::Handle(); |
| |
| // Run once to ensure everything we touch is compiled. |
| val = lib.Evaluate(expression, Array::empty_array(), Array::empty_array()); |
| EXPECT(!val.IsNull()); |
| EXPECT(!val.IsError()); |
| EXPECT(val.IsInteger()); |
| EXPECT_EQ(7, Integer::Cast(val).AsInt64Value()); |
| |
| intptr_t initial_class_table_size = |
| Isolate::Current()->class_table()->NumCids(); |
| |
| val = lib.Evaluate(expression, Array::empty_array(), Array::empty_array()); |
| EXPECT(!val.IsNull()); |
| EXPECT(!val.IsError()); |
| EXPECT(val.IsInteger()); |
| EXPECT_EQ(7, Integer::Cast(val).AsInt64Value()); |
| |
| intptr_t final_class_table_size = |
| Isolate::Current()->class_table()->NumCids(); |
| // Eval should not eat into this non-renewable resource. |
| EXPECT_EQ(initial_class_table_size, final_class_table_size); |
| } |
| |
| } // namespace dart |