diff options
author | Marc Mutz <marc.mutz@kdab.com> | 2011-06-30 11:36:03 +0200 |
---|---|---|
committer | Marc Mutz <marc.mutz@kdab.com> | 2014-08-05 18:04:56 +0200 |
commit | c55aec2ed78a5c436a522ad571b10caaffa2295d (patch) | |
tree | 2765307ae1bf32fc2c6c237e28eb973231563c72 /src/corelib/thread/qfutureinterface.cpp | |
parent | a258ac8feb7f82446debbe2d5d6aa405942ceab4 (diff) |
QFutureInterface: allow to work with a QThreadPool != globalInstance()
Background:
It is often necessary/advisable to schedule tasks on thread pools !=
globalInstance(). As Herb Sutter writes in
http://www.drdobbs.com/parallel/use-thread-pools-correctly-keep-tasks-sh/216500409
and the Qt Training Material stresses, tasks you schedule on a (global)
thread pool should be non-blocking, which currently rules out using any of
the QtConcurrent functions for, say, file I/O.
Nonetheless it's often convenient to have thread pools also for file I/O, as
the thumbnail viewer exercise in the Qt Training Material shows. In this
case, you'd use a dedicated thead pool, leaving the global thread pool for
CPU-bound tasks.
Yet, none of the QtConcurrent functions allow to pick the QThreadPool
instance on which to schedule the work created with them.
This patch prepares for them to do so.
This is the first part of the forward-port of
https://qt.gitorious.org/qt/qt/merge_requests/1281.
Implement by using a new QThreadPool* member that defaults to nullptr,
and adding setThreadPool to set this member, then using it in lieu of
QThreadPool::globalInstance() everywhere.
I chose to leave m_pool == nullptr to mean globalInstance() to avoid
creating the global instance whenever a QFuture is created, even if the
future represents the result of a calculation not run on the global thread
pool.
[ChangeLog][QtCore][QFuture] Can now be used with any QThreadPool, not
just globalInstance().
Task-number: QTBUG-17220
Change-Id: I4e1dc18d55cf60141b2fa3d14e2d44a3e9e74858
Reviewed-by: Olivier Goffart <ogoffart@woboq.com>
Diffstat (limited to 'src/corelib/thread/qfutureinterface.cpp')
-rw-r--r-- | src/corelib/thread/qfutureinterface.cpp | 14 |
1 files changed, 9 insertions, 5 deletions
diff --git a/src/corelib/thread/qfutureinterface.cpp b/src/corelib/thread/qfutureinterface.cpp index ee3d113196..4f51fd7a0a 100644 --- a/src/corelib/thread/qfutureinterface.cpp +++ b/src/corelib/thread/qfutureinterface.cpp @@ -48,7 +48,6 @@ #include <QtCore/qatomic.h> #include <QtCore/qthread.h> -#include <QtCore/qthreadpool.h> #include <private/qthreadpool_p.h> QT_BEGIN_NAMESPACE @@ -195,7 +194,7 @@ void QFutureInterfaceBase::waitForResume() return; // decrease active thread count since this thread will wait. - const ThreadPoolThreadReleaser releaser(QThreadPool::globalInstance()); + const ThreadPoolThreadReleaser releaser(d->pool()); d->pausedWaitCondition.wait(&d->m_mutex); } @@ -301,7 +300,7 @@ void QFutureInterfaceBase::waitForResult(int resultIndex) // To avoid deadlocks and reduce the number of threads used, try to // run the runnable in the current thread. - QThreadPool::globalInstance()->d_func()->stealRunnable(d->runnable); + d->pool()->d_func()->stealRunnable(d->runnable); lock.relock(); @@ -322,7 +321,7 @@ void QFutureInterfaceBase::waitForFinished() lock.unlock(); if (!alreadyFinished) { - QThreadPool::globalInstance()->d_func()->stealRunnable(d->runnable); + d->pool()->d_func()->stealRunnable(d->runnable); lock.relock(); @@ -364,6 +363,11 @@ void QFutureInterfaceBase::setRunnable(QRunnable *runnable) d->runnable = runnable; } +void QFutureInterfaceBase::setThreadPool(QThreadPool *pool) +{ + d->m_pool = pool; +} + void QFutureInterfaceBase::setFilterMode(bool enable) { QMutexLocker locker(&d->m_mutex); @@ -444,7 +448,7 @@ bool QFutureInterfaceBase::derefT() const QFutureInterfaceBasePrivate::QFutureInterfaceBasePrivate(QFutureInterfaceBase::State initialState) : refCount(1), m_progressValue(0), m_progressMinimum(0), m_progressMaximum(0), state(initialState), - manualProgress(false), m_expectedResultCount(0), runnable(0) + manualProgress(false), m_expectedResultCount(0), runnable(0), m_pool(0) { progressTime.invalidate(); } |