Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(615)

Unified Diff: third_party/WebKit/Source/core/mojo/MojoWatcher.cpp

Issue 2725133002: Mojo: Armed Watchers (Closed)
Patch Set: rebase Created 3 years, 9 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: third_party/WebKit/Source/core/mojo/MojoWatcher.cpp
diff --git a/third_party/WebKit/Source/core/mojo/MojoWatcher.cpp b/third_party/WebKit/Source/core/mojo/MojoWatcher.cpp
index 804536d40d710b92a73294c1b43e30129550f58e..2175d06ca7899b0c915eef13c811fea95380d79c 100644
--- a/third_party/WebKit/Source/core/mojo/MojoWatcher.cpp
+++ b/third_party/WebKit/Source/core/mojo/MojoWatcher.cpp
@@ -20,6 +20,7 @@ static void runWatchCallback(MojoWatchCallback* callback,
callback->call(wrappable, result);
}
+// static
MojoWatcher* MojoWatcher::create(mojo::Handle handle,
const MojoHandleSignals& signalsDict,
MojoWatchCallback* callback,
@@ -27,8 +28,8 @@ MojoWatcher* MojoWatcher::create(mojo::Handle handle,
MojoWatcher* watcher = new MojoWatcher(context, callback);
MojoResult result = watcher->watch(handle, signalsDict);
// TODO(alokp): Consider raising an exception.
- // Current clients expect to recieve the initial error returned by MojoWatch
- // via watch callback.
+ // Current clients expect to recieve the initial error returned by
+ // MojoRegisterWatcher via watch callback.
//
// Note that the usage of wrapPersistent is intentional so that the intial
// error is guaranteed to be reported to the client in case where the given
@@ -39,45 +40,21 @@ MojoWatcher* MojoWatcher::create(mojo::Handle handle,
BLINK_FROM_HERE, WTF::bind(&runWatchCallback, wrapPersistent(callback),
wrapPersistent(watcher), result));
}
+
return watcher;
}
-MojoWatcher::MojoWatcher(ExecutionContext* context, MojoWatchCallback* callback)
- : ContextLifecycleObserver(context),
- m_taskRunner(TaskRunnerHelper::get(TaskType::UnspecedTimer, context)),
- m_callback(this, callback) {}
-
MojoWatcher::~MojoWatcher() {
DCHECK(!m_handle.is_valid());
}
-MojoResult MojoWatcher::watch(mojo::Handle handle,
- const MojoHandleSignals& signalsDict) {
- ::MojoHandleSignals signals = MOJO_HANDLE_SIGNAL_NONE;
- if (signalsDict.readable())
- signals |= MOJO_HANDLE_SIGNAL_READABLE;
- if (signalsDict.writable())
- signals |= MOJO_HANDLE_SIGNAL_WRITABLE;
- if (signalsDict.peerClosed())
- signals |= MOJO_HANDLE_SIGNAL_PEER_CLOSED;
-
- MojoResult result =
- MojoWatch(handle.value(), signals, &MojoWatcher::onHandleReady,
- reinterpret_cast<uintptr_t>(this));
- if (result == MOJO_RESULT_OK) {
- m_handle = handle;
- }
- return result;
-}
-
MojoResult MojoWatcher::cancel() {
- if (!m_handle.is_valid())
- return MOJO_RESULT_OK;
+ if (!m_watcherHandle.is_valid())
+ return MOJO_RESULT_INVALID_ARGUMENT;
- MojoResult result =
- MojoCancelWatch(m_handle.value(), reinterpret_cast<uintptr_t>(this));
+ m_watcherHandle.reset();
m_handle = mojo::Handle();
- return result;
+ return MOJO_RESULT_OK;
}
DEFINE_TRACE(MojoWatcher) {
@@ -90,25 +67,87 @@ DEFINE_TRACE_WRAPPERS(MojoWatcher) {
}
bool MojoWatcher::hasPendingActivity() const {
- return m_handle.is_valid();
+ return !!m_context;
}
void MojoWatcher::contextDestroyed(ExecutionContext*) {
cancel();
}
+MojoWatcher::MojoWatcher(ExecutionContext* context, MojoWatchCallback* callback)
+ : ContextLifecycleObserver(context),
+ m_taskRunner(TaskRunnerHelper::get(TaskType::UnspecedTimer, context)),
+ m_callback(this, callback) {}
+
+MojoResult MojoWatcher::watch(mojo::Handle handle,
+ const MojoHandleSignals& signalsDict) {
+ ::MojoHandleSignals signals = MOJO_HANDLE_SIGNAL_NONE;
+ if (signalsDict.readable())
+ signals |= MOJO_HANDLE_SIGNAL_READABLE;
+ if (signalsDict.writable())
+ signals |= MOJO_HANDLE_SIGNAL_WRITABLE;
+ if (signalsDict.peerClosed())
+ signals |= MOJO_HANDLE_SIGNAL_PEER_CLOSED;
+
+ m_handle = handle;
+
+ MojoResult rv =
+ mojo::CreateWatcher(&MojoWatcher::onHandleReady, &m_watcherHandle);
+ DCHECK_EQ(MOJO_RESULT_OK, rv);
+
+ m_context.reset(new CrossThreadPersistent<MojoWatcher>(this));
+ rv = MojoWatch(m_watcherHandle.get().value(), m_handle.value(), signals,
+ reinterpret_cast<uintptr_t>(m_context.get()));
+ if (rv != MOJO_RESULT_OK) {
+ m_context.reset();
+ return rv;
+ }
+
+ MojoResult readyResult;
+ rv = arm(&readyResult);
+ if (rv == MOJO_RESULT_OK)
+ return rv;
+
+ // We couldn't arm the watcher because the handle is already ready to
+ // trigger a success notification. Post a notification manually.
+ DCHECK_EQ(MOJO_RESULT_FAILED_PRECONDITION, rv);
+ m_taskRunner->postTask(BLINK_FROM_HERE,
+ WTF::bind(&MojoWatcher::runReadyCallback,
+ wrapPersistent(this), readyResult));
+ return MOJO_RESULT_OK;
+}
+
+MojoResult MojoWatcher::arm(MojoResult* readyResult) {
+ // Nothing to do if the watcher is inactive.
+ if (!m_handle.is_valid())
+ return MOJO_RESULT_OK;
+
+ uint32_t numReadyContexts = 1;
+ uintptr_t readyContext;
+ MojoResult localReadyResult;
+ MojoHandleSignalsState readySignals;
+ MojoResult rv =
+ MojoArmWatcher(m_watcherHandle.get().value(), &numReadyContexts,
+ &readyContext, &localReadyResult, &readySignals);
+ if (rv == MOJO_RESULT_OK)
+ return MOJO_RESULT_OK;
+
+ DCHECK_EQ(MOJO_RESULT_FAILED_PRECONDITION, rv);
+ DCHECK_EQ(1u, numReadyContexts);
+ DCHECK_EQ(reinterpret_cast<uintptr_t>(m_context.get()), readyContext);
+ *readyResult = localReadyResult;
+ return rv;
+}
+
void MojoWatcher::onHandleReady(uintptr_t context,
MojoResult result,
MojoHandleSignalsState,
- MojoWatchNotificationFlags) {
- // It is safe to assume the MojoWatcher still exists because this
- // callback will never be run after MojoWatcher destructor,
- // which cancels the watch.
- MojoWatcher* watcher = reinterpret_cast<MojoWatcher*>(context);
- watcher->m_taskRunner->postTask(
+ MojoWatcherNotificationFlags) {
+ auto* watcher =
+ reinterpret_cast<CrossThreadPersistent<MojoWatcher>*>(context);
+ (*watcher)->m_taskRunner->postTask(
BLINK_FROM_HERE,
- crossThreadBind(&MojoWatcher::runReadyCallback,
- wrapCrossThreadWeakPersistent(watcher), result));
+ crossThreadBind(&MojoWatcher::runReadyCallback, *watcher, result));
}
void MojoWatcher::runReadyCallback(MojoResult result) {
@@ -116,13 +155,40 @@ void MojoWatcher::runReadyCallback(MojoResult result) {
if (!m_handle.is_valid())
return;
- // MOJO_RESULT_CANCELLED indicates that the handle has been closed, in which
- // case watch has been implicitly cancelled. There is no need to explicitly
- // cancel the watch.
- if (result == MOJO_RESULT_CANCELLED)
+ // MOJO_RESULT_CANCELLED indicates that the watch has been cancelled, either
+ // explicitly or implicitly due to handle closure. The context object can be
+ // safely deleted at this point.
+ if (result == MOJO_RESULT_CANCELLED) {
m_handle = mojo::Handle();
+ m_context.reset();
+
+ // If cancellation was due to an explicit |cancel()|, don't run the watch
+ // callback for this notification.
+ if (!m_watcherHandle.is_valid())
+ return;
+
+ m_watcherHandle.reset();
+ runWatchCallback(m_callback, this, MOJO_RESULT_CANCELLED);
+ return;
+ }
runWatchCallback(m_callback, this, result);
+
+ // Rearm the watcher so another notification can fire.
+ //
+ // TODO(rockot): MojoWatcher should expose some better approximation of the
+ // new watcher API, including explicit add and removal of handles from the
+ // watcher, as well as explicit arming.
+ MojoResult readyResult;
+ MojoResult rv = arm(&readyResult);
+ if (rv == MOJO_RESULT_OK)
+ return;
+
+ DCHECK_EQ(MOJO_RESULT_FAILED_PRECONDITION, rv);
+
+ m_taskRunner->postTask(BLINK_FROM_HERE,
+ WTF::bind(&MojoWatcher::runReadyCallback,
+ wrapPersistent(this), readyResult));
}
} // namespace blink
« mojo/public/cpp/system/simple_watcher.cc ('K') | « third_party/WebKit/Source/core/mojo/MojoWatcher.h ('k') | no next file » | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698