This only affects the !QT_CONFIG(thread) case and requires the rewrite of the old-style static cleanup() deleter protocol into the modern operator()() one. As a drive-by, mark the deleter noexcept, like destructors are supposed to be. Task-number: QTBUG-132213 Change-Id: I8839865880647d76b77eb9a3f2858067db86234e Reviewed-by: Thiago Macieira <thiago.macieira@intel.com> (cherry picked from commit 6da1f72311b844b2232da3067ad6e1e24614e67c) Reviewed-by: Qt Cherry-pick Bot <cherrypick_bot@qt-project.org> (cherry picked from commit 7aadf25ccd70fd739a7b06f6b6bccbdee03847a5)
195 lines
3.9 KiB
C++
195 lines
3.9 KiB
C++
// Copyright (C) 2016 The Qt Company Ltd.
|
|
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
|
|
|
|
#ifndef QTHREADSTORAGE_H
|
|
#define QTHREADSTORAGE_H
|
|
|
|
#include <QtCore/qglobal.h>
|
|
|
|
QT_BEGIN_NAMESPACE
|
|
|
|
#if QT_CONFIG(thread)
|
|
|
|
class Q_CORE_EXPORT QThreadStorageData
|
|
{
|
|
public:
|
|
explicit QThreadStorageData(void (*func)(void *));
|
|
~QThreadStorageData();
|
|
|
|
void** get() const;
|
|
void** set(void* p);
|
|
|
|
static void finish(void**);
|
|
int id;
|
|
};
|
|
|
|
#if !defined(QT_MOC_CPP)
|
|
// MOC_SKIP_BEGIN
|
|
|
|
// pointer specialization
|
|
template <typename T>
|
|
inline
|
|
T *&qThreadStorage_localData(QThreadStorageData &d, T **)
|
|
{
|
|
void **v = d.get();
|
|
if (!v) v = d.set(nullptr);
|
|
return *(reinterpret_cast<T**>(v));
|
|
}
|
|
|
|
template <typename T>
|
|
inline
|
|
T *qThreadStorage_localData_const(const QThreadStorageData &d, T **)
|
|
{
|
|
void **v = d.get();
|
|
return v ? *(reinterpret_cast<T**>(v)) : 0;
|
|
}
|
|
|
|
template <typename T>
|
|
inline
|
|
void qThreadStorage_setLocalData(QThreadStorageData &d, T **t)
|
|
{ (void) d.set(*t); }
|
|
|
|
template <typename T>
|
|
inline
|
|
void qThreadStorage_deleteData(void *d, T **)
|
|
{ delete static_cast<T *>(d); }
|
|
|
|
// value-based specialization
|
|
template <typename T>
|
|
inline
|
|
T &qThreadStorage_localData(QThreadStorageData &d, T *)
|
|
{
|
|
void **v = d.get();
|
|
if (!v) v = d.set(new T());
|
|
return *(reinterpret_cast<T*>(*v));
|
|
}
|
|
|
|
template <typename T>
|
|
inline
|
|
T qThreadStorage_localData_const(const QThreadStorageData &d, T *)
|
|
{
|
|
void **v = d.get();
|
|
return v ? *(reinterpret_cast<T*>(*v)) : T();
|
|
}
|
|
|
|
template <typename T>
|
|
inline
|
|
void qThreadStorage_setLocalData(QThreadStorageData &d, T *t)
|
|
{ (void) d.set(new T(*t)); }
|
|
|
|
template <typename T>
|
|
inline
|
|
void qThreadStorage_deleteData(void *d, T *)
|
|
{ delete static_cast<T *>(d); }
|
|
|
|
|
|
// MOC_SKIP_END
|
|
#endif
|
|
|
|
template <class T>
|
|
class QThreadStorage
|
|
{
|
|
private:
|
|
QThreadStorageData d;
|
|
|
|
Q_DISABLE_COPY(QThreadStorage)
|
|
|
|
static inline void deleteData(void *x)
|
|
{ qThreadStorage_deleteData(x, reinterpret_cast<T*>(0)); }
|
|
|
|
public:
|
|
inline QThreadStorage() : d(deleteData) { }
|
|
inline ~QThreadStorage() { }
|
|
|
|
inline bool hasLocalData() const
|
|
{ return d.get() != nullptr; }
|
|
|
|
inline T& localData()
|
|
{ return qThreadStorage_localData(d, reinterpret_cast<T*>(0)); }
|
|
inline T localData() const
|
|
{ return qThreadStorage_localData_const(d, reinterpret_cast<T*>(0)); }
|
|
|
|
inline void setLocalData(T t)
|
|
{ qThreadStorage_setLocalData(d, &t); }
|
|
};
|
|
|
|
#else // !QT_CONFIG(thread)
|
|
|
|
#include <memory>
|
|
#include <type_traits>
|
|
|
|
template <typename T, typename U>
|
|
inline bool qThreadStorage_hasLocalData(const std::unique_ptr<T, U> &data)
|
|
{
|
|
return !!data;
|
|
}
|
|
|
|
template <typename T, typename U>
|
|
inline bool qThreadStorage_hasLocalData(const std::unique_ptr<T*, U> &data)
|
|
{
|
|
return !!data ? *data != nullptr : false;
|
|
}
|
|
|
|
template <typename T>
|
|
inline void qThreadStorage_deleteLocalData(T *t)
|
|
{
|
|
delete t;
|
|
}
|
|
|
|
template <typename T>
|
|
inline void qThreadStorage_deleteLocalData(T **t)
|
|
{
|
|
delete *t;
|
|
delete t;
|
|
}
|
|
|
|
template <class T>
|
|
class QThreadStorage
|
|
{
|
|
private:
|
|
struct ScopedPointerThreadStorageDeleter
|
|
{
|
|
void operator()(T *t) const noexcept
|
|
{
|
|
if (t == nullptr)
|
|
return;
|
|
qThreadStorage_deleteLocalData(t);
|
|
}
|
|
};
|
|
std::unique_ptr<T, ScopedPointerThreadStorageDeleter> data;
|
|
|
|
public:
|
|
QThreadStorage() = default;
|
|
~QThreadStorage() = default;
|
|
QThreadStorage(const QThreadStorage &rhs) = delete;
|
|
QThreadStorage &operator=(const QThreadStorage &rhs) = delete;
|
|
|
|
inline bool hasLocalData() const
|
|
{
|
|
return qThreadStorage_hasLocalData(data);
|
|
}
|
|
|
|
inline T &localData()
|
|
{
|
|
if (!data)
|
|
data.reset(new T());
|
|
return *data;
|
|
}
|
|
|
|
inline T localData() const
|
|
{
|
|
return !!data ? *data : T();
|
|
}
|
|
|
|
inline void setLocalData(T t)
|
|
{
|
|
data.reset(new T(t));
|
|
}
|
|
};
|
|
|
|
#endif // QT_CONFIG(thread)
|
|
|
|
QT_END_NAMESPACE
|
|
|
|
#endif // QTHREADSTORAGE_H
|