| Index: runtime/bin/eventhandler_fuchsia.cc
|
| diff --git a/runtime/bin/eventhandler_fuchsia.cc b/runtime/bin/eventhandler_fuchsia.cc
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..69c2190017b0aa14cb832b65b4e566c5def44818
|
| --- /dev/null
|
| +++ b/runtime/bin/eventhandler_fuchsia.cc
|
| @@ -0,0 +1,172 @@
|
| +// Copyright (c) 2016, 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.
|
| +
|
| +#if !defined(DART_IO_DISABLED)
|
| +
|
| +#include "platform/globals.h"
|
| +#if defined(TARGET_OS_FUCHSIA)
|
| +
|
| +#include "bin/eventhandler.h"
|
| +#include "bin/eventhandler_fuchsia.h"
|
| +
|
| +#include <magenta/syscalls.h>
|
| +#include <runtime/status.h>
|
| +
|
| +#include "bin/thread.h"
|
| +#include "bin/utils.h"
|
| +
|
| +namespace dart {
|
| +namespace bin {
|
| +
|
| +EventHandlerImplementation::EventHandlerImplementation() {
|
| + mx_status_t status = mx_message_pipe_create(interrupt_handles_, 0);
|
| + if (status != NO_ERROR) {
|
| + FATAL1("mx_message_pipe_create failed: %s\n", mx_strstatus(status));
|
| + }
|
| +}
|
| +
|
| +
|
| +EventHandlerImplementation::~EventHandlerImplementation() {
|
| + mx_status_t status = mx_handle_close(interrupt_handles_[0]);
|
| + if (status != NO_ERROR) {
|
| + FATAL1("mx_handle_close failed: %s\n", mx_strstatus(status));
|
| + }
|
| + status = mx_handle_close(interrupt_handles_[1]);
|
| + if (status != NO_ERROR) {
|
| + FATAL1("mx_handle_close failed: %s\n", mx_strstatus(status));
|
| + }
|
| +}
|
| +
|
| +
|
| +void EventHandlerImplementation::WakeupHandler(intptr_t id,
|
| + Dart_Port dart_port,
|
| + int64_t data) {
|
| + InterruptMessage msg;
|
| + msg.id = id;
|
| + msg.dart_port = dart_port;
|
| + msg.data = data;
|
| +
|
| + mx_status_t status =
|
| + mx_message_write(interrupt_handles_[1], &msg, sizeof(msg), NULL, 0, 0);
|
| + if (status != NO_ERROR) {
|
| + FATAL1("mx_message_write failed: %s\n", mx_strstatus(status));
|
| + }
|
| +}
|
| +
|
| +
|
| +void EventHandlerImplementation::HandleInterruptFd() {
|
| + InterruptMessage msg;
|
| + uint32_t bytes = kInterruptMessageSize;
|
| + mx_status_t status;
|
| + while (true) {
|
| + status = mx_message_read(
|
| + interrupt_handles_[0], &msg, &bytes, NULL, NULL, 0);
|
| + if (status != NO_ERROR) {
|
| + break;
|
| + }
|
| + ASSERT(bytes == kInterruptMessageSize);
|
| + if (msg.id == kTimerId) {
|
| + timeout_queue_.UpdateTimeout(msg.dart_port, msg.data);
|
| + } else if (msg.id == kShutdownId) {
|
| + shutdown_ = true;
|
| + } else {
|
| + UNIMPLEMENTED();
|
| + }
|
| + }
|
| + // status == ERR_BAD_STATE when we try to read and there are no messages
|
| + // available, so it is an error if we get here and status != ERR_BAD_STATE.
|
| + if (status != ERR_BAD_STATE) {
|
| + FATAL1("mx_message_read failed: %s\n", mx_strstatus(status));
|
| + }
|
| +}
|
| +
|
| +
|
| +void EventHandlerImplementation::HandleEvents() {
|
| + // TODO(zra): Handle events from other handles. At the moment we are only
|
| + // interrupted when there is a message on interrupt_handles_[0].
|
| + HandleInterruptFd();
|
| +}
|
| +
|
| +
|
| +int64_t EventHandlerImplementation::GetTimeout() const {
|
| + if (!timeout_queue_.HasTimeout()) {
|
| + return kInfinityTimeout;
|
| + }
|
| + int64_t millis = timeout_queue_.CurrentTimeout() -
|
| + TimerUtils::GetCurrentMonotonicMillis();
|
| + return (millis < 0) ? 0 : millis;
|
| +}
|
| +
|
| +
|
| +void EventHandlerImplementation::HandleTimeout() {
|
| + if (timeout_queue_.HasTimeout()) {
|
| + int64_t millis = timeout_queue_.CurrentTimeout() -
|
| + TimerUtils::GetCurrentMonotonicMillis();
|
| + if (millis <= 0) {
|
| + DartUtils::PostNull(timeout_queue_.CurrentPort());
|
| + timeout_queue_.RemoveCurrent();
|
| + }
|
| + }
|
| +}
|
| +
|
| +
|
| +void EventHandlerImplementation::Poll(uword args) {
|
| + EventHandler* handler = reinterpret_cast<EventHandler*>(args);
|
| + EventHandlerImplementation* handler_impl = &handler->delegate_;
|
| + ASSERT(handler_impl != NULL);
|
| +
|
| + while (!handler_impl->shutdown_) {
|
| + int64_t millis = handler_impl->GetTimeout();
|
| + ASSERT((millis == kInfinityTimeout) || (millis >= 0));
|
| +
|
| + mx_time_t timeout =
|
| + millis * kMicrosecondsPerMillisecond * kNanosecondsPerMicrosecond;
|
| + mx_signals_state_t signals_state;
|
| + mx_status_t status = mx_handle_wait_one(
|
| + handler_impl->interrupt_handles_[0],
|
| + MX_SIGNAL_READABLE | MX_SIGNAL_PEER_CLOSED,
|
| + timeout,
|
| + &signals_state);
|
| + if ((status != NO_ERROR) && (status != ERR_TIMED_OUT)) {
|
| + FATAL1("mx_handle_wait_one failed: %s\n", mx_strstatus(status));
|
| + } else {
|
| + handler_impl->HandleTimeout();
|
| + if ((signals_state.satisfied & MX_SIGNAL_READABLE) != 0) {
|
| + handler_impl->HandleEvents();
|
| + }
|
| + if ((signals_state.satisfied & MX_SIGNAL_PEER_CLOSED) != 0) {
|
| + FATAL("EventHandlerImplementation::Poll: Unexpected peer closed\n");
|
| + }
|
| + }
|
| + }
|
| + handler->NotifyShutdownDone();
|
| +}
|
| +
|
| +
|
| +void EventHandlerImplementation::Start(EventHandler* handler) {
|
| + int result = Thread::Start(&EventHandlerImplementation::Poll,
|
| + reinterpret_cast<uword>(handler));
|
| + if (result != 0) {
|
| + FATAL1("Failed to start event handler thread %d", result);
|
| + }
|
| +}
|
| +
|
| +
|
| +void EventHandlerImplementation::Shutdown() {
|
| + SendData(kShutdownId, 0, 0);
|
| +}
|
| +
|
| +
|
| +void EventHandlerImplementation::SendData(intptr_t id,
|
| + Dart_Port dart_port,
|
| + int64_t data) {
|
| + WakeupHandler(id, dart_port, data);
|
| +}
|
| +
|
| +} // namespace bin
|
| +} // namespace dart
|
| +
|
| +#endif // defined(TARGET_OS_FUCHSIA)
|
| +
|
| +#endif // !defined(DART_IO_DISABLED)
|
|
|