Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2016 The Chromium Authors. All rights reserved. | 1 // Copyright 2016 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 "base/task_scheduler/scheduler_worker.h" | 5 #include "base/task_scheduler/scheduler_worker.h" |
| 6 | 6 |
| 7 #include <stddef.h> | 7 #include <stddef.h> |
| 8 | 8 |
| 9 #include <utility> | 9 #include <utility> |
| 10 | 10 |
| 11 #include "base/logging.h" | 11 #include "base/logging.h" |
| 12 #include "base/task_scheduler/task_tracker.h" | 12 #include "base/task_scheduler/task_tracker.h" |
| 13 #include "build/build_config.h" | |
| 13 | 14 |
| 14 namespace base { | 15 namespace base { |
| 15 namespace internal { | 16 namespace internal { |
| 16 | 17 |
| 17 class SchedulerWorker::Thread : public PlatformThread::Delegate { | 18 class SchedulerWorker::Thread : public PlatformThread::Delegate { |
| 18 public: | 19 public: |
| 19 ~Thread() override = default; | 20 ~Thread() override = default; |
| 20 | 21 |
| 21 static std::unique_ptr<Thread> Create(SchedulerWorker* outer) { | 22 static std::unique_ptr<Thread> Create(SchedulerWorker* outer) { |
| 22 std::unique_ptr<Thread> thread(new Thread(outer)); | 23 std::unique_ptr<Thread> thread(new Thread(outer)); |
| 23 thread->Initialize(); | 24 thread->Initialize(); |
| 24 if (thread->thread_handle_.is_null()) | 25 if (thread->thread_handle_.is_null()) |
| 25 return nullptr; | 26 return nullptr; |
| 26 return thread; | 27 return thread; |
| 27 } | 28 } |
| 28 | 29 |
| 29 // PlatformThread::Delegate. | 30 // PlatformThread::Delegate. |
| 30 void ThreadMain() override { | 31 void ThreadMain() override { |
| 31 // Set if this thread was detached. | 32 // Set if this thread was detached. |
| 32 std::unique_ptr<Thread> detached_thread; | 33 std::unique_ptr<Thread> detached_thread; |
| 33 | 34 |
| 34 outer_->delegate_->OnMainEntry(outer_); | 35 outer_->delegate_->OnMainEntry(outer_); |
| 35 | 36 |
| 36 // A SchedulerWorker starts out waiting for work. | 37 // A SchedulerWorker starts out waiting for work. |
| 37 WaitForWork(); | 38 WaitForWork(); |
| 38 | 39 |
| 39 while (!outer_->task_tracker_->IsShutdownComplete() && | 40 while (!outer_->task_tracker_->IsShutdownComplete() && |
| 40 !outer_->ShouldExitForTesting()) { | 41 !outer_->ShouldExitForTesting()) { |
| 41 DCHECK(outer_); | 42 DCHECK(outer_); |
| 43 | |
| 44 SetThreadPriority(GetDesiredThreadPriority()); | |
| 45 | |
| 42 // Get the sequence containing the next task to execute. | 46 // Get the sequence containing the next task to execute. |
| 43 scoped_refptr<Sequence> sequence = outer_->delegate_->GetWork(outer_); | 47 scoped_refptr<Sequence> sequence = outer_->delegate_->GetWork(outer_); |
| 44 if (!sequence) { | 48 if (!sequence) { |
| 45 if (outer_->delegate_->CanDetach(outer_)) { | 49 if (outer_->delegate_->CanDetach(outer_)) { |
| 46 detached_thread = outer_->Detach(); | 50 detached_thread = outer_->Detach(); |
| 47 if (detached_thread) { | 51 if (detached_thread) { |
| 48 DCHECK_EQ(detached_thread.get(), this); | 52 DCHECK_EQ(detached_thread.get(), this); |
| 49 PlatformThread::Detach(thread_handle_); | 53 PlatformThread::Detach(thread_handle_); |
| 50 outer_ = nullptr; | 54 outer_ = nullptr; |
| 51 break; | 55 break; |
| (...skipping 32 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 84 } | 88 } |
| 85 | 89 |
| 86 void Join() { PlatformThread::Join(thread_handle_); } | 90 void Join() { PlatformThread::Join(thread_handle_); } |
| 87 | 91 |
| 88 void WakeUp() { wake_up_event_.Signal(); } | 92 void WakeUp() { wake_up_event_.Signal(); } |
| 89 | 93 |
| 90 bool IsWakeUpPending() { return wake_up_event_.IsSignaled(); } | 94 bool IsWakeUpPending() { return wake_up_event_.IsSignaled(); } |
| 91 | 95 |
| 92 private: | 96 private: |
| 93 Thread(SchedulerWorker* outer) | 97 Thread(SchedulerWorker* outer) |
| 94 : outer_(outer), | 98 : outer_(outer), |
| 95 wake_up_event_(WaitableEvent::ResetPolicy::MANUAL, | 99 wake_up_event_(WaitableEvent::ResetPolicy::MANUAL, |
| 96 WaitableEvent::InitialState::NOT_SIGNALED) { | 100 WaitableEvent::InitialState::NOT_SIGNALED), |
| 101 current_thread_priority_(GetDesiredThreadPriority()) { | |
| 97 DCHECK(outer_); | 102 DCHECK(outer_); |
| 98 } | 103 } |
| 99 | 104 |
| 100 void Initialize() { | 105 void Initialize() { |
| 101 constexpr size_t kDefaultStackSize = 0; | 106 constexpr size_t kDefaultStackSize = 0; |
| 102 PlatformThread::CreateWithPriority(kDefaultStackSize, this, | 107 PlatformThread::CreateWithPriority(kDefaultStackSize, this, &thread_handle_, |
| 103 &thread_handle_, | 108 current_thread_priority_); |
| 104 outer_->thread_priority_); | |
| 105 } | 109 } |
| 106 | 110 |
| 107 void WaitForWork() { | 111 void WaitForWork() { |
| 108 DCHECK(outer_); | 112 DCHECK(outer_); |
| 109 const TimeDelta sleep_time = outer_->delegate_->GetSleepTimeout(); | 113 const TimeDelta sleep_time = outer_->delegate_->GetSleepTimeout(); |
| 110 if (sleep_time.is_max()) { | 114 if (sleep_time.is_max()) { |
| 111 // Calling TimedWait with TimeDelta::Max is not recommended per | 115 // Calling TimedWait with TimeDelta::Max is not recommended per |
| 112 // http://crbug.com/465948. | 116 // http://crbug.com/465948. |
| 113 wake_up_event_.Wait(); | 117 wake_up_event_.Wait(); |
| 114 } else { | 118 } else { |
| 115 wake_up_event_.TimedWait(sleep_time); | 119 wake_up_event_.TimedWait(sleep_time); |
| 116 } | 120 } |
| 117 wake_up_event_.Reset(); | 121 wake_up_event_.Reset(); |
| 118 } | 122 } |
| 119 | 123 |
| 124 // Returns the desired thread priority based on the worker priority and the | |
| 125 // current shutdown state. | |
| 126 ThreadPriority GetDesiredThreadPriority() { | |
| 127 DCHECK(outer_); | |
| 128 | |
| 129 if (outer_->task_tracker_->HasShutdownStarted() && | |
| 130 static_cast<int>(outer_->thread_priority_) < | |
| 131 static_cast<int>(ThreadPriority::NORMAL)) { | |
| 132 return ThreadPriority::NORMAL; | |
| 133 } | |
| 134 return outer_->thread_priority_; | |
| 135 } | |
| 136 | |
| 137 void SetThreadPriority(ThreadPriority desired_thread_priority) { | |
| 138 if (desired_thread_priority == current_thread_priority_) | |
| 139 return; | |
| 140 | |
| 141 #if defined(OS_LINUX) | |
| 142 if (static_cast<int>(desired_thread_priority) > | |
| 143 static_cast<int>(current_thread_priority_)) { | |
|
gab
2016/07/21 21:29:51
If we can't bring it up, we shouldn't bring it dow
fdoray
2016/07/22 12:56:32
Done. I disabled this method completely on Linux.
gab
2016/07/22 13:33:26
An alternate option is to detach the BACKGROUND th
| |
| 144 // Increasing the thread priority requires the CAP_SYS_NICE capability on | |
| 145 // Linux. | |
| 146 return; | |
| 147 } | |
| 148 #endif // defined(OS_LINUX) | |
| 149 | |
| 150 PlatformThread::SetCurrentThreadPriority(desired_thread_priority); | |
| 151 current_thread_priority_ = desired_thread_priority; | |
| 152 } | |
| 153 | |
| 120 PlatformThreadHandle thread_handle_; | 154 PlatformThreadHandle thread_handle_; |
| 121 | 155 |
| 122 SchedulerWorker* outer_; | 156 SchedulerWorker* outer_; |
| 123 | 157 |
| 124 // Event signaled to wake up this thread. | 158 // Event signaled to wake up this thread. |
| 125 WaitableEvent wake_up_event_; | 159 WaitableEvent wake_up_event_; |
| 126 | 160 |
| 161 // Current priority of this thread. May be different from | |
| 162 // |outer_->thread_priority_| during shutdown. | |
| 163 ThreadPriority current_thread_priority_; | |
| 164 | |
| 127 DISALLOW_COPY_AND_ASSIGN(Thread); | 165 DISALLOW_COPY_AND_ASSIGN(Thread); |
| 128 }; | 166 }; |
| 129 | 167 |
| 130 std::unique_ptr<SchedulerWorker> SchedulerWorker::Create( | 168 std::unique_ptr<SchedulerWorker> SchedulerWorker::Create( |
| 131 ThreadPriority thread_priority, | 169 ThreadPriority thread_priority, |
| 132 std::unique_ptr<Delegate> delegate, | 170 std::unique_ptr<Delegate> delegate, |
| 133 TaskTracker* task_tracker, | 171 TaskTracker* task_tracker, |
| 134 InitialState initial_state) { | 172 InitialState initial_state) { |
| 135 std::unique_ptr<SchedulerWorker> worker( | 173 std::unique_ptr<SchedulerWorker> worker( |
| 136 new SchedulerWorker(thread_priority, std::move(delegate), task_tracker)); | 174 new SchedulerWorker(thread_priority, std::move(delegate), task_tracker)); |
| (...skipping 76 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 213 CreateThread(); | 251 CreateThread(); |
| 214 } | 252 } |
| 215 | 253 |
| 216 bool SchedulerWorker::ShouldExitForTesting() const { | 254 bool SchedulerWorker::ShouldExitForTesting() const { |
| 217 AutoSchedulerLock auto_lock(should_exit_for_testing_lock_); | 255 AutoSchedulerLock auto_lock(should_exit_for_testing_lock_); |
| 218 return should_exit_for_testing_; | 256 return should_exit_for_testing_; |
| 219 } | 257 } |
| 220 | 258 |
| 221 } // namespace internal | 259 } // namespace internal |
| 222 } // namespace base | 260 } // namespace base |
| OLD | NEW |