diff options
author | Lars Knoll <lars.knoll@nokia.com> | 2009-03-23 09:18:55 (GMT) |
---|---|---|
committer | Simon Hausmann <simon.hausmann@nokia.com> | 2009-03-23 09:18:55 (GMT) |
commit | e5fcad302d86d316390c6b0f62759a067313e8a9 (patch) | |
tree | c2afbf6f1066b6ce261f14341cf6d310e5595bc1 /src/corelib/thread/qthreadstorage.cpp | |
download | Qt-e5fcad302d86d316390c6b0f62759a067313e8a9.zip Qt-e5fcad302d86d316390c6b0f62759a067313e8a9.tar.gz Qt-e5fcad302d86d316390c6b0f62759a067313e8a9.tar.bz2 |
Long live Qt 4.5!
Diffstat (limited to 'src/corelib/thread/qthreadstorage.cpp')
-rw-r--r-- | src/corelib/thread/qthreadstorage.cpp | 320 |
1 files changed, 320 insertions, 0 deletions
diff --git a/src/corelib/thread/qthreadstorage.cpp b/src/corelib/thread/qthreadstorage.cpp new file mode 100644 index 0000000..35c55c1 --- /dev/null +++ b/src/corelib/thread/qthreadstorage.cpp @@ -0,0 +1,320 @@ +/**************************************************************************** +** +** Copyright (C) 2009 Nokia Corporation and/or its subsidiary(-ies). +** Contact: Qt Software Information (qt-info@nokia.com) +** +** This file is part of the QtCore module of the Qt Toolkit. +** +** $QT_BEGIN_LICENSE:LGPL$ +** No Commercial Usage +** This file contains pre-release code and may not be distributed. +** You may use this file in accordance with the terms and conditions +** contained in the either Technology Preview License Agreement or the +** Beta Release License Agreement. +** +** GNU Lesser General Public License Usage +** Alternatively, this file may be used under the terms of the GNU Lesser +** General Public License version 2.1 as published by the Free Software +** Foundation and appearing in the file LICENSE.LGPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU Lesser General Public License version 2.1 requirements +** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. +** +** In addition, as a special exception, Nokia gives you certain +** additional rights. These rights are described in the Nokia Qt LGPL +** Exception version 1.0, included in the file LGPL_EXCEPTION.txt in this +** package. +** +** GNU General Public License Usage +** Alternatively, this file may be used under the terms of the GNU +** General Public License version 3.0 as published by the Free Software +** Foundation and appearing in the file LICENSE.GPL included in the +** packaging of this file. Please review the following information to +** ensure the GNU General Public License version 3.0 requirements will be +** met: http://www.gnu.org/copyleft/gpl.html. +** +** If you are unsure which license is appropriate for your use, please +** contact the sales department at qt-sales@nokia.com. +** $QT_END_LICENSE$ +** +****************************************************************************/ + +#include "qthreadstorage.h" + +#ifndef QT_NO_THREAD +#include "qthread.h" +#include "qthread_p.h" +#include "qmutex.h" + +#include <string.h> + +QT_BEGIN_NAMESPACE + +// #define THREADSTORAGE_DEBUG +#ifdef THREADSTORAGE_DEBUG +# define DEBUG_MSG qtsDebug + +# include <stdio.h> +# include <stdarg.h> +void qtsDebug(const char *fmt, ...) +{ + va_list va; + va_start(va, fmt); + + fprintf(stderr, "QThreadStorage: "); + vfprintf(stderr, fmt, va); + fprintf(stderr, "\n"); + + va_end(va); +} +#else +# define DEBUG_MSG if(false)qDebug +#endif + +static QBasicAtomicInt idCounter = Q_BASIC_ATOMIC_INITIALIZER(INT_MAX); +Q_GLOBAL_STATIC(QMutex, mutex) +typedef QMap<int, void (*)(void *)> DestructorMap; +Q_GLOBAL_STATIC(DestructorMap, destructors) + +QThreadStorageData::QThreadStorageData(void (*func)(void *)) + : id(idCounter.fetchAndAddRelaxed(-1)) +{ + QMutexLocker locker(mutex()); + destructors()->insert(id, func); + + DEBUG_MSG("QThreadStorageData: Allocated id %d, destructor %p", id, func); +} + +QThreadStorageData::~QThreadStorageData() +{ + QMutexLocker locker(mutex()); + if (destructors()) + destructors()->remove(id); + + DEBUG_MSG("QThreadStorageData: Released id %d", id); +} + +void **QThreadStorageData::get() const +{ + QThreadData *data = QThreadData::current(); + if (!data) { + qWarning("QThreadStorage::get: QThreadStorage can only be used with threads started with QThread"); + return 0; + } + QMap<int, void *>::iterator it = data->tls.find(id); + DEBUG_MSG("QThreadStorageData: Returning storage %d, data %p, for thread %p", + id, + it != data->tls.end() ? it.value() : 0, + data->thread); + return it != data->tls.end() && it.value() != 0 ? &it.value() : 0; +} + +void **QThreadStorageData::set(void *p) +{ + QThreadData *data = QThreadData::current(); + if (!data) { + qWarning("QThreadStorage::set: QThreadStorage can only be used with threads started with QThread"); + return 0; + } + + QMap<int, void *>::iterator it = data->tls.find(id); + if (it != data->tls.end()) { + // delete any previous data + if (it.value() != 0) { + DEBUG_MSG("QThreadStorageData: Deleting previous storage %d, data %p, for thread %p", + id, + it.value(), + data->thread); + + void *q = it.value(); + it.value() = 0; + + mutex()->lock(); + void (*destructor)(void *) = destructors()->value(id); + mutex()->unlock(); + + destructor(q); + } + + // store new data + it.value() = p; + DEBUG_MSG("QThreadStorageData: Set storage %d for thread %p to %p", id, data->thread, p); + } else { + it = data->tls.insert(id, p); + DEBUG_MSG("QThreadStorageData: Inserted storage %d, data %p, for thread %p", id, p, data->thread); + } + + return &it.value(); +} + +void QThreadStorageData::finish(void **p) +{ + QMap<int, void *> *tls = reinterpret_cast<QMap<int, void *> *>(p); + if (!tls || tls->isEmpty() || !mutex()) + return; // nothing to do + + DEBUG_MSG("QThreadStorageData: Destroying storage for thread %p", QThread::currentThread()); + + QMap<int, void *>::iterator it = tls->begin(); + while (it != tls->end()) { + int id = it.key(); + void *q = it.value(); + it.value() = 0; + ++it; + + if (!q) { + // data already deleted + continue; + } + + mutex()->lock(); + void (*destructor)(void *) = destructors()->value(id); + mutex()->unlock(); + + if (!destructor) { + if (QThread::currentThread()) + qWarning("QThreadStorage: Thread %p exited after QThreadStorage %d destroyed", + QThread::currentThread(), id); + continue; + } + destructor(q); + } + tls->clear(); +} + +/*! + \class QThreadStorage + \brief The QThreadStorage class provides per-thread data storage. + + \threadsafe + + \ingroup thread + \ingroup environment + \mainclass + + QThreadStorage is a template class that provides per-thread data + storage. + + \e{Note that due to compiler limitations, QThreadStorage can only + store pointers.} + + The setLocalData() function stores a single thread-specific value + for the calling thread. The data can be accessed later using + localData(). QThreadStorage takes ownership of the data (which + must be created on the heap with \c new) and deletes it when the + thread exits, either normally or via termination. + + The hasLocalData() function allows the programmer to determine if + data has previously been set using the setLocalData() function. + This is also useful for lazy initializiation. + + For example, the following code uses QThreadStorage to store a + single cache for each thread that calls the cacheObject() and + removeFromCache() functions. The cache is automatically + deleted when the calling thread exits. + + \snippet doc/src/snippets/threads/threads.cpp 7 + \snippet doc/src/snippets/threads/threads.cpp 8 + \snippet doc/src/snippets/threads/threads.cpp 9 + + \section1 Caveats + + \list + + \o As noted above, QThreadStorage can only store pointers due to + compiler limitations. + + \o The QThreadStorage destructor does not delete per-thread data. + QThreadStorage only deletes per-thread data when the thread exits + or when setLocalData() is called multiple times. + + \o QThreadStorage can be used to store data for the \c main() + thread. QThreadStorage deletes all data set for the \c main() + thread when QApplication is destroyed, regardless of whether or + not the \c main() thread has actually finished. + + \endlist + + \sa QThread +*/ + +/*! + \fn QThreadStorage::QThreadStorage() + + Constructs a new per-thread data storage object. +*/ + +/*! + \fn QThreadStorage::~QThreadStorage() + + Destroys the per-thread data storage object. + + Note: The per-thread data stored is not deleted. Any data left + in QThreadStorage is leaked. Make sure that all threads using + QThreadStorage have exited before deleting the QThreadStorage. + + \sa hasLocalData() +*/ + +/*! + \fn bool QThreadStorage::hasLocalData() const + + Returns true if the calling thread has non-zero data available; + otherwise returns false. + + \sa localData() +*/ + +/*! + \fn T &QThreadStorage::localData() + + Returns a reference to the data that was set by the calling + thread. + + Note: QThreadStorage can only store pointers. This function + returns a reference to the pointer that was set by the calling + thread. The value of this reference is 0 if no data was set by + the calling thread, + + \sa hasLocalData() +*/ + +/*! + \fn const T QThreadStorage::localData() const + \overload + + Returns a copy of the data that was set by the calling thread. + + Note: QThreadStorage can only store pointers. This function + returns a pointer to the data that was set by the calling thread. + If no data was set by the calling thread, this function returns 0. + + \sa hasLocalData() +*/ + +/*! + \fn void QThreadStorage::setLocalData(T data) + + Sets the local data for the calling thread to \a data. It can be + accessed later using the localData() functions. + + If \a data is 0, this function deletes the previous data (if + any) and returns immediately. + + If \a data is non-zero, QThreadStorage takes ownership of the \a + data and deletes it automatically either when the thread exits + (either normally or via termination) or when setLocalData() is + called again. + + Note: QThreadStorage can only store pointers. The \a data + argument must be either a pointer to an object created on the heap + (i.e. using \c new) or 0. You should not delete \a data + yourself; QThreadStorage takes ownership and will delete the \a + data itself. + + \sa localData(), hasLocalData() +*/ + +#endif // QT_NO_THREAD + +QT_END_NAMESPACE |