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

Unified Diff: Source/core/dom/AsyncInitializerResolver.h

Issue 311733004: Introduce KeepAliveWhilePending to ScriptPromiseResolverWithContext. (Closed) Base URL: https://chromium.googlesource.com/chromium/blink.git@refactor-webmidi-initialization
Patch Set: Created 6 years, 6 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
« no previous file with comments | « Source/core/core.gypi ('k') | Source/modules/webmidi/MIDIAccess.h » ('j') | no next file with comments »
Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
Index: Source/core/dom/AsyncInitializerResolver.h
diff --git a/Source/core/dom/AsyncInitializerResolver.h b/Source/core/dom/AsyncInitializerResolver.h
new file mode 100644
index 0000000000000000000000000000000000000000..1a5e05f3fb009904fdaaac30e67c772298d7636f
--- /dev/null
+++ b/Source/core/dom/AsyncInitializerResolver.h
@@ -0,0 +1,116 @@
+// Copyright 2014 The Chromium Authors. All rights reserved.
+// Use of this source code is governed by a BSD-style license that can be
+// found in the LICENSE file.
+
+#ifndef AsyncInitializerResolver_h
+#define AsyncInitializerResolver_h
+
+#include "bindings/v8/ScriptPromiseResolverWithContext.h"
+#include "core/dom/ContextLifecycleObserver.h"
+#include "core/dom/ExecutionContext.h"
+#include "platform/heap/Heap.h"
+#include "wtf/OwnPtr.h"
+#include "wtf/RefPtr.h"
+#include "wtf/WeakPtr.h"
+
+namespace WebCore {
+
+// AsyncInitializerResolver is a helper class of asynchronous initialization
+// using Promises. Once AsyncInitializerResolver<T> is created, the resolver
+// will live until one of the following conditions is met:
+// - the associated promise is resolved
+// - the associated promise is rejected
+// - the associated execution context is destroyed
+//
+// AsyncInitializerResolver<Initializer> takes an actual initializer.
+// It is guaranteed the resolver stays alive while the initializer is alive
+// after start() is called,
+// i.e. You can access the resolver pointer set by start() in the initializer
+// If you want to access the resolver outside of the initializer,
+// use a weak pointer.
+//
+template<typename T>
+class AsyncInitializerResolver : public ContextLifecycleObserver {
+public:
+ typedef T Initializer;
+
+ static WeakPtr<AsyncInitializerResolver<T> > create(ScriptState* scriptState)
+ {
+ AsyncInitializerResolver* resolver = new AsyncInitializerResolver(scriptState);
+ return resolver->weakPtr();
+ }
+
+ void start(PassOwnPtr<Initializer> initializer)
+ {
+ ASSERT(!m_initializer);
+ m_initializer = initializer;
+ m_initializer->start(this);
+ }
+
+ static ScriptPromise createAndStart(ScriptState* scriptState, PassOwnPtr<T> initializer)
+ {
+ AsyncInitializerResolver* resolver = new AsyncInitializerResolver(scriptState);
+ ScriptPromise promise = resolver->promise();
+ resolver->start(initializer);
+ return promise;
+ }
+
+ virtual ~AsyncInitializerResolver() { }
+
+ ScriptPromise promise() { return m_resolver->promise(); }
+ ScriptState* scriptState() const { return m_resolver->scriptState(); }
+ template<typename U> void resolve(U u)
+ {
+ m_resolver->resolve(u);
+ m_protect.clear();
+ }
+ template<typename U> void reject(U u)
+ {
+ m_resolver->reject(u);
+ m_protect.clear();
+ }
+
+ // LifecycleObserver implementation.
+ // Because the lifecycle context is held in the associated execution
+ // context, this function is called when the execution context is
+ // destroyed.
+ virtual void contextDestroyed() OVERRIDE
+ {
+ ContextLifecycleObserver::contextDestroyed();
+ m_initializer->contextDestroyed();
+ // |this| will be deleted in the next Oilpan GC.
+ m_protect.clear();
+ }
+
+ WeakPtr<AsyncInitializerResolver<Initializer> > weakPtr() { return m_factory.createWeakPtr(); }
+
+private:
+ class Holder : public GarbageCollectedFinalized<Holder> {
+ public:
+ Holder(PassOwnPtr<AsyncInitializerResolver<T> > target) : m_protect(target) { }
+ virtual void trace(Visitor*) { }
+
+ private:
+ OwnPtr<AsyncInitializerResolver<T> > m_protect;
+ };
+
+ AsyncInitializerResolver(ScriptState* scriptState)
+ : ContextLifecycleObserver(scriptState->executionContext())
+ , m_resolver(ScriptPromiseResolverWithContext::create(scriptState))
+ , m_factory(this)
+ , m_protect(new Holder(adoptPtr(this)))
+ {
+ }
+
+ RefPtr<ScriptPromiseResolverWithContext> m_resolver;
+ OwnPtr<Initializer> m_initializer;
+ WeakPtrFactory<AsyncInitializerResolver<Initializer> > m_factory;
+ // In order to keep alive self until one of the termination conditions
+ // is met, AsyncInitizlierResolver has an OwnPtr referencing self.
+ // Clearing it means deleting it.
+ Persistent<Holder> m_protect;
kouhei (in TOK) 2014/06/05 07:54:47 Why don't we make AsyncInitializerResolver Garbage
yhirano 2014/06/05 09:09:19 My compiler complains at having Persistent in Garb
+};
+
+} // namespace WebCore
+
+#endif // #ifndef AsyncInitializerResolver_h
« no previous file with comments | « Source/core/core.gypi ('k') | Source/modules/webmidi/MIDIAccess.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698