15821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Copyright (c) 2012 The Chromium Authors. All rights reserved.
25821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Use of this source code is governed by a BSD-style license that can be
35821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// found in the LICENSE file.
45821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
55821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/shared_impl/tracked_callback.h"
65821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
75821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/bind.h"
85821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/compiler_specific.h"
95821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/logging.h"
109ab5563a3196760eb381d102cbb2bc0f7abc6a50Ben Murdoch#include "base/message_loop/message_loop.h"
115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "base/synchronization/lock.h"
125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/pp_completion_callback.h"
135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/c/pp_errors.h"
142a99a7e74a7f215066514fe81d2bfa6639d9edddTorne (Richard Coles)#include "ppapi/c/ppb_message_loop.h"
155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/shared_impl/callback_tracker.h"
165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/shared_impl/ppapi_globals.h"
175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/shared_impl/ppb_message_loop_shared.h"
185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/shared_impl/proxy_lock.h"
195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)#include "ppapi/shared_impl/resource.h"
205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace ppapi {
225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)namespace {
245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool IsMainThread() {
26a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)  return PpapiGlobals::Get()
27a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      ->GetMainThreadMessageLoop()
28a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)      ->BelongsToCurrentThread();
295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
31ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdochint32_t RunCompletionTask(TrackedCallback::CompletionTask completion_task,
32ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch                          int32_t result) {
33ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch  int32_t task_result = completion_task.Run(result);
34ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch  if (result != PP_ERROR_ABORTED)
35ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    result = task_result;
36ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch  return result;
37ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch}
38ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch
395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace
405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// TrackedCallback -------------------------------------------------------------
425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
435821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// Note: don't keep a Resource* since it may go out of scope before us.
44a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)TrackedCallback::TrackedCallback(Resource* resource,
45a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)                                 const PP_CompletionCallback& callback)
465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    : is_scheduled_(false),
475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      resource_id_(resource ? resource->pp_resource() : 0),
485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      completed_(false),
495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      aborted_(false),
505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      callback_(callback),
515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      target_loop_(PpapiGlobals::Get()->GetCurrentMessageLoop()),
525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      result_for_blocked_callback_(PP_OK) {
535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Note that target_loop_ may be NULL at this point, if the plugin has not
545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // attached a loop to this thread, or if this is an in-process plugin.
555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // The Enter class should handle checking this for us.
565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // TODO(dmichael): Add tracking at the instance level, for callbacks that only
585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // have an instance (e.g. for MouseLock).
595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (resource) {
605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    tracker_ = PpapiGlobals::Get()->GetCallbackTrackerForInstance(
615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)        resource->pp_instance());
625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    tracker_->Add(make_scoped_refptr(this));
635821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
645821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
65d0247b1b59f9c528cb6df88b4f2b9afaf80d181eTorne (Richard Coles)  base::Lock* proxy_lock = ProxyLock::Get();
665821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (proxy_lock) {
675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // If the proxy_lock is valid, we're running out-of-process, and locking
685821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // is enabled.
695821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (is_blocking()) {
705821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // This is a blocking completion callback, so we will need a condition
715821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // variable for blocking & signalling the calling thread.
725821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      operation_completed_condvar_.reset(
735821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)          new base::ConditionVariable(proxy_lock));
745821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    } else {
755821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // It's a non-blocking callback, so we should have a MessageLoopResource
765821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // to dispatch to. Note that we don't error check here, though. Later,
775821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // EnterResource::SetResult will check to make sure the callback is valid
785821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      // and take appropriate action.
795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
83a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)TrackedCallback::~TrackedCallback() {}
845821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
85a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)void TrackedCallback::Abort() { Run(PP_ERROR_ABORTED); }
865821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
87a1401311d1ab56c4ed0a474bd38c108f75cb0cd9Torne (Richard Coles)void TrackedCallback::PostAbort() { PostRun(PP_ERROR_ABORTED); }
885821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void TrackedCallback::Run(int32_t result) {
905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Only allow the callback to be run once. Note that this also covers the case
915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // where the callback was previously Aborted because its associated Resource
925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // went away. The callback may live on for a while because of a reference from
935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // a Closure. But when the Closure runs, Run() quietly does nothing, and the
945821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // callback will go away when all referring Closures go away.
955821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (completed())
965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (result == PP_ERROR_ABORTED)
985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    aborted_ = true;
995821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1005821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Note that this call of Run() may have been scheduled prior to Abort() or
1015821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // PostAbort() being called. If we have been told to Abort, that always
1025821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // trumps a result that was scheduled before, so we should make sure to pass
1035821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // PP_ERROR_ABORTED.
1045821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (aborted())
1055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    result = PP_ERROR_ABORTED;
1065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (is_blocking()) {
1085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // If the condition variable is invalid, there are two possibilities. One,
1095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // we're running in-process, in which case the call should have come in on
1105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // the main thread and we should have returned PP_ERROR_BLOCKS_MAIN_THREAD
1115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // well before this. Otherwise, this callback was not created as a
1125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // blocking callback. Either way, there's some internal error.
1135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    if (!operation_completed_condvar_.get()) {
1145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      NOTREACHED();
1155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return;
1165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
1175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    result_for_blocked_callback_ = result;
1185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Retain ourselves, since MarkAsCompleted will remove us from the
1195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // tracker. Then MarkAsCompleted before waking up the blocked thread,
1205821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // which could potentially re-enter.
1215821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    scoped_refptr<TrackedCallback> thiz(this);
1225821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    MarkAsCompleted();
1235821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Wake up the blocked thread. See BlockUntilComplete for where the thread
1245821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // Wait()s.
1255821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    operation_completed_condvar_->Signal();
1265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  } else {
1275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // If there's a target_loop_, and we're not on the right thread, we need to
1285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // post to target_loop_.
129868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)    if (target_loop_.get() &&
130868fa2fe829687343ffae624259930155e16dbd8Torne (Richard Coles)        target_loop_.get() != PpapiGlobals::Get()->GetCurrentMessageLoop()) {
1315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      PostRun(result);
1325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)      return;
1335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    }
134ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch
135ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    // Copy callback fields now, since |MarkAsCompleted()| may delete us.
1365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    PP_CompletionCallback callback = callback_;
137ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    CompletionTask completion_task = completion_task_;
138ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    completion_task_.Reset();
139ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    // Do this before running the callback in case of reentrancy from running
140ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    // the completion task.
1415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    MarkAsCompleted();
142ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch
143ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    if (!completion_task.is_null())
144ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch      result = RunCompletionTask(completion_task, result);
145ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch
1465821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // TODO(dmichael): Associate a message loop with the callback; if it's not
1475821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    // the same as the current thread's loop, then post it to the right loop.
1485821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    CallWhileUnlocked(PP_RunCompletionCallback, &callback, result);
1495821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1505821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1515821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
1525821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void TrackedCallback::PostRun(int32_t result) {
1535821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (completed()) {
1545821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    NOTREACHED();
1555821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return;
1565821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1575821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (result == PP_ERROR_ABORTED)
1585821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    aborted_ = true;
1595821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // We might abort when there's already a scheduled callback, but callers
1605821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // should never try to PostRun more than once otherwise.
1615821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(result == PP_ERROR_ABORTED || !is_scheduled_);
1625821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
16358537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)  if (is_blocking()) {
16458537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    // We might not have a MessageLoop to post to, so we must call Run()
16558537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    // directly.
16658537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    Run(result);
1675821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  } else {
16858537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    base::Closure callback_closure(
16958537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)        RunWhileLocked(base::Bind(&TrackedCallback::Run, this, result)));
1701320f92c476a1ad9d19dba2a48c72b75566198e9Primiano Tucci    if (target_loop_.get()) {
17158537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)      target_loop_->PostClosure(FROM_HERE, callback_closure, 0);
17258537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    } else {
17358537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)      // We must be running in-process and on the main thread (the Enter
17458537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)      // classes protect against having a null target_loop_ otherwise).
17558537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)      DCHECK(IsMainThread());
17658537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)      DCHECK(PpapiGlobals::Get()->IsHostGlobals());
17758537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)      base::MessageLoop::current()->PostTask(FROM_HERE, callback_closure);
17858537e28ecd584eab876aee8be7156509866d23aTorne (Richard Coles)    }
1795821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
1805821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  is_scheduled_ = true;
1815821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1825821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
183ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdochvoid TrackedCallback::set_completion_task(
184ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    const CompletionTask& completion_task) {
185ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch  DCHECK(completion_task_.is_null());
186ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch  completion_task_ = completion_task;
187ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch}
188ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch
1895821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)// static
1905821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)bool TrackedCallback::IsPending(
1915821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    const scoped_refptr<TrackedCallback>& callback) {
1925821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!callback.get())
1935821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return false;
194fb250657ef40d7500f20882d5c9909c1013367d3Ben Murdoch  if (callback->aborted())
195fb250657ef40d7500f20882d5c9909c1013367d3Ben Murdoch    return false;
1965821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return !callback->completed();
1975821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
1985821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
199558790d6acca3451cf3a6b497803a5f07d0bec58Ben Murdoch// static
200558790d6acca3451cf3a6b497803a5f07d0bec58Ben Murdochbool TrackedCallback::IsScheduledToRun(
201558790d6acca3451cf3a6b497803a5f07d0bec58Ben Murdoch    const scoped_refptr<TrackedCallback>& callback) {
202558790d6acca3451cf3a6b497803a5f07d0bec58Ben Murdoch  return IsPending(callback) && callback->is_scheduled_;
203558790d6acca3451cf3a6b497803a5f07d0bec58Ben Murdoch}
204558790d6acca3451cf3a6b497803a5f07d0bec58Ben Murdoch
2055821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)int32_t TrackedCallback::BlockUntilComplete() {
2065821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // Note, we are already holding the proxy lock in all these methods, including
2075821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // this one (see ppapi/thunk/enter.cc for where it gets acquired).
2085821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2095821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // It doesn't make sense to wait on a non-blocking callback. Furthermore,
2105821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // BlockUntilComplete should never be called for in-process plugins, where
2115821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // blocking callbacks are not supported.
2125821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  CHECK(operation_completed_condvar_.get());
2135821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (!is_blocking() || !operation_completed_condvar_.get()) {
2145821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    NOTREACHED();
2155821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    return PP_ERROR_FAILED;
2165821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  }
2175821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2185821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  while (!completed())
2195821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    operation_completed_condvar_->Wait();
220ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch
221ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch  if (!completion_task_.is_null()) {
222ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    result_for_blocked_callback_ =
223ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch        RunCompletionTask(completion_task_, result_for_blocked_callback_);
224ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch    completion_task_.Reset();
225ba5b9a6411cb1792fd21f0a078d7a25cd1ceec16Ben Murdoch  }
2265821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  return result_for_blocked_callback_;
2275821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2285821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2295821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)void TrackedCallback::MarkAsCompleted() {
2305821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  DCHECK(!completed());
2315821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2325821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // We will be removed; maintain a reference to ensure we won't be deleted
2335821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // until we're done.
2345821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  scoped_refptr<TrackedCallback> thiz = this;
2355821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  completed_ = true;
2365821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  // We may not have a valid resource, in which case we're not in the tracker.
2375821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  if (resource_id_)
2385821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)    tracker_->Remove(thiz);
2395821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)  tracker_ = NULL;
2405821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}
2415821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)
2425821806d5e7f356e8fa4b058a389a808ea183019Torne (Richard Coles)}  // namespace ppapi
243