|  | // 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 "include/dart_native_api.h" | 
|  |  | 
|  | #include "platform/assert.h" | 
|  | #include "vm/dart_api_impl.h" | 
|  | #include "vm/dart_api_message.h" | 
|  | #include "vm/dart_api_state.h" | 
|  | #include "vm/message.h" | 
|  | #include "vm/native_message_handler.h" | 
|  | #include "vm/port.h" | 
|  |  | 
|  | namespace dart { | 
|  |  | 
|  | // --- Message sending/receiving from native code --- | 
|  |  | 
|  | class IsolateSaver { | 
|  | public: | 
|  | explicit IsolateSaver(Isolate* current_isolate) | 
|  | : saved_isolate_(current_isolate) { | 
|  | if (current_isolate != NULL) { | 
|  | ASSERT(current_isolate == Isolate::Current()); | 
|  | Dart_ExitIsolate(); | 
|  | } | 
|  | } | 
|  | ~IsolateSaver() { | 
|  | if (saved_isolate_ != NULL) { | 
|  | Dart_Isolate I = reinterpret_cast<Dart_Isolate>(saved_isolate_); | 
|  | Dart_EnterIsolate(I); | 
|  | } | 
|  | } | 
|  |  | 
|  | private: | 
|  | Isolate* saved_isolate_; | 
|  |  | 
|  | DISALLOW_COPY_AND_ASSIGN(IsolateSaver); | 
|  | }; | 
|  |  | 
|  | static bool PostCObjectHelper(Dart_Port port_id, Dart_CObject* message) { | 
|  | ApiMessageWriter writer; | 
|  | Message* msg = | 
|  | writer.WriteCMessage(message, port_id, Message::kNormalPriority); | 
|  |  | 
|  | if (msg == NULL) { | 
|  | return false; | 
|  | } | 
|  |  | 
|  | // Post the message at the given port. | 
|  | return PortMap::PostMessage(msg); | 
|  | } | 
|  |  | 
|  | DART_EXPORT bool Dart_PostCObject(Dart_Port port_id, Dart_CObject* message) { | 
|  | return PostCObjectHelper(port_id, message); | 
|  | } | 
|  |  | 
|  | DART_EXPORT bool Dart_PostInteger(Dart_Port port_id, int64_t message) { | 
|  | if (Smi::IsValid(message)) { | 
|  | return PortMap::PostMessage( | 
|  | new Message(port_id, Smi::New(message), Message::kNormalPriority)); | 
|  | } | 
|  | Dart_CObject cobj; | 
|  | cobj.type = Dart_CObject_kInt64; | 
|  | cobj.value.as_int64 = message; | 
|  | return PostCObjectHelper(port_id, &cobj); | 
|  | } | 
|  |  | 
|  | DART_EXPORT Dart_Port Dart_NewNativePort(const char* name, | 
|  | Dart_NativeMessageHandler handler, | 
|  | bool handle_concurrently) { | 
|  | if (name == NULL) { | 
|  | name = "<UnnamedNativePort>"; | 
|  | } | 
|  | if (handler == NULL) { | 
|  | OS::PrintErr("%s expects argument 'handler' to be non-null.\n", | 
|  | CURRENT_FUNC); | 
|  | return ILLEGAL_PORT; | 
|  | } | 
|  | // Start the native port without a current isolate. | 
|  | IsolateSaver saver(Isolate::Current()); | 
|  |  | 
|  | NativeMessageHandler* nmh = new NativeMessageHandler(name, handler); | 
|  | Dart_Port port_id = PortMap::CreatePort(nmh); | 
|  | PortMap::SetPortState(port_id, PortMap::kLivePort); | 
|  | nmh->Run(Dart::thread_pool(), NULL, NULL, 0); | 
|  | return port_id; | 
|  | } | 
|  |  | 
|  | DART_EXPORT bool Dart_CloseNativePort(Dart_Port native_port_id) { | 
|  | // Close the native port without a current isolate. | 
|  | IsolateSaver saver(Isolate::Current()); | 
|  |  | 
|  | // TODO(turnidge): Check that the port is native before trying to close. | 
|  | return PortMap::ClosePort(native_port_id); | 
|  | } | 
|  |  | 
|  | // --- Verification tools --- | 
|  |  | 
|  | DART_EXPORT Dart_Handle Dart_CompileAll() { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return Api::NewError("%s: Cannot compile on an AOT runtime.", CURRENT_FUNC); | 
|  | #else | 
|  | DARTSCOPE(Thread::Current()); | 
|  | API_TIMELINE_DURATION(T); | 
|  | Dart_Handle result = Api::CheckAndFinalizePendingClasses(T); | 
|  | if (::Dart_IsError(result)) { | 
|  | return result; | 
|  | } | 
|  | CHECK_CALLBACK_STATE(T); | 
|  | const Error& error = Error::Handle(T->zone(), Library::CompileAll()); | 
|  | if (!error.IsNull()) { | 
|  | return Api::NewHandle(T, error.raw()); | 
|  | } | 
|  | return Api::Success(); | 
|  | #endif  // defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | DART_EXPORT Dart_Handle Dart_ParseAll() { | 
|  | #if defined(DART_PRECOMPILED_RUNTIME) | 
|  | return Api::NewError("%s: Cannot compile on an AOT runtime.", CURRENT_FUNC); | 
|  | #else | 
|  | DARTSCOPE(Thread::Current()); | 
|  | API_TIMELINE_DURATION(T); | 
|  | Dart_Handle result = Api::CheckAndFinalizePendingClasses(T); | 
|  | if (::Dart_IsError(result)) { | 
|  | return result; | 
|  | } | 
|  | CHECK_CALLBACK_STATE(T); | 
|  | const Error& error = Error::Handle(T->zone(), Library::ParseAll(T)); | 
|  | if (!error.IsNull()) { | 
|  | return Api::NewHandle(T, error.raw()); | 
|  | } | 
|  | return Api::Success(); | 
|  | #endif  // defined(DART_PRECOMPILED_RUNTIME) | 
|  | } | 
|  |  | 
|  | }  // namespace dart |