Don't crash by modifying read-only shared_null

Functions that modify the d-pointer must detach or otherwise take
measures to not modify the const, read-only shared_null.

The setSharable(bool) function takes care to detach when setting
sharable to false, but should avoid setting the sharable data member
unless d is not the shared null.

Similarly, QMap<Key, T>::setInsertInOrder() needs to detach if it is
shared with the shared_null (the logic has been updated to be the same
as setSharable()).

Change-Id: Ida5cb9818b86695f1b9f0264418b955c56424898
Reviewed-on: http://codereview.qt-project.org/5929
Reviewed-by: Qt Sanity Bot <qt_sanity_bot@ovi.com>
Reviewed-by: Bradley T. Hughes <bradley.hughes@nokia.com>
Reviewed-by: Jan-Arve Sæther <jan-arve.saether@nokia.com>
This commit is contained in:
Bradley T. Hughes 2011-10-03 15:21:02 +02:00 committed by Qt by Nokia
parent 22948130e8
commit 992a37234c
9 changed files with 62 additions and 6 deletions

View File

@ -291,7 +291,7 @@ public:
inline void detach() { if (d->ref != 1) detach_helper(); } inline void detach() { if (d->ref != 1) detach_helper(); }
inline bool isDetached() const { return d->ref == 1; } inline bool isDetached() const { return d->ref == 1; }
inline void setSharable(bool sharable) { if (!sharable) detach(); d->sharable = sharable; } inline void setSharable(bool sharable) { if (!sharable) detach(); if (d != &QHashData::shared_null) d->sharable = sharable; }
inline bool isSharedWith(const QHash<Key, T> &other) const { return d == other.d; } inline bool isSharedWith(const QHash<Key, T> &other) const { return d == other.d; }
void clear(); void clear();

View File

@ -97,7 +97,7 @@ public:
inline void detach() inline void detach()
{ if (d->ref != 1) detach_helper(); } { if (d->ref != 1) detach_helper(); }
inline bool isDetached() const { return d->ref == 1; } inline bool isDetached() const { return d->ref == 1; }
inline void setSharable(bool sharable) { if (!sharable) detach(); d->sharable = sharable; } inline void setSharable(bool sharable) { if (!sharable) detach(); if (d != &QLinkedListData::shared_null) d->sharable = sharable; }
inline bool isSharedWith(const QLinkedList<T> &other) const { return d == other.d; } inline bool isSharedWith(const QLinkedList<T> &other) const { return d == other.d; }
inline bool isEmpty() const { return d->size == 0; } inline bool isEmpty() const { return d->size == 0; }

View File

@ -143,7 +143,7 @@ public:
} }
inline bool isDetached() const { return d->ref == 1; } inline bool isDetached() const { return d->ref == 1; }
inline void setSharable(bool sharable) { if (!sharable) detach(); d->sharable = sharable; } inline void setSharable(bool sharable) { if (!sharable) detach(); if (d != &QListData::shared_null) d->sharable = sharable; }
inline bool isSharedWith(const QList<T> &other) const { return d == other.d; } inline bool isSharedWith(const QList<T> &other) const { return d == other.d; }
inline bool isEmpty() const { return p.isEmpty(); } inline bool isEmpty() const { return p.isEmpty(); }

View File

@ -204,9 +204,9 @@ public:
inline void detach() { if (d->ref != 1) detach_helper(); } inline void detach() { if (d->ref != 1) detach_helper(); }
inline bool isDetached() const { return d->ref == 1; } inline bool isDetached() const { return d->ref == 1; }
inline void setSharable(bool sharable) { if (!sharable) detach(); d->sharable = sharable; } inline void setSharable(bool sharable) { if (!sharable) detach(); if (d != &QMapData::shared_null) d->sharable = sharable; }
inline bool isSharedWith(const QMap<Key, T> &other) const { return d == other.d; } inline bool isSharedWith(const QMap<Key, T> &other) const { return d == other.d; }
inline void setInsertInOrder(bool ordered) { d->insertInOrder = ordered; } inline void setInsertInOrder(bool ordered) { if (ordered) detach(); if (d != &QMapData::shared_null) d->insertInOrder = ordered; }
void clear(); void clear();

View File

@ -146,7 +146,7 @@ public:
inline void detach() { if (d->ref != 1) detach_helper(); } inline void detach() { if (d->ref != 1) detach_helper(); }
inline bool isDetached() const { return d->ref == 1; } inline bool isDetached() const { return d->ref == 1; }
inline void setSharable(bool sharable) { if (!sharable) detach(); d->sharable = sharable; } inline void setSharable(bool sharable) { if (!sharable) detach(); if (d != &QVectorData::shared_null) d->sharable = sharable; }
inline bool isSharedWith(const QVector<T> &other) const { return d == other.d; } inline bool isSharedWith(const QVector<T> &other) const { return d == other.d; }
inline T *data() { detach(); return p->array; } inline T *data() { detach(); return p->array; }

View File

@ -79,6 +79,8 @@ private slots:
void iterators(); // sligthly modified from tst_QMap void iterators(); // sligthly modified from tst_QMap
void keys_values_uniqueKeys(); // slightly modified from tst_QMap void keys_values_uniqueKeys(); // slightly modified from tst_QMap
void noNeedlessRehashes(); void noNeedlessRehashes();
void const_shared_null();
}; };
struct Foo { struct Foo {
@ -1237,5 +1239,16 @@ void tst_QHash::noNeedlessRehashes()
} }
} }
void tst_QHash::const_shared_null()
{
QHash<int, QString> hash1;
hash1.setSharable(false);
QVERIFY(hash1.isDetached());
QHash<int, QString> hash2;
hash2.setSharable(true);
QVERIFY(!hash2.isDetached());
}
QTEST_APPLESS_MAIN(tst_QHash) QTEST_APPLESS_MAIN(tst_QHash)
#include "tst_qhash.moc" #include "tst_qhash.moc"

View File

@ -91,6 +91,8 @@ private slots:
void testOperators() const; void testOperators() const;
void initializeList() const; void initializeList() const;
void const_shared_null() const;
}; };
void tst_QList::length() const void tst_QList::length() const
@ -688,5 +690,16 @@ void tst_QList::initializeList() const
#endif #endif
} }
void tst_QList::const_shared_null() const
{
QList<int> list1;
list1.setSharable(false);
QVERIFY(list1.isDetached());
QList<int> list2;
list2.setSharable(true);
QVERIFY(!list2.isDetached());
}
QTEST_APPLESS_MAIN(tst_QList) QTEST_APPLESS_MAIN(tst_QList)
#include "tst_qlist.moc" #include "tst_qlist.moc"

View File

@ -80,6 +80,8 @@ private slots:
void iterators(); void iterators();
void keys_values_uniqueKeys(); void keys_values_uniqueKeys();
void qmultimap_specific(); void qmultimap_specific();
void const_shared_null();
}; };
tst_QMap::tst_QMap() tst_QMap::tst_QMap()
@ -871,5 +873,20 @@ void tst_QMap::qmultimap_specific()
} }
} }
void tst_QMap::const_shared_null()
{
QMap<int, QString> map1;
map1.setSharable(false);
QVERIFY(map1.isDetached());
QMap<int, QString> map2;
map2.setSharable(true);
QVERIFY(!map2.isDetached());
QMap<int, QString> map3;
map3.setInsertInOrder(true);
map3.setInsertInOrder(false);
}
QTEST_APPLESS_MAIN(tst_QMap) QTEST_APPLESS_MAIN(tst_QMap)
#include "tst_qmap.moc" #include "tst_qmap.moc"

View File

@ -92,6 +92,8 @@ private slots:
void QTBUG11763_data(); void QTBUG11763_data();
void QTBUG11763(); void QTBUG11763();
void initializeList(); void initializeList();
void const_shared_null();
}; };
void tst_QVector::constructors() const void tst_QVector::constructors() const
@ -940,5 +942,16 @@ void tst_QVector::initializeList()
#endif #endif
} }
void tst_QVector::const_shared_null()
{
QVector<int> v1;
v1.setSharable(false);
QVERIFY(v1.isDetached());
QVector<int> v2;
v2.setSharable(true);
QVERIFY(!v2.isDetached());
}
QTEST_APPLESS_MAIN(tst_QVector) QTEST_APPLESS_MAIN(tst_QVector)
#include "tst_qvector.moc" #include "tst_qvector.moc"