/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the QtCore module of the Qt Toolkit. ** ** $QT_BEGIN_LICENSE:LGPL$ ** 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 Digia. For licensing terms and ** conditions see http://qt.digia.com/licensing. For further information ** use the contact form at http://qt.digia.com/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 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, Digia gives you certain additional ** rights. These rights are described in the Digia Qt LGPL Exception ** version 1.1, 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. ** ** ** $QT_END_LICENSE$ ** ****************************************************************************/ #ifndef QSCOPEDPOINTER_H #define QSCOPEDPOINTER_H #include #include QT_BEGIN_NAMESPACE template struct QScopedPointerDeleter { static inline void cleanup(T *pointer) { // Enforce a complete type. // If you get a compile error here, read the section on forward declared // classes in the QScopedPointer documentation. typedef char IsIncompleteType[ sizeof(T) ? 1 : -1 ]; (void) sizeof(IsIncompleteType); delete pointer; } }; template struct QScopedPointerArrayDeleter { static inline void cleanup(T *pointer) { // Enforce a complete type. // If you get a compile error here, read the section on forward declared // classes in the QScopedPointer documentation. typedef char IsIncompleteType[ sizeof(T) ? 1 : -1 ]; (void) sizeof(IsIncompleteType); delete [] pointer; } }; struct QScopedPointerPodDeleter { static inline void cleanup(void *pointer) { if (pointer) free(pointer); } }; template > class QScopedPointer { typedef T *QScopedPointer:: *RestrictedBool; public: explicit inline QScopedPointer(T *p = 0) : d(p) { } inline ~QScopedPointer() { T *oldD = this->d; Cleanup::cleanup(oldD); } inline T &operator*() const { Q_ASSERT(d); return *d; } inline T *operator->() const { Q_ASSERT(d); return d; } inline bool operator!() const { return !d; } #if defined(Q_QDOC) inline operator bool() const { return isNull() ? 0 : &QScopedPointer::d; } #else inline operator RestrictedBool() const { return isNull() ? 0 : &QScopedPointer::d; } #endif inline T *data() const { return d; } inline bool isNull() const { return !d; } inline void reset(T *other = 0) { if (d == other) return; T *oldD = d; d = other; Cleanup::cleanup(oldD); } inline T *take() { T *oldD = d; d = 0; return oldD; } inline void swap(QScopedPointer &other) { qSwap(d, other.d); } typedef T *pointer; protected: T *d; private: Q_DISABLE_COPY(QScopedPointer) }; template inline bool operator==(const QScopedPointer &lhs, const QScopedPointer &rhs) { return lhs.data() == rhs.data(); } template inline bool operator!=(const QScopedPointer &lhs, const QScopedPointer &rhs) { return lhs.data() != rhs.data(); } template Q_INLINE_TEMPLATE void qSwap(QScopedPointer &p1, QScopedPointer &p2) { p1.swap(p2); } QT_END_NAMESPACE namespace std { template Q_INLINE_TEMPLATE void swap(QT_PREPEND_NAMESPACE(QScopedPointer) &p1, QT_PREPEND_NAMESPACE(QScopedPointer) &p2) { p1.swap(p2); } } QT_BEGIN_NAMESPACE namespace QtPrivate { template struct QScopedArrayEnsureSameType; template struct QScopedArrayEnsureSameType { typedef X* Type; }; template struct QScopedArrayEnsureSameType { typedef X* Type; }; } template > class QScopedArrayPointer : public QScopedPointer { public: inline QScopedArrayPointer() : QScopedPointer(0) {} template explicit inline QScopedArrayPointer(D *p, typename QtPrivate::QScopedArrayEnsureSameType::Type = 0) : QScopedPointer(p) { } inline T &operator[](int i) { return this->d[i]; } inline const T &operator[](int i) const { return this->d[i]; } private: explicit inline QScopedArrayPointer(void *) { // Enforce the same type. // If you get a compile error here, make sure you declare // QScopedArrayPointer with the same template type as you pass to the // constructor. See also the QScopedPointer documentation. // Storing a scalar array as a pointer to a different type is not // allowed and results in undefined behavior. } Q_DISABLE_COPY(QScopedArrayPointer) }; QT_END_NAMESPACE #endif // QSCOPEDPOINTER_H