2012-10-02 09:12:39 +02:00
|
|
|
/****************************************************************************
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2016-01-15 14:58:39 +01:00
|
|
|
** Copyright (C) 2016 The Qt Company Ltd.
|
|
|
|
|
** Contact: https://www.qt.io/licensing/
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** This file is part of Qt Creator.
|
2008-12-02 12:01:29 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
** Commercial License Usage
|
|
|
|
|
** Licensees holding valid commercial Qt licenses may use this file in
|
|
|
|
|
** accordance with the commercial license agreement provided with the
|
|
|
|
|
** Software or, alternatively, in accordance with the terms contained in
|
2016-01-15 14:58:39 +01:00
|
|
|
** a written agreement between you and The Qt Company. For licensing terms
|
|
|
|
|
** and conditions see https://www.qt.io/terms-conditions. For further
|
|
|
|
|
** information use the contact form at https://www.qt.io/contact-us.
|
2008-12-02 14:17:16 +01:00
|
|
|
**
|
2016-01-15 14:58:39 +01:00
|
|
|
** GNU General Public License Usage
|
|
|
|
|
** Alternatively, this file may be used under the terms of the GNU
|
|
|
|
|
** General Public License version 3 as published by the Free Software
|
|
|
|
|
** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT
|
|
|
|
|
** included in the packaging of this file. Please review the following
|
|
|
|
|
** information to ensure the GNU General Public License requirements will
|
|
|
|
|
** be met: https://www.gnu.org/licenses/gpl-3.0.html.
|
2010-12-17 16:01:08 +01:00
|
|
|
**
|
2012-10-02 09:12:39 +02:00
|
|
|
****************************************************************************/
|
2008-12-02 14:09:21 +01:00
|
|
|
|
2013-03-15 12:35:25 +01:00
|
|
|
#ifndef RUNEXTENSIONS_H
|
|
|
|
|
#define RUNEXTENSIONS_H
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2015-12-16 14:58:04 +01:00
|
|
|
#include "qtcassert.h"
|
2016-01-28 09:54:39 +01:00
|
|
|
#include "utils_global.h"
|
2015-12-16 14:58:04 +01:00
|
|
|
|
2016-01-27 15:44:31 +01:00
|
|
|
#include <QCoreApplication>
|
2016-01-26 16:48:21 +01:00
|
|
|
#include <QFuture>
|
|
|
|
|
#include <QFutureInterface>
|
|
|
|
|
#include <QRunnable>
|
2016-01-28 09:54:39 +01:00
|
|
|
#include <QThread>
|
2016-01-26 16:48:21 +01:00
|
|
|
#include <QThreadPool>
|
2008-12-02 12:01:29 +01:00
|
|
|
|
2015-12-16 14:58:04 +01:00
|
|
|
#include <chrono>
|
2014-09-04 11:55:36 +02:00
|
|
|
#include <functional>
|
2015-12-16 14:58:04 +01:00
|
|
|
#include <future>
|
|
|
|
|
#include <thread>
|
|
|
|
|
#include <vector>
|
2014-09-04 11:55:36 +02:00
|
|
|
|
2008-12-02 12:01:29 +01:00
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
|
|
|
|
|
|
namespace QtConcurrent {
|
|
|
|
|
|
|
|
|
|
template <typename T, typename FunctionPointer>
|
|
|
|
|
class StoredInterfaceFunctionCall0 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2014-09-04 11:55:36 +02:00
|
|
|
StoredInterfaceFunctionCall0(const FunctionPointer &fn)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
fn(futureInterface);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Class>
|
|
|
|
|
class StoredInterfaceMemberFunctionCall0 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
StoredInterfaceMemberFunctionCall0(void (Class::*fn)(QFutureInterface<T> &), Class *object)
|
|
|
|
|
: fn(fn), object(object) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
(object->*fn)(futureInterface);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Class *object;
|
|
|
|
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Arg1>
|
|
|
|
|
class StoredInterfaceFunctionCall1 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceFunctionCall1(void (fn)(QFutureInterface<T> &, Arg1), const Arg1 &arg1)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), arg1(arg1) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
fn(futureInterface, arg1);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Arg1 arg1;
|
|
|
|
|
};
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Class, typename Arg1>
|
|
|
|
|
class StoredInterfaceMemberFunctionCall1 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceMemberFunctionCall1(void (Class::*fn)(QFutureInterface<T> &, Arg1), Class *object, const Arg1 &arg1)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), object(object), arg1(arg1) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
(object->*fn)(futureInterface, arg1);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Class *object;
|
|
|
|
|
Arg1 arg1;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Arg1, typename Arg2>
|
|
|
|
|
class StoredInterfaceFunctionCall2 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceFunctionCall2(void (fn)(QFutureInterface<T> &, Arg1, Arg2), const Arg1 &arg1, const Arg2 &arg2)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), arg1(arg1), arg2(arg2) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
fn(futureInterface, arg1, arg2);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Arg1 arg1; Arg2 arg2;
|
|
|
|
|
};
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Class, typename Arg1, typename Arg2>
|
|
|
|
|
class StoredInterfaceMemberFunctionCall2 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceMemberFunctionCall2(void (Class::*fn)(QFutureInterface<T> &, Arg1, Arg2), Class *object, const Arg1 &arg1, const Arg2 &arg2)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), object(object), arg1(arg1), arg2(arg2) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
(object->*fn)(futureInterface, arg1, arg2);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Class *object;
|
|
|
|
|
Arg1 arg1; Arg2 arg2;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Arg1, typename Arg2, typename Arg3>
|
|
|
|
|
class StoredInterfaceFunctionCall3 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceFunctionCall3(void (fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3), const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), arg1(arg1), arg2(arg2), arg3(arg3) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
fn(futureInterface, arg1, arg2, arg3);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Arg1 arg1; Arg2 arg2; Arg3 arg3;
|
|
|
|
|
};
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Class, typename Arg1, typename Arg2, typename Arg3>
|
|
|
|
|
class StoredInterfaceMemberFunctionCall3 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceMemberFunctionCall3(void (Class::*fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3), Class *object, const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), object(object), arg1(arg1), arg2(arg2), arg3(arg3) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
(object->*fn)(futureInterface, arg1, arg2, arg3);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Class *object;
|
|
|
|
|
Arg1 arg1; Arg2 arg2; Arg3 arg3;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Arg1, typename Arg2, typename Arg3, typename Arg4>
|
|
|
|
|
class StoredInterfaceFunctionCall4 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceFunctionCall4(void (fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4), const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3, const Arg4 &arg4)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), arg1(arg1), arg2(arg2), arg3(arg3), arg4(arg4) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
fn(futureInterface, arg1, arg2, arg3, arg4);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Arg1 arg1; Arg2 arg2; Arg3 arg3; Arg4 arg4;
|
|
|
|
|
};
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Class, typename Arg1, typename Arg2, typename Arg3, typename Arg4>
|
|
|
|
|
class StoredInterfaceMemberFunctionCall4 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceMemberFunctionCall4(void (Class::*fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4), Class *object, const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3, const Arg4 &arg4)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), object(object), arg1(arg1), arg2(arg2), arg3(arg3), arg4(arg4) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
(object->*fn)(futureInterface, arg1, arg2, arg3, arg4);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Class *object;
|
|
|
|
|
Arg1 arg1; Arg2 arg2; Arg3 arg3; Arg4 arg4;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Arg1, typename Arg2, typename Arg3, typename Arg4, typename Arg5>
|
|
|
|
|
class StoredInterfaceFunctionCall5 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceFunctionCall5(void (fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4, Arg5), const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3, const Arg4 &arg4, const Arg5 &arg5)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), arg1(arg1), arg2(arg2), arg3(arg3), arg4(arg4), arg5(arg5) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
fn(futureInterface, arg1, arg2, arg3, arg4, arg5);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Arg1 arg1; Arg2 arg2; Arg3 arg3; Arg4 arg4; Arg5 arg5;
|
|
|
|
|
};
|
|
|
|
|
template <typename T, typename FunctionPointer, typename Class, typename Arg1, typename Arg2, typename Arg3, typename Arg4, typename Arg5>
|
|
|
|
|
class StoredInterfaceMemberFunctionCall5 : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
2012-02-16 17:27:46 +01:00
|
|
|
StoredInterfaceMemberFunctionCall5(void (Class::*fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4, Arg5), Class *object, const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3, const Arg4 &arg4, const Arg5 &arg5)
|
2008-12-02 12:01:29 +01:00
|
|
|
: fn(fn), object(object), arg1(arg1), arg2(arg2), arg3(arg3), arg4(arg4), arg5(arg5) { }
|
|
|
|
|
|
|
|
|
|
QFuture<T> start()
|
|
|
|
|
{
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
QFuture<T> future = futureInterface.future();
|
|
|
|
|
QThreadPool::globalInstance()->start(this);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void run()
|
|
|
|
|
{
|
|
|
|
|
(object->*fn)(futureInterface, arg1, arg2, arg3, arg4, arg5);
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
private:
|
|
|
|
|
QFutureInterface<T> futureInterface;
|
|
|
|
|
FunctionPointer fn;
|
|
|
|
|
Class *object;
|
|
|
|
|
Arg1 arg1; Arg2 arg2; Arg3 arg3; Arg4 arg4; Arg5 arg5;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
QFuture<T> run(void (*functionPointer)(QFutureInterface<T> &))
|
|
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceFunctionCall0<T, void (*)(QFutureInterface<T> &)>(functionPointer))->start();
|
|
|
|
|
}
|
|
|
|
|
template <typename T, typename Arg1>
|
2012-02-16 17:27:46 +01:00
|
|
|
QFuture<T> run(void (*functionPointer)(QFutureInterface<T> &, Arg1), const Arg1 &arg1)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceFunctionCall1<T, void (*)(QFutureInterface<T> &, Arg1), Arg1>(functionPointer, arg1))->start();
|
|
|
|
|
}
|
|
|
|
|
template <typename T, typename Arg1, typename Arg2>
|
2012-02-16 17:27:46 +01:00
|
|
|
QFuture<T> run(void (*functionPointer)(QFutureInterface<T> &, Arg1, Arg2), const Arg1 &arg1, const Arg2 &arg2)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceFunctionCall2<T, void (*)(QFutureInterface<T> &, Arg1, Arg2), Arg1, Arg2>(functionPointer, arg1, arg2))->start();
|
|
|
|
|
}
|
|
|
|
|
template <typename T, typename Arg1, typename Arg2, typename Arg3>
|
2012-02-16 17:27:46 +01:00
|
|
|
QFuture<T> run(void (*functionPointer)(QFutureInterface<T> &, Arg1, Arg2, Arg3), const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceFunctionCall3<T, void (*)(QFutureInterface<T> &, Arg1, Arg2, Arg3), Arg1, Arg2, Arg3>(functionPointer, arg1, arg2, arg3))->start();
|
|
|
|
|
}
|
|
|
|
|
template <typename T, typename Arg1, typename Arg2, typename Arg3, typename Arg4>
|
2012-02-16 17:27:46 +01:00
|
|
|
QFuture<T> run(void (*functionPointer)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4), const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3, const Arg4 &arg4)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceFunctionCall4<T, void (*)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4), Arg1, Arg2, Arg3, Arg4>(functionPointer, arg1, arg2, arg3, arg4))->start();
|
|
|
|
|
}
|
|
|
|
|
template <typename T, typename Arg1, typename Arg2, typename Arg3, typename Arg4, typename Arg5>
|
2012-02-16 17:27:46 +01:00
|
|
|
QFuture<T> run(void (*functionPointer)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4, Arg5), const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3, const Arg4 &arg4, const Arg5 &arg5)
|
2008-12-02 12:01:29 +01:00
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceFunctionCall5<T, void (*)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4, Arg5), Arg1, Arg2, Arg3, Arg4, Arg5>(functionPointer, arg1, arg2, arg3, arg4, arg5))->start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename Class, typename T>
|
|
|
|
|
QFuture<T> run(void (Class::*fn)(QFutureInterface<T> &), Class *object)
|
|
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceMemberFunctionCall0<T, void (Class::*)(QFutureInterface<T> &), Class>(fn, object))->start();
|
|
|
|
|
}
|
|
|
|
|
|
2012-02-16 17:27:46 +01:00
|
|
|
template <typename Class, typename T, typename Arg1>
|
|
|
|
|
QFuture<T> run(void (Class::*fn)(QFutureInterface<T> &, Arg1), Class *object, Arg1 arg1)
|
|
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceMemberFunctionCall1<T, void (Class::*)(QFutureInterface<T> &, Arg1), Class, Arg1>(fn, object, arg1))->start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename Class, typename T, typename Arg1, typename Arg2>
|
|
|
|
|
QFuture<T> run(void (Class::*fn)(QFutureInterface<T> &, Arg1, Arg2), Class *object, const Arg1 &arg1, const Arg2 &arg2)
|
|
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceMemberFunctionCall2<T, void (Class::*)(QFutureInterface<T> &, Arg1, Arg2), Class, Arg1, Arg2>(fn, object, arg1, arg2))->start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename Class, typename T, typename Arg1, typename Arg2, typename Arg3>
|
|
|
|
|
QFuture<T> run(void (Class::*fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3), Class *object, const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3)
|
|
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceMemberFunctionCall3<T, void (Class::*)(QFutureInterface<T> &, Arg1, Arg2, Arg3), Class, Arg1, Arg2, Arg3>(fn, object, arg1, arg2, arg3))->start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename Class, typename T, typename Arg1, typename Arg2, typename Arg3, typename Arg4>
|
|
|
|
|
QFuture<T> run(void (Class::*fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4), Class *object, const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3, const Arg4 &arg4)
|
|
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceMemberFunctionCall4<T, void (Class::*)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4), Class, Arg1, Arg2, Arg3, Arg4>(fn, object, arg1, arg2, arg3, arg4))->start();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename Class, typename T, typename Arg1, typename Arg2, typename Arg3, typename Arg4, typename Arg5>
|
|
|
|
|
QFuture<T> run(void (Class::*fn)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4, Arg5), Class *object, const Arg1 &arg1, const Arg2 &arg2, const Arg3 &arg3, const Arg4 &arg4, const Arg5 &arg5)
|
|
|
|
|
{
|
|
|
|
|
return (new StoredInterfaceMemberFunctionCall5<T, void (Class::*)(QFutureInterface<T> &, Arg1, Arg2, Arg3, Arg4, Arg5), Class, Arg1, Arg2, Arg3, Arg4, Arg5>(fn, object, arg1, arg2, arg3, arg4, arg5))->start();
|
|
|
|
|
}
|
2014-09-04 11:55:36 +02:00
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
|
QFuture<T> run(const std::function<void (QFutureInterface<T> &)> &fn)
|
|
|
|
|
{
|
2015-02-16 15:47:43 +01:00
|
|
|
return (new StoredInterfaceFunctionCall0<T, std::function<void (QFutureInterface<T> &)>>(fn))->start();
|
2014-09-04 11:55:36 +02:00
|
|
|
}
|
|
|
|
|
|
2008-12-02 14:09:21 +01:00
|
|
|
} // namespace QtConcurrent
|
2008-12-02 12:01:29 +01:00
|
|
|
|
|
|
|
|
QT_END_NAMESPACE
|
|
|
|
|
|
2015-12-16 14:58:04 +01:00
|
|
|
namespace Utils {
|
|
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
|
typename std::vector<std::future<T>>::iterator
|
|
|
|
|
waitForAny(std::vector<std::future<T>> &futures)
|
|
|
|
|
{
|
|
|
|
|
// Wait for any future to have a result ready.
|
|
|
|
|
// Unfortunately we have to do that in a busy loop because future doesn't have a feature to
|
|
|
|
|
// wait for any of a set of futures (yet? possibly when_any in C++17).
|
|
|
|
|
auto end = futures.end();
|
|
|
|
|
QTC_ASSERT(!futures.empty(), return end);
|
|
|
|
|
auto futureIterator = futures.begin();
|
|
|
|
|
forever {
|
|
|
|
|
if (futureIterator->wait_for(std::chrono::duration<quint64>::zero()) == std::future_status::ready)
|
|
|
|
|
return futureIterator;
|
|
|
|
|
++futureIterator;
|
|
|
|
|
if (futureIterator == end)
|
|
|
|
|
futureIterator = futures.begin();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
namespace Internal {
|
|
|
|
|
|
|
|
|
|
template<typename T>
|
|
|
|
|
void swapErase(std::vector<T> &vec, typename std::vector<T>::iterator it)
|
|
|
|
|
{
|
|
|
|
|
// efficient erasing by swapping with back element
|
|
|
|
|
*it = std::move(vec.back());
|
|
|
|
|
vec.pop_back();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename MapResult, typename State, typename ReduceResult, typename ReduceFunction>
|
|
|
|
|
void reduceOne(QFutureInterface<ReduceResult> &futureInterface,
|
|
|
|
|
std::vector<std::future<MapResult>> &futures,
|
|
|
|
|
State &state, const ReduceFunction &reduce)
|
|
|
|
|
{
|
|
|
|
|
auto futureIterator = waitForAny(futures);
|
|
|
|
|
if (futureIterator != futures.end()) {
|
|
|
|
|
reduce(futureInterface, state, futureIterator->get());
|
|
|
|
|
swapErase(futures, futureIterator);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This together with reduceOne can be replaced by std::transformReduce (parallelism TS)
|
|
|
|
|
// when that becomes widely available in C++ implementations
|
|
|
|
|
template <typename Container, typename MapFunction, typename State, typename ReduceResult, typename ReduceFunction>
|
|
|
|
|
void mapReduceLoop(QFutureInterface<ReduceResult> &futureInterface, const Container &container,
|
|
|
|
|
const MapFunction &map, State &state, const ReduceFunction &reduce)
|
|
|
|
|
{
|
|
|
|
|
const unsigned MAX_THREADS = std::thread::hardware_concurrency();
|
|
|
|
|
using MapResult = typename std::result_of<MapFunction(QFutureInterface<ReduceResult>,typename Container::value_type)>::type;
|
|
|
|
|
std::vector<std::future<MapResult>> futures;
|
|
|
|
|
futures.reserve(MAX_THREADS);
|
|
|
|
|
auto fileIterator = container.begin();
|
|
|
|
|
auto end = container.end();
|
|
|
|
|
while (!futureInterface.isCanceled() && (fileIterator != end || futures.size() != 0)) {
|
|
|
|
|
if (futures.size() >= MAX_THREADS || fileIterator == end) {
|
|
|
|
|
// We don't want to start a new thread (yet), so try to find a future that is ready and
|
|
|
|
|
// handle its result.
|
|
|
|
|
reduceOne(futureInterface, futures, state, reduce);
|
|
|
|
|
} else { // start a new thread
|
|
|
|
|
futures.push_back(std::async(std::launch::async,
|
|
|
|
|
map, futureInterface, *fileIterator));
|
|
|
|
|
++fileIterator;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename Container, typename InitFunction, typename MapFunction, typename ReduceResult,
|
|
|
|
|
typename ReduceFunction, typename CleanUpFunction>
|
|
|
|
|
void blockingMapReduce(QFutureInterface<ReduceResult> futureInterface, const Container &container,
|
|
|
|
|
const InitFunction &init, const MapFunction &map,
|
|
|
|
|
const ReduceFunction &reduce, const CleanUpFunction &cleanup)
|
|
|
|
|
{
|
|
|
|
|
auto state = init(futureInterface);
|
|
|
|
|
mapReduceLoop(futureInterface, container, map, state, reduce);
|
|
|
|
|
cleanup(futureInterface, state);
|
|
|
|
|
if (futureInterface.isPaused())
|
|
|
|
|
futureInterface.waitForResume();
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-18 12:48:09 +01:00
|
|
|
template <typename ResultType, typename Function, typename Obj, typename... Args>
|
|
|
|
|
typename std::enable_if<std::is_member_pointer<typename std::decay<Function>::type>::value>::type
|
2016-02-02 08:25:55 +01:00
|
|
|
runAsyncImpl(QFutureInterface<ResultType> futureInterface, Function &&function, Obj &&obj, Args&&... args)
|
2016-01-18 12:48:09 +01:00
|
|
|
{
|
2016-02-02 08:25:55 +01:00
|
|
|
std::mem_fn(std::forward<Function>(function))(std::forward<Obj>(obj),
|
|
|
|
|
futureInterface, std::forward<Args>(args)...);
|
2016-01-18 12:48:09 +01:00
|
|
|
if (futureInterface.isPaused())
|
|
|
|
|
futureInterface.waitForResume();
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-12 14:05:05 +01:00
|
|
|
template <typename ResultType, typename Function, typename... Args>
|
2016-01-18 12:48:09 +01:00
|
|
|
typename std::enable_if<!std::is_member_pointer<typename std::decay<Function>::type>::value>::type
|
2016-02-02 08:25:55 +01:00
|
|
|
runAsyncImpl(QFutureInterface<ResultType> futureInterface, Function &&function, Args&&... args)
|
2016-01-12 14:05:05 +01:00
|
|
|
{
|
2016-02-02 08:25:55 +01:00
|
|
|
function(futureInterface, std::forward<Args>(args)...);
|
2016-01-12 14:05:05 +01:00
|
|
|
if (futureInterface.isPaused())
|
|
|
|
|
futureInterface.waitForResume();
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-26 16:48:21 +01:00
|
|
|
// can be replaced with std::(make_)index_sequence with C++14
|
|
|
|
|
template <std::size_t...>
|
|
|
|
|
struct indexSequence { };
|
|
|
|
|
template <std::size_t N, std::size_t... S>
|
|
|
|
|
struct makeIndexSequence : makeIndexSequence<N-1, N-1, S...> { };
|
|
|
|
|
template <std::size_t... S>
|
|
|
|
|
struct makeIndexSequence<0, S...> { typedef indexSequence<S...> type; };
|
|
|
|
|
|
|
|
|
|
template <class T>
|
|
|
|
|
typename std::decay<T>::type
|
|
|
|
|
decayCopy(T&& v)
|
|
|
|
|
{
|
|
|
|
|
return std::forward<T>(v);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename ResultType, typename Function, typename... Args>
|
|
|
|
|
class AsyncJob : public QRunnable
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
AsyncJob(Function &&function, Args&&... args)
|
|
|
|
|
// decay copy like std::thread
|
|
|
|
|
: data(decayCopy(std::forward<Function>(function)), decayCopy(std::forward<Args>(args))...)
|
|
|
|
|
{
|
|
|
|
|
// we need to report it as started even though it isn't yet, because someone might
|
|
|
|
|
// call waitForFinished on the future, which does _not_ block if the future is not started
|
|
|
|
|
futureInterface.setRunnable(this);
|
|
|
|
|
futureInterface.reportStarted();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
~AsyncJob()
|
|
|
|
|
{
|
|
|
|
|
// QThreadPool can delete runnables even if they were never run (e.g. QThreadPool::clear).
|
|
|
|
|
// Since we reported them as started, we make sure that we always report them as finished.
|
|
|
|
|
// reportFinished only actually sends the signal if it wasn't already finished.
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
QFuture<ResultType> future() { return futureInterface.future(); }
|
|
|
|
|
|
|
|
|
|
void run() override
|
|
|
|
|
{
|
2016-01-27 15:44:31 +01:00
|
|
|
if (priority != QThread::InheritPriority)
|
|
|
|
|
if (QThread *thread = QThread::currentThread())
|
|
|
|
|
if (thread != qApp->thread())
|
|
|
|
|
thread->setPriority(priority);
|
2016-01-26 16:48:21 +01:00
|
|
|
if (futureInterface.isCanceled()) {
|
|
|
|
|
futureInterface.reportFinished();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
runHelper(typename makeIndexSequence<std::tuple_size<Data>::value>::type());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void setThreadPool(QThreadPool *pool)
|
|
|
|
|
{
|
|
|
|
|
futureInterface.setThreadPool(pool);
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-27 15:44:31 +01:00
|
|
|
void setThreadPriority(QThread::Priority p)
|
|
|
|
|
{
|
|
|
|
|
priority = p;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-26 16:48:21 +01:00
|
|
|
private:
|
|
|
|
|
using Data = std::tuple<typename std::decay<Function>::type, typename std::decay<Args>::type...>;
|
|
|
|
|
|
|
|
|
|
template <std::size_t... index>
|
|
|
|
|
void runHelper(indexSequence<index...>)
|
|
|
|
|
{
|
|
|
|
|
// invalidates data, which is moved into the call
|
|
|
|
|
runAsyncImpl(futureInterface, std::move(std::get<index>(data))...);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Data data;
|
|
|
|
|
QFutureInterface<ResultType> futureInterface;
|
2016-01-27 15:44:31 +01:00
|
|
|
QThread::Priority priority = QThread::InheritPriority;
|
2016-01-26 16:48:21 +01:00
|
|
|
};
|
|
|
|
|
|
2016-01-28 09:54:39 +01:00
|
|
|
class QTCREATOR_UTILS_EXPORT RunnableThread : public QThread
|
|
|
|
|
{
|
|
|
|
|
public:
|
|
|
|
|
explicit RunnableThread(QRunnable *runnable, QObject *parent = 0);
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
|
void run();
|
|
|
|
|
|
|
|
|
|
private:
|
|
|
|
|
QRunnable *m_runnable;
|
|
|
|
|
};
|
|
|
|
|
|
2015-12-16 14:58:04 +01:00
|
|
|
} // Internal
|
|
|
|
|
|
|
|
|
|
template <typename ReduceResult, typename Container, typename InitFunction, typename MapFunction,
|
|
|
|
|
typename ReduceFunction, typename CleanUpFunction>
|
|
|
|
|
QFuture<ReduceResult> mapReduce(std::reference_wrapper<Container> containerWrapper,
|
|
|
|
|
const InitFunction &init, const MapFunction &map,
|
|
|
|
|
const ReduceFunction &reduce, const CleanUpFunction &cleanup)
|
|
|
|
|
{
|
|
|
|
|
auto fi = QFutureInterface<ReduceResult>();
|
|
|
|
|
QFuture<ReduceResult> future = fi.future();
|
2016-01-14 15:52:49 +01:00
|
|
|
fi.reportStarted();
|
2015-12-16 14:58:04 +01:00
|
|
|
std::thread(Internal::blockingMapReduce<Container, InitFunction, MapFunction, ReduceResult, ReduceFunction, CleanUpFunction>,
|
|
|
|
|
fi, containerWrapper, init, map, reduce, cleanup).detach();
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
template <typename ReduceResult, typename Container, typename InitFunction, typename MapFunction,
|
|
|
|
|
typename ReduceFunction, typename CleanUpFunction>
|
|
|
|
|
QFuture<ReduceResult> mapReduce(const Container &container, const InitFunction &init, const MapFunction &map,
|
|
|
|
|
const ReduceFunction &reduce, const CleanUpFunction &cleanup)
|
|
|
|
|
{
|
|
|
|
|
auto fi = QFutureInterface<ReduceResult>();
|
|
|
|
|
QFuture<ReduceResult> future = fi.future();
|
|
|
|
|
std::thread(Internal::blockingMapReduce<Container, InitFunction, MapFunction, ReduceResult, ReduceFunction, CleanUpFunction>,
|
|
|
|
|
fi, container, init, map, reduce, cleanup).detach();
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-18 12:48:09 +01:00
|
|
|
/*!
|
|
|
|
|
The interface of \c {runAsync} is similar to the std::thread constructor and \c {std::invoke}.
|
|
|
|
|
|
|
|
|
|
The \a function argument can be a member function, an object with \c {operator()}, a
|
|
|
|
|
\c {std::function}, lambda, function pointer or function reference.
|
|
|
|
|
They need to take a \c {QFutureInterface<ResultType>&} as their first argument, followed by
|
|
|
|
|
other custom arguments which need to be passed to this function.
|
|
|
|
|
If \a function is a (non-static) member function, the first argument in \a args is expected
|
|
|
|
|
to be the object that the function is called on.
|
|
|
|
|
|
|
|
|
|
\sa std::thread
|
|
|
|
|
\sa std::invoke
|
|
|
|
|
*/
|
2016-02-02 13:46:08 +01:00
|
|
|
template <typename ResultType, typename Function, typename... Args>
|
|
|
|
|
QFuture<ResultType> runAsync(QThread::Priority priority, Function &&function, Args&&... args)
|
2016-01-12 14:05:05 +01:00
|
|
|
{
|
2016-01-28 09:54:39 +01:00
|
|
|
auto job = new Internal::AsyncJob<ResultType,Function,Args...>
|
|
|
|
|
(std::forward<Function>(function), std::forward<Args>(args)...);
|
2016-02-02 13:46:08 +01:00
|
|
|
job->setThreadPriority(priority);
|
2016-01-28 09:54:39 +01:00
|
|
|
QFuture<ResultType> future = job->future();
|
2016-01-29 10:42:55 +01:00
|
|
|
auto thread = new Internal::RunnableThread(job);
|
|
|
|
|
thread->moveToThread(qApp->thread()); // make sure thread gets deleteLater on main thread
|
|
|
|
|
QObject::connect(thread, &QThread::finished, thread, &QObject::deleteLater);
|
|
|
|
|
thread->start();
|
2016-01-28 09:54:39 +01:00
|
|
|
return future;
|
2016-01-12 14:05:05 +01:00
|
|
|
}
|
|
|
|
|
|
2016-02-02 13:46:08 +01:00
|
|
|
template <typename ResultType, typename Function, typename... Args,
|
|
|
|
|
typename = typename std::enable_if<
|
|
|
|
|
!std::is_same<typename std::decay<Function>::type, QThreadPool>::value
|
|
|
|
|
&& !std::is_same<typename std::decay<Function>::type, QThread::Priority>::value
|
|
|
|
|
>::type>
|
|
|
|
|
QFuture<ResultType> runAsync(Function &&function, Args&&... args)
|
|
|
|
|
{
|
|
|
|
|
return runAsync<ResultType>(QThread::InheritPriority, std::forward<Function>(function),
|
|
|
|
|
std::forward<Args>(args)...);
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-26 16:48:21 +01:00
|
|
|
template <typename ResultType, typename Function, typename... Args>
|
2016-01-27 15:44:31 +01:00
|
|
|
QFuture<ResultType> runAsync(QThreadPool *pool, QThread::Priority priority,
|
|
|
|
|
Function &&function, Args&&... args)
|
2016-01-26 16:48:21 +01:00
|
|
|
{
|
|
|
|
|
auto job = new Internal::AsyncJob<ResultType,Function,Args...>
|
|
|
|
|
(std::forward<Function>(function), std::forward<Args>(args)...);
|
|
|
|
|
job->setThreadPool(pool);
|
2016-01-27 15:44:31 +01:00
|
|
|
job->setThreadPriority(priority);
|
2016-01-26 16:48:21 +01:00
|
|
|
QFuture<ResultType> future = job->future();
|
|
|
|
|
pool->start(job);
|
|
|
|
|
return future;
|
|
|
|
|
}
|
|
|
|
|
|
2016-01-27 15:44:31 +01:00
|
|
|
template <typename ResultType, typename Function, typename... Args,
|
|
|
|
|
typename = typename std::enable_if<
|
|
|
|
|
!std::is_same<typename std::decay<Function>::type, QThread::Priority>::value
|
|
|
|
|
>::type>
|
|
|
|
|
QFuture<ResultType> runAsync(QThreadPool *pool,
|
|
|
|
|
Function &&function, Args&&... args)
|
|
|
|
|
{
|
|
|
|
|
return runAsync<ResultType>(pool, QThread::InheritPriority, std::forward<Function>(function),
|
|
|
|
|
std::forward<Args>(args)...);
|
|
|
|
|
}
|
|
|
|
|
|
2015-12-16 14:58:04 +01:00
|
|
|
} // Utils
|
|
|
|
|
|
2013-03-15 12:35:25 +01:00
|
|
|
#endif // RUNEXTENSIONS_H
|