summaryrefslogtreecommitdiffstats
path: root/base/message_loop.h
diff options
context:
space:
mode:
Diffstat (limited to 'base/message_loop.h')
-rw-r--r--base/message_loop.h172
1 files changed, 57 insertions, 115 deletions
diff --git a/base/message_loop.h b/base/message_loop.h
index babc17a..53c832a 100644
--- a/base/message_loop.h
+++ b/base/message_loop.h
@@ -80,24 +80,34 @@ class MessageLoop : public base::MessagePump::Delegate {
// DestructionObserver is receiving a notification callback.
void RemoveDestructionObserver(DestructionObserver* destruction_observer);
- // Call the task's Run method asynchronously from within a message loop at
- // some point in the future. With the PostTask variant, tasks are invoked in
- // FIFO order, inter-mixed with normal UI event processing. With the
- // PostDelayedTask variant, tasks are called after at least approximately
- // 'delay_ms' have elapsed.
+ // The "PostTask" family of methods call the task's Run method asynchronously
+ // from within a message loop at some point in the future.
//
- // The MessageLoop takes ownership of the Task, and deletes it after it
- // has been Run().
+ // With the PostTask variant, tasks are invoked in FIFO order, inter-mixed
+ // with normal UI or IO event processing. With the PostDelayedTask variant,
+ // tasks are called after at least approximately 'delay_ms' have elapsed.
//
- // NOTE: This method may be called on any thread. The Task will be invoked
+ // The NonNestable variants work similarly except that they promise never to
+ // dispatch the task from a nested invocation of MessageLoop::Run. Instead,
+ // such tasks get deferred until the top-most MessageLoop::Run is executing.
+ //
+ // The MessageLoop takes ownership of the Task, and deletes it after it has
+ // been Run().
+ //
+ // NOTE: These methods may be called on any thread. The Task will be invoked
// on the thread that executes MessageLoop::Run().
+
+ void PostTask(
+ const tracked_objects::Location& from_here, Task* task);
+
+ void PostDelayedTask(
+ const tracked_objects::Location& from_here, Task* task, int delay_ms);
- void PostTask(const tracked_objects::Location& from_here, Task* task) {
- PostDelayedTask(from_here, task, 0);
- }
+ void PostNonNestableTask(
+ const tracked_objects::Location& from_here, Task* task);
- void PostDelayedTask(const tracked_objects::Location& from_here, Task* task,
- int delay_ms);
+ void PostNonNestableDelayedTask(
+ const tracked_objects::Location& from_here, Task* task, int delay_ms);
// A variant on PostTask that deletes the given object. This is useful
// if the object needs to live until the next run of the MessageLoop (for
@@ -110,7 +120,7 @@ class MessageLoop : public base::MessagePump::Delegate {
// from RefCountedThreadSafe<T>!
template <class T>
void DeleteSoon(const tracked_objects::Location& from_here, T* object) {
- PostTask(from_here, new DeleteTask<T>(object));
+ PostNonNestableTask(from_here, new DeleteTask<T>(object));
}
// A variant on PostTask that releases the given reference counted object
@@ -125,7 +135,7 @@ class MessageLoop : public base::MessagePump::Delegate {
// RefCountedThreadSafe<T>!
template <class T>
void ReleaseSoon(const tracked_objects::Location& from_here, T* object) {
- PostTask(from_here, new ReleaseTask<T>(object));
+ PostNonNestableTask(from_here, new ReleaseTask<T>(object));
}
// Run the message loop.
@@ -199,10 +209,6 @@ class MessageLoop : public base::MessagePump::Delegate {
return loop;
}
- // Returns the TimerManager object for the current thread. This getter is
- // deprecated. Please use OneShotTimer or RepeatingTimer instead.
- base::TimerManager* timer_manager_deprecated() { return &timer_manager_; }
-
// Enables or disables the recursive task processing. This happens in the case
// of recursive message loops. Some unwanted message loop may occurs when
// using common controls or printer functions. By default, recursive task
@@ -229,11 +235,8 @@ class MessageLoop : public base::MessagePump::Delegate {
exception_restoration_ = restore;
}
-
//----------------------------------------------------------------------------
protected:
- friend class base::TimerManager; // So it can call DidChangeNextTimerExpiry
-
struct RunState {
// Used to count how many Run() invocations are on the stack.
int run_depth;
@@ -256,70 +259,23 @@ class MessageLoop : public base::MessagePump::Delegate {
RunState* previous_state_;
};
- // A prioritized queue with interface that mostly matches std::queue<>.
- // For debugging/performance testing, you can swap in std::queue<Task*>.
- class PrioritizedTaskQueue {
- public:
- PrioritizedTaskQueue() : next_sequence_number_(0) {}
- ~PrioritizedTaskQueue() {}
- void pop() { queue_.pop(); }
- bool empty() { return queue_.empty(); }
- size_t size() { return queue_.size(); }
- Task* front() { return queue_.top().task(); }
- void push(Task * task);
+ // This structure is copied around by value.
+ struct PendingTask {
+ Task* task; // The task to run.
+ Time delayed_run_time; // The time when the task should be run.
+ int sequence_num; // Used to facilitate sorting by run time.
+ bool nestable; // True if OK to dispatch from a nested loop.
- private:
- class PrioritizedTask {
- public:
- PrioritizedTask(Task* task, int sequence_number)
- : task_(task),
- sequence_number_(sequence_number),
- priority_(task->priority()) {}
- Task* task() const { return task_; }
- bool operator < (PrioritizedTask const & right) const ;
-
- private:
- Task* task_;
- // Number to ensure (default) FIFO ordering in a PriorityQueue.
- int sequence_number_;
- // Priority of task when pushed.
- int priority_;
- }; // class PrioritizedTask
-
- std::priority_queue<PrioritizedTask> queue_;
- // Default sequence number used when push'ing (monotonically decreasing).
- int next_sequence_number_;
- DISALLOW_EVIL_CONSTRUCTORS(PrioritizedTaskQueue);
- };
-
- // Implementation of a TaskQueue as a null terminated list, with end pointers.
- class TaskQueue {
- public:
- TaskQueue() : first_(NULL), last_(NULL) {}
- void Push(Task* task);
- Task* Pop(); // Extract the next Task from the queue, and return it.
- bool Empty() const { return !first_; }
- private:
- Task* first_;
- Task* last_;
+ PendingTask(Task* task, bool nestable)
+ : task(task), sequence_num(0), nestable(nestable) {
+ }
+
+ // Used to support sorting.
+ bool operator<(const PendingTask& other) const;
};
- // Implementation of a Task queue that automatically switches into a priority
- // queue if it observes any non-zero priorities in tasks.
- class OptionallyPrioritizedTaskQueue {
- public:
- OptionallyPrioritizedTaskQueue() : use_priority_queue_(false) {}
- void Push(Task* task);
- Task* Pop(); // Extract next Task from queue, and return it.
- bool Empty();
- bool use_priority_queue() const { return use_priority_queue_; }
-
- private:
- bool use_priority_queue_;
- PrioritizedTaskQueue prioritized_queue_;
- TaskQueue queue_;
- DISALLOW_EVIL_CONSTRUCTORS(OptionallyPrioritizedTaskQueue);
- };
+ typedef std::queue<PendingTask> TaskQueue;
+ typedef std::priority_queue<PendingTask> DelayedTaskQueue;
#if defined(OS_WIN)
base::MessagePumpWin* pump_win() {
@@ -356,10 +312,9 @@ class MessageLoop : public base::MessagePump::Delegate {
// Runs the specified task and deletes it.
void RunTask(Task* task);
- // Make state adjustments just before and after running tasks so that we can
- // continue to work if a native message loop is employed during a task.
- void BeforeTaskRunSetup();
- void AfterTaskRunRestore();
+ // Calls RunTask or queues the pending_task on the deferred task list if it
+ // cannot be run right now. Returns true if the task was run.
+ bool DeferOrRunPendingTask(const PendingTask& pending_task);
// Load tasks from the incoming_queue_ into work_queue_ if the latter is
// empty. The former requires a lock to access, while the latter is directly
@@ -371,20 +326,8 @@ class MessageLoop : public base::MessagePump::Delegate {
void DeletePendingTasks();
// Post a task to our incomming queue.
- void PostTaskInternal(Task* task);
-
- // Called by the TimerManager when its next timer changes.
- void DidChangeNextTimerExpiry();
-
- // Entry point for TimerManager to request the Run() of a task. If we
- // created the task during an PostTask(FROM_HERE, ), then we will also
- // perform destructions, and we'll have the option of queueing the task. If
- // we didn't create the timer, then we will Run it immediately.
- bool RunTimerTask(Timer* timer);
-
- // Since some Timer's are owned by MessageLoop, the TimerManager (when it is
- // being destructed) passses us the timers to discard (without doing a Run()).
- void DiscardTimer(Timer* timer);
+ void PostTask_Helper(const tracked_objects::Location& from_here, Task* task,
+ int delay_ms, bool nestable);
// base::MessagePump::Delegate methods:
virtual bool DoWork();
@@ -406,16 +349,17 @@ class MessageLoop : public base::MessagePump::Delegate {
Type type_;
- base::TimerManager timer_manager_;
+ // A list of tasks that need to be processed by this instance. Note that
+ // this queue is only accessed (push/pop) by our current thread.
+ TaskQueue work_queue_;
+
+ // Contains delayed tasks, sorted by their 'delayed_run_time' property.
+ DelayedTaskQueue delayed_work_queue_;
- // A list of tasks that need to be processed by this instance. Note that this
- // queue is only accessed (push/pop) by our current thread.
- // As an optimization, when we don't need to use the prioritization of
- // work_queue_, we use a null terminated list (TaskQueue) as our
- // implementation of the queue. This saves on memory (list uses pointers
- // internal to Task) and probably runs faster than the priority queue when
- // there was no real prioritization.
- OptionallyPrioritizedTaskQueue work_queue_;
+ // A queue of non-nestable tasks that we had to defer because when it came
+ // time to execute them we were in a nested message loop. They will execute
+ // once we're out of nested message loops.
+ TaskQueue deferred_non_nestable_work_queue_;
scoped_refptr<base::MessagePump> pump_;
@@ -438,13 +382,11 @@ class MessageLoop : public base::MessagePump::Delegate {
// Protect access to incoming_queue_.
Lock incoming_queue_lock_;
- // A null terminated list of non-nestable tasks that we had to delay because
- // when it came time to execute them we were in a nested message loop. They
- // will execute once we're out of nested message loops.
- TaskQueue delayed_non_nestable_queue_;
-
RunState* state_;
+ // The next sequence number to use for delayed tasks.
+ int next_sequence_num_;
+
DISALLOW_COPY_AND_ASSIGN(MessageLoop);
};