// Copyright 2014 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 CC_RESOURCES_TASK_GRAPH_RUNNER_H_
#define CC_RESOURCES_TASK_GRAPH_RUNNER_H_
#include <map>
#include <vector>
#include "base/logging.h"
#include "base/memory/ref_counted.h"
#include "base/synchronization/condition_variable.h"
#include "cc/base/cc_export.h"
namespace cc {
class CC_EXPORT Task : public base::RefCountedThreadSafe<Task> {
public:
typedef std::vector<scoped_refptr<Task> > Vector;
virtual void RunOnWorkerThread() = 0;
void WillRun();
void DidRun();
bool HasFinishedRunning() const;
protected:
friend class base::RefCountedThreadSafe<Task>;
Task();
virtual ~Task();
bool will_run_;
bool did_run_;
};
// Dependencies are represented as edges in a task graph. Each graph node is
// assigned a priority and a run count that matches the number of dependencies.
// Priority range from 0 (most favorable scheduling) to UINT_MAX (least
// favorable).
struct CC_EXPORT TaskGraph {
struct Node {
class TaskComparator {
public:
explicit TaskComparator(const Task* task) : task_(task) {}
bool operator()(const Node& node) const { return node.task == task_; }
private:
const Task* task_;
};
typedef std::vector<Node> Vector;
Node(Task* task, unsigned priority, size_t dependencies)
: task(task), priority(priority), dependencies(dependencies) {}
Task* task;
unsigned priority;
size_t dependencies;
};
struct Edge {
typedef std::vector<Edge> Vector;
Edge(const Task* task, Task* dependent)
: task(task), dependent(dependent) {}
const Task* task;
Task* dependent;
};
TaskGraph();
~TaskGraph();
void Swap(TaskGraph* other);
void Reset();
Node::Vector nodes;
Edge::Vector edges;
};
class TaskGraphRunner;
// Opaque identifier that defines a namespace of tasks.
class CC_EXPORT NamespaceToken {
public:
NamespaceToken() : id_(0) {}
~NamespaceToken() {}
bool IsValid() const { return id_ != 0; }
private:
friend class TaskGraphRunner;
explicit NamespaceToken(int id) : id_(id) {}
int id_;
};
// A TaskGraphRunner is used to process tasks with dependencies. There can
// be any number of TaskGraphRunner instances per thread. Tasks can be scheduled
// from any thread and they can be run on any thread.
class CC_EXPORT TaskGraphRunner {
public:
TaskGraphRunner();
virtual ~TaskGraphRunner();
// Returns a unique token that can be used to pass a task graph to
// ScheduleTasks(). Valid tokens are always nonzero.
NamespaceToken GetNamespaceToken();
// Schedule running of tasks in |graph|. Tasks previously scheduled but no
// longer needed will be canceled unless already running. Canceled tasks are
// moved to |completed_tasks| without being run. The result is that once
// scheduled, a task is guaranteed to end up in the |completed_tasks| queue
// even if it later gets canceled by another call to ScheduleTasks().
void ScheduleTasks(NamespaceToken token, TaskGraph* graph);
// Wait for all scheduled tasks to finish running.
void WaitForTasksToFinishRunning(NamespaceToken token);
// Collect all completed tasks in |completed_tasks|.
void CollectCompletedTasks(NamespaceToken token,
Task::Vector* completed_tasks);
// Run tasks until Shutdown() is called.
void Run();
// Process all pending tasks, but don't wait/sleep. Return as soon as all
// tasks that can be run are taken care of.
void RunUntilIdle();
// Signals the Run method to return when it becomes idle. It will continue to
// process tasks and future tasks as long as they are scheduled.
// Warning: if the TaskGraphRunner remains busy, it may never quit.
void Shutdown();
private:
struct PrioritizedTask {
typedef std::vector<PrioritizedTask> Vector;
PrioritizedTask(Task* task, unsigned priority)
: task(task), priority(priority) {}
Task* task;
unsigned priority;
};
typedef std::vector<const Task*> TaskVector;
struct TaskNamespace {
typedef std::vector<TaskNamespace*> Vector;
TaskNamespace();
~TaskNamespace();
// Current task graph.
TaskGraph graph;
// Ordered set of tasks that are ready to run.
PrioritizedTask::Vector ready_to_run_tasks;
// Completed tasks not yet collected by origin thread.
Task::Vector completed_tasks;
// This set contains all currently running tasks.
TaskVector running_tasks;
};
typedef std::map<int, TaskNamespace> TaskNamespaceMap;
static bool CompareTaskPriority(const PrioritizedTask& a,
const PrioritizedTask& b) {
// In this system, numerically lower priority is run first.
return a.priority > b.priority;
}
static bool CompareTaskNamespacePriority(const TaskNamespace* a,
const TaskNamespace* b) {
DCHECK(!a->ready_to_run_tasks.empty());
DCHECK(!b->ready_to_run_tasks.empty());
// Compare based on task priority of the ready_to_run_tasks heap .front()
// will hold the max element of the heap, except after pop_heap, when max
// element is moved to .back().
return CompareTaskPriority(a->ready_to_run_tasks.front(),
b->ready_to_run_tasks.front());
}
static bool HasFinishedRunningTasksInNamespace(
const TaskNamespace* task_namespace) {
return task_namespace->running_tasks.empty() &&
task_namespace->ready_to_run_tasks.empty();
}
// Run next task. Caller must acquire |lock_| prior to calling this function
// and make sure at least one task is ready to run.
void RunTaskWithLockAcquired();
// This lock protects all members of this class. Do not read or modify
// anything without holding this lock. Do not block while holding this lock.
mutable base::Lock lock_;
// Condition variable that is waited on by Run() until new tasks are ready to
// run or shutdown starts.
base::ConditionVariable has_ready_to_run_tasks_cv_;
// Condition variable that is waited on by origin threads until a namespace
// has finished running all associated tasks.
base::ConditionVariable has_namespaces_with_finished_running_tasks_cv_;
// Provides a unique id to each NamespaceToken.
int next_namespace_id_;
// This set contains all namespaces with pending, running or completed tasks
// not yet collected.
TaskNamespaceMap namespaces_;
// Ordered set of task namespaces that have ready to run tasks.
TaskNamespace::Vector ready_to_run_namespaces_;
// Set during shutdown. Tells Run() to return when no more tasks are pending.
bool shutdown_;
DISALLOW_COPY_AND_ASSIGN(TaskGraphRunner);
};
} // namespace cc
#endif // CC_RESOURCES_TASK_GRAPH_RUNNER_H_