// Copyright (c) 2012 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 "ppapi/shared_impl/tracked_callback.h" #include "base/bind.h" #include "base/compiler_specific.h" #include "base/location.h" #include "base/logging.h" #include "base/single_thread_task_runner.h" #include "base/synchronization/lock.h" #include "base/thread_task_runner_handle.h" #include "ppapi/c/pp_completion_callback.h" #include "ppapi/c/pp_errors.h" #include "ppapi/c/ppb_message_loop.h" #include "ppapi/shared_impl/callback_tracker.h" #include "ppapi/shared_impl/ppapi_globals.h" #include "ppapi/shared_impl/ppb_message_loop_shared.h" #include "ppapi/shared_impl/proxy_lock.h" #include "ppapi/shared_impl/resource.h" namespace ppapi { namespace { bool IsMainThread() { return PpapiGlobals::Get() ->GetMainThreadMessageLoop() ->BelongsToCurrentThread(); } int32_t RunCompletionTask(TrackedCallback::CompletionTask completion_task, int32_t result) { ProxyLock::AssertAcquired(); int32_t task_result = completion_task.Run(result); if (result != PP_ERROR_ABORTED) result = task_result; return result; } } // namespace // TrackedCallback ------------------------------------------------------------- // Note: don't keep a Resource* since it may go out of scope before us. TrackedCallback::TrackedCallback(Resource* resource, const PP_CompletionCallback& callback) : is_scheduled_(false), resource_id_(resource ? resource->pp_resource() : 0), completed_(false), aborted_(false), callback_(callback), target_loop_(PpapiGlobals::Get()->GetCurrentMessageLoop()), result_for_blocked_callback_(PP_OK) { // Note that target_loop_ may be NULL at this point, if the plugin has not // attached a loop to this thread, or if this is an in-process plugin. // The Enter class should handle checking this for us. // TODO(dmichael): Add tracking at the instance level, for callbacks that only // have an instance (e.g. for MouseLock). if (resource) { tracker_ = PpapiGlobals::Get()->GetCallbackTrackerForInstance( resource->pp_instance()); tracker_->Add(make_scoped_refptr(this)); } base::Lock* proxy_lock = ProxyLock::Get(); if (proxy_lock) { ProxyLock::AssertAcquired(); // If the proxy_lock is valid, we're running out-of-process, and locking // is enabled. if (is_blocking()) { // This is a blocking completion callback, so we will need a condition // variable for blocking & signalling the calling thread. operation_completed_condvar_.reset( new base::ConditionVariable(proxy_lock)); } else { // It's a non-blocking callback, so we should have a MessageLoopResource // to dispatch to. Note that we don't error check here, though. Later, // EnterResource::SetResult will check to make sure the callback is valid // and take appropriate action. } } } TrackedCallback::~TrackedCallback() {} void TrackedCallback::Abort() { Run(PP_ERROR_ABORTED); } void TrackedCallback::PostAbort() { PostRun(PP_ERROR_ABORTED); } void TrackedCallback::Run(int32_t result) { // Retain ourselves, since SignalBlockingCallback and MarkAsCompleted might // otherwise cause |this| to be deleted. Do this before acquiring lock_ so // that |this| is definitely valid at the time we release |lock_|. scoped_refptr thiz(this); base::AutoLock acquire(lock_); // Only allow the callback to be run once. Note that this also covers the case // where the callback was previously Aborted because its associated Resource // went away. The callback may live on for a while because of a reference from // a Closure. But when the Closure runs, Run() quietly does nothing, and the // callback will go away when all referring Closures go away. if (completed_) return; if (result == PP_ERROR_ABORTED) aborted_ = true; // Note that this call of Run() may have been scheduled prior to Abort() or // PostAbort() being called. If we have been told to Abort, that always // trumps a result that was scheduled before, so we should make sure to pass // PP_ERROR_ABORTED. if (aborted_) result = PP_ERROR_ABORTED; if (is_blocking()) { // This is a blocking callback; signal the condvar to wake up the thread. SignalBlockingCallback(result); } else { // If there's a target_loop_, and we're not on the right thread, we need to // post to target_loop_. if (target_loop_ && target_loop_.get() != PpapiGlobals::Get()->GetCurrentMessageLoop()) { PostRunWithLock(result); return; } // Do this before running the callback in case of reentrancy from running // the completion callback. MarkAsCompletedWithLock(); if (!completion_task_.is_null()) result = RunCompletionTask(completion_task_, result); { base::AutoUnlock release(lock_); // Call the callback without lock_ and without the ProxyLock. CallWhileUnlocked(PP_RunCompletionCallback, &callback_, result); } } } void TrackedCallback::PostRun(int32_t result) { base::AutoLock acquire(lock_); PostRunWithLock(result); } void TrackedCallback::set_completion_task( const CompletionTask& completion_task) { base::AutoLock acquire(lock_); DCHECK(completion_task_.is_null()); completion_task_ = completion_task; } // static bool TrackedCallback::IsPending( const scoped_refptr& callback) { if (!callback) return false; base::AutoLock acquire(callback->lock_); if (callback->aborted_) return false; return !callback->completed_; } // static bool TrackedCallback::IsScheduledToRun( const scoped_refptr& callback) { if (!callback) return false; base::AutoLock acquire(callback->lock_); if (callback->aborted_) return false; return !callback->completed_ && callback->is_scheduled_; } int32_t TrackedCallback::BlockUntilComplete() { // Note, we are already holding the proxy lock in this method and many others // (see ppapi/thunk/enter.cc for where it gets acquired). ProxyLock::AssertAcquired(); base::AutoLock acquire(lock_); // It doesn't make sense to wait on a non-blocking callback. Furthermore, // BlockUntilComplete should never be called for in-process plugins, where // blocking callbacks are not supported. CHECK(is_blocking() && operation_completed_condvar_); // Protect us from being deleted to ensure operation_completed_condvar_ is // available to wait on when we drop our lock. scoped_refptr thiz(this); while (!completed_) { // Unlock our lock temporarily; any thread that tries to signal us will need // the lock. lock_.Release(); operation_completed_condvar_->Wait(); // Note that the condvar releases the ProxyLock during Wait and re-acquires // the ProxyLock when it's signaled. We reacquire lock_ immediately after, // preserving lock order. ProxyLock::AssertAcquired(); lock_.Acquire(); } if (!completion_task_.is_null()) { result_for_blocked_callback_ = RunCompletionTask(completion_task_, result_for_blocked_callback_); completion_task_.Reset(); } return result_for_blocked_callback_; } void TrackedCallback::MarkAsCompleted() { base::AutoLock acquire(lock_); MarkAsCompletedWithLock(); } void TrackedCallback::MarkAsCompletedWithLock() { lock_.AssertAcquired(); DCHECK(!completed_); // We will be removed; maintain a reference to ensure we won't be deleted // until we're done. scoped_refptr thiz = this; completed_ = true; // We may not have a valid resource, in which case we're not in the tracker. if (resource_id_) tracker_->Remove(thiz); tracker_ = NULL; target_loop_ = NULL; } void TrackedCallback::PostRunWithLock(int32_t result) { lock_.AssertAcquired(); if (completed_) { NOTREACHED(); return; } if (result == PP_ERROR_ABORTED) aborted_ = true; // We might abort when there's already a scheduled callback, but callers // should never try to PostRun more than once otherwise. DCHECK(result == PP_ERROR_ABORTED || !is_scheduled_); if (is_blocking()) { // We might not have a MessageLoop to post to, so we must Signal // directly. SignalBlockingCallback(result); } else { base::Closure callback_closure( RunWhileLocked(base::Bind(&TrackedCallback::Run, this, result))); if (target_loop_) { target_loop_->PostClosure(FROM_HERE, callback_closure, 0); } else { // We must be running in-process and on the main thread (the Enter // classes protect against having a null target_loop_ otherwise). DCHECK(IsMainThread()); DCHECK(PpapiGlobals::Get()->IsHostGlobals()); base::ThreadTaskRunnerHandle::Get()->PostTask(FROM_HERE, callback_closure); } } is_scheduled_ = true; } void TrackedCallback::SignalBlockingCallback(int32_t result) { lock_.AssertAcquired(); DCHECK(is_blocking()); if (!operation_completed_condvar_) { // If the condition variable is invalid, there are two possibilities. One, // we're running in-process, in which case the call should have come in on // the main thread and we should have returned PP_ERROR_BLOCKS_MAIN_THREAD // well before this. Otherwise, this callback was not created as a // blocking callback. Either way, there's some internal error. NOTREACHED(); return; } result_for_blocked_callback_ = result; // Retain ourselves, since MarkAsCompleted will remove us from the // tracker. Then MarkAsCompleted before waking up the blocked thread, // which could potentially re-enter. scoped_refptr thiz(this); MarkAsCompletedWithLock(); // Wake up the blocked thread. See BlockUntilComplete for where the thread // Wait()s. operation_completed_condvar_->Signal(); } } // namespace ppapi