qtconcurrentthreadengine.h 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276
  1. /****************************************************************************
  2. **
  3. ** Copyright (C) 2016 The Qt Company Ltd.
  4. ** Contact: https://www.qt.io/licensing/
  5. **
  6. ** This file is part of the QtCore module of the Qt Toolkit.
  7. **
  8. ** $QT_BEGIN_LICENSE:LGPL$
  9. ** Commercial License Usage
  10. ** Licensees holding valid commercial Qt licenses may use this file in
  11. ** accordance with the commercial license agreement provided with the
  12. ** Software or, alternatively, in accordance with the terms contained in
  13. ** a written agreement between you and The Qt Company. For licensing terms
  14. ** and conditions see https://www.qt.io/terms-conditions. For further
  15. ** information use the contact form at https://www.qt.io/contact-us.
  16. **
  17. ** GNU Lesser General Public License Usage
  18. ** Alternatively, this file may be used under the terms of the GNU Lesser
  19. ** General Public License version 3 as published by the Free Software
  20. ** Foundation and appearing in the file LICENSE.LGPL3 included in the
  21. ** packaging of this file. Please review the following information to
  22. ** ensure the GNU Lesser General Public License version 3 requirements
  23. ** will be met: https://www.gnu.org/licenses/lgpl-3.0.html.
  24. **
  25. ** GNU General Public License Usage
  26. ** Alternatively, this file may be used under the terms of the GNU
  27. ** General Public License version 2.0 or (at your option) the GNU General
  28. ** Public license version 3 or any later version approved by the KDE Free
  29. ** Qt Foundation. The licenses are as published by the Free Software
  30. ** Foundation and appearing in the file LICENSE.GPL2 and LICENSE.GPL3
  31. ** included in the packaging of this file. Please review the following
  32. ** information to ensure the GNU General Public License requirements will
  33. ** be met: https://www.gnu.org/licenses/gpl-2.0.html and
  34. ** https://www.gnu.org/licenses/gpl-3.0.html.
  35. **
  36. ** $QT_END_LICENSE$
  37. **
  38. ****************************************************************************/
  39. #ifndef QTCONCURRENT_THREADENGINE_H
  40. #define QTCONCURRENT_THREADENGINE_H
  41. #include <QtConcurrent/qtconcurrent_global.h>
  42. #ifndef QT_NO_CONCURRENT
  43. #include <QtCore/qthreadpool.h>
  44. #include <QtCore/qfuture.h>
  45. #include <QtCore/qdebug.h>
  46. #include <QtCore/qexception.h>
  47. #include <QtCore/qwaitcondition.h>
  48. #include <QtCore/qatomic.h>
  49. #include <QtCore/qsemaphore.h>
  50. QT_BEGIN_NAMESPACE
  51. #ifndef Q_QDOC
  52. namespace QtConcurrent {
  53. // The ThreadEngineBarrier counts worker threads, and allows one
  54. // thread to wait for all others to finish. Tested for its use in
  55. // QtConcurrent, requires more testing for use as a general class.
  56. class ThreadEngineBarrier
  57. {
  58. private:
  59. // The thread count is maintained as an integer in the count atomic
  60. // variable. The count can be either positive or negative - a negative
  61. // count signals that a thread is waiting on the barrier.
  62. QAtomicInt count;
  63. QSemaphore semaphore;
  64. public:
  65. ThreadEngineBarrier();
  66. void acquire();
  67. int release();
  68. void wait();
  69. int currentCount();
  70. bool releaseUnlessLast();
  71. };
  72. enum ThreadFunctionResult { ThrottleThread, ThreadFinished };
  73. // The ThreadEngine controls the threads used in the computation.
  74. // Can be run in three modes: single threaded, multi-threaded blocking
  75. // and multi-threaded asynchronous.
  76. // The code for the single threaded mode is
  77. class Q_CONCURRENT_EXPORT ThreadEngineBase: public QRunnable
  78. {
  79. public:
  80. // Public API:
  81. ThreadEngineBase();
  82. virtual ~ThreadEngineBase();
  83. void startSingleThreaded();
  84. void startBlocking();
  85. void startThread();
  86. bool isCanceled();
  87. void waitForResume();
  88. bool isProgressReportingEnabled();
  89. void setProgressValue(int progress);
  90. void setProgressRange(int minimum, int maximum);
  91. void acquireBarrierSemaphore();
  92. protected: // The user overrides these:
  93. virtual void start() {}
  94. virtual void finish() {}
  95. virtual ThreadFunctionResult threadFunction() { return ThreadFinished; }
  96. virtual bool shouldStartThread() { return futureInterface ? !futureInterface->isPaused() : true; }
  97. virtual bool shouldThrottleThread() { return futureInterface ? futureInterface->isPaused() : false; }
  98. private:
  99. bool startThreadInternal();
  100. void startThreads();
  101. void threadExit();
  102. bool threadThrottleExit();
  103. void run() Q_DECL_OVERRIDE;
  104. virtual void asynchronousFinish() = 0;
  105. #ifndef QT_NO_EXCEPTIONS
  106. void handleException(const QException &exception);
  107. #endif
  108. protected:
  109. QFutureInterfaceBase *futureInterface;
  110. QThreadPool *threadPool;
  111. ThreadEngineBarrier barrier;
  112. QtPrivate::ExceptionStore exceptionStore;
  113. };
  114. template <typename T>
  115. class ThreadEngine : public virtual ThreadEngineBase
  116. {
  117. public:
  118. typedef T ResultType;
  119. virtual T *result() { return Q_NULLPTR; }
  120. QFutureInterface<T> *futureInterfaceTyped()
  121. {
  122. return static_cast<QFutureInterface<T> *>(futureInterface);
  123. }
  124. // Runs the user algorithm using a single thread.
  125. T *startSingleThreaded()
  126. {
  127. ThreadEngineBase::startSingleThreaded();
  128. return result();
  129. }
  130. // Runs the user algorithm using multiple threads.
  131. // This function blocks until the algorithm is finished,
  132. // and then returns the result.
  133. T *startBlocking()
  134. {
  135. ThreadEngineBase::startBlocking();
  136. return result();
  137. }
  138. // Runs the user algorithm using multiple threads.
  139. // Does not block, returns a future.
  140. QFuture<T> startAsynchronously()
  141. {
  142. futureInterface = new QFutureInterface<T>();
  143. // reportStart() must be called before starting threads, otherwise the
  144. // user algorithm might finish while reportStart() is running, which
  145. // is very bad.
  146. futureInterface->reportStarted();
  147. QFuture<T> future = QFuture<T>(futureInterfaceTyped());
  148. start();
  149. acquireBarrierSemaphore();
  150. threadPool->start(this);
  151. return future;
  152. }
  153. void asynchronousFinish() Q_DECL_OVERRIDE
  154. {
  155. finish();
  156. futureInterfaceTyped()->reportFinished(result());
  157. delete futureInterfaceTyped();
  158. delete this;
  159. }
  160. void reportResult(const T *_result, int index = -1)
  161. {
  162. if (futureInterface)
  163. futureInterfaceTyped()->reportResult(_result, index);
  164. }
  165. void reportResults(const QVector<T> &_result, int index = -1, int count = -1)
  166. {
  167. if (futureInterface)
  168. futureInterfaceTyped()->reportResults(_result, index, count);
  169. }
  170. };
  171. // The ThreadEngineStarter class ecapsulates the return type
  172. // from the thread engine.
  173. // Depending on how the it is used, it will run
  174. // the engine in either blocking mode or asynchronous mode.
  175. template <typename T>
  176. class ThreadEngineStarterBase
  177. {
  178. public:
  179. ThreadEngineStarterBase(ThreadEngine<T> *_threadEngine)
  180. : threadEngine(_threadEngine) { }
  181. inline ThreadEngineStarterBase(const ThreadEngineStarterBase &other)
  182. : threadEngine(other.threadEngine) { }
  183. QFuture<T> startAsynchronously()
  184. {
  185. return threadEngine->startAsynchronously();
  186. }
  187. operator QFuture<T>()
  188. {
  189. return startAsynchronously();
  190. }
  191. protected:
  192. ThreadEngine<T> *threadEngine;
  193. };
  194. // We need to factor out the code that dereferences the T pointer,
  195. // with a specialization where T is void. (code that dereferences a void *
  196. // won't compile)
  197. template <typename T>
  198. class ThreadEngineStarter : public ThreadEngineStarterBase<T>
  199. {
  200. typedef ThreadEngineStarterBase<T> Base;
  201. typedef ThreadEngine<T> TypedThreadEngine;
  202. public:
  203. ThreadEngineStarter(TypedThreadEngine *eng)
  204. : Base(eng) { }
  205. T startBlocking()
  206. {
  207. T t = *this->threadEngine->startBlocking();
  208. delete this->threadEngine;
  209. return t;
  210. }
  211. };
  212. // Full template specialization where T is void.
  213. template <>
  214. class ThreadEngineStarter<void> : public ThreadEngineStarterBase<void>
  215. {
  216. public:
  217. ThreadEngineStarter<void>(ThreadEngine<void> *_threadEngine)
  218. :ThreadEngineStarterBase<void>(_threadEngine) {}
  219. void startBlocking()
  220. {
  221. this->threadEngine->startBlocking();
  222. delete this->threadEngine;
  223. }
  224. };
  225. template <typename ThreadEngine>
  226. inline ThreadEngineStarter<typename ThreadEngine::ResultType> startThreadEngine(ThreadEngine *threadEngine)
  227. {
  228. return ThreadEngineStarter<typename ThreadEngine::ResultType>(threadEngine);
  229. }
  230. } // namespace QtConcurrent
  231. #endif //Q_QDOC
  232. QT_END_NAMESPACE
  233. #endif // QT_NO_CONCURRENT
  234. #endif