QVarLengthArray: fix UBs in emplace()/insert() ([basic.life], broken class invariant)

There are two problems in emplace_impl() (the same code exists as
rvalue insert() since 5.10):

First, the old code updated size() at the end of the function.

However, if, after constructing the new end element, one of the
subsequent move-assignments fail (throws), then the class invariant
that size() be the number of alive elements in the container is
broken, with the immediate consequence that the QVLA dtor would not
destroy this element, but surely other unpleasantness (UB) that the
C++ lifetime rules decide to throw our way.

Similarly, in the trivially-relocatable case, the memmove() starts the
life-time of the new end object, so if the following placement new
fails, we're in the same situation.

The latter case is worse, though, since here we leave *b in some weird
zombie state: the memmove() effectively ended its lifetime in the
sense that one mustn't call the destructor on the source object after
trivial relocation, but C++ doesn't agree and QVLA's dtor will happily
call b->~T() as part of its cleanup.

The other ugly thing is that we're using placement new into an object
that C++ says is still alive. QString is trivially relocatable, but
not trivially copyable, so we can't end a QString's lifetime by
placement-new'ing a new QString instance into it without first having
ended the old object's lifetime.

The fix for both of these is, fortunately, the same: It's a rotate!™

By using emplace_back() + std::rotate(), we always place the new
object in a spot that didn't contain an alive object (in the C++
sense) before, we always update the size() right after doing so,
maintaining that invariant, and we then rotate() it into place, which
doesn't leave zombie objects around.

std::rotate() is such a fundamental algorithm that we should trust the
STL implementors to have optimized it well:
https://stackoverflow.com/questions/21160875/why-is-stdrotate-so-fast

We know we can do better only for trivially-relocatable, but
non-trivially-copyable types (ex: QString), so in order to not lose
the memmove() optimization, we now fall back to std::rotate on raw
memory for that case.

Amends dd58ddd5d9.

Pick-to: 6.5 6.4 6.2 5.15
Change-Id: Iacce4488ca649502861e0ed4e084c9fad38cab47
Reviewed-by: Thiago Macieira <thiago.macieira@intel.com>
Reviewed-by: Fabian Kosmale <fabian.kosmale@qt.io>
This commit is contained in:
Marc Mutz 2023-02-10 09:16:51 +01:00
parent 7346de6491
commit e24df8bc72

View File

@ -910,29 +910,17 @@ Q_OUTOFLINE_TEMPLATE auto QVLABase<T>::emplace_impl(qsizetype prealloc, void *ar
Q_ASSERT(size() <= capacity()); Q_ASSERT(size() <= capacity());
Q_ASSERT(capacity() > 0); Q_ASSERT(capacity() > 0);
qsizetype offset = qsizetype(before - cbegin()); const qsizetype offset = qsizetype(before - cbegin());
if (size() == capacity()) emplace_back_impl(prealloc, array, std::forward<Args>(args)...);
growBy(prealloc, array, 1); const auto b = begin() + offset;
if constexpr (!QTypeInfo<T>::isRelocatable) { const auto e = end();
T *b = begin() + offset; if constexpr (QTypeInfo<T>::isRelocatable) {
T *i = end(); auto cast = [](T *p) { return reinterpret_cast<uchar*>(p); };
T *j = i + 1; std::rotate(cast(b), cast(e - 1), cast(e));
// The new end-element needs to be constructed, the rest must be move assigned
if (i != b) {
q20::construct_at(--j, std::move(*--i));
while (i != b)
*--j = std::move(*--i);
*b = T(std::forward<Args>(args)...);
} else {
q20::construct_at(b, std::forward<Args>(args)...);
}
} else { } else {
T *b = begin() + offset; std::rotate(b, e - 1, e);
memmove(static_cast<void *>(b + 1), static_cast<const void *>(b), (size() - offset) * sizeof(T));
q20::construct_at(b, std::forward<Args>(args)...);
} }
this->s += 1; return b;
return data() + offset;
} }
template <class T> template <class T>