Files
RMHook-Win/paho-mqtt3as-proxy/Qt/include/QtCore/qcontiguouscache.h
2026-05-07 16:41:01 +02:00

432 lines
12 KiB
C++

// Copyright (C) 2016 The Qt Company Ltd.
// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
#ifndef QCONTIGUOUSCACHE_H
#define QCONTIGUOUSCACHE_H
#include <QtCore/qatomic.h>
#include <limits.h>
#include <new>
QT_BEGIN_NAMESPACE
#undef QT_QCONTIGUOUSCACHE_DEBUG
struct Q_CORE_EXPORT QContiguousCacheData
{
QBasicAtomicInt ref;
qsizetype alloc;
qsizetype count;
qsizetype start;
qsizetype offset;
static QContiguousCacheData *allocateData(qsizetype size, qsizetype alignment);
static void freeData(QContiguousCacheData *data);
#ifdef QT_QCONTIGUOUSCACHE_DEBUG
void dump() const;
#endif
};
template <typename T>
struct QContiguousCacheTypedData : public QContiguousCacheData
{
T array[1];
};
template<typename T>
class QContiguousCache {
static_assert(std::is_nothrow_destructible_v<T>, "Types with throwing destructors are not supported in Qt containers.");
typedef QContiguousCacheTypedData<T> Data;
Data *d;
public:
// STL compatibility
typedef T value_type;
typedef value_type* pointer;
typedef const value_type* const_pointer;
typedef value_type& reference;
typedef const value_type& const_reference;
typedef qptrdiff difference_type;
typedef qsizetype size_type;
explicit QContiguousCache(qsizetype capacity = 0);
QContiguousCache(const QContiguousCache<T> &v) : d(v.d) { d->ref.ref(); }
inline ~QContiguousCache() { if (!d) return; if (!d->ref.deref()) freeData(d); }
inline void detach() { if (d->ref.loadRelaxed() != 1) detach_helper(); }
inline bool isDetached() const { return d->ref.loadRelaxed() == 1; }
QContiguousCache<T> &operator=(const QContiguousCache<T> &other);
QT_MOVE_ASSIGNMENT_OPERATOR_IMPL_VIA_PURE_SWAP(QContiguousCache)
void swap(QContiguousCache &other) noexcept { qt_ptr_swap(d, other.d); }
#ifndef Q_QDOC
template <typename U = T>
QTypeTraits::compare_eq_result<U> operator==(const QContiguousCache<T> &other) const
{
if (other.d == d)
return true;
if (other.d->start != d->start
|| other.d->count != d->count
|| other.d->offset != d->offset
|| other.d->alloc != d->alloc)
return false;
for (qsizetype i = firstIndex(); i <= lastIndex(); ++i)
if (!(at(i) == other.at(i)))
return false;
return true;
}
template <typename U = T>
QTypeTraits::compare_eq_result<U> operator!=(const QContiguousCache<T> &other) const
{ return !(*this == other); }
#else
bool operator==(const QContiguousCache &other) const;
bool operator!=(const QContiguousCache &other) const;
#endif // Q_QDOC
inline qsizetype capacity() const {return d->alloc; }
inline qsizetype count() const { return d->count; }
inline qsizetype size() const { return d->count; }
inline bool isEmpty() const { return d->count == 0; }
inline bool isFull() const { return d->count == d->alloc; }
inline qsizetype available() const { return d->alloc - d->count; }
void clear();
void setCapacity(qsizetype size);
const T &at(qsizetype pos) const;
T &operator[](qsizetype i);
const T &operator[](qsizetype i) const;
void append(T &&value);
void append(const T &value);
void prepend(T &&value);
void prepend(const T &value);
void insert(qsizetype pos, T &&value);
void insert(qsizetype pos, const T &value);
inline bool containsIndex(qsizetype pos) const { return pos >= d->offset && pos - d->offset < d->count; }
inline qsizetype firstIndex() const { return d->offset; }
inline qsizetype lastIndex() const { return d->offset + d->count - 1; }
inline const T &first() const { Q_ASSERT(!isEmpty()); return d->array[d->start]; }
inline const T &last() const { Q_ASSERT(!isEmpty()); return d->array[(d->start + d->count -1) % d->alloc]; }
inline T &first() { Q_ASSERT(!isEmpty()); detach(); return d->array[d->start]; }
inline T &last() { Q_ASSERT(!isEmpty()); detach(); return d->array[(d->start + d->count -1) % d->alloc]; }
void removeFirst();
T takeFirst();
void removeLast();
T takeLast();
// Use extra parentheses around max to avoid expanding it if it is a macro.
inline bool areIndexesValid() const
{ return d->offset >= 0 && d->offset < (std::numeric_limits<qsizetype>::max)() - d->count && (d->offset % d->alloc) == d->start; }
inline void normalizeIndexes() { d->offset = d->start; }
#ifdef QT_QCONTIGUOUSCACHE_DEBUG
void dump() const { d->dump(); }
#endif
private:
void detach_helper();
Data *allocateData(qsizetype aalloc);
void freeData(Data *x);
};
template <typename T>
void QContiguousCache<T>::detach_helper()
{
Data *x = allocateData(d->alloc);
x->ref.storeRelaxed(1);
x->count = d->count;
x->start = d->start;
x->offset = d->offset;
x->alloc = d->alloc;
T *dest = x->array + x->start;
T *src = d->array + d->start;
qsizetype oldcount = x->count;
while (oldcount--) {
new (dest) T(*src);
dest++;
if (dest == x->array + x->alloc)
dest = x->array;
src++;
if (src == d->array + d->alloc)
src = d->array;
}
if (!d->ref.deref())
freeData(d);
d = x;
}
template <typename T>
void QContiguousCache<T>::setCapacity(qsizetype asize)
{
Q_ASSERT(asize >= 0);
if (asize == d->alloc)
return;
detach();
Data *x = allocateData(asize);
x->ref.storeRelaxed(1);
x->alloc = asize;
x->count = qMin(d->count, asize);
x->offset = d->offset + d->count - x->count;
if (asize)
x->start = x->offset % x->alloc;
else
x->start = 0;
qsizetype oldcount = x->count;
if (oldcount)
{
T *dest = x->array + (x->start + x->count-1) % x->alloc;
T *src = d->array + (d->start + d->count-1) % d->alloc;
while (oldcount--) {
new (dest) T(*src);
if (dest == x->array)
dest = x->array + x->alloc;
dest--;
if (src == d->array)
src = d->array + d->alloc;
src--;
}
}
/* free old */
freeData(d);
d = x;
}
template <typename T>
void QContiguousCache<T>::clear()
{
if (d->ref.loadRelaxed() == 1) {
if (QTypeInfo<T>::isComplex) {
qsizetype oldcount = d->count;
T * i = d->array + d->start;
T * e = d->array + d->alloc;
while (oldcount--) {
i->~T();
i++;
if (i == e)
i = d->array;
}
}
d->count = d->start = d->offset = 0;
} else {
Data *x = allocateData(d->alloc);
x->ref.storeRelaxed(1);
x->alloc = d->alloc;
x->count = x->start = x->offset = 0;
if (!d->ref.deref())
freeData(d);
d = x;
}
}
template <typename T>
inline typename QContiguousCache<T>::Data *QContiguousCache<T>::allocateData(qsizetype aalloc)
{
return static_cast<Data *>(QContiguousCacheData::allocateData(sizeof(Data) + (aalloc - 1) * sizeof(T), alignof(Data)));
}
template <typename T>
QContiguousCache<T>::QContiguousCache(qsizetype cap)
{
Q_ASSERT(cap >= 0);
d = allocateData(cap);
d->ref.storeRelaxed(1);
d->alloc = cap;
d->count = d->start = d->offset = 0;
}
template <typename T>
QContiguousCache<T> &QContiguousCache<T>::operator=(const QContiguousCache<T> &other)
{
other.d->ref.ref();
if (!d->ref.deref())
freeData(d);
d = other.d;
return *this;
}
template <typename T>
void QContiguousCache<T>::freeData(Data *x)
{
if (QTypeInfo<T>::isComplex) {
qsizetype oldcount = d->count;
T * i = d->array + d->start;
T * e = d->array + d->alloc;
while (oldcount--) {
i->~T();
i++;
if (i == e)
i = d->array;
}
}
Data::freeData(x);
}
template <typename T>
void QContiguousCache<T>::append(T &&value)
{
if (!d->alloc)
return; // zero capacity
detach();
if (d->count == d->alloc)
(d->array + (d->start+d->count) % d->alloc)->~T();
new (d->array + (d->start+d->count) % d->alloc) T(std::move(value));
if (d->count == d->alloc) {
d->start++;
d->start %= d->alloc;
d->offset++;
} else {
d->count++;
}
}
template <typename T>
void QContiguousCache<T>::append(const T &value)
{
if (!d->alloc)
return; // zero capacity
detach();
if (d->count == d->alloc)
(d->array + (d->start+d->count) % d->alloc)->~T();
new (d->array + (d->start+d->count) % d->alloc) T(value);
if (d->count == d->alloc) {
d->start++;
d->start %= d->alloc;
d->offset++;
} else {
d->count++;
}
}
template<typename T>
void QContiguousCache<T>::prepend(T &&value)
{
if (!d->alloc)
return; // zero capacity
detach();
if (d->start)
d->start--;
else
d->start = d->alloc-1;
d->offset--;
if (d->count != d->alloc)
d->count++;
else
(d->array + d->start)->~T();
new (d->array + d->start) T(std::move(value));
}
template<typename T>
void QContiguousCache<T>::prepend(const T &value)
{
if (!d->alloc)
return; // zero capacity
detach();
if (d->start)
d->start--;
else
d->start = d->alloc-1;
d->offset--;
if (d->count != d->alloc)
d->count++;
else
(d->array + d->start)->~T();
new (d->array + d->start) T(value);
}
template<typename T>
void QContiguousCache<T>::insert(qsizetype pos, T &&value)
{
Q_ASSERT_X(pos >= 0, "QContiguousCache<T>::insert", "index out of range");
if (!d->alloc)
return; // zero capacity
detach();
if (containsIndex(pos)) {
d->array[pos % d->alloc] = std::move(value);
} else if (pos == d->offset-1)
prepend(value);
else if (pos == d->offset+d->count)
append(value);
else {
// we don't leave gaps.
clear();
d->offset = pos;
d->start = pos % d->alloc;
d->count = 1;
new (d->array + d->start) T(std::move(value));
}
}
template<typename T>
void QContiguousCache<T>::insert(qsizetype pos, const T &value)
{
return insert(pos, T(value));
}
template <typename T>
inline const T &QContiguousCache<T>::at(qsizetype pos) const
{ Q_ASSERT_X(pos >= d->offset && pos - d->offset < d->count, "QContiguousCache<T>::at", "index out of range"); return d->array[pos % d->alloc]; }
template <typename T>
inline const T &QContiguousCache<T>::operator[](qsizetype pos) const
{ return at(pos); }
template <typename T>
inline T &QContiguousCache<T>::operator[](qsizetype pos)
{
detach();
if (!containsIndex(pos))
insert(pos, T());
return d->array[pos % d->alloc];
}
template <typename T>
inline void QContiguousCache<T>::removeFirst()
{
Q_ASSERT(d->count > 0);
detach();
d->count--;
if (QTypeInfo<T>::isComplex)
(d->array + d->start)->~T();
d->start = (d->start + 1) % d->alloc;
d->offset++;
}
template <typename T>
inline void QContiguousCache<T>::removeLast()
{
Q_ASSERT(d->count > 0);
detach();
d->count--;
if (QTypeInfo<T>::isComplex)
(d->array + (d->start + d->count) % d->alloc)->~T();
}
template <typename T>
inline T QContiguousCache<T>::takeFirst()
{ T t = std::move(first()); removeFirst(); return t; }
template <typename T>
inline T QContiguousCache<T>::takeLast()
{ T t = std::move(last()); removeLast(); return t; }
QT_END_NAMESPACE
#endif