Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2014 The Chromium Authors. All rights reserved. | 1 // Copyright 2014 The Chromium Authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "config.h" | 5 #include "config.h" |
| 6 #include "platform/scheduler/Scheduler.h" | 6 #include "platform/scheduler/Scheduler.h" |
| 7 | 7 |
| 8 #include "platform/PlatformThreadData.h" | |
| 8 #include "platform/Task.h" | 9 #include "platform/Task.h" |
| 10 #include "platform/ThreadTimers.h" | |
| 9 #include "platform/TraceEvent.h" | 11 #include "platform/TraceEvent.h" |
| 10 #include "platform/TraceLocation.h" | |
| 11 #include "public/platform/Platform.h" | 12 #include "public/platform/Platform.h" |
| 12 #include "public/platform/WebThread.h" | 13 #include "wtf/ThreadingPrimitives.h" |
| 13 | 14 |
| 14 namespace blink { | 15 namespace blink { |
| 15 | 16 |
| 16 namespace { | 17 namespace { |
| 17 | 18 |
| 18 class MainThreadTaskAdapter : public blink::WebThread::Task { | |
| 19 public: | |
| 20 explicit MainThreadTaskAdapter(const TraceLocation& location, const Schedule r::Task& task) | |
| 21 : m_location(location) | |
| 22 , m_task(task) | |
| 23 { | |
| 24 } | |
| 25 | |
| 26 // WebThread::Task implementation. | |
| 27 virtual void run() OVERRIDE | |
| 28 { | |
| 29 TRACE_EVENT2("blink", "MainThreadTaskAdapter::run", | |
| 30 "src_file", m_location.fileName(), | |
| 31 "src_func", m_location.functionName()); | |
| 32 m_task(); | |
| 33 } | |
| 34 | |
| 35 private: | |
| 36 const TraceLocation m_location; | |
| 37 Scheduler::Task m_task; | |
| 38 }; | |
| 39 | |
| 40 class MainThreadIdleTaskAdapter : public blink::WebThread::Task { | 19 class MainThreadIdleTaskAdapter : public blink::WebThread::Task { |
| 41 public: | 20 public: |
| 42 MainThreadIdleTaskAdapter(const Scheduler::IdleTask& idleTask, double allott edTimeMs) | 21 MainThreadIdleTaskAdapter(const Scheduler::IdleTask& idleTask, double allott edTimeMs) |
| 43 : m_idleTask(idleTask) | 22 : m_idleTask(idleTask) |
| 44 , m_allottedTimeMs(allottedTimeMs) | 23 , m_allottedTimeMs(allottedTimeMs) |
| 45 { | 24 { |
| 46 } | 25 } |
| 47 | 26 |
| 48 // WebThread::Task implementation. | 27 // WebThread::Task implementation. |
| 49 virtual void run() OVERRIDE | 28 virtual void run() OVERRIDE |
| 50 { | 29 { |
| 51 TRACE_EVENT1("blink", "MainThreadIdleTaskAdapter::run", "allottedTime", m_allottedTimeMs); | 30 TRACE_EVENT1("blink", "MainThreadIdleTaskAdapter::run", "allottedTime", m_allottedTimeMs); |
| 52 m_idleTask(m_allottedTimeMs); | 31 m_idleTask(m_allottedTimeMs); |
| 53 } | 32 } |
| 54 | 33 |
| 55 private: | 34 private: |
| 56 Scheduler::IdleTask m_idleTask; | 35 Scheduler::IdleTask m_idleTask; |
| 57 double m_allottedTimeMs; | 36 double m_allottedTimeMs; |
| 58 }; | 37 }; |
| 59 | 38 |
| 60 } | 39 } // namespace |
| 40 | |
| 41 class Scheduler::MainThreadPendingTaskRunner : public blink::WebThread::Task { | |
| 42 public: | |
| 43 MainThreadPendingTaskRunner(volatile int* taskRunnerCount) | |
| 44 : m_taskRunnerCount(taskRunnerCount) | |
| 45 { | |
| 46 atomicIncrement(m_taskRunnerCount); | |
| 47 } | |
| 48 | |
| 49 ~MainThreadPendingTaskRunner() | |
| 50 { | |
| 51 atomicDecrement(m_taskRunnerCount); | |
| 52 } | |
| 53 | |
| 54 // WebThread::Task implementation. | |
| 55 virtual void run() OVERRIDE | |
| 56 { | |
| 57 Scheduler::shared()->runPendingTasks(); | |
|
Sami
2014/08/07 15:51:26
It would be slightly neater to pass in the schedul
alexclarke
2014/08/07 16:01:29
Done.
| |
| 58 } | |
| 59 | |
| 60 volatile int* m_taskRunnerCount; // Not owned. | |
| 61 }; | |
| 61 | 62 |
| 62 Scheduler* Scheduler::s_sharedScheduler = nullptr; | 63 Scheduler* Scheduler::s_sharedScheduler = nullptr; |
| 63 | 64 |
| 64 void Scheduler::initializeOnMainThread() | 65 void Scheduler::initializeOnMainThread() |
| 65 { | 66 { |
| 66 s_sharedScheduler = new Scheduler(); | 67 s_sharedScheduler = new Scheduler(); |
| 67 } | 68 } |
| 68 | 69 |
| 69 void Scheduler::shutdown() | 70 void Scheduler::shutdown() |
| 70 { | 71 { |
| 71 delete s_sharedScheduler; | 72 delete s_sharedScheduler; |
| 72 s_sharedScheduler = nullptr; | 73 s_sharedScheduler = nullptr; |
| 73 } | 74 } |
| 74 | 75 |
| 75 Scheduler* Scheduler::shared() | 76 Scheduler* Scheduler::shared() |
| 76 { | 77 { |
| 77 return s_sharedScheduler; | 78 return s_sharedScheduler; |
| 78 } | 79 } |
| 79 | 80 |
| 80 Scheduler::Scheduler() | 81 Scheduler::Scheduler() |
| 81 : m_mainThread(blink::Platform::current()->currentThread()) | 82 : m_mainThread(blink::Platform::current()->currentThread()) |
| 82 , m_sharedTimerFunction(nullptr) | 83 , m_sharedTimerFunction(nullptr) |
| 84 , m_mainThreadInFlight(0) | |
|
Sami
2014/08/07 15:51:26
Bikeshedding again: m_mainThreadTaskRunnerCount?
alexclarke
2014/08/07 16:01:29
Done.
| |
| 83 { | 85 { |
| 84 } | 86 } |
| 85 | 87 |
| 86 Scheduler::~Scheduler() | 88 Scheduler::~Scheduler() |
| 87 { | 89 { |
| 88 } | 90 while (!m_pendingLowPriorityTasks.isEmpty()) |
| 89 | 91 runPendingTasks(); |
| 90 void Scheduler::scheduleTask(const TraceLocation& location, const Task& task) | |
| 91 { | |
| 92 m_mainThread->postTask(new MainThreadTaskAdapter(location, task)); | |
| 93 } | 92 } |
| 94 | 93 |
| 95 void Scheduler::scheduleIdleTask(const IdleTask& idleTask) | 94 void Scheduler::scheduleIdleTask(const IdleTask& idleTask) |
| 96 { | 95 { |
| 97 // TODO: send a real allottedTime here. | 96 // TODO: send a real allottedTime here. |
| 98 m_mainThread->postTask(new MainThreadIdleTaskAdapter(idleTask, 0)); | 97 m_mainThread->postTask(new MainThreadIdleTaskAdapter(idleTask, 0)); |
| 99 } | 98 } |
| 100 | 99 |
| 101 void Scheduler::postTask(const TraceLocation& location, const Task& task) | 100 void Scheduler::postTask(const TraceLocation& location, const Task& task) |
| 102 { | 101 { |
| 103 scheduleTask(location, task); | 102 m_pendingLowPriorityTasks.append(TracedTask(task, location)); |
| 103 maybePostTaskLoopOnMainThread(); | |
| 104 } | 104 } |
| 105 | 105 |
| 106 void Scheduler::postInputTask(const TraceLocation& location, const Task& task) | 106 void Scheduler::postInputTask(const TraceLocation& location, const Task& task) |
| 107 { | 107 { |
| 108 scheduleTask(location, task); | 108 m_pendingInputTasks.append(TracedTask(task, location)); |
| 109 maybePostTaskLoopOnMainThread(); | |
| 109 } | 110 } |
| 110 | 111 |
| 111 void Scheduler::postCompositorTask(const TraceLocation& location, const Task& ta sk) | 112 void Scheduler::postCompositorTask(const TraceLocation& location, const Task& ta sk) |
| 112 { | 113 { |
| 113 scheduleTask(location, task); | 114 m_pendingCompositorTasks.append(TracedTask(task, location)); |
| 115 maybePostTaskLoopOnMainThread(); | |
| 114 } | 116 } |
| 115 | 117 |
| 116 void Scheduler::postIdleTask(const IdleTask& idleTask) | 118 void Scheduler::postIdleTask(const IdleTask& idleTask) |
| 117 { | 119 { |
| 118 scheduleIdleTask(idleTask); | 120 scheduleIdleTask(idleTask); |
| 119 } | 121 } |
| 120 | 122 |
| 121 void Scheduler::tickSharedTimer() | 123 void Scheduler::tickSharedTimer() |
| 122 { | 124 { |
| 123 TRACE_EVENT0("blink", "Scheduler::tickSharedTimer"); | 125 TRACE_EVENT0("blink", "Scheduler::tickSharedTimer"); |
| 126 | |
| 127 runHighPriorityTasks(); | |
| 124 m_sharedTimerFunction(); | 128 m_sharedTimerFunction(); |
| 129 runHighPriorityTasks(); | |
| 130 } | |
| 131 | |
| 132 void Scheduler::runPendingTasks() | |
| 133 { | |
| 134 TRACE_EVENT0("blink", "Scheduler::runPendingTasks"); | |
| 135 | |
| 136 // Execute high priority tasks first. | |
| 137 runHighPriorityTasks(); | |
| 138 | |
| 139 if (!m_pendingLowPriorityTasks.runFirstTaskIfNotEmpty()) | |
| 140 return; | |
| 141 | |
| 142 // To avoid staving main loop tasks that don't (yet?) go through the schedul er, we only | |
| 143 // execute one low priority task rather than all of them. | |
| 144 if (!m_pendingLowPriorityTasks.isEmpty()) | |
| 145 m_mainThread->postTask(new MainThreadPendingTaskRunner(&m_mainThreadInFl ight)); | |
|
Sami
2014/08/07 15:51:26
Why not just call maybePostTaskLoopOnMainThread he
alexclarke
2014/08/07 16:01:29
Because the reference count is not zero.
| |
| 146 } | |
| 147 | |
| 148 void Scheduler::runHighPriorityTasks() | |
| 149 { | |
| 150 TRACE_EVENT0("blink", "Scheduler::runHighPriorityTasks"); | |
| 151 | |
| 152 for (;;) { | |
| 153 if (m_pendingInputTasks.runFirstTaskIfNotEmpty()) | |
| 154 continue; | |
| 155 | |
| 156 if (!m_pendingCompositorTasks.runFirstTaskIfNotEmpty()) | |
| 157 break; | |
| 158 } | |
| 159 } | |
| 160 | |
| 161 void Scheduler::maybePostTaskLoopOnMainThread() | |
|
Sami
2014/08/07 15:51:26
My bikeshed sense is tingling again :) I like havi
alexclarke
2014/08/07 16:01:29
Done.
| |
| 162 { | |
| 163 if (acquireLoad(&m_mainThreadInFlight)) | |
| 164 return; | |
| 165 | |
| 166 m_mainThread->postTask(new MainThreadPendingTaskRunner(&m_mainThreadInFlight )); | |
|
Sami
2014/08/07 15:51:26
It's worth noting that this can cause several task
alexclarke
2014/08/07 16:01:29
Done.
| |
| 125 } | 167 } |
| 126 | 168 |
| 127 void Scheduler::sharedTimerAdapter() | 169 void Scheduler::sharedTimerAdapter() |
| 128 { | 170 { |
| 129 shared()->tickSharedTimer(); | 171 shared()->tickSharedTimer(); |
| 130 } | 172 } |
| 131 | 173 |
| 132 void Scheduler::setSharedTimerFiredFunction(void (*function)()) | 174 void Scheduler::setSharedTimerFiredFunction(void (*function)()) |
| 133 { | 175 { |
| 134 m_sharedTimerFunction = function; | 176 m_sharedTimerFunction = function; |
| 135 blink::Platform::current()->setSharedTimerFiredFunction(function ? &Schedule r::sharedTimerAdapter : nullptr); | 177 blink::Platform::current()->setSharedTimerFiredFunction(function ? &Schedule r::sharedTimerAdapter : nullptr); |
| 136 } | 178 } |
| 137 | 179 |
| 138 void Scheduler::setSharedTimerFireInterval(double interval) | 180 void Scheduler::setSharedTimerFireInterval(double interval) |
| 139 { | 181 { |
| 140 blink::Platform::current()->setSharedTimerFireInterval(interval); | 182 blink::Platform::current()->setSharedTimerFireInterval(interval); |
| 141 } | 183 } |
| 142 | 184 |
| 143 void Scheduler::stopSharedTimer() | 185 void Scheduler::stopSharedTimer() |
| 144 { | 186 { |
| 145 blink::Platform::current()->stopSharedTimer(); | 187 blink::Platform::current()->stopSharedTimer(); |
| 146 } | 188 } |
| 147 | 189 |
| 148 bool Scheduler::shouldYieldForHighPriorityWork() | 190 bool Scheduler::shouldYieldForHighPriorityWork() |
| 149 { | 191 { |
| 150 return false; | 192 return !m_pendingInputTasks.isEmpty() || !m_pendingCompositorTasks.isEmpty() ; |
| 193 } | |
| 194 | |
| 195 void Scheduler::TracedTask::run() | |
| 196 { | |
| 197 TRACE_EVENT2("blink", "TracedTask::run", | |
| 198 "src_file", m_location.fileName(), | |
| 199 "src_func", m_location.functionName()); | |
| 200 m_task(); | |
| 201 } | |
| 202 | |
| 203 void Scheduler::LockingTracedTaskDeque::append(const TracedTask& value) | |
| 204 { | |
| 205 Locker<Mutex> lock(m_mutex); | |
| 206 m_queue.append(value); | |
| 207 } | |
| 208 | |
| 209 bool Scheduler::LockingTracedTaskDeque::isEmpty() | |
| 210 { | |
| 211 Locker<Mutex> lock(m_mutex); | |
| 212 return m_queue.isEmpty(); | |
| 213 } | |
| 214 | |
| 215 bool Scheduler::LockingTracedTaskDeque::runFirstTaskIfNotEmpty() | |
| 216 { | |
| 217 m_mutex.lock(); | |
| 218 if (m_queue.isEmpty()) { | |
| 219 m_mutex.unlock(); | |
| 220 return false; | |
| 221 } | |
| 222 TracedTask task = m_queue.takeFirst(); | |
| 223 m_mutex.unlock(); | |
| 224 | |
| 225 task.run(); | |
| 226 return true; | |
| 151 } | 227 } |
| 152 | 228 |
| 153 } // namespace blink | 229 } // namespace blink |
| OLD | NEW |