| /**************************************************************************** |
| ** |
| ** Copyright (C) 2015 The Qt Company Ltd. |
| ** Contact: http://www.qt.io/licensing/ |
| ** |
| ** This file is part of the QtCore module of the Qt Toolkit. |
| ** |
| ** $QT_BEGIN_LICENSE:LGPL21$ |
| ** Commercial License Usage |
| ** Licensees holding valid commercial Qt licenses may use this file in |
| ** accordance with the commercial license agreement provided with the |
| ** Software or, alternatively, in accordance with the terms contained in |
| ** a written agreement between you and The Qt Company. For licensing terms |
| ** and conditions see http://www.qt.io/terms-conditions. For further |
| ** information use the contact form at http://www.qt.io/contact-us. |
| ** |
| ** 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 or version 3 as published by the Free |
| ** Software Foundation and appearing in the file LICENSE.LGPLv21 and |
| ** LICENSE.LGPLv3 included in the packaging of this file. Please review the |
| ** following information to ensure the GNU Lesser General Public License |
| ** requirements will be met: https://www.gnu.org/licenses/lgpl.html and |
| ** http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html. |
| ** |
| ** As a special exception, The Qt Company gives you certain additional |
| ** rights. These rights are described in The Qt Company LGPL Exception |
| ** version 1.1, included in the file LGPL_EXCEPTION.txt in this package. |
| ** |
| ** $QT_END_LICENSE$ |
| ** |
| ****************************************************************************/ |
| |
| #ifndef QSHAREDDATA_H |
| #define QSHAREDDATA_H |
| |
| #include <QtCore/qglobal.h> |
| #include <QtCore/qatomic.h> |
| #include <QtCore/qhash.h> |
| |
| QT_BEGIN_NAMESPACE |
| |
| |
| template <class T> class QSharedDataPointer; |
| |
| class Q_CORE_EXPORT QSharedData |
| { |
| public: |
| mutable QAtomicInt ref; |
| |
| inline QSharedData() : ref(0) { } |
| inline QSharedData(const QSharedData &) : ref(0) { } |
| |
| private: |
| // using the assignment operator would lead to corruption in the ref-counting |
| QSharedData &operator=(const QSharedData &); |
| }; |
| |
| template <class T> class QSharedDataPointer |
| { |
| public: |
| typedef T Type; |
| typedef T *pointer; |
| |
| inline void detach() { if (d && d->ref.load() != 1) detach_helper(); } |
| inline T &operator*() { detach(); return *d; } |
| inline const T &operator*() const { return *d; } |
| inline T *operator->() { detach(); return d; } |
| inline const T *operator->() const { return d; } |
| inline operator T *() { detach(); return d; } |
| inline operator const T *() const { return d; } |
| inline T *data() { detach(); return d; } |
| inline const T *data() const { return d; } |
| inline const T *constData() const { return d; } |
| |
| inline bool operator==(const QSharedDataPointer<T> &other) const { return d == other.d; } |
| inline bool operator!=(const QSharedDataPointer<T> &other) const { return d != other.d; } |
| |
| inline QSharedDataPointer() { d = 0; } |
| inline ~QSharedDataPointer() { if (d && !d->ref.deref()) delete d; } |
| |
| explicit QSharedDataPointer(T *data) Q_DECL_NOTHROW; |
| inline QSharedDataPointer(const QSharedDataPointer<T> &o) : d(o.d) { if (d) d->ref.ref(); } |
| inline QSharedDataPointer<T> & operator=(const QSharedDataPointer<T> &o) { |
| if (o.d != d) { |
| if (o.d) |
| o.d->ref.ref(); |
| T *old = d; |
| d = o.d; |
| if (old && !old->ref.deref()) |
| delete old; |
| } |
| return *this; |
| } |
| inline QSharedDataPointer &operator=(T *o) { |
| if (o != d) { |
| if (o) |
| o->ref.ref(); |
| T *old = d; |
| d = o; |
| if (old && !old->ref.deref()) |
| delete old; |
| } |
| return *this; |
| } |
| #ifdef Q_COMPILER_RVALUE_REFS |
| QSharedDataPointer(QSharedDataPointer &&o) Q_DECL_NOTHROW : d(o.d) { o.d = Q_NULLPTR; } |
| inline QSharedDataPointer<T> &operator=(QSharedDataPointer<T> &&other) Q_DECL_NOTHROW |
| { qSwap(d, other.d); return *this; } |
| #endif |
| |
| inline bool operator!() const { return !d; } |
| |
| inline void swap(QSharedDataPointer &other) Q_DECL_NOTHROW |
| { qSwap(d, other.d); } |
| |
| protected: |
| T *clone(); |
| |
| private: |
| void detach_helper(); |
| |
| T *d; |
| }; |
| |
| template <class T> class QExplicitlySharedDataPointer |
| { |
| public: |
| typedef T Type; |
| typedef T *pointer; |
| |
| inline T &operator*() const { return *d; } |
| inline T *operator->() { return d; } |
| inline T *operator->() const { return d; } |
| inline T *data() const { return d; } |
| inline const T *constData() const { return d; } |
| |
| inline void detach() { if (d && d->ref.load() != 1) detach_helper(); } |
| |
| inline void reset() |
| { |
| if(d && !d->ref.deref()) |
| delete d; |
| |
| d = 0; |
| } |
| |
| inline operator bool () const { return d != 0; } |
| |
| inline bool operator==(const QExplicitlySharedDataPointer<T> &other) const { return d == other.d; } |
| inline bool operator!=(const QExplicitlySharedDataPointer<T> &other) const { return d != other.d; } |
| inline bool operator==(const T *ptr) const { return d == ptr; } |
| inline bool operator!=(const T *ptr) const { return d != ptr; } |
| |
| inline QExplicitlySharedDataPointer() { d = 0; } |
| inline ~QExplicitlySharedDataPointer() { if (d && !d->ref.deref()) delete d; } |
| |
| explicit QExplicitlySharedDataPointer(T *data) Q_DECL_NOTHROW; |
| inline QExplicitlySharedDataPointer(const QExplicitlySharedDataPointer<T> &o) : d(o.d) { if (d) d->ref.ref(); } |
| |
| template<class X> |
| inline QExplicitlySharedDataPointer(const QExplicitlySharedDataPointer<X> &o) |
| #ifdef QT_ENABLE_QEXPLICITLYSHAREDDATAPOINTER_STATICCAST |
| : d(static_cast<T *>(o.data())) |
| #else |
| : d(o.data()) |
| #endif |
| { |
| if(d) |
| d->ref.ref(); |
| } |
| |
| inline QExplicitlySharedDataPointer<T> & operator=(const QExplicitlySharedDataPointer<T> &o) { |
| if (o.d != d) { |
| if (o.d) |
| o.d->ref.ref(); |
| T *old = d; |
| d = o.d; |
| if (old && !old->ref.deref()) |
| delete old; |
| } |
| return *this; |
| } |
| inline QExplicitlySharedDataPointer &operator=(T *o) { |
| if (o != d) { |
| if (o) |
| o->ref.ref(); |
| T *old = d; |
| d = o; |
| if (old && !old->ref.deref()) |
| delete old; |
| } |
| return *this; |
| } |
| #ifdef Q_COMPILER_RVALUE_REFS |
| inline QExplicitlySharedDataPointer(QExplicitlySharedDataPointer &&o) Q_DECL_NOTHROW : d(o.d) { o.d = Q_NULLPTR; } |
| inline QExplicitlySharedDataPointer<T> &operator=(QExplicitlySharedDataPointer<T> &&other) Q_DECL_NOTHROW |
| { qSwap(d, other.d); return *this; } |
| #endif |
| |
| inline bool operator!() const { return !d; } |
| |
| inline void swap(QExplicitlySharedDataPointer &other) Q_DECL_NOTHROW |
| { qSwap(d, other.d); } |
| |
| protected: |
| T *clone(); |
| |
| private: |
| void detach_helper(); |
| |
| T *d; |
| }; |
| |
| template <class T> |
| Q_INLINE_TEMPLATE QSharedDataPointer<T>::QSharedDataPointer(T *adata) Q_DECL_NOTHROW |
| : d(adata) |
| { if (d) d->ref.ref(); } |
| |
| template <class T> |
| Q_INLINE_TEMPLATE T *QSharedDataPointer<T>::clone() |
| { |
| return new T(*d); |
| } |
| |
| template <class T> |
| Q_OUTOFLINE_TEMPLATE void QSharedDataPointer<T>::detach_helper() |
| { |
| T *x = clone(); |
| x->ref.ref(); |
| if (!d->ref.deref()) |
| delete d; |
| d = x; |
| } |
| |
| template <class T> |
| Q_INLINE_TEMPLATE T *QExplicitlySharedDataPointer<T>::clone() |
| { |
| return new T(*d); |
| } |
| |
| template <class T> |
| Q_OUTOFLINE_TEMPLATE void QExplicitlySharedDataPointer<T>::detach_helper() |
| { |
| T *x = clone(); |
| x->ref.ref(); |
| if (!d->ref.deref()) |
| delete d; |
| d = x; |
| } |
| |
| template <class T> |
| Q_INLINE_TEMPLATE QExplicitlySharedDataPointer<T>::QExplicitlySharedDataPointer(T *adata) Q_DECL_NOTHROW |
| : d(adata) |
| { if (d) d->ref.ref(); } |
| |
| template <class T> |
| Q_INLINE_TEMPLATE void qSwap(QSharedDataPointer<T> &p1, QSharedDataPointer<T> &p2) |
| { p1.swap(p2); } |
| |
| template <class T> |
| Q_INLINE_TEMPLATE void qSwap(QExplicitlySharedDataPointer<T> &p1, QExplicitlySharedDataPointer<T> &p2) |
| { p1.swap(p2); } |
| |
| QT_END_NAMESPACE |
| namespace std { |
| template <class T> |
| Q_INLINE_TEMPLATE void swap(QT_PREPEND_NAMESPACE(QSharedDataPointer)<T> &p1, QT_PREPEND_NAMESPACE(QSharedDataPointer)<T> &p2) |
| { p1.swap(p2); } |
| |
| template <class T> |
| Q_INLINE_TEMPLATE void swap(QT_PREPEND_NAMESPACE(QExplicitlySharedDataPointer)<T> &p1, QT_PREPEND_NAMESPACE(QExplicitlySharedDataPointer)<T> &p2) |
| { p1.swap(p2); } |
| } |
| QT_BEGIN_NAMESPACE |
| |
| template <class T> |
| Q_INLINE_TEMPLATE uint qHash(const QSharedDataPointer<T> &ptr, uint seed = 0) Q_DECL_NOTHROW |
| { |
| return qHash(ptr.data(), seed); |
| } |
| template <class T> |
| Q_INLINE_TEMPLATE uint qHash(const QExplicitlySharedDataPointer<T> &ptr, uint seed = 0) Q_DECL_NOTHROW |
| { |
| return qHash(ptr.data(), seed); |
| } |
| |
| template<typename T> Q_DECLARE_TYPEINFO_BODY(QSharedDataPointer<T>, Q_MOVABLE_TYPE); |
| template<typename T> Q_DECLARE_TYPEINFO_BODY(QExplicitlySharedDataPointer<T>, Q_MOVABLE_TYPE); |
| |
| QT_END_NAMESPACE |
| |
| #endif // QSHAREDDATA_H |