| Mathieu Chartier | 93c21ba | 2018-12-10 13:08:30 -0800 | [diff] [blame] | 1 |  | 
| Elliott Hughes | 1aa246d | 2012-12-13 09:29:36 -0800 | [diff] [blame] | 2 | /* | 
 | 3 |  * Copyright (C) 2012 The Android Open Source Project | 
 | 4 |  * | 
 | 5 |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
 | 6 |  * you may not use this file except in compliance with the License. | 
 | 7 |  * You may obtain a copy of the License at | 
 | 8 |  * | 
 | 9 |  *      http://www.apache.org/licenses/LICENSE-2.0 | 
 | 10 |  * | 
 | 11 |  * Unless required by applicable law or agreed to in writing, software | 
 | 12 |  * distributed under the License is distributed on an "AS IS" BASIS, | 
 | 13 |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
 | 14 |  * See the License for the specific language governing permissions and | 
 | 15 |  * limitations under the License. | 
 | 16 |  */ | 
 | 17 |  | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 18 | #include "thread_pool.h" | 
 | 19 |  | 
| Andreas Gampe | d490129 | 2017-05-30 18:41:34 -0700 | [diff] [blame] | 20 | #include <sys/mman.h> | 
| Andreas Gampe | 9e927f5 | 2016-02-29 20:49:38 -0800 | [diff] [blame] | 21 | #include <sys/resource.h> | 
| Andreas Gampe | 8cf9cb3 | 2017-07-19 09:28:38 -0700 | [diff] [blame] | 22 | #include <sys/time.h> | 
 | 23 |  | 
 | 24 | #include <pthread.h> | 
| Andreas Gampe | 9e927f5 | 2016-02-29 20:49:38 -0800 | [diff] [blame] | 25 |  | 
| Andreas Gampe | 5794381 | 2017-12-06 21:39:13 -0800 | [diff] [blame] | 26 | #include <android-base/logging.h> | 
 | 27 | #include <android-base/stringprintf.h> | 
| Andreas Gampe | 46ee31b | 2016-12-14 10:11:49 -0800 | [diff] [blame] | 28 |  | 
| Vladimir Marko | 0b6e283 | 2015-09-24 10:41:33 +0100 | [diff] [blame] | 29 | #include "base/bit_utils.h" | 
| Elliott Hughes | 1aa246d | 2012-12-13 09:29:36 -0800 | [diff] [blame] | 30 | #include "base/casts.h" | 
 | 31 | #include "base/stl_util.h" | 
| Vladimir Marko | 80afd02 | 2015-05-19 18:08:00 +0100 | [diff] [blame] | 32 | #include "base/time_utils.h" | 
| David Sehr | c431b9d | 2018-03-02 12:01:51 -0800 | [diff] [blame] | 33 | #include "base/utils.h" | 
| Elliott Hughes | 1aa246d | 2012-12-13 09:29:36 -0800 | [diff] [blame] | 34 | #include "runtime.h" | 
| Andreas Gampe | b486a98 | 2017-06-01 13:45:54 -0700 | [diff] [blame] | 35 | #include "thread-current-inl.h" | 
| Elliott Hughes | 1aa246d | 2012-12-13 09:29:36 -0800 | [diff] [blame] | 36 |  | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 37 | namespace art { | 
 | 38 |  | 
| Andreas Gampe | 46ee31b | 2016-12-14 10:11:49 -0800 | [diff] [blame] | 39 | using android::base::StringPrintf; | 
 | 40 |  | 
| Mathieu Chartier | 720ef76 | 2013-08-17 14:46:54 -0700 | [diff] [blame] | 41 | static constexpr bool kMeasureWaitTime = false; | 
| Mathieu Chartier | 94c32c5 | 2013-08-09 11:14:04 -0700 | [diff] [blame] | 42 |  | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 43 | ThreadPoolWorker::ThreadPoolWorker(ThreadPool* thread_pool, const std::string& name, | 
 | 44 |                                    size_t stack_size) | 
 | 45 |     : thread_pool_(thread_pool), | 
| Mathieu Chartier | bcd5e9d | 2013-11-13 14:33:28 -0800 | [diff] [blame] | 46 |       name_(name) { | 
| Vladimir Marko | 0b6e283 | 2015-09-24 10:41:33 +0100 | [diff] [blame] | 47 |   // Add an inaccessible page to catch stack overflow. | 
 | 48 |   stack_size += kPageSize; | 
| Mathieu Chartier | bcd5e9d | 2013-11-13 14:33:28 -0800 | [diff] [blame] | 49 |   std::string error_msg; | 
| Vladimir Marko | c34bebf | 2018-08-16 16:12:49 +0100 | [diff] [blame] | 50 |   stack_ = MemMap::MapAnonymous(name.c_str(), | 
| Vladimir Marko | c34bebf | 2018-08-16 16:12:49 +0100 | [diff] [blame] | 51 |                                 stack_size, | 
 | 52 |                                 PROT_READ | PROT_WRITE, | 
| Vladimir Marko | 1130659 | 2018-10-26 14:22:59 +0100 | [diff] [blame] | 53 |                                 /*low_4gb=*/ false, | 
| Vladimir Marko | c34bebf | 2018-08-16 16:12:49 +0100 | [diff] [blame] | 54 |                                 &error_msg); | 
 | 55 |   CHECK(stack_.IsValid()) << error_msg; | 
 | 56 |   CHECK_ALIGNED(stack_.Begin(), kPageSize); | 
| Mathieu Chartier | 8d8de0c | 2017-10-04 09:35:30 -0700 | [diff] [blame] | 57 |   CheckedCall(mprotect, | 
 | 58 |               "mprotect bottom page of thread pool worker stack", | 
| Vladimir Marko | c34bebf | 2018-08-16 16:12:49 +0100 | [diff] [blame] | 59 |               stack_.Begin(), | 
| Mathieu Chartier | 8d8de0c | 2017-10-04 09:35:30 -0700 | [diff] [blame] | 60 |               kPageSize, | 
 | 61 |               PROT_NONE); | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 62 |   const char* reason = "new thread pool worker thread"; | 
| Brian Carlstrom | bcc2926 | 2012-11-02 11:36:03 -0700 | [diff] [blame] | 63 |   pthread_attr_t attr; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 64 |   CHECK_PTHREAD_CALL(pthread_attr_init, (&attr), reason); | 
| Vladimir Marko | c34bebf | 2018-08-16 16:12:49 +0100 | [diff] [blame] | 65 |   CHECK_PTHREAD_CALL(pthread_attr_setstack, (&attr, stack_.Begin(), stack_.Size()), reason); | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 66 |   CHECK_PTHREAD_CALL(pthread_create, (&pthread_, &attr, &Callback, this), reason); | 
 | 67 |   CHECK_PTHREAD_CALL(pthread_attr_destroy, (&attr), reason); | 
 | 68 | } | 
 | 69 |  | 
 | 70 | ThreadPoolWorker::~ThreadPoolWorker() { | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 71 |   CHECK_PTHREAD_CALL(pthread_join, (pthread_, nullptr), "thread pool worker shutdown"); | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 72 | } | 
 | 73 |  | 
| Andreas Gampe | 9e927f5 | 2016-02-29 20:49:38 -0800 | [diff] [blame] | 74 | void ThreadPoolWorker::SetPthreadPriority(int priority) { | 
 | 75 |   CHECK_GE(priority, PRIO_MIN); | 
 | 76 |   CHECK_LE(priority, PRIO_MAX); | 
| Bilyan Borisov | bb661c0 | 2016-04-04 16:27:32 +0100 | [diff] [blame] | 77 | #if defined(ART_TARGET_ANDROID) | 
| Andreas Gampe | 9e927f5 | 2016-02-29 20:49:38 -0800 | [diff] [blame] | 78 |   int result = setpriority(PRIO_PROCESS, pthread_gettid_np(pthread_), priority); | 
 | 79 |   if (result != 0) { | 
 | 80 |     PLOG(ERROR) << "Failed to setpriority to :" << priority; | 
 | 81 |   } | 
 | 82 | #else | 
 | 83 |   UNUSED(priority); | 
 | 84 | #endif | 
 | 85 | } | 
 | 86 |  | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 87 | void ThreadPoolWorker::Run() { | 
 | 88 |   Thread* self = Thread::Current(); | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 89 |   Task* task = nullptr; | 
| Mathieu Chartier | 93c21ba | 2018-12-10 13:08:30 -0800 | [diff] [blame] | 90 |   thread_pool_->creation_barier_.Pass(self); | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 91 |   while ((task = thread_pool_->GetTask(self)) != nullptr) { | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 92 |     task->Run(self); | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 93 |     task->Finalize(); | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 94 |   } | 
 | 95 | } | 
 | 96 |  | 
 | 97 | void* ThreadPoolWorker::Callback(void* arg) { | 
 | 98 |   ThreadPoolWorker* worker = reinterpret_cast<ThreadPoolWorker*>(arg); | 
 | 99 |   Runtime* runtime = Runtime::Current(); | 
| Andreas Gampe | b15de0c | 2017-01-24 13:12:19 -0800 | [diff] [blame] | 100 |   CHECK(runtime->AttachCurrentThread(worker->name_.c_str(), | 
 | 101 |                                      true, | 
 | 102 |                                      nullptr, | 
 | 103 |                                      worker->thread_pool_->create_peers_)); | 
| Nicolas Geoffray | 340dafa | 2016-11-18 16:03:10 +0000 | [diff] [blame] | 104 |   worker->thread_ = Thread::Current(); | 
| Alex Light | e9f6103 | 2018-09-24 16:04:51 -0700 | [diff] [blame] | 105 |   // Mark thread pool workers as runtime-threads. | 
 | 106 |   worker->thread_->SetIsRuntimeThread(true); | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 107 |   // Do work until its time to shut down. | 
 | 108 |   worker->Run(); | 
 | 109 |   runtime->DetachCurrentThread(); | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 110 |   return nullptr; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 111 | } | 
 | 112 |  | 
| Brian Carlstrom | 2ce745c | 2013-07-17 17:44:30 -0700 | [diff] [blame] | 113 | void ThreadPool::AddTask(Thread* self, Task* task) { | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 114 |   MutexLock mu(self, task_queue_lock_); | 
 | 115 |   tasks_.push_back(task); | 
 | 116 |   // If we have any waiters, signal one. | 
| Mathieu Chartier | 94c32c5 | 2013-08-09 11:14:04 -0700 | [diff] [blame] | 117 |   if (started_ && waiting_count_ != 0) { | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 118 |     task_queue_condition_.Signal(self); | 
 | 119 |   } | 
 | 120 | } | 
 | 121 |  | 
| Nicolas Geoffray | 629e935 | 2015-11-04 17:22:16 +0000 | [diff] [blame] | 122 | void ThreadPool::RemoveAllTasks(Thread* self) { | 
 | 123 |   MutexLock mu(self, task_queue_lock_); | 
 | 124 |   tasks_.clear(); | 
 | 125 | } | 
 | 126 |  | 
| Mathieu Chartier | eac4d6a | 2018-12-05 12:33:46 -0800 | [diff] [blame] | 127 | ThreadPool::ThreadPool(const char* name, | 
 | 128 |                        size_t num_threads, | 
 | 129 |                        bool create_peers, | 
 | 130 |                        size_t worker_stack_size) | 
| Mathieu Chartier | bcd5e9d | 2013-11-13 14:33:28 -0800 | [diff] [blame] | 131 |   : name_(name), | 
 | 132 |     task_queue_lock_("task queue lock"), | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 133 |     task_queue_condition_("task queue condition", task_queue_lock_), | 
 | 134 |     completion_condition_("task completion condition", task_queue_lock_), | 
 | 135 |     started_(false), | 
 | 136 |     shutting_down_(false), | 
| Mathieu Chartier | 35883cc | 2012-11-13 14:08:12 -0800 | [diff] [blame] | 137 |     waiting_count_(0), | 
| Ian Rogers | d914eb2 | 2013-04-18 16:11:15 -0700 | [diff] [blame] | 138 |     start_time_(0), | 
 | 139 |     total_wait_time_(0), | 
| Nicolas Geoffray | ce9ed36 | 2018-11-29 03:19:28 +0000 | [diff] [blame] | 140 |     creation_barier_(0), | 
| Andreas Gampe | b15de0c | 2017-01-24 13:12:19 -0800 | [diff] [blame] | 141 |     max_active_workers_(num_threads), | 
| Nicolas Geoffray | ce9ed36 | 2018-11-29 03:19:28 +0000 | [diff] [blame] | 142 |     create_peers_(create_peers), | 
 | 143 |     worker_stack_size_(worker_stack_size) { | 
 | 144 |   CreateThreads(); | 
 | 145 | } | 
 | 146 |  | 
 | 147 | void ThreadPool::CreateThreads() { | 
 | 148 |   CHECK(threads_.empty()); | 
 | 149 |   Thread* self = Thread::Current(); | 
 | 150 |   { | 
 | 151 |     MutexLock mu(self, task_queue_lock_); | 
 | 152 |     shutting_down_ = false; | 
 | 153 |     // Add one since the caller of constructor waits on the barrier too. | 
| Mathieu Chartier | 93c21ba | 2018-12-10 13:08:30 -0800 | [diff] [blame] | 154 |     creation_barier_.Init(self, max_active_workers_); | 
| Nicolas Geoffray | ce9ed36 | 2018-11-29 03:19:28 +0000 | [diff] [blame] | 155 |     while (GetThreadCount() < max_active_workers_) { | 
 | 156 |       const std::string worker_name = StringPrintf("%s worker thread %zu", name_.c_str(), | 
 | 157 |                                                    GetThreadCount()); | 
 | 158 |       threads_.push_back( | 
 | 159 |           new ThreadPoolWorker(this, worker_name, worker_stack_size_)); | 
 | 160 |     } | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 161 |   } | 
| Mathieu Chartier | 93c21ba | 2018-12-10 13:08:30 -0800 | [diff] [blame] | 162 | } | 
 | 163 |  | 
 | 164 | void ThreadPool::WaitForWorkersToBeCreated() { | 
 | 165 |   creation_barier_.Increment(Thread::Current(), 0); | 
 | 166 | } | 
 | 167 |  | 
 | 168 | const std::vector<ThreadPoolWorker*>& ThreadPool::GetWorkers() { | 
 | 169 |   // Wait for all the workers to be created before returning them. | 
 | 170 |   WaitForWorkersToBeCreated(); | 
 | 171 |   return threads_; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 172 | } | 
 | 173 |  | 
| Nicolas Geoffray | ce9ed36 | 2018-11-29 03:19:28 +0000 | [diff] [blame] | 174 | void ThreadPool::DeleteThreads() { | 
| Mathieu Chartier | e46cd75 | 2012-10-31 16:56:18 -0700 | [diff] [blame] | 175 |   { | 
 | 176 |     Thread* self = Thread::Current(); | 
 | 177 |     MutexLock mu(self, task_queue_lock_); | 
 | 178 |     // Tell any remaining workers to shut down. | 
 | 179 |     shutting_down_ = true; | 
| Mathieu Chartier | e46cd75 | 2012-10-31 16:56:18 -0700 | [diff] [blame] | 180 |     // Broadcast to everyone waiting. | 
 | 181 |     task_queue_condition_.Broadcast(self); | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 182 |     completion_condition_.Broadcast(self); | 
| Mathieu Chartier | e46cd75 | 2012-10-31 16:56:18 -0700 | [diff] [blame] | 183 |   } | 
| Nicolas Geoffray | ce9ed36 | 2018-11-29 03:19:28 +0000 | [diff] [blame] | 184 |   // Wait for the threads to finish. We expect the user of the pool | 
 | 185 |   // not to run multi-threaded calls to `CreateThreads` and `DeleteThreads`, | 
 | 186 |   // so we don't guard the field here. | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 187 |   STLDeleteElements(&threads_); | 
 | 188 | } | 
 | 189 |  | 
| Nicolas Geoffray | ce9ed36 | 2018-11-29 03:19:28 +0000 | [diff] [blame] | 190 | void ThreadPool::SetMaxActiveWorkers(size_t max_workers) { | 
 | 191 |   MutexLock mu(Thread::Current(), task_queue_lock_); | 
 | 192 |   CHECK_LE(max_workers, GetThreadCount()); | 
 | 193 |   max_active_workers_ = max_workers; | 
 | 194 | } | 
 | 195 |  | 
 | 196 | ThreadPool::~ThreadPool() { | 
 | 197 |   DeleteThreads(); | 
 | 198 | } | 
 | 199 |  | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 200 | void ThreadPool::StartWorkers(Thread* self) { | 
 | 201 |   MutexLock mu(self, task_queue_lock_); | 
 | 202 |   started_ = true; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 203 |   task_queue_condition_.Broadcast(self); | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 204 |   start_time_ = NanoTime(); | 
 | 205 |   total_wait_time_ = 0; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 206 | } | 
 | 207 |  | 
 | 208 | void ThreadPool::StopWorkers(Thread* self) { | 
 | 209 |   MutexLock mu(self, task_queue_lock_); | 
 | 210 |   started_ = false; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 211 | } | 
 | 212 |  | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 213 | Task* ThreadPool::GetTask(Thread* self) { | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 214 |   MutexLock mu(self, task_queue_lock_); | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 215 |   while (!IsShuttingDown()) { | 
| Mathieu Chartier | 2775ee4 | 2013-08-20 17:43:47 -0700 | [diff] [blame] | 216 |     const size_t thread_count = GetThreadCount(); | 
 | 217 |     // Ensure that we don't use more threads than the maximum active workers. | 
 | 218 |     const size_t active_threads = thread_count - waiting_count_; | 
 | 219 |     // <= since self is considered an active worker. | 
 | 220 |     if (active_threads <= max_active_workers_) { | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 221 |       Task* task = TryGetTaskLocked(); | 
 | 222 |       if (task != nullptr) { | 
| Mathieu Chartier | 2775ee4 | 2013-08-20 17:43:47 -0700 | [diff] [blame] | 223 |         return task; | 
 | 224 |       } | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 225 |     } | 
 | 226 |  | 
| Mathieu Chartier | 2775ee4 | 2013-08-20 17:43:47 -0700 | [diff] [blame] | 227 |     ++waiting_count_; | 
| Andreas Gampe | 6f3a70f | 2016-11-16 13:58:05 -0800 | [diff] [blame] | 228 |     if (waiting_count_ == GetThreadCount() && !HasOutstandingTasks()) { | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 229 |       // We may be done, lets broadcast to the completion condition. | 
 | 230 |       completion_condition_.Broadcast(self); | 
 | 231 |     } | 
| Mathieu Chartier | 94c32c5 | 2013-08-09 11:14:04 -0700 | [diff] [blame] | 232 |     const uint64_t wait_start = kMeasureWaitTime ? NanoTime() : 0; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 233 |     task_queue_condition_.Wait(self); | 
| Mathieu Chartier | 94c32c5 | 2013-08-09 11:14:04 -0700 | [diff] [blame] | 234 |     if (kMeasureWaitTime) { | 
 | 235 |       const uint64_t wait_end = NanoTime(); | 
 | 236 |       total_wait_time_ += wait_end - std::max(wait_start, start_time_); | 
 | 237 |     } | 
 | 238 |     --waiting_count_; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 239 |   } | 
 | 240 |  | 
| Mathieu Chartier | 2cebb24 | 2015-04-21 16:50:40 -0700 | [diff] [blame] | 241 |   // We are shutting down, return null to tell the worker thread to stop looping. | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 242 |   return nullptr; | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 243 | } | 
 | 244 |  | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 245 | Task* ThreadPool::TryGetTask(Thread* self) { | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 246 |   MutexLock mu(self, task_queue_lock_); | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 247 |   return TryGetTaskLocked(); | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 248 | } | 
 | 249 |  | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 250 | Task* ThreadPool::TryGetTaskLocked() { | 
| Andreas Gampe | 6f3a70f | 2016-11-16 13:58:05 -0800 | [diff] [blame] | 251 |   if (HasOutstandingTasks()) { | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 252 |     Task* task = tasks_.front(); | 
 | 253 |     tasks_.pop_front(); | 
 | 254 |     return task; | 
 | 255 |   } | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 256 |   return nullptr; | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 257 | } | 
 | 258 |  | 
| Ian Rogers | 1d54e73 | 2013-05-02 21:10:01 -0700 | [diff] [blame] | 259 | void ThreadPool::Wait(Thread* self, bool do_work, bool may_hold_locks) { | 
 | 260 |   if (do_work) { | 
| Andreas Gampe | b15de0c | 2017-01-24 13:12:19 -0800 | [diff] [blame] | 261 |     CHECK(!create_peers_); | 
| Ian Rogers | 6a3c1fc | 2014-10-31 00:33:20 -0700 | [diff] [blame] | 262 |     Task* task = nullptr; | 
 | 263 |     while ((task = TryGetTask(self)) != nullptr) { | 
| Ian Rogers | 1d54e73 | 2013-05-02 21:10:01 -0700 | [diff] [blame] | 264 |       task->Run(self); | 
 | 265 |       task->Finalize(); | 
 | 266 |     } | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 267 |   } | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 268 |   // Wait until each thread is waiting and the task list is empty. | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 269 |   MutexLock mu(self, task_queue_lock_); | 
| Andreas Gampe | 6f3a70f | 2016-11-16 13:58:05 -0800 | [diff] [blame] | 270 |   while (!shutting_down_ && (waiting_count_ != GetThreadCount() || HasOutstandingTasks())) { | 
| Ian Rogers | 1d54e73 | 2013-05-02 21:10:01 -0700 | [diff] [blame] | 271 |     if (!may_hold_locks) { | 
 | 272 |       completion_condition_.Wait(self); | 
 | 273 |     } else { | 
 | 274 |       completion_condition_.WaitHoldingLocks(self); | 
 | 275 |     } | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 276 |   } | 
 | 277 | } | 
 | 278 |  | 
| Brian Carlstrom | 2ce745c | 2013-07-17 17:44:30 -0700 | [diff] [blame] | 279 | size_t ThreadPool::GetTaskCount(Thread* self) { | 
| Mathieu Chartier | 02b6a78 | 2012-10-26 13:51:26 -0700 | [diff] [blame] | 280 |   MutexLock mu(self, task_queue_lock_); | 
 | 281 |   return tasks_.size(); | 
 | 282 | } | 
 | 283 |  | 
| Andreas Gampe | 9e927f5 | 2016-02-29 20:49:38 -0800 | [diff] [blame] | 284 | void ThreadPool::SetPthreadPriority(int priority) { | 
 | 285 |   for (ThreadPoolWorker* worker : threads_) { | 
 | 286 |     worker->SetPthreadPriority(priority); | 
 | 287 |   } | 
 | 288 | } | 
 | 289 |  | 
| Mathieu Chartier | 0e4627e | 2012-10-23 16:13:36 -0700 | [diff] [blame] | 290 | }  // namespace art |