|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#ifndef TTHREAD_H
|
|
Toshihiro Shimizu |
890ddd |
#define TTHREAD_H
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#include "tsmartpointer.h"
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#include <qthread></qthread>
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#undef DVAPI
|
|
Toshihiro Shimizu |
890ddd |
#undef DVVAR
|
|
Toshihiro Shimizu |
890ddd |
#ifdef TNZCORE_EXPORTS
|
|
Toshihiro Shimizu |
890ddd |
#define DVAPI DV_EXPORT_API
|
|
Toshihiro Shimizu |
890ddd |
#define DVVAR DV_EXPORT_VAR
|
|
Toshihiro Shimizu |
890ddd |
#else
|
|
Toshihiro Shimizu |
890ddd |
#define DVAPI DV_IMPORT_API
|
|
Toshihiro Shimizu |
890ddd |
#define DVVAR DV_IMPORT_VAR
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
namespace TThread
|
|
Toshihiro Shimizu |
890ddd |
{
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
//------------------------------------------------------------------------------
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
//! Initializes all TThread namespace components.
|
|
Toshihiro Shimizu |
890ddd |
void DVAPI init();
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
//! Closes all TThread components as soon as possible in a safe manner.
|
|
Toshihiro Shimizu |
890ddd |
//! \sa Executor::shutdown() method
|
|
Toshihiro Shimizu |
890ddd |
void DVAPI shutdown();
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
//------------------------------------------------------------------------------
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
//Forward declarations
|
|
Toshihiro Shimizu |
890ddd |
class ExecutorId; //Private
|
|
Toshihiro Shimizu |
890ddd |
class Runnable;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#ifndef MACOSX
|
|
Toshihiro Shimizu |
890ddd |
template class TSmartPointerT<runnable>;</runnable>
|
|
Toshihiro Shimizu |
890ddd |
#endif
|
|
Toshihiro Shimizu |
890ddd |
typedef TSmartPointerT<runnable> RunnableP;</runnable>
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
//------------------------------------------------------------------------------
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
/*!
|
|
Toshihiro Shimizu |
890ddd |
Runnable class is the abstract model for user-defined tasks which can be
|
|
Toshihiro Shimizu |
890ddd |
scheduled for execution into separate application threads.
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
A user must first implement the run() method to provide a task with the
|
|
Toshihiro Shimizu |
890ddd |
very execution code. Worker threads created internally by the task manager will
|
|
Toshihiro Shimizu |
890ddd |
take ownership of the task and make it run() at the most appropriate time,
|
|
Toshihiro Shimizu |
890ddd |
depending on the task load, insertion time and its scheduling priority.
|
|
Toshihiro Shimizu |
890ddd |
\n \n
|
|
Toshihiro Shimizu |
890ddd |
The scheduling priority of a task can be set by reimplementing the
|
|
Toshihiro Shimizu |
890ddd |
\b schedulingPriority() method. Tasks whose scheduling priority is higher are
|
|
Toshihiro Shimizu |
890ddd |
always started before compared to the others added by the same executor.
|
|
Toshihiro Shimizu |
890ddd |
\n \n
|
|
Toshihiro Shimizu |
890ddd |
The hosting thread's running priority may also be set reimplementing the
|
|
Toshihiro Shimizu |
890ddd |
runningPriority() method; see QThread class documentation in Qt manual.
|
|
Toshihiro Shimizu |
890ddd |
\n \n
|
|
Toshihiro Shimizu |
890ddd |
A task's load is an important property that should be reimplemented in all
|
|
Toshihiro Shimizu |
890ddd |
resource-consuming tasks. It enables the user to declare the approximate
|
|
Toshihiro Shimizu |
890ddd |
CPU load produced by a task, allowing the task manager to effectively
|
|
Toshihiro Shimizu |
890ddd |
calculate the most appropriate moment to run it.
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
\warning \n All built-in signals about the Runnable class are emitted in a
|
|
Toshihiro Shimizu |
890ddd |
mutex-protected environment in order to make sure that signal emission is consistent
|
|
Toshihiro Shimizu |
890ddd |
with the task status - in other words, so that \a queued controller-emitted canceled()
|
|
Toshihiro Shimizu |
890ddd |
and terminated() signals are not delivered \a before started() or \a after
|
|
Toshihiro Shimizu |
890ddd |
finished() and exception() worker-emitted signals.
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
\warning Thus, setting up blocking connections or blocking direct slots is \b not \b
|
|
Toshihiro Shimizu |
890ddd |
supported and will typically result in deadlocks; furthermore, the same also applies to
|
|
Toshihiro Shimizu |
890ddd |
slot calls to the Executor API, which would have the aforementioned mutex to be
|
|
Toshihiro Shimizu |
890ddd |
locked again.
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
\warning In case the above blocking strategies are mandatory, the user should add
|
|
Toshihiro Shimizu |
890ddd |
custom signals to be emitted inside the mutex-free run() block, just before or after
|
|
Toshihiro Shimizu |
890ddd |
the actual code to be executed, like this:
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
\code
|
|
Toshihiro Shimizu |
890ddd |
void MyTask::run()
|
|
Toshihiro Shimizu |
890ddd |
{
|
|
Toshihiro Shimizu |
890ddd |
try
|
|
Toshihiro Shimizu |
890ddd |
{
|
|
Toshihiro Shimizu |
890ddd |
emit myStarted(this);
|
|
Toshihiro Shimizu |
890ddd |
theRunCode();
|
|
Toshihiro Shimizu |
890ddd |
emit myFinished(this);
|
|
Toshihiro Shimizu |
890ddd |
}
|
|
Toshihiro Shimizu |
890ddd |
catch(...)
|
|
Toshihiro Shimizu |
890ddd |
{
|
|
Toshihiro Shimizu |
890ddd |
emit myException(this);
|
|
Toshihiro Shimizu |
890ddd |
throw;
|
|
Toshihiro Shimizu |
890ddd |
}
|
|
Toshihiro Shimizu |
890ddd |
}
|
|
Toshihiro Shimizu |
890ddd |
\endcode
|
|
Toshihiro Shimizu |
890ddd |
\code
|
|
Toshihiro Shimizu |
890ddd |
..
|
|
Toshihiro Shimizu |
890ddd |
MyTask* myTask = new MyTask;
|
|
Toshihiro Shimizu |
890ddd |
connect(myTask, SIGNAL(myStarted(TThread::RunnableP)), myTask, SLOT(onStarted(TThread::RunnableP)),
|
|
Toshihiro Shimizu |
890ddd |
Qt::BlockingQueuedConnection) //theRunCode() waits for onStarted() to complete
|
|
Toshihiro Shimizu |
890ddd |
..
|
|
Toshihiro Shimizu |
890ddd |
\endcode
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
\sa Executor class.
|
|
Toshihiro Shimizu |
890ddd |
*/
|
|
Toshihiro Shimizu |
890ddd |
class DVAPI Runnable : public QObject, public TSmartObject
|
|
Toshihiro Shimizu |
890ddd |
{
|
|
Toshihiro Shimizu |
890ddd |
Q_OBJECT
|
|
Toshihiro Shimizu |
890ddd |
DECLARE_CLASS_CODE
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
ExecutorId *m_id;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
int m_load;
|
|
Toshihiro Shimizu |
890ddd |
int m_schedulingPriority;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
friend class Executor; //Needed to confront Executor's and Runnable's ids
|
|
Toshihiro Shimizu |
890ddd |
friend class ExecutorImp; //The internal task manager needs full control over the task
|
|
Toshihiro Shimizu |
890ddd |
friend class Worker; //Workers force tasks to emit state signals
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
public:
|
|
Toshihiro Shimizu |
890ddd |
Runnable();
|
|
Toshihiro Shimizu |
890ddd |
virtual ~Runnable();
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
//! The central code of the task that is executed by a worker thread.
|
|
Toshihiro Shimizu |
890ddd |
virtual void run() = 0;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
virtual int taskLoad();
|
|
Toshihiro Shimizu |
890ddd |
virtual int schedulingPriority();
|
|
Toshihiro Shimizu |
890ddd |
virtual QThread::Priority runningPriority();
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
Q_SIGNALS :
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
void
|
|
Toshihiro Shimizu |
890ddd |
started(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
void finished(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
void exception(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
void canceled(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
void terminated(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
public Q_SLOTS:
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
virtual void onStarted(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
virtual void onFinished(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
virtual void onException(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
virtual void onCanceled(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
virtual void onTerminated(TThread::RunnableP sender);
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
private:
|
|
Toshihiro Shimizu |
890ddd |
inline bool needsAccumulation();
|
|
Toshihiro Shimizu |
890ddd |
inline bool customConditions();
|
|
Toshihiro Shimizu |
890ddd |
};
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
//------------------------------------------------------------------------------
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
/*!
|
|
Toshihiro Shimizu |
890ddd |
Executor class provides an effective way for planning the execution of
|
|
Toshihiro Shimizu |
890ddd |
user-defined tasks that require separate working threads.
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
When an application needs to perform a resource-consuming task, it is often
|
|
Toshihiro Shimizu |
890ddd |
a good idea to dedicate a separate thread for it, especially in GUI applications;
|
|
Toshihiro Shimizu |
890ddd |
however, doing so eventually raises the problem of managing such intensive
|
|
Toshihiro Shimizu |
890ddd |
tasks in a way that constantly ensures the correct use of the machine resources -
|
|
Toshihiro Shimizu |
890ddd |
so that in any given time the CPU usage is maximal, but not overloaded.
|
|
Toshihiro Shimizu |
890ddd |
Additional requests by the user may arise, including preferenced ordering
|
|
Toshihiro Shimizu |
890ddd |
among tasks, the necessity of salvaging some CPU resources or threads for
|
|
Toshihiro Shimizu |
890ddd |
incoming tasks, and so on.
|
|
Toshihiro Shimizu |
890ddd |
\n \n
|
|
Toshihiro Shimizu |
890ddd |
The TThread namespace API contains two main classes, \b Runnable and \b Executor,
|
|
Toshihiro Shimizu |
890ddd |
which provide a way for implementing consistent threading strategies.
|
|
Toshihiro Shimizu |
890ddd |
\n \n
|
|
Toshihiro Shimizu |
890ddd |
In order to use the Executor class it is first necessary to install the thread manager
|
|
Toshihiro Shimizu |
890ddd |
into application code by calling the static method init() appropriately.
|
|
Toshihiro Shimizu |
890ddd |
\n \n
|
|
Toshihiro Shimizu |
890ddd |
Executors are then used to submit - or eventually remove - tasks for execution into a
|
|
Toshihiro Shimizu |
890ddd |
separate working thread, by means of the \b addTask(), \b removeTask() and \b cancelAll() methods.
|
|
Toshihiro Shimizu |
890ddd |
Each Executor's visibility is always limited to the tasks that it submits -
|
|
Toshihiro Shimizu |
890ddd |
so calling removeTask() or cancelAll() only affects the tasks previously
|
|
Toshihiro Shimizu |
890ddd |
added by that same Executor - which easily reflects the idea of an Executor representing
|
|
Toshihiro Shimizu |
890ddd |
a group of tasks.
|
|
Toshihiro Shimizu |
890ddd |
\n \n
|
|
Toshihiro Shimizu |
890ddd |
Basic control over the execution strategy for the group of tasks submitted by an Executor
|
|
Toshihiro Shimizu |
890ddd |
can be acquired using the setMaxActiveTasks() and setMaxActiveLoad() methods, both granting
|
|
Toshihiro Shimizu |
890ddd |
the possibility to bound the execution of tasks to custom maximum conditions.
|
|
Toshihiro Shimizu |
890ddd |
For example, use setMaxActiveTasks(1) to force the execution of 1 task only at a time,
|
|
Toshihiro Shimizu |
890ddd |
or setMaxActiveLoad(100) to set a single CPU core available for the group.
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
\sa \b Runnable class documentation.
|
|
Toshihiro Shimizu |
890ddd |
*/
|
|
Toshihiro Shimizu |
890ddd |
class DVAPI Executor
|
|
Toshihiro Shimizu |
890ddd |
{
|
|
Toshihiro Shimizu |
890ddd |
ExecutorId *m_id;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
friend class ExecutorImp;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
public:
|
|
Toshihiro Shimizu |
890ddd |
Executor();
|
|
Toshihiro Shimizu |
890ddd |
~Executor();
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
static void init();
|
|
Toshihiro Shimizu |
890ddd |
static void shutdown();
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
void addTask(RunnableP task);
|
|
Toshihiro Shimizu |
890ddd |
void removeTask(RunnableP task);
|
|
Toshihiro Shimizu |
890ddd |
void cancelAll();
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
void setMaxActiveTasks(int count);
|
|
Toshihiro Shimizu |
890ddd |
void setMaxActiveLoad(int load);
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
int maxActiveTasks() const;
|
|
Toshihiro Shimizu |
890ddd |
int maxActiveLoad() const;
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
void setDedicatedThreads(bool dedicated, bool persistent = true);
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
private:
|
|
Toshihiro Shimizu |
890ddd |
// not implemented
|
|
Toshihiro Shimizu |
890ddd |
Executor &operator=(const Executor &);
|
|
Toshihiro Shimizu |
890ddd |
Executor(const Executor &);
|
|
Toshihiro Shimizu |
890ddd |
};
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
} // namespace TThread
|
|
Toshihiro Shimizu |
890ddd |
|
|
Toshihiro Shimizu |
890ddd |
#endif //TTHREAD_H
|