/**************************************************************************** ** $Id$ ** ** QThread class for Unix ** ** Created : 20000913 ** ** Copyright (C) 1992-2000 Trolltech AS. All rights reserved. ** ** This file is part of the kernel module of the Qt GUI Toolkit. ** ** This file may be distributed under the terms of the Q Public License ** as defined by Trolltech AS of Norway and appearing in the file ** LICENSE.QPL included in the packaging of this file. ** ** This file may be distributed and/or modified under the terms of the ** GNU General Public License version 2 as published by the Free Software ** Foundation and appearing in the file LICENSE.GPL included in the ** packaging of this file. ** ** Licensees holding valid Qt Enterprise Edition or Qt Professional Edition ** licenses for Unix/X11 or for Qt/Embedded 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 ** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. ** ** See http://www.trolltech.com/pricing.html or email sales@trolltech.com for ** information about Qt Commercial License Agreements. ** See http://www.trolltech.com/qpl/ for QPL licensing information. ** See http://www.trolltech.com/gpl/ for GPL licensing information. ** ** Contact info@trolltech.com if any conditions of this licensing are ** not clear to you. ** **********************************************************************/ #if defined(QT_THREAD_SUPPORT) #include "qplatformdefs.h" // Solaris redefines connect -> __xnet_connect with _XOPEN_SOURCE_EXTENDED. #if defined(connect) #undef connect #endif #include "qthread.h" #include "qmutex.h" #include "qwaitcondition.h" #include "qptrlist.h" #ifndef QT_H #if defined( QWS ) || defined( Q_WS_MACX ) #include "qptrdict.h" #else #include "qintdict.h" #endif #endif // QT_H #include static QMutex *dictMutex = 0; #if defined( QWS ) || defined( Q_WS_MACX ) static QPtrDict *thrDict = 0; #else static QIntDict *thrDict = 0; #endif extern "C" { static void *start_thread(void *t); } #ifdef Q_OS_SOLARIS // Solaris typedef thread_t Q_THREAD_T; // helpers #define Q_THREAD_SELF() thr_self() #define Q_THREAD_EXIT(a) thr_exit((a)) #define Q_THREAD_CREATE(a) (a) = thr_create(0, 0, start_thread, that, THR_DETACHED, \ &thread_id); #else // !Q_OS_SOLARIS // Pthreads typedef pthread_t Q_THREAD_T; // helpers #define Q_THREAD_SELF() pthread_self() #define Q_THREAD_EXIT(a) pthread_exit((a)) #define Q_THREAD_CREATE(a) pthread_attr_t attr; \ pthread_attr_init(&attr); \ pthread_attr_setinheritsched(&attr, PTHREAD_INHERIT_SCHED); \ pthread_attr_setdetachstate(&attr, PTHREAD_CREATE_DETACHED); \ (a) = pthread_create(&thread_id, &attr, start_thread, \ (void *) that); \ pthread_attr_destroy(&attr); #endif // Q_OS_SOLARIS class QThreadPrivate { public: Q_THREAD_T thread_id; QWaitCondition thread_done; // Used for QThread::wait() bool finished, running; QThreadPrivate() : thread_id(0), finished(FALSE), running(FALSE) { if (! dictMutex) dictMutex = new QMutex; if (! thrDict) { #if defined( QWS ) || defined( Q_WS_MACX ) thrDict = new QPtrDict; #else thrDict = new QIntDict; #endif } } ~QThreadPrivate() { dictMutex->lock(); if (thread_id) thrDict->remove((Qt::HANDLE) thread_id); dictMutex->unlock(); thread_id = 0; } void init(QThread *that) { that->d->running = TRUE; that->d->finished = FALSE; int ret; Q_THREAD_CREATE(ret); #ifdef QT_CHECK_RANGE if (ret) qWarning("QThread::start: thread creation error: %s", strerror(ret)); #endif } static void internalRun(QThread *that) { dictMutex->lock(); thrDict->insert(QThread::currentThread(), that); dictMutex->unlock(); that->run(); dictMutex->lock(); QThread *there = thrDict->find(QThread::currentThread()); if (there) { there->d->running = FALSE; there->d->finished = TRUE; there->d->thread_done.wakeAll(); } thrDict->remove(QThread::currentThread()); dictMutex->unlock(); } }; extern "C" { static void *start_thread(void *t) { QThreadPrivate::internalRun( (QThread *) t ); return 0; } } /************************************************************************** ** QThread *************************************************************************/ /*! \class QThread qthread.h \brief The QThread class provides platform-independent threads. \ingroup thread \ingroup environment A QThread represents a separate thread of control within the program; it shares data with all the other threads within the process but executes independently in the way that a separate program does on a multitasking operating system. Instead of starting in main(), QThreads begin executing in run(). You inherit run() to include your code. For example: \code class MyThread : public QThread { public: virtual void run(); }; void MyThread::run() { for( int count = 0; count < 20; count++ ) { sleep( 1 ); qDebug( "Ping!" ); } } int main() { MyThread a; MyThread b; a.start(); b.start(); a.wait(); b.wait(); } \endcode This will start two threads, each of which writes Ping! 20 times to the screen and exits. The wait() calls at the end of main() are necessary because exiting main() ends the program, unceremoniously killing all other threads. Each MyThread stops executing when it reaches the end of MyThread::run(), just as an application does when it leaves main(). \sa \link threads.html Thread Support in Qt\endlink. */ /*! This returns the thread handle of the currently executing thread. \warning The handle returned by this function is used for internal purposes and should \e not be used in any application code. On Windows, the returned value is a pseudo handle for the current thread, and it cannot be used for numerical comparison. */ Qt::HANDLE QThread::currentThread() { return (HANDLE) Q_THREAD_SELF(); } /*! \internal Initializes the QThread system. */ void QThread::initialize() { } /*! \internal Cleans up the QThread system. */ void QThread::cleanup() { } // helper function to do thread sleeps, since usleep()/nanosleep() aren't reliable // enough (in terms of behavior and availability) static void thread_sleep(struct timespec *ti) { #ifdef Q_OS_SOLARIS mutex_t mtx; cond_t cnd; mutex_init(&mtx, 0, 0); cond_init(&cnd, 0, 0); mutex_lock(&mtx); (void) cond_timedwait(&cnd, &mtx, ti); mutex_unlock(&mtx); cond_destroy(&cnd); mutex_destroy(&mtx); #else // !Q_OS_SOLARIS pthread_mutex_t mtx; pthread_cond_t cnd; pthread_mutex_init(&mtx, 0); pthread_cond_init(&cnd, 0); pthread_mutex_lock(&mtx); (void) pthread_cond_timedwait(&cnd, &mtx, ti); pthread_mutex_unlock(&mtx); pthread_cond_destroy(&cnd); pthread_mutex_destroy(&mtx); #endif // Q_OS_SOLARIS } /*! System independent sleep. This causes the current thread to sleep for \a secs seconds. */ void QThread::sleep( unsigned long secs ) { struct timeval tv; gettimeofday(&tv, 0); struct timespec ti; ti.tv_sec = tv.tv_sec + secs; ti.tv_nsec = (tv.tv_usec * 1000); thread_sleep(&ti); } /*! System independent sleep. This causes the current thread to sleep for \a msecs milliseconds */ void QThread::msleep( unsigned long msecs ) { struct timeval tv; gettimeofday(&tv, 0); struct timespec ti; ti.tv_nsec = (tv.tv_usec * 1000) + (msecs % 1000) * 1000000; ti.tv_sec = tv.tv_sec + (msecs / 1000) + (ti.tv_nsec / 1000000000); ti.tv_nsec %= 1000000000; thread_sleep(&ti); } /*! System independent sleep. This causes the current thread to sleep for \a usecs microseconds */ void QThread::usleep( unsigned long usecs ) { struct timeval tv; gettimeofday(&tv, 0); struct timespec ti; ti.tv_nsec = (tv.tv_usec * 1000) + (usecs % 1000000) * 1000; ti.tv_sec = tv.tv_sec + (usecs / 1000000) + (ti.tv_nsec / 1000000000); ti.tv_nsec %= 1000000000; thread_sleep(&ti); } /*! Constructs a new thread. The thread does not begin executing until start() is called. */ QThread::QThread() { d = new QThreadPrivate; } /*! QThread destructor. Note that deleting a QThread object will not stop the execution of the thread it represents. Deleting a running QThread (i.e. finished() returns FALSE) will probably result in a program crash. You can wait() on a thread to make sure that it has finished. */ QThread::~QThread() { #ifndef QT_NO_DEBUG if( d->running && !d->finished ) { qWarning("QThread object destroyed while thread is still running."); } else #endif delete d; } /*! Ends the execution of the calling thread and wakes up any threads waiting for its termination. */ void QThread::exit() { dictMutex->lock(); QThread *there = thrDict->find(QThread::currentThread()); if (there) { there->d->running = FALSE; there->d->finished = TRUE; there->d->thread_done.wakeAll(); } dictMutex->unlock(); Q_THREAD_EXIT(0); } /*! This begins the execution of the thread by calling run(), which should be reimplemented in a QThread subclass to contain your code. If you try to start a thread that is already running, this call will wait until the thread has finished, and then restart the thread. */ void QThread::start() { if (d->running) { #ifdef QT_CHECK_RANGE qWarning("Attempt to start a thread already running"); #endif wait(); } d->init(this); } /*! This provides similar functionality to POSIX pthread_join. A thread calling this will block until either of these conditions is met: \list \i The thread associated with this QThread object has finished execution (i.e. when it returns from \l{run()}). This function will return TRUE if the thread has finished. It also returns TRUE if the thread has not been started yet. \i \a time milliseconds has elapsed. If \a time is ULONG_MAX (the default), then the wait will never timeout (the thread must return from \l{run()}). This function will return FALSE if the wait timed out. \endlist */ bool QThread::wait(unsigned long time) { if (d->finished || ! d->running) return TRUE; return d->thread_done.wait(time); } /*! Returns TRUE is the thread is finished; otherwise returns FALSE. */ bool QThread::finished() const { return d->finished; } /*! Returns TRUE if the thread is running; otherwise returns FALSE. */ bool QThread::running() const { return d->running; } /*! \fn void QThread::run() This method is pure virtual, and must be implemented in derived classes in order to do useful work. Returning from this method will end the execution of the thread. \sa wait() */ #endif // QT_THREAD_SUPPORT