• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /*
2  * libjingle
3  * Copyright 2004--2006, Google Inc.
4  *
5  * Redistribution and use in source and binary forms, with or without
6  * modification, are permitted provided that the following conditions are met:
7  *
8  *  1. Redistributions of source code must retain the above copyright notice,
9  *     this list of conditions and the following disclaimer.
10  *  2. Redistributions in binary form must reproduce the above copyright notice,
11  *     this list of conditions and the following disclaimer in the documentation
12  *     and/or other materials provided with the distribution.
13  *  3. The name of the author may not be used to endorse or promote products
14  *     derived from this software without specific prior written permission.
15  *
16  * THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR IMPLIED
17  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
18  * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO
19  * EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
20  * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
21  * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS;
22  * OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
23  * WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR
24  * OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
25  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
26  */
27 
28 #include <algorithm>
29 
30 #include "talk/base/taskrunner.h"
31 
32 #include "talk/base/common.h"
33 #include "talk/base/scoped_ptr.h"
34 #include "talk/base/task.h"
35 #include "talk/base/logging.h"
36 
37 namespace talk_base {
38 
TaskRunner()39 TaskRunner::TaskRunner()
40   : TaskParent(this),
41     next_timeout_task_(NULL),
42     tasks_running_(false)
43 #ifdef _DEBUG
44     , abort_count_(0),
45     deleting_task_(NULL)
46 #endif
47 {
48 }
49 
~TaskRunner()50 TaskRunner::~TaskRunner() {
51   // this kills and deletes children silently!
52   AbortAllChildren();
53   InternalRunTasks(true);
54 }
55 
StartTask(Task * task)56 void TaskRunner::StartTask(Task * task) {
57   tasks_.push_back(task);
58 
59   // the task we just started could be about to timeout --
60   // make sure our "next timeout task" is correct
61   UpdateTaskTimeout(task, 0);
62 
63   WakeTasks();
64 }
65 
RunTasks()66 void TaskRunner::RunTasks() {
67   InternalRunTasks(false);
68 }
69 
InternalRunTasks(bool in_destructor)70 void TaskRunner::InternalRunTasks(bool in_destructor) {
71   // This shouldn't run while an abort is happening.
72   // If that occurs, then tasks may be deleted in this method,
73   // but pointers to them will still be in the
74   // "ChildSet copy" in TaskParent::AbortAllChildren.
75   // Subsequent use of those task may cause data corruption or crashes.
76   ASSERT(!abort_count_);
77   // Running continues until all tasks are Blocked (ok for a small # of tasks)
78   if (tasks_running_) {
79     return;  // don't reenter
80   }
81 
82   tasks_running_ = true;
83 
84   int64 previous_timeout_time = next_task_timeout();
85 
86   int did_run = true;
87   while (did_run) {
88     did_run = false;
89     // use indexing instead of iterators because tasks_ may grow
90     for (size_t i = 0; i < tasks_.size(); ++i) {
91       while (!tasks_[i]->Blocked()) {
92         tasks_[i]->Step();
93         did_run = true;
94       }
95     }
96   }
97   // Tasks are deleted when running has paused
98   bool need_timeout_recalc = false;
99   for (size_t i = 0; i < tasks_.size(); ++i) {
100     if (tasks_[i]->IsDone()) {
101       Task* task = tasks_[i];
102       if (next_timeout_task_ &&
103           task->unique_id() == next_timeout_task_->unique_id()) {
104         next_timeout_task_ = NULL;
105         need_timeout_recalc = true;
106       }
107 
108 #ifdef _DEBUG
109       deleting_task_ = task;
110 #endif
111       delete task;
112 #ifdef _DEBUG
113       deleting_task_ = NULL;
114 #endif
115       tasks_[i] = NULL;
116     }
117   }
118   // Finally, remove nulls
119   std::vector<Task *>::iterator it;
120   it = std::remove(tasks_.begin(),
121                    tasks_.end(),
122                    reinterpret_cast<Task *>(NULL));
123 
124   tasks_.erase(it, tasks_.end());
125 
126   if (need_timeout_recalc)
127     RecalcNextTimeout(NULL);
128 
129   // Make sure that adjustments are done to account
130   // for any timeout changes (but don't call this
131   // while being destroyed since it calls a pure virtual function).
132   if (!in_destructor)
133     CheckForTimeoutChange(previous_timeout_time);
134 
135   tasks_running_ = false;
136 }
137 
PollTasks()138 void TaskRunner::PollTasks() {
139   // see if our "next potentially timed-out task" has indeed timed out.
140   // If it has, wake it up, then queue up the next task in line
141   // Repeat while we have new timed-out tasks.
142   // TODO: We need to guard against WakeTasks not updating
143   // next_timeout_task_. Maybe also add documentation in the header file once
144   // we understand this code better.
145   Task* old_timeout_task = NULL;
146   while (next_timeout_task_ &&
147       old_timeout_task != next_timeout_task_ &&
148       next_timeout_task_->TimedOut()) {
149     old_timeout_task = next_timeout_task_;
150     next_timeout_task_->Wake();
151     WakeTasks();
152   }
153 }
154 
next_task_timeout() const155 int64 TaskRunner::next_task_timeout() const {
156   if (next_timeout_task_) {
157     return next_timeout_task_->timeout_time();
158   }
159   return 0;
160 }
161 
162 // this function gets called frequently -- when each task changes
163 // state to something other than DONE, ERROR or BLOCKED, it calls
164 // ResetTimeout(), which will call this function to make sure that
165 // the next timeout-able task hasn't changed.  The logic in this function
166 // prevents RecalcNextTimeout() from getting called in most cases,
167 // effectively making the task scheduler O-1 instead of O-N
168 
UpdateTaskTimeout(Task * task,int64 previous_task_timeout_time)169 void TaskRunner::UpdateTaskTimeout(Task* task,
170                                    int64 previous_task_timeout_time) {
171   ASSERT(task != NULL);
172   int64 previous_timeout_time = next_task_timeout();
173   bool task_is_timeout_task = next_timeout_task_ != NULL &&
174       task->unique_id() == next_timeout_task_->unique_id();
175   if (task_is_timeout_task) {
176     previous_timeout_time = previous_task_timeout_time;
177   }
178 
179   // if the relevant task has a timeout, then
180   // check to see if it's closer than the current
181   // "about to timeout" task
182   if (task->timeout_time()) {
183     if (next_timeout_task_ == NULL ||
184         (task->timeout_time() <= next_timeout_task_->timeout_time())) {
185       next_timeout_task_ = task;
186     }
187   } else if (task_is_timeout_task) {
188     // otherwise, if the task doesn't have a timeout,
189     // and it used to be our "about to timeout" task,
190     // walk through all the tasks looking for the real
191     // "about to timeout" task
192     RecalcNextTimeout(task);
193   }
194 
195   // Note when task_running_, then the running routine
196   // (TaskRunner::InternalRunTasks) is responsible for calling
197   // CheckForTimeoutChange.
198   if (!tasks_running_) {
199     CheckForTimeoutChange(previous_timeout_time);
200   }
201 }
202 
RecalcNextTimeout(Task * exclude_task)203 void TaskRunner::RecalcNextTimeout(Task *exclude_task) {
204   // walk through all the tasks looking for the one
205   // which satisfies the following:
206   //   it's not finished already
207   //   we're not excluding it
208   //   it has the closest timeout time
209 
210   int64 next_timeout_time = 0;
211   next_timeout_task_ = NULL;
212 
213   for (size_t i = 0; i < tasks_.size(); ++i) {
214     Task *task = tasks_[i];
215     // if the task isn't complete, and it actually has a timeout time
216     if (!task->IsDone() && (task->timeout_time() > 0))
217       // if it doesn't match our "exclude" task
218       if (exclude_task == NULL ||
219           exclude_task->unique_id() != task->unique_id())
220         // if its timeout time is sooner than our current timeout time
221         if (next_timeout_time == 0 ||
222             task->timeout_time() <= next_timeout_time) {
223           // set this task as our next-to-timeout
224           next_timeout_time = task->timeout_time();
225           next_timeout_task_ = task;
226         }
227   }
228 }
229 
CheckForTimeoutChange(int64 previous_timeout_time)230 void TaskRunner::CheckForTimeoutChange(int64 previous_timeout_time) {
231   int64 next_timeout = next_task_timeout();
232   bool timeout_change = (previous_timeout_time == 0 && next_timeout != 0) ||
233       next_timeout < previous_timeout_time ||
234       (previous_timeout_time <= CurrentTime() &&
235        previous_timeout_time != next_timeout);
236   if (timeout_change) {
237     OnTimeoutChange();
238   }
239 }
240 
241 } // namespace talk_base
242