// Copyright 2013 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 CHROME_BROWSER_SYNC_FILE_SYSTEM_SYNC_TASK_MANAGER_H_ #define CHROME_BROWSER_SYNC_FILE_SYSTEM_SYNC_TASK_MANAGER_H_ #include #include #include "base/callback.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "base/threading/non_thread_safe.h" #include "chrome/browser/sync_file_system/sync_callbacks.h" #include "chrome/browser/sync_file_system/sync_status_code.h" #include "chrome/browser/sync_file_system/sync_task.h" namespace tracked_objects { class Location; } namespace sync_file_system { class SyncTaskManager : public base::NonThreadSafe, public base::SupportsWeakPtr { public: class TaskToken; typedef base::Callback Task; enum Priority { PRIORITY_LOW, PRIORITY_MED, PRIORITY_HIGH, }; class Client { public: virtual ~Client() {} // Called when the manager is idle. virtual void MaybeScheduleNextTask() = 0; // Called when the manager is notified a task is done. virtual void NotifyLastOperationStatus( SyncStatusCode last_operation_status, bool last_operation_used_network) = 0; }; explicit SyncTaskManager(base::WeakPtr client); virtual ~SyncTaskManager(); // This needs to be called to start task scheduling. // If |status| is not SYNC_STATUS_OK calling this may change the // service status. This should not be called more than once. void Initialize(SyncStatusCode status); // Schedules a task at PRIORITY_MED. void ScheduleTask(const Task& task, const SyncStatusCallback& callback); void ScheduleSyncTask(scoped_ptr task, const SyncStatusCallback& callback); // Schedules a task at the given priority. void ScheduleTaskAtPriority(const Task& task, Priority priority, const SyncStatusCallback& callback); void ScheduleSyncTaskAtPriority(scoped_ptr task, Priority priority, const SyncStatusCallback& callback); // Runs the posted task only when we're idle. Returns true if tha task is // scheduled. bool ScheduleTaskIfIdle(const Task& task, const SyncStatusCallback& callback); bool ScheduleSyncTaskIfIdle(scoped_ptr task, const SyncStatusCallback& callback); void NotifyTaskDone(scoped_ptr token, SyncStatusCode status); private: struct PendingTask { base::Closure task; Priority priority; int64 seq; PendingTask(); PendingTask(const base::Closure& task, Priority pri, int seq); ~PendingTask(); }; struct PendingTaskComparator { bool operator()(const PendingTask& left, const PendingTask& right) const; }; // This should be called when an async task needs to get a task token. scoped_ptr GetToken(const tracked_objects::Location& from_here); // Creates a completion callback that calls NotifyTaskDone. // It is ok to give null |callback|. SyncStatusCallback CreateCompletionCallback( scoped_ptr token, const SyncStatusCallback& callback); void PushPendingTask(const base::Closure& closure, Priority priority); base::WeakPtr client_; SyncStatusCode last_operation_status_; scoped_ptr running_task_; SyncStatusCallback current_callback_; std::priority_queue, PendingTaskComparator> pending_tasks_; int64 pending_task_seq_; // Absence of |token_| implies a task is running. Incoming tasks should // wait for the task to finish in |pending_tasks_| if |token_| is null. // Each task must take TaskToken instance from |token_| and must hold it // until it finished. And the task must return the instance through // NotifyTaskDone when the task finished. scoped_ptr token_; DISALLOW_COPY_AND_ASSIGN(SyncTaskManager); }; } // namespace sync_file_system #endif // CHROME_BROWSER_SYNC_FILE_SYSTEM_SYNC_TASK_MANAGER_H_