// Copyright 2011 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 CCScopedThreadProxy_h #define CCScopedThreadProxy_h #include "base/logging.h" #include "base/threading/platform_thread.h" #include "cc/thread_task.h" #include #include #include namespace cc { // This class is a proxy used to post tasks to an target thread from any other thread. The proxy may be shut down at // any point from the target thread after which no more tasks posted to the proxy will run. In other words, all // tasks posted via a proxy are scoped to the lifecycle of the proxy. Use this when posting tasks to an object that // might die with tasks in flight. // // The proxy must be created and shut down from the target thread, tasks may be posted from any thread. // // Implementation note: Unlike ScopedRunnableMethodFactory in Chromium, pending tasks are not cancelled by actually // destroying the proxy. Instead each pending task holds a reference to the proxy to avoid maintaining an explicit // list of outstanding tasks. class ScopedThreadProxy : public ThreadSafeRefCounted { public: static PassRefPtr create(Thread* targetThread) { DCHECK(base::PlatformThread::CurrentId() == targetThread->threadID()); return adoptRef(new ScopedThreadProxy(targetThread)); } ~ScopedThreadProxy(); // Can be called from any thread. Posts a task to the target thread that runs unless // shutdown() is called before it runs. void postTask(PassOwnPtr task) { ref(); m_targetThread->postTask(createThreadTask(this, &ScopedThreadProxy::runTaskIfNotShutdown, task)); } void shutdown() { DCHECK(base::PlatformThread::CurrentId() == m_targetThread->threadID()); DCHECK(!m_shutdown); m_shutdown = true; } private: explicit ScopedThreadProxy(Thread* targetThread); void runTaskIfNotShutdown(PassOwnPtr popTask) { OwnPtr task = popTask; // If our shutdown flag is set, it's possible that m_targetThread has already been destroyed so don't // touch it. if (m_shutdown) { deref(); return; } DCHECK(base::PlatformThread::CurrentId() == m_targetThread->threadID()); task->performTask(); deref(); } Thread* m_targetThread; bool m_shutdown; // Only accessed on the target thread }; } #endif