| Index: Source/modules/compositorworker/CompositorWorkerManager.cpp
 | 
| diff --git a/Source/modules/compositorworker/CompositorWorkerManager.cpp b/Source/modules/compositorworker/CompositorWorkerManager.cpp
 | 
| new file mode 100644
 | 
| index 0000000000000000000000000000000000000000..dc7fd62952dbcae3837c6ab1c628aa83409e4ddd
 | 
| --- /dev/null
 | 
| +++ b/Source/modules/compositorworker/CompositorWorkerManager.cpp
 | 
| @@ -0,0 +1,158 @@
 | 
| +// Copyright 2015 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.
 | 
| +
 | 
| +#include "config.h"
 | 
| +#include "modules/compositorworker/CompositorWorkerManager.h"
 | 
| +
 | 
| +#include "platform/ThreadSafeFunctional.h"
 | 
| +#include "platform/WebThreadSupportingGC.h"
 | 
| +#include "wtf/MainThread.h"
 | 
| +#include "wtf/ThreadingPrimitives.h"
 | 
| +
 | 
| +namespace blink {
 | 
| +
 | 
| +namespace {
 | 
| +
 | 
| +class SharedV8Isolate final : public WorkerIsolateWrapper {
 | 
| +    WTF_MAKE_NONCOPYABLE(SharedV8Isolate);
 | 
| +public:
 | 
| +    explicit SharedV8Isolate(v8::Isolate* isolate)
 | 
| +        : m_isolate(isolate) { }
 | 
| +    ~SharedV8Isolate() override
 | 
| +    {
 | 
| +        CompositorWorkerManager::instance()->destroyIsolate();
 | 
| +    }
 | 
| +
 | 
| +private:
 | 
| +    // WorkerIsolateWrapper:
 | 
| +    v8::Isolate* isolate() const override { return m_isolate; }
 | 
| +    void willDestroy() override
 | 
| +    {
 | 
| +        CompositorWorkerManager::instance()->willDestroyIsolate();
 | 
| +    }
 | 
| +
 | 
| +    void terminateExecution() override
 | 
| +    {
 | 
| +        CompositorWorkerManager::instance()->terminateIsolate();
 | 
| +    }
 | 
| +
 | 
| +    v8::Isolate* m_isolate;
 | 
| +};
 | 
| +
 | 
| +static CompositorWorkerManager* s_instance = nullptr;
 | 
| +
 | 
| +static Mutex& singletonMutex()
 | 
| +{
 | 
| +    AtomicallyInitializedStaticReference(Mutex, mutex, new Mutex);
 | 
| +    return mutex;
 | 
| +}
 | 
| +
 | 
| +static void destroyThread(WebThreadSupportingGC* thread)
 | 
| +{
 | 
| +    delete thread;
 | 
| +}
 | 
| +
 | 
| +} // namespace
 | 
| +
 | 
| +void CompositorWorkerManager::initialize()
 | 
| +{
 | 
| +    MutexLocker lock(singletonMutex());
 | 
| +    ASSERT(!s_instance);
 | 
| +    s_instance = new CompositorWorkerManager();
 | 
| +}
 | 
| +
 | 
| +void CompositorWorkerManager::shutdown()
 | 
| +{
 | 
| +    MutexLocker lock(singletonMutex());
 | 
| +    ASSERT(s_instance);
 | 
| +    delete s_instance;
 | 
| +    s_instance = nullptr;
 | 
| +}
 | 
| +
 | 
| +CompositorWorkerManager* CompositorWorkerManager::instance()
 | 
| +{
 | 
| +    MutexLocker lock(singletonMutex());
 | 
| +    ASSERT(s_instance);
 | 
| +    return s_instance;
 | 
| +}
 | 
| +
 | 
| +CompositorWorkerManager::CompositorWorkerManager()
 | 
| +{
 | 
| +}
 | 
| +
 | 
| +CompositorWorkerManager::~CompositorWorkerManager()
 | 
| +{
 | 
| +}
 | 
| +
 | 
| +WebThreadSupportingGC& CompositorWorkerManager::compositorWorkerThread()
 | 
| +{
 | 
| +    MutexLocker lock(m_mutex);
 | 
| +    if (!m_thread) {
 | 
| +        ASSERT(isMainThread());
 | 
| +        ASSERT(!m_workerCount);
 | 
| +        m_thread = WebThreadSupportingGC::create("CompositorWorker Thread");
 | 
| +    }
 | 
| +    return *m_thread.get();
 | 
| +}
 | 
| +
 | 
| +void CompositorWorkerManager::initializeBackingThread()
 | 
| +{
 | 
| +    ASSERT(m_thread->isCurrentThread());
 | 
| +    MutexLocker lock(m_mutex);
 | 
| +    ++m_workerCount;
 | 
| +    if (m_workerCount == 1)
 | 
| +        m_thread->initialize();
 | 
| +}
 | 
| +
 | 
| +void CompositorWorkerManager::shutdownBackingThread()
 | 
| +{
 | 
| +    MutexLocker lock(m_mutex);
 | 
| +    ASSERT(m_thread->isCurrentThread());
 | 
| +    ASSERT(m_workerCount > 0);
 | 
| +    --m_workerCount;
 | 
| +    if (m_workerCount == 0) {
 | 
| +        m_thread->shutdown();
 | 
| +        Platform::current()->mainThread()->postTask(FROM_HERE, threadSafeBind(destroyThread, AllowCrossThreadAccess(m_thread.leakPtr())));
 | 
| +    }
 | 
| +}
 | 
| +
 | 
| +PassOwnPtr<WorkerIsolateWrapper> CompositorWorkerManager::createIsolate()
 | 
| +{
 | 
| +    MutexLocker lock(m_mutex);
 | 
| +    ASSERT(m_thread->isCurrentThread());
 | 
| +    if (!m_isolate)
 | 
| +        m_isolate = WorkerIsolateWrapper::createDefault();
 | 
| +    // It is safe to use the existing isolate even if TerminateExecution() has been
 | 
| +    // called on it, without calling CancelTerminateExecution().
 | 
| +    return adoptPtr(new SharedV8Isolate(m_isolate->isolate()));
 | 
| +}
 | 
| +
 | 
| +void CompositorWorkerManager::willDestroyIsolate()
 | 
| +{
 | 
| +    MutexLocker lock(m_mutex);
 | 
| +    ASSERT(m_thread->isCurrentThread());
 | 
| +    if (m_workerCount == 1)
 | 
| +        m_isolate->willDestroy();
 | 
| +}
 | 
| +
 | 
| +void CompositorWorkerManager::destroyIsolate()
 | 
| +{
 | 
| +    MutexLocker lock(m_mutex);
 | 
| +    if (!m_thread) {
 | 
| +        ASSERT(m_workerCount == 0);
 | 
| +        m_isolate.clear();
 | 
| +    }
 | 
| +}
 | 
| +
 | 
| +void CompositorWorkerManager::terminateIsolate()
 | 
| +{
 | 
| +    MutexLocker lock(m_mutex);
 | 
| +    ASSERT(isMainThread());
 | 
| +    if (m_workerCount > 1)
 | 
| +        return;
 | 
| +
 | 
| +    m_isolate->terminateExecution();
 | 
| +}
 | 
| +
 | 
| +} // namespace blink
 | 
| 
 |