|  | // Copyright (c) 2011, 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 "include/dart_api.h" | 
|  | #include "include/dart_native_api.h" | 
|  |  | 
|  | #include "vm/unit_test.h" | 
|  |  | 
|  | // Custom Isolate Test. | 
|  | // | 
|  | // This mid-size test uses the Dart Embedding Api to create a custom | 
|  | // isolate abstraction.  Instead of having a dedicated thread for each | 
|  | // isolate, as is the case normally, this implementation shares a | 
|  | // single thread among the isolates using an event queue. | 
|  |  | 
|  | namespace dart { | 
|  |  | 
|  | DECLARE_FLAG(bool, trace_shutdown); | 
|  |  | 
|  | static void native_echo(Dart_NativeArguments args); | 
|  | static void CustomIsolateImpl_start(Dart_NativeArguments args); | 
|  | static Dart_NativeFunction NativeLookup(Dart_Handle name, | 
|  | int argc, | 
|  | bool* auto_setup_scope); | 
|  |  | 
|  | static const char* kCustomIsolateScriptChars = | 
|  | "import 'dart:isolate';\n" | 
|  | "\n" | 
|  | "final RawReceivePort mainPort = new RawReceivePort();\n" | 
|  | "final SendPort mainSendPort = mainPort.sendPort;\n" | 
|  | "\n" | 
|  | "echo(arg) native \"native_echo\";\n" | 
|  | "\n" | 
|  | "class CustomIsolateImpl implements CustomIsolate {\n" | 
|  | "  CustomIsolateImpl(String entry) : _entry = entry{\n" | 
|  | "    echo('Constructing isolate');\n" | 
|  | "  }\n" | 
|  | "\n" | 
|  | "  SendPort spawn() {\n" | 
|  | "    return _start(_entry);\n" | 
|  | "  }\n" | 
|  | "\n" | 
|  | "  static SendPort _start(entry)\n" | 
|  | "      native \"CustomIsolateImpl_start\";\n" | 
|  | "\n" | 
|  | "  String _entry;\n" | 
|  | "}\n" | 
|  | "\n" | 
|  | "abstract class CustomIsolate {\n" | 
|  | "  factory CustomIsolate(String entry) = CustomIsolateImpl;\n" | 
|  | "\n" | 
|  | "  SendPort spawn();\n" | 
|  | "}\n" | 
|  | "\n" | 
|  | "isolateMain() {\n" | 
|  | "   echo('Running isolateMain');\n" | 
|  | "   mainPort.handler = (message) {\n" | 
|  | "     var data = message[0];\n" | 
|  | "     var replyTo = message[1];\n" | 
|  | "     echo('Received: $data');\n" | 
|  | "     replyTo.send(data + 1);\n" | 
|  | "     mainPort.close();\n" | 
|  | "   };\n" | 
|  | "}\n" | 
|  | "\n" | 
|  | "main() {\n" | 
|  | "  var isolate = new CustomIsolate(\"isolateMain\");\n" | 
|  | "  var receivePort = new RawReceivePort();\n" | 
|  | "  SendPort port = isolate.spawn();\n" | 
|  | "  port.send([42, receivePort.sendPort]);\n" | 
|  | "  receivePort.handler = (message) {\n" | 
|  | "    receivePort.close();\n" | 
|  | "    echo('Received: $message');\n" | 
|  | "  };\n" | 
|  | "  return 'success';\n" | 
|  | "}\n"; | 
|  |  | 
|  | // An entry in our event queue. | 
|  | class Event { | 
|  | protected: | 
|  | explicit Event(Dart_Isolate isolate) : isolate_(isolate), next_(NULL) {} | 
|  |  | 
|  | public: | 
|  | virtual ~Event() {} | 
|  | virtual void Process() = 0; | 
|  |  | 
|  | Dart_Isolate isolate() const { return isolate_; } | 
|  |  | 
|  | private: | 
|  | friend class EventQueue; | 
|  | Dart_Isolate isolate_; | 
|  | Event* next_; | 
|  | }; | 
|  |  | 
|  | // A simple event queue for our test. | 
|  | class EventQueue { | 
|  | public: | 
|  | EventQueue() { head_ = NULL; } | 
|  |  | 
|  | void Add(Event* event) { | 
|  | if (head_ == NULL) { | 
|  | head_ = event; | 
|  | tail_ = event; | 
|  | } else { | 
|  | tail_->next_ = event; | 
|  | tail_ = event; | 
|  | } | 
|  | } | 
|  |  | 
|  | Event* Get() { | 
|  | if (head_ == NULL) { | 
|  | return NULL; | 
|  | } | 
|  | Event* tmp = head_; | 
|  | head_ = head_->next_; | 
|  | if (head_ == NULL) { | 
|  | // Not necessary, but why not. | 
|  | tail_ = NULL; | 
|  | } | 
|  |  | 
|  | return tmp; | 
|  | } | 
|  |  | 
|  | void RemoveEventsForIsolate(Dart_Isolate isolate) { | 
|  | Event* cur = head_; | 
|  | Event* prev = NULL; | 
|  | while (cur != NULL) { | 
|  | Event* next = cur->next_; | 
|  | if (cur->isolate() == isolate) { | 
|  | // Remove matching event. | 
|  | if (prev != NULL) { | 
|  | prev->next_ = next; | 
|  | } else { | 
|  | head_ = next; | 
|  | } | 
|  | delete cur; | 
|  | } else { | 
|  | // Advance. | 
|  | prev = cur; | 
|  | } | 
|  | cur = next; | 
|  | } | 
|  | tail_ = prev; | 
|  | } | 
|  |  | 
|  | private: | 
|  | Event* head_; | 
|  | Event* tail_; | 
|  | }; | 
|  | EventQueue* event_queue; | 
|  |  | 
|  | // Start an isolate. | 
|  | class StartEvent : public Event { | 
|  | public: | 
|  | StartEvent(Dart_Isolate isolate, const char* main) | 
|  | : Event(isolate), main_(main) {} | 
|  |  | 
|  | virtual void Process(); | 
|  |  | 
|  | private: | 
|  | const char* main_; | 
|  | }; | 
|  |  | 
|  | void StartEvent::Process() { | 
|  | OS::PrintErr(">> StartEvent with isolate(%p)--\n", isolate()); | 
|  | Dart_EnterIsolate(isolate()); | 
|  | Dart_EnterScope(); | 
|  | Dart_Handle result; | 
|  |  | 
|  | Dart_Handle lib = Dart_LookupLibrary(NewString(TestCase::url())); | 
|  | EXPECT_VALID(lib); | 
|  |  | 
|  | result = Dart_Invoke(lib, NewString(main_), 0, NULL); | 
|  | EXPECT_VALID(result); | 
|  | free(const_cast<char*>(main_)); | 
|  | main_ = NULL; | 
|  |  | 
|  | Dart_SetMessageNotifyCallback(NULL); | 
|  | Dart_ExitScope(); | 
|  | Dart_ExitIsolate(); | 
|  | } | 
|  |  | 
|  | // Notify an isolate of a pending message. | 
|  | class MessageEvent : public Event { | 
|  | public: | 
|  | explicit MessageEvent(Dart_Isolate isolate) : Event(isolate) {} | 
|  |  | 
|  | ~MessageEvent() {} | 
|  |  | 
|  | virtual void Process(); | 
|  | }; | 
|  |  | 
|  | void MessageEvent::Process() { | 
|  | OS::PrintErr("$$ MessageEvent with isolate(%p)\n", isolate()); | 
|  | Dart_EnterIsolate(isolate()); | 
|  | Dart_EnterScope(); | 
|  |  | 
|  | Dart_Handle result = Dart_HandleMessage(); | 
|  | EXPECT_VALID(result); | 
|  |  | 
|  | if (!Dart_HasLivePorts()) { | 
|  | OS::PrintErr("<< Shutting down isolate(%p)\n", isolate()); | 
|  | event_queue->RemoveEventsForIsolate(isolate()); | 
|  | Dart_SetMessageNotifyCallback(NULL); | 
|  | Dart_ExitScope(); | 
|  | Dart_ShutdownIsolate(); | 
|  | } else { | 
|  | Dart_ExitScope(); | 
|  | Dart_ExitIsolate(); | 
|  | } | 
|  | ASSERT(Dart_CurrentIsolate() == NULL); | 
|  | } | 
|  |  | 
|  | static void NotifyMessage(Dart_Isolate dest_isolate) { | 
|  | OS::PrintErr("-- Notify isolate(%p) of pending message --\n", dest_isolate); | 
|  | OS::PrintErr("-- Adding MessageEvent to queue --\n"); | 
|  | event_queue->Add(new MessageEvent(dest_isolate)); | 
|  | } | 
|  |  | 
|  | static Dart_NativeFunction NativeLookup(Dart_Handle name, | 
|  | int argc, | 
|  | bool* auto_setup_scope) { | 
|  | ASSERT(auto_setup_scope != NULL); | 
|  | *auto_setup_scope = true; | 
|  | const char* name_str = NULL; | 
|  | EXPECT(Dart_IsString(name)); | 
|  | EXPECT_VALID(Dart_StringToCString(name, &name_str)); | 
|  | if (strcmp(name_str, "native_echo") == 0) { | 
|  | return &native_echo; | 
|  | } else if (strcmp(name_str, "CustomIsolateImpl_start") == 0) { | 
|  | return &CustomIsolateImpl_start; | 
|  | } | 
|  | return NULL; | 
|  | } | 
|  |  | 
|  | char* saved_echo = NULL; | 
|  | static void native_echo(Dart_NativeArguments args) { | 
|  | Dart_EnterScope(); | 
|  | Dart_Handle arg = Dart_GetNativeArgument(args, 0); | 
|  | Dart_Handle toString = Dart_ToString(arg); | 
|  | EXPECT_VALID(toString); | 
|  | const char* c_str = NULL; | 
|  | EXPECT_VALID(Dart_StringToCString(toString, &c_str)); | 
|  | if (saved_echo != nullptr) { | 
|  | free(saved_echo); | 
|  | } | 
|  | saved_echo = Utils::StrDup(c_str); | 
|  | OS::PrintErr("-- (isolate=%p) %s\n", Dart_CurrentIsolate(), c_str); | 
|  | Dart_ExitScope(); | 
|  | } | 
|  |  | 
|  | static void CustomIsolateImpl_start(Dart_NativeArguments args) { | 
|  | OS::PrintErr("-- Enter: CustomIsolateImpl_start --\n"); | 
|  |  | 
|  | // We would probably want to pass in the this pointer too, so we | 
|  | // could associate the CustomIsolateImpl instance with the | 
|  | // Dart_Isolate by storing it in a native field. | 
|  | EXPECT_EQ(1, Dart_GetNativeArgumentCount(args)); | 
|  | Dart_Handle param = Dart_GetNativeArgument(args, 0); | 
|  | EXPECT_VALID(param); | 
|  | EXPECT(Dart_IsString(param)); | 
|  | const char* isolate_main = NULL; | 
|  | EXPECT_VALID(Dart_StringToCString(param, &isolate_main)); | 
|  | isolate_main = Utils::StrDup(isolate_main); | 
|  |  | 
|  | // Save current isolate. | 
|  | Dart_Isolate saved_isolate = Dart_CurrentIsolate(); | 
|  | Dart_ExitIsolate(); | 
|  |  | 
|  | // Create a new Dart_Isolate. | 
|  | Dart_Isolate new_isolate = TestCase::CreateTestIsolate(); | 
|  | EXPECT(new_isolate != NULL); | 
|  | Dart_SetMessageNotifyCallback(&NotifyMessage); | 
|  | Dart_EnterScope(); | 
|  | // Reload all the test classes here. | 
|  | // | 
|  | // TODO(turnidge): Use the create isolate callback instead? | 
|  | Dart_Handle lib = | 
|  | TestCase::LoadTestScript(kCustomIsolateScriptChars, NativeLookup); | 
|  | EXPECT_VALID(lib); | 
|  |  | 
|  | Dart_Handle main_send_port = Dart_GetField(lib, NewString("mainSendPort")); | 
|  | EXPECT_VALID(main_send_port); | 
|  | Dart_Port main_port_id; | 
|  | Dart_Handle err = Dart_SendPortGetId(main_send_port, &main_port_id); | 
|  | EXPECT_VALID(err); | 
|  |  | 
|  | OS::PrintErr("-- Adding StartEvent to queue --\n"); | 
|  | event_queue->Add(new StartEvent(new_isolate, isolate_main)); | 
|  |  | 
|  | // Restore the original isolate. | 
|  | Dart_ExitScope(); | 
|  | Dart_ExitIsolate(); | 
|  | Dart_EnterIsolate(saved_isolate); | 
|  | Dart_EnterScope(); | 
|  |  | 
|  | Dart_Handle send_port = Dart_NewSendPort(main_port_id); | 
|  | EXPECT_VALID(send_port); | 
|  | Dart_SetReturnValue(args, send_port); | 
|  |  | 
|  | OS::PrintErr("-- Exit: CustomIsolateImpl_start --\n"); | 
|  | Dart_ExitScope(); | 
|  | } | 
|  |  | 
|  | VM_UNIT_TEST_CASE(CustomIsolates) { | 
|  | bool saved_flag = FLAG_trace_shutdown; | 
|  | FLAG_trace_shutdown = true; | 
|  | FLAG_verify_handles = true; | 
|  | event_queue = new EventQueue(); | 
|  |  | 
|  | Dart_Isolate dart_isolate = TestCase::CreateTestIsolate(); | 
|  | EXPECT(dart_isolate != NULL); | 
|  | Dart_SetMessageNotifyCallback(&NotifyMessage); | 
|  | Dart_EnterScope(); | 
|  | Dart_Handle result; | 
|  |  | 
|  | // Create a test library. | 
|  | Dart_Handle lib = | 
|  | TestCase::LoadTestScript(kCustomIsolateScriptChars, NativeLookup); | 
|  | EXPECT_VALID(lib); | 
|  |  | 
|  | // Run main. | 
|  | result = Dart_Invoke(lib, NewString("main"), 0, NULL); | 
|  | EXPECT_VALID(result); | 
|  | EXPECT(Dart_IsString(result)); | 
|  | const char* result_str = NULL; | 
|  | EXPECT_VALID(Dart_StringToCString(result, &result_str)); | 
|  | EXPECT_STREQ("success", result_str); | 
|  |  | 
|  | Dart_ExitScope(); | 
|  | Dart_ExitIsolate(); | 
|  |  | 
|  | OS::PrintErr("-- Starting event loop --\n"); | 
|  | Event* event = event_queue->Get(); | 
|  | while (event != nullptr) { | 
|  | event->Process(); | 
|  | delete event; | 
|  | event = event_queue->Get(); | 
|  | } | 
|  | OS::PrintErr("-- Finished event loop --\n"); | 
|  | EXPECT_STREQ("Received: 43", saved_echo); | 
|  | free(saved_echo); | 
|  |  | 
|  | delete event_queue; | 
|  | event_queue = NULL; | 
|  | FLAG_trace_shutdown = saved_flag; | 
|  | } | 
|  |  | 
|  | }  // namespace dart |