Vulkan SDK for Android 1.1.1 Mali Developer Center
All Classes Functions Variables Enumerations Enumerator Pages
thread_pool.cpp
1 /* Copyright (c) 2016-2017, ARM Limited and Contributors
2  *
3  * SPDX-License-Identifier: MIT
4  *
5  * Permission is hereby granted, free of charge,
6  * to any person obtaining a copy of this software and associated documentation files (the "Software"),
7  * to deal in the Software without restriction, including without limitation the rights to
8  * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software,
9  * and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
10  *
11  * The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
12  *
13  * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED,
14  * INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
15  * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.
16  * IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
17  * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
18  * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
19  */
20 
21 #include "thread_pool.hpp"
22 #include <utility>
23 
24 using namespace std;
25 
26 namespace MaliSDK
27 {
28 void ThreadPool::setWorkerThreadCount(unsigned workerThreadCount)
29 {
30  workerThreads.clear();
31  for (unsigned i = 0; i < workerThreadCount; i++)
32  workerThreads.emplace_back(new Worker);
33 }
34 
35 void ThreadPool::pushWorkToThread(unsigned threadIndex, std::function<void()> func)
36 {
37  workerThreads[threadIndex]->pushWork(move(func));
38 }
39 
40 void ThreadPool::waitIdle()
41 {
42  for (auto &worker : workerThreads)
43  worker->waitIdle();
44 }
45 
46 ThreadPool::Worker::Worker()
47 {
48  workerThread = thread(&ThreadPool::Worker::threadEntry, this);
49 }
50 
51 ThreadPool::Worker::~Worker()
52 {
53  if (workerThread.joinable())
54  {
55  waitIdle();
56 
57  lock.lock();
58  threadIsAlive = false;
59  cond.notify_one();
60  lock.unlock();
61 
62  workerThread.join();
63  }
64 }
65 
66 void ThreadPool::Worker::pushWork(std::function<void()> func)
67 {
68  lock_guard<mutex> holder{ lock };
69  workQueue.push(move(func));
70  cond.notify_one();
71 }
72 
73 void ThreadPool::Worker::waitIdle()
74 {
75  unique_lock<mutex> holder{ lock };
76  cond.wait(holder, [this] { return workQueue.empty(); });
77 }
78 
79 void ThreadPool::Worker::threadEntry()
80 {
81  for (;;)
82  {
83  function<void()> *pWork = nullptr;
84  {
85  unique_lock<mutex> holder{ lock };
86  cond.wait(holder, [this] { return !workQueue.empty() || !threadIsAlive; });
87  if (!threadIsAlive)
88  break;
89 
90  pWork = &workQueue.front();
91  }
92 
93  (*pWork)();
94 
95  {
96  lock_guard<mutex> holder{ lock };
97  workQueue.pop();
98  cond.notify_one();
99  }
100  }
101 }
102 }