/**************************************************************************** ** ** Copyright (C) 2013 Digia Plc and/or its subsidiary(-ies). ** Contact: http://www.qt-project.org/legal ** ** This file is part of the test suite 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$ ** ****************************************************************************/ #include #include #ifndef QT_NO_WIDGETS #include #endif class tst_QPointer : public QObject { Q_OBJECT public: inline tst_QPointer *me() const { return const_cast(this); } private slots: void constructors(); void destructor(); void assignment_operators(); void equality_operators(); void isNull(); void dereference_operators(); void disconnect(); void castDuringDestruction(); void threadSafety(); void qvariantCast(); void constPointer(); }; void tst_QPointer::constructors() { QPointer p1; QPointer p2(this); QPointer p3(p2); QCOMPARE(p1, QPointer(0)); QCOMPARE(p2, QPointer(this)); QCOMPARE(p3, QPointer(this)); } void tst_QPointer::destructor() { // Make two QPointer's to the same object QObject *object = new QObject; QPointer p1 = object; QPointer p2 = object; // Check that they point to the correct object QCOMPARE(p1, QPointer(object)); QCOMPARE(p2, QPointer(object)); QCOMPARE(p1, p2); // Destroy the guarded object delete object; // Check that both pointers were zeroed QCOMPARE(p1, QPointer(0)); QCOMPARE(p2, QPointer(0)); QCOMPARE(p1, p2); } void tst_QPointer::assignment_operators() { QPointer p1; QPointer p2; // Test assignment with a QObject-derived object pointer p1 = this; p2 = p1; QCOMPARE(p1, QPointer(this)); QCOMPARE(p2, QPointer(this)); QCOMPARE(p1, QPointer(p2)); // Test assignment with a null pointer p1 = 0; p2 = p1; QCOMPARE(p1, QPointer(0)); QCOMPARE(p2, QPointer(0)); QCOMPARE(p1, QPointer(p2)); // Test assignment with a real QObject pointer QObject *object = new QObject; p1 = object; p2 = p1; QCOMPARE(p1, QPointer(object)); QCOMPARE(p2, QPointer(object)); QCOMPARE(p1, QPointer(p2)); // Test assignment with the same pointer that's already guarded p1 = object; p2 = p1; QCOMPARE(p1, QPointer(object)); QCOMPARE(p2, QPointer(object)); QCOMPARE(p1, QPointer(p2)); // Cleanup delete object; } void tst_QPointer::equality_operators() { QPointer p1; QPointer p2; QVERIFY(p1 == p2); QObject *object = 0; #ifndef QT_NO_WIDGETS QWidget *widget = 0; #endif p1 = object; QVERIFY(p1 == p2); QVERIFY(p1 == object); p2 = object; QVERIFY(p2 == p1); QVERIFY(p2 == object); p1 = this; QVERIFY(p1 != p2); p2 = p1; QVERIFY(p1 == p2); // compare to zero p1 = 0; QVERIFY(p1 == 0); QVERIFY(0 == p1); QVERIFY(p2 != 0); QVERIFY(0 != p2); QVERIFY(p1 == object); QVERIFY(object == p1); QVERIFY(p2 != object); QVERIFY(object != p2); #ifndef QT_NO_WIDGETS QVERIFY(p1 == widget); QVERIFY(widget == p1); QVERIFY(p2 != widget); QVERIFY(widget != p2); #endif } void tst_QPointer::isNull() { QPointer p1; QVERIFY(p1.isNull()); p1 = this; QVERIFY(!p1.isNull()); p1 = 0; QVERIFY(p1.isNull()); } void tst_QPointer::dereference_operators() { QPointer p1 = this; QPointer p2; // operator->() -- only makes sense if not null QObject *object = p1->me(); QCOMPARE(object, this); // operator*() -- only makes sense if not null QObject &ref = *p1; QCOMPARE(&ref, this); // operator T*() QCOMPARE(static_cast(p1), this); QCOMPARE(static_cast(p2), static_cast(0)); // data() QCOMPARE(p1.data(), this); QCOMPARE(p2.data(), static_cast(0)); } void tst_QPointer::disconnect() { // Verify that pointer remains guarded when all signals are disconencted. QPointer p1 = new QObject; QVERIFY(!p1.isNull()); p1->disconnect(); QVERIFY(!p1.isNull()); delete static_cast(p1); QVERIFY(p1.isNull()); } class ChildObject : public QObject { QPointer guardedPointer; public: ChildObject(QObject *parent) : QObject(parent), guardedPointer(parent) { } ~ChildObject(); }; ChildObject::~ChildObject() { QCOMPARE(static_cast(guardedPointer), static_cast(0)); QCOMPARE(qobject_cast(guardedPointer), static_cast(0)); } #ifndef QT_NO_WIDGETS class ChildWidget : public QWidget { QPointer guardedPointer; public: ChildWidget(QWidget *parent) : QWidget(parent), guardedPointer(parent) { } ~ChildWidget(); }; ChildWidget::~ChildWidget() { QCOMPARE(static_cast(guardedPointer), parentWidget()); QCOMPARE(qobject_cast(guardedPointer), parentWidget()); } #endif class DerivedChild; class DerivedParent : public QObject { Q_OBJECT DerivedChild *derivedChild; public: DerivedParent(); ~DerivedParent(); }; class DerivedChild : public QObject { Q_OBJECT DerivedParent *parentPointer; QPointer guardedParentPointer; public: DerivedChild(DerivedParent *parent) : QObject(parent), parentPointer(parent), guardedParentPointer(parent) { } ~DerivedChild(); }; DerivedParent::DerivedParent() : QObject() { derivedChild = new DerivedChild(this); } DerivedParent::~DerivedParent() { delete derivedChild; } DerivedChild::~DerivedChild() { QCOMPARE(static_cast(guardedParentPointer), parentPointer); QCOMPARE(qobject_cast(guardedParentPointer), parentPointer); } void tst_QPointer::castDuringDestruction() { { QObject *parentObject = new QObject(); (void) new ChildObject(parentObject); delete parentObject; } #ifndef QT_NO_WIDGETS { QWidget *parentWidget = new QWidget(); (void) new ChildWidget(parentWidget); delete parentWidget; } #endif { delete new DerivedParent(); } } class TestRunnable : public QObject, public QRunnable { void run() { QPointer obj1 = new QObject; QPointer obj2 = new QObject; obj1->moveToThread(thread()); // this is the owner thread obj1->deleteLater(); // the delete will happen in the owner thread obj2->moveToThread(thread()); // this is the owner thread obj2->deleteLater(); // the delete will happen in the owner thread } }; void tst_QPointer::threadSafety() { QThread owner; owner.start(); QThreadPool pool; for (int i = 0; i < 300; i++) { QPointer task = new TestRunnable; task->setAutoDelete(true); task->moveToThread(&owner); pool.start(task); } pool.waitForDone(); owner.quit(); owner.wait(); } void tst_QPointer::qvariantCast() { QPointer tracking = new QFile; tracking->setObjectName("A test name"); QVariant v = QVariant::fromValue(tracking); { QPointer other = qPointerFromVariant(v); QCOMPARE(other->objectName(), QString::fromLatin1("A test name")); } { QPointer other = qPointerFromVariant(v); QCOMPARE(other->objectName(), QString::fromLatin1("A test name")); } { QPointer other = qPointerFromVariant(v); QCOMPARE(other->objectName(), QString::fromLatin1("A test name")); } { QPointer other = qPointerFromVariant(v); QVERIFY(!other); } { QPointer toBeDeleted = new QFile; QVariant deletedVariant = QVariant::fromValue(toBeDeleted); delete toBeDeleted; QPointer deleted = qPointerFromVariant(deletedVariant); QVERIFY(!deleted); } // Intentionally does not compile. // QPointer sop = qPointerFromVariant(v); } void tst_QPointer::constPointer() { // Compile-time test that QPointer works. QPointer fp = new QFile; delete fp.data(); } QTEST_MAIN(tst_QPointer) #include "tst_qpointer.moc"