Blame xpcom/threads/TaskQueue.cpp

Packit f0b94e
/* -*- Mode: C++; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
Packit f0b94e
/* vim: set ts=8 sts=2 et sw=2 tw=80: */
Packit f0b94e
/* This Source Code Form is subject to the terms of the Mozilla Public
Packit f0b94e
 * License, v. 2.0. If a copy of the MPL was not distributed with this
Packit f0b94e
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
Packit f0b94e
Packit f0b94e
#include "mozilla/TaskQueue.h"
Packit f0b94e
Packit f0b94e
#include "nsISerialEventTarget.h"
Packit f0b94e
#include "nsThreadUtils.h"
Packit f0b94e
Packit f0b94e
namespace mozilla {
Packit f0b94e
Packit f0b94e
class TaskQueue::EventTargetWrapper final : public nsISerialEventTarget {
Packit f0b94e
  RefPtr<TaskQueue> mTaskQueue;
Packit f0b94e
Packit f0b94e
  ~EventTargetWrapper() {}
Packit f0b94e
Packit f0b94e
 public:
Packit f0b94e
  explicit EventTargetWrapper(TaskQueue* aTaskQueue) : mTaskQueue(aTaskQueue) {
Packit f0b94e
    MOZ_ASSERT(mTaskQueue);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  NS_IMETHOD
Packit f0b94e
  DispatchFromScript(nsIRunnable* aEvent, uint32_t aFlags) override {
Packit f0b94e
    nsCOMPtr<nsIRunnable> ref = aEvent;
Packit f0b94e
    return Dispatch(ref.forget(), aFlags);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  NS_IMETHOD
Packit f0b94e
  Dispatch(already_AddRefed<nsIRunnable> aEvent, uint32_t aFlags) override {
Packit f0b94e
    nsCOMPtr<nsIRunnable> runnable = aEvent;
Packit f0b94e
    MonitorAutoLock mon(mTaskQueue->mQueueMonitor);
Packit f0b94e
    return mTaskQueue->DispatchLocked(/* passed by ref */ runnable,
Packit f0b94e
                                      NormalDispatch);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  NS_IMETHOD
Packit f0b94e
  DelayedDispatch(already_AddRefed<nsIRunnable>, uint32_t aFlags) override {
Packit f0b94e
    return NS_ERROR_NOT_IMPLEMENTED;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  NS_IMETHOD
Packit f0b94e
  IsOnCurrentThread(bool* aResult) override {
Packit f0b94e
    *aResult = mTaskQueue->IsCurrentThreadIn();
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  NS_IMETHOD_(bool)
Packit f0b94e
  IsOnCurrentThreadInfallible() override {
Packit f0b94e
    return mTaskQueue->mTarget->IsOnCurrentThread();
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  NS_DECL_THREADSAFE_ISUPPORTS
Packit f0b94e
};
Packit f0b94e
Packit f0b94e
NS_IMPL_ISUPPORTS(TaskQueue::EventTargetWrapper, nsIEventTarget,
Packit f0b94e
                  nsISerialEventTarget)
Packit f0b94e
Packit f0b94e
TaskQueue::TaskQueue(already_AddRefed<nsIEventTarget> aTarget,
Packit f0b94e
                     const char* aName, bool aRequireTailDispatch)
Packit f0b94e
    : AbstractThread(aRequireTailDispatch),
Packit f0b94e
      mTarget(aTarget),
Packit f0b94e
      mQueueMonitor("TaskQueue::Queue"),
Packit f0b94e
      mTailDispatcher(nullptr),
Packit f0b94e
      mIsRunning(false),
Packit f0b94e
      mIsShutdown(false),
Packit f0b94e
      mName(aName) {}
Packit f0b94e
Packit f0b94e
TaskQueue::TaskQueue(already_AddRefed<nsIEventTarget> aTarget,
Packit f0b94e
                     bool aSupportsTailDispatch)
Packit f0b94e
    : TaskQueue(Move(aTarget), "Unnamed", aSupportsTailDispatch) {}
Packit f0b94e
Packit f0b94e
TaskQueue::~TaskQueue() {
Packit f0b94e
  MonitorAutoLock mon(mQueueMonitor);
Packit f0b94e
  MOZ_ASSERT(mIsShutdown);
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
TaskDispatcher& TaskQueue::TailDispatcher() {
Packit f0b94e
  MOZ_ASSERT(IsCurrentThreadIn());
Packit f0b94e
  MOZ_ASSERT(mTailDispatcher);
Packit f0b94e
  return *mTailDispatcher;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
// Note aRunnable is passed by ref to support conditional ownership transfer.
Packit f0b94e
// See Dispatch() in TaskQueue.h for more details.
Packit f0b94e
nsresult TaskQueue::DispatchLocked(nsCOMPtr<nsIRunnable>& aRunnable,
Packit f0b94e
                                   DispatchReason aReason) {
Packit f0b94e
  mQueueMonitor.AssertCurrentThreadOwns();
Packit f0b94e
  if (mIsShutdown) {
Packit f0b94e
    return NS_ERROR_FAILURE;
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  AbstractThread* currentThread;
Packit f0b94e
  if (aReason != TailDispatch && (currentThread = GetCurrent()) &&
Packit f0b94e
      RequiresTailDispatch(currentThread)) {
Packit f0b94e
    return currentThread->TailDispatcher().AddTask(this, aRunnable.forget());
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  mTasks.push(aRunnable.forget());
Packit f0b94e
  if (mIsRunning) {
Packit f0b94e
    return NS_OK;
Packit f0b94e
  }
Packit f0b94e
  RefPtr<nsIRunnable> runner(new Runner(this));
Packit f0b94e
  nsresult rv = mTarget->Dispatch(runner.forget(), NS_DISPATCH_NORMAL);
Packit f0b94e
  if (NS_FAILED(rv)) {
Packit f0b94e
    NS_WARNING("Failed to dispatch runnable to run TaskQueue");
Packit f0b94e
    return rv;
Packit f0b94e
  }
Packit f0b94e
  mIsRunning = true;
Packit f0b94e
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void TaskQueue::AwaitIdle() {
Packit f0b94e
  MonitorAutoLock mon(mQueueMonitor);
Packit f0b94e
  AwaitIdleLocked();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void TaskQueue::AwaitIdleLocked() {
Packit f0b94e
  // Make sure there are no tasks for this queue waiting in the caller's tail
Packit f0b94e
  // dispatcher.
Packit f0b94e
  MOZ_ASSERT_IF(AbstractThread::GetCurrent(),
Packit f0b94e
                !AbstractThread::GetCurrent()->HasTailTasksFor(this));
Packit f0b94e
Packit f0b94e
  mQueueMonitor.AssertCurrentThreadOwns();
Packit f0b94e
  MOZ_ASSERT(mIsRunning || mTasks.empty());
Packit f0b94e
  while (mIsRunning) {
Packit f0b94e
    mQueueMonitor.Wait();
Packit f0b94e
  }
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
void TaskQueue::AwaitShutdownAndIdle() {
Packit f0b94e
  MOZ_ASSERT(!IsCurrentThreadIn());
Packit f0b94e
  // Make sure there are no tasks for this queue waiting in the caller's tail
Packit f0b94e
  // dispatcher.
Packit f0b94e
  MOZ_ASSERT_IF(AbstractThread::GetCurrent(),
Packit f0b94e
                !AbstractThread::GetCurrent()->HasTailTasksFor(this));
Packit f0b94e
Packit f0b94e
  MonitorAutoLock mon(mQueueMonitor);
Packit f0b94e
  while (!mIsShutdown) {
Packit f0b94e
    mQueueMonitor.Wait();
Packit f0b94e
  }
Packit f0b94e
  AwaitIdleLocked();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
RefPtr<ShutdownPromise> TaskQueue::BeginShutdown() {
Packit f0b94e
  // Dispatch any tasks for this queue waiting in the caller's tail dispatcher,
Packit f0b94e
  // since this is the last opportunity to do so.
Packit f0b94e
  if (AbstractThread* currentThread = AbstractThread::GetCurrent()) {
Packit f0b94e
    currentThread->TailDispatchTasksFor(this);
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  MonitorAutoLock mon(mQueueMonitor);
Packit f0b94e
  mIsShutdown = true;
Packit f0b94e
  RefPtr<ShutdownPromise> p = mShutdownPromise.Ensure(__func__);
Packit f0b94e
  MaybeResolveShutdown();
Packit f0b94e
  mon.NotifyAll();
Packit f0b94e
  return p;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
bool TaskQueue::IsEmpty() {
Packit f0b94e
  MonitorAutoLock mon(mQueueMonitor);
Packit f0b94e
  return mTasks.empty();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
uint32_t TaskQueue::ImpreciseLengthForHeuristics() {
Packit f0b94e
  MonitorAutoLock mon(mQueueMonitor);
Packit f0b94e
  return mTasks.size();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
bool TaskQueue::IsCurrentThreadIn() {
Packit f0b94e
  bool in = mRunningThread == GetCurrentPhysicalThread();
Packit f0b94e
  return in;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
already_AddRefed<nsISerialEventTarget> TaskQueue::WrapAsEventTarget() {
Packit f0b94e
  nsCOMPtr<nsISerialEventTarget> ref = new EventTargetWrapper(this);
Packit f0b94e
  return ref.forget();
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
nsresult TaskQueue::Runner::Run() {
Packit f0b94e
  RefPtr<nsIRunnable> event;
Packit f0b94e
  {
Packit f0b94e
    MonitorAutoLock mon(mQueue->mQueueMonitor);
Packit f0b94e
    MOZ_ASSERT(mQueue->mIsRunning);
Packit f0b94e
    if (mQueue->mTasks.empty()) {
Packit f0b94e
      mQueue->mIsRunning = false;
Packit f0b94e
      mQueue->MaybeResolveShutdown();
Packit f0b94e
      mon.NotifyAll();
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
    event = mQueue->mTasks.front().forget();
Packit f0b94e
    mQueue->mTasks.pop();
Packit f0b94e
  }
Packit f0b94e
  MOZ_ASSERT(event);
Packit f0b94e
Packit f0b94e
  // Note that dropping the queue monitor before running the task, and
Packit f0b94e
  // taking the monitor again after the task has run ensures we have memory
Packit f0b94e
  // fences enforced. This means that if the object we're calling wasn't
Packit f0b94e
  // designed to be threadsafe, it will be, provided we're only calling it
Packit f0b94e
  // in this task queue.
Packit f0b94e
  {
Packit f0b94e
    AutoTaskGuard g(mQueue);
Packit f0b94e
    event->Run();
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // Drop the reference to event. The event will hold a reference to the
Packit f0b94e
  // object it's calling, and we don't want to keep it alive, it may be
Packit f0b94e
  // making assumptions what holds references to it. This is especially
Packit f0b94e
  // the case if the object is waiting for us to shutdown, so that it
Packit f0b94e
  // can shutdown (like in the MediaDecoderStateMachine's SHUTDOWN case).
Packit f0b94e
  event = nullptr;
Packit f0b94e
Packit f0b94e
  {
Packit f0b94e
    MonitorAutoLock mon(mQueue->mQueueMonitor);
Packit f0b94e
    if (mQueue->mTasks.empty()) {
Packit f0b94e
      // No more events to run. Exit the task runner.
Packit f0b94e
      mQueue->mIsRunning = false;
Packit f0b94e
      mQueue->MaybeResolveShutdown();
Packit f0b94e
      mon.NotifyAll();
Packit f0b94e
      return NS_OK;
Packit f0b94e
    }
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  // There's at least one more event that we can run. Dispatch this Runner
Packit f0b94e
  // to the target again to ensure it runs again. Note that we don't just
Packit f0b94e
  // run in a loop here so that we don't hog the target. This means we may
Packit f0b94e
  // run on another thread next time, but we rely on the memory fences from
Packit f0b94e
  // mQueueMonitor for thread safety of non-threadsafe tasks.
Packit f0b94e
  nsresult rv = mQueue->mTarget->Dispatch(this, NS_DISPATCH_AT_END);
Packit f0b94e
  if (NS_FAILED(rv)) {
Packit f0b94e
    // Failed to dispatch, shutdown!
Packit f0b94e
    MonitorAutoLock mon(mQueue->mQueueMonitor);
Packit f0b94e
    mQueue->mIsRunning = false;
Packit f0b94e
    mQueue->mIsShutdown = true;
Packit f0b94e
    mQueue->MaybeResolveShutdown();
Packit f0b94e
    mon.NotifyAll();
Packit f0b94e
  }
Packit f0b94e
Packit f0b94e
  return NS_OK;
Packit f0b94e
}
Packit f0b94e
Packit f0b94e
}  // namespace mozilla