summaryrefslogtreecommitdiffstats
path: root/doc/threads.doc
blob: dc1a61d49f4eb77e831df32814ed5f475d82e380 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
/****************************************************************************
**
**
** TQt thread support
**
** Copyright (C) 1992-2008 Trolltech ASA.  All rights reserved.
**
** This file is part of the TQt GUI Toolkit.
**
** This file may be used under the terms of the GNU General
** Public License versions 2.0 or 3.0 as published by the Free
** Software Foundation and appearing in the files LICENSE.GPL2
** and LICENSE.GPL3 included in the packaging of this file.
** Alternatively you may (at your option) use any later version
** of the GNU General Public License if such license has been
** publicly approved by Trolltech ASA (or its successors, if any)
** and the KDE Free TQt Foundation.
**
** Please review the following information to ensure GNU General
** Public Licensing requirements will be met:
** http://trolltech.com/products/qt/licenses/licensing/opensource/.
** If you are unsure which license is appropriate for your use, please
** review the following information:
** http://trolltech.com/products/qt/licenses/licensing/licensingoverview
** or contact the sales department at sales@trolltech.com.
**
** This file may be used under the terms of the Q Public License as
** defined by Trolltech ASA and appearing in the file LICENSE.QPL
** included in the packaging of this file.  Licensees holding valid Qt
** Commercial licenses may use this file in accordance with the Qt
** Commercial License Agreement provided with the Software.
**
** This file is provided "AS IS" with NO WARRANTY OF ANY KIND,
** INCLUDING THE WARRANTIES OF DESIGN, MERCHANTABILITY AND FITNESS FOR
** A PARTICULAR PURPOSE. Trolltech reserves all rights not granted
** herein.
**
**********************************************************************/

/*! \page threads.html

\title Thread Support in Qt

\tableofcontents

\section1 Introduction

Qt provides thread support in the form of basic platform-independent
threading classes, a thread-safe way of posting events, and a global
Qt library lock that allows you to call TQt methods from different
threads.

This document is intended for an audience that has knowledge of, and
experience with, multithreaded applications. If you are new to
threading see our \link #reading Recommended Reading\endlink list.


\section1 Enabling Thread Support

When TQt is installed on Windows, thread support is an option on some
compilers. 

On Mac OS X and Unix, thread support is enabled by adding the
\c{-thread} option when running the \c{configure} script.  On Unix
platforms where multithreaded programs must be linked in special ways,
such as with a special libc, installation will create a separate
library, \c{libtqt-mt} and hence threaded programs must be linked
against this library (with \c{-ltqt-mt}) rather than the standard Qt
library.

On both platforms, you should compile with the macro \c
TQT_THREAD_SUPPORT defined (e.g. compile with
\c{-DTQT_THREAD_SUPPORT}). On Windows, this is usually done by an
entry in \c{ntqconfig.h}.

\section1 The Thread Classes

These classes are built into the TQt library when thread support is enabled:

\list
\i TQThread - Provides the means to start a new thread, which begins
execution in your reimplementation of TQThread::run().  This is similar
to the Java thread class.

\i TQThreadStorage - Provides per-thread data storage. This class can
only be used with threads started with TQThread; it cannot be used with
threads started with platform-specific APIs.

\i TQMutex - Provides a mutual exclusion lock (also know as a mutex).
\i TQMutexLocker - A convenience class which automatically locks and
unlocks a TQMutex.  TQMutexLocker is useful in complicated code, or in
code which uses exceptions.  See the documentation for more details.
\i TQWaitCondition - Provides a way for threads to go to sleep until
woken up by another thread.
\i TQSemaphore - Provides a simple integer semaphore.
\endlist

\section1 Important Definitions

\target reentrant
\target threadsafe

When using TQt in a multithreaded program, it is important to
understand the definition of the terms \e reentrant and \e
thread-safe:

\list
\i \e reentrant - Describes a function which can be called
simultaneously by multiple threads when each invocation of the
function references unique data.  Calling a reentrant function
simultaneously with the same data is not safe, and such invocations
should be serialized.
\i \e thread-safe - Describes a function which can be called
simultaneously by multiple threads when each invocation references
shared data.  Calling a thread-safe function simultaneously with the
same data is safe, since all access to the shared data are serialized.
\endlist

Note that TQt provides both implictly and explicitly shared
classes. For more information, see the \link
threads.html#threads-shared Threads and Shared Data \endlink
section.

Most C++ member functions are inherently reentrant, since they only
reference class member data.  Any thread can call such a member
function on an instance, as long as no other thread is calling a
member function on the same instance.  For example, given the class \c
Number below:

\code
    class Number
    {
    public:
        inline Number( int n ) : num( n ) { }

        inline int number() const { return num; }
        inline void setNumber( int n ) { num = n; }

    private:
        int num;
    };
\endcode

The methods \c Number::number() and \c Number::setNumber() are
reentrant, since they only reference unique data.  Only one thread at
a time can call member functions on each instance of \c Number.
However, multiple threads can call member functions on separate
instances of \c Number.

Thread-safe functions usually use a mutex (e.g a TQMutex) to serialize
access to shared data.  Because of this, thread-safe functions are
usually slower than reentrant functions, because of the extra overhead
of locking and unlocking the mutex.  For example, given the class \c
Counter below:

\code
    class Counter
    {
    public:
        inline Counter()  { ++instances; }
        inline ~Counter() { --instances; }

    private:
        static int instances;
    };
\endcode

Since the modifications of the static \c instances integer are not
serialized, this class is not thread-safe.  So make it threadsafe, a
mutex must be used:

\code
    class Counter
    {
    public:
        inline Counter()
        {
            mutex.lock();
            ++instances;
            mutex.unlock();
        }

        ...
    private:
        static TQMutex mutex;
        static int instances;
    };
\endcode

\section1 Thread-safe Event Posting

In Qt, one thread is always the GUI or event thread.  This is the
thread that creates a QApplication object and calls
QApplication::exec().  This is also the initial thread that calls
main() at program start.  This thread is the only thread that is
allowed to perform GUI operations, including generating and receiving
events from the window system.  TQt does not support creating
QApplication and running the event loop (with QApplication::exec()) in
a secondary thread.  You must create the QApplication object and call
QApplication::exec() from the main() function in your program.

Threads that wish to display data in a widget cannot modify the widget
directly, so they must post an event to the widget using
QApplication::postEvent().  The event will be delivered later on by
the GUI thread.

Normally, the programmer would like to include some information in the
event sent to the widget.  See the documentation for TQCustomEvent for
more information on user-defined events.

\section1 Threads and TQObject subclasses

The TQObject class itself is \e reentrant.  However, certain rules
apply when creating and using TQObjects in a thread that is not the GUI
thread.

\list 1

\i \e None of the TQObject based classes included in the TQt library are
\e reentrant.  This includes all widgets (e.g. TQWidget and
subclasses), OS kernel classes (e.g. TQProcess, TQAccel, TQTimer), and
all networking classes (e.g. TQSocket, TQDns).

\i TQObject and all of its subclasses are \e not \e thread-safe.  This
includes the entire event delivery system.  It is important to
remember that the GUI thread may be delivering events to your TQObject
subclass while you are accessing the object from another thread.  If
you are using TQObject in a thread that is not the GUI thread, and you
are handling events sent to this object, you \e must protect all
access to your data with a mutex; otherwise you may experience crashes
or other undesired behavior.

\i As a corollary to the above, deleting a TQObject while pending
events are waiting to be delivered can cause a crash.  You must not
delete the TQObject directly from a thread that is not the GUI thread.
Use the TQObject::deleteLater() method instead, which will cause the
event loop to delete the object after all pending events have been
delivered to the object.

\endlist

\section1 The TQt Library Mutex

QApplication includes a mutex that is used to protect access to window
system functions.  This mutex is locked while the event loop is
running (e.g. during event delivery) and unlocked when the eventloop
goes to sleep.  Note: The TQt event loop is recursive, and the library
mutex is \e not unlocked when re-entering the event loop (e.g. when
executing a modal dialog with TQDialog::exec()).

If another thread locks the TQt library mutex, then the event loop will
stop processing events, and the locking thread may do simple GUI
operations.  Operations such as creating a TQPainter and drawing a line
are examples of simple GUI operations:

\code
    ...
    tqApp->lock();

    TQPainter p;
    p.begin( mywidget );
    p.setPen( TQColor( "red" ) );
    p.drawLine( 0,0,100,100 );
    p.end();

    tqApp->unlock();
    ...
\endcode

Any operations that generate events must not be called by any thread
other than the GUI thread.  Examples of such operations are:

\list
\i creating a TQWidget, TQTimer, TQSocketNotifier, TQSocket or other network class.
\i moving, resizing, showing or hiding a TQWidget.
\i starting or stoping a TQTimer.
\i enabling or disabling a TQSocketNotifier.
\i using a TQSocket or other network class.
\endlist

Events generated by these operations will be lost on some platforms.

\section1 Threads and Signals and Slots

The Signals and Slots mechanism can be used in separate threads, as
long as the rules for TQObject based classes are followed.  The Signals
and Slots mechanism is synchronous: when a signal is emitted, all
slots are called immediately.  The slots are executed in the thread
context that emitted the signal.

\warning Slots that generate window system events or use window system
functions \e must \e not be connected to a signal that is emitted from
a thread that is not the GUI thread.  See the TQt Library Mutex section
above for more details.

\target threads-shared
\section1 Threads and Shared Data

Qt provides many implicitly shared and explicitly shared classes.  In
a multithreaded program, multiple instances of a shared class can
reference shared data, which is dangerous if one or more threads
attempt to modify the data.  TQt provides the TQDeepCopy class, which
ensures that shared classes reference unique data.

See the description of \link shclass.html implicit sharing\endlink for more
information.

\target threads-sql
\section1 Threads and the SQL Module

A connection can only be used from within the thread that created it.
Moving connections between threads or creating queries from a different
thread is not supported.

In addition, the third party libraries used by the TQSqlDrivers can impose
further restrictions on using the SQL Module in a multithreaded program.
Consult the manual of your database client for more information.

\section1 Caveats

Some things to watch out for when programming with threads:

 \list 

\i As mentioned above, TQObject based classes are neither thread-safe
nor reentrant.  This includes all widgets (e.g. TQWidget and
subclasses), OS kernel classes (e.g. TQProcess, TQAccel), and all
networking classes (e.g. TQSocket, TQDns).

\i Deleting a TQObject while pending events are waiting to be delivered
will cause a crash.  If you are creating TQObjects in a thread that is
not the GUI thread and posting events to these objects, you should not
delete the TQObject directly.  Use the TQObject::deleteLater() method
instead, which will cause the event loop to delete the object after
all pending events have been delivered to the object.

\i Don't do any blocking operations while holding the TQt library
mutex.  This will freeze up the event loop.

\i Make sure you unlock a recursive TQMutex as many times as you lock
it, no more and no less.

\i Don't mix the normal TQt library and the threaded TQt library in your
application.  This means that if your application uses the threaded Qt
library, you should not link with the normal TQt library, dynamically
load the normal TQt library or dynamically load another library or
plugin that depends on the normal TQt library.  On some systems, doing
this can corrupt the static data used in the TQt library.

\i TQt does not support creating QApplication and running the event
loop (with QApplication::exec()) in a secondary thread.  You must
create the QApplication object and call QApplication::exec() from the
main() function in your program.

\endlist

\target reading
\section1 Recommended Reading

\list
\i \link http://www.amazon.com/exec/obidos/ASIN/0134436989/trolltech/t
Threads Primer: A Guide to Multithreaded Programming\endlink
\i \link http://www.amazon.com/exec/obidos/ASIN/0131900676/trolltech/t
Thread Time: The Multithreaded Programming Guide\endlink
\i \link http://www.amazon.com/exec/obidos/ASIN/1565921151/trolltech/t
Pthreads Programming: A POSIX Standard for Better Multiprocessing (O'Reilly Nutshell)\endlink
\i \link http://www.amazon.com/exec/obidos/ASIN/1565922964/trolltech/t
Win32 Multithreaded Programming\endlink
\endlist

*/