QWeakPointer: fix the converting constructor from rvalues
When constructing a QWeakPointer<T> from a rvalue QWeakPointer<X>, even if X* is convertible to T*, actually doing the conversion requires access to the pointee's vtable in case of virtual inheritance. For instance: class Base { virtual ~Base(); }; class Derived : public virtual Base {}; Now given a `Derived *ptr`, then a conversion of `ptr` to `Base *` is implicit (it's a public base), but the compiler needs to dereference `ptr` to find out where the Base sub-object is. This access to the pointee requires protection, because by the time we attempt the cast the pointee may have already been destroyed, or it's being destroyed by another thread. Do that by going through a shared pointer. (This matches the existing code for the converting assignment.) This requires changing the private assign() method, used by QPointer, to avoid going through a converting move assignment/construction, because one can't upgrade a QWeakPointer tracking a QObject to a QSharedPointer. Given it's the caller's responsibility to guard the lifetime of the pointee passed into assign(), I can simply build a QWeakPointer<T> and use ordinary (i.e. non-converting) move assignment instead. Change-Id: I7743b334d479de7cefa6999395a33df06814c8f1 Pick-to: 6.5 6.6 Fixes: QTBUG-117483 Reviewed-by: Thiago Macieira <thiago.macieira@intel.com>
This commit is contained in:
parent
230ff021a1
commit
89b6ad3ab5
@ -569,7 +569,7 @@ public:
|
|||||||
template <class X, IfCompatible<X> = true>
|
template <class X, IfCompatible<X> = true>
|
||||||
Q_NODISCARD_CTOR
|
Q_NODISCARD_CTOR
|
||||||
QWeakPointer(QWeakPointer<X> &&other) noexcept
|
QWeakPointer(QWeakPointer<X> &&other) noexcept
|
||||||
: d(other.d), value(other.value)
|
: d(other.d), value(other.toStrongRef().get()) // must go through QSharedPointer, see below
|
||||||
{
|
{
|
||||||
other.d = nullptr;
|
other.d = nullptr;
|
||||||
other.value = nullptr;
|
other.value = nullptr;
|
||||||
@ -677,7 +677,7 @@ private:
|
|||||||
|
|
||||||
template <class X>
|
template <class X>
|
||||||
inline QWeakPointer &assign(X *ptr)
|
inline QWeakPointer &assign(X *ptr)
|
||||||
{ return *this = QWeakPointer<X>(ptr, true); }
|
{ return *this = QWeakPointer<T>(ptr, true); }
|
||||||
|
|
||||||
#ifndef QT_NO_QOBJECT
|
#ifndef QT_NO_QOBJECT
|
||||||
template <class X, IfCompatible<X> = true>
|
template <class X, IfCompatible<X> = true>
|
||||||
|
@ -1275,6 +1275,22 @@ void tst_QSharedPointer::virtualBaseDifferentPointers()
|
|||||||
QVERIFY(baseptr == aBase);
|
QVERIFY(baseptr == aBase);
|
||||||
}
|
}
|
||||||
safetyCheck();
|
safetyCheck();
|
||||||
|
{
|
||||||
|
VirtualDerived *aData = new VirtualDerived;
|
||||||
|
|
||||||
|
QSharedPointer<VirtualDerived> ptr = QSharedPointer<VirtualDerived>(aData);
|
||||||
|
QWeakPointer<VirtualDerived> wptr = ptr;
|
||||||
|
|
||||||
|
ptr.reset();
|
||||||
|
QVERIFY(wptr.toStrongRef().isNull());
|
||||||
|
|
||||||
|
QWeakPointer<Data> wptr2 = wptr;
|
||||||
|
QVERIFY(wptr2.toStrongRef().isNull());
|
||||||
|
|
||||||
|
QWeakPointer<Data> wptr3 = std::move(wptr);
|
||||||
|
QVERIFY(wptr3.toStrongRef().isNull());
|
||||||
|
}
|
||||||
|
safetyCheck();
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifndef QTEST_NO_RTTI
|
#ifndef QTEST_NO_RTTI
|
||||||
|
Loading…
Reference in New Issue
Block a user