File indexing completed on 2025-09-17 09:09:39
0001
0002
0003
0004
0005
0006 #ifndef QVERSIONNUMBER_H
0007 #define QVERSIONNUMBER_H
0008
0009 #include <QtCore/qcompare.h>
0010 #include <QtCore/qcontainertools_impl.h>
0011 #include <QtCore/qlist.h>
0012 #include <QtCore/qmetatype.h>
0013 #include <QtCore/qnamespace.h>
0014 #include <QtCore/qspan.h>
0015 #include <QtCore/qstring.h>
0016 #include <QtCore/qtypeinfo.h>
0017 #if !defined(QT_LEAN_HEADERS) || QT_LEAN_HEADERS < 2
0018 #include <QtCore/qtyperevision.h>
0019 #endif
0020
0021 QT_BEGIN_NAMESPACE
0022
0023 class QVersionNumber;
0024 Q_CORE_EXPORT size_t qHash(const QVersionNumber &key, size_t seed = 0);
0025
0026 #ifndef QT_NO_DATASTREAM
0027 Q_CORE_EXPORT QDataStream &operator<<(QDataStream &out, const QVersionNumber &version);
0028 Q_CORE_EXPORT QDataStream &operator>>(QDataStream &in, QVersionNumber &version);
0029 #endif
0030
0031 class QVersionNumber
0032 {
0033
0034
0035
0036
0037
0038
0039
0040 enum {
0041
0042
0043 InlineSegmentMarker = Q_BYTE_ORDER == Q_LITTLE_ENDIAN ? 0 : sizeof(void *) - 1,
0044 InlineSegmentStartIdx = !InlineSegmentMarker,
0045 InlineSegmentCount = sizeof(void *) - 1
0046 };
0047 static_assert(InlineSegmentCount >= 3);
0048
0049 struct SegmentStorage
0050 {
0051
0052
0053
0054 union {
0055 quintptr dummy;
0056 qint8 inline_segments[sizeof(void *)];
0057 QList<int> *pointer_segments;
0058 };
0059
0060
0061 SegmentStorage() noexcept : dummy(1) {}
0062
0063 SegmentStorage(const QList<int> &seg)
0064 {
0065 if (dataFitsInline(seg.data(), seg.size()))
0066 setInlineData(seg.data(), seg.size());
0067 else
0068 setListData(seg);
0069 }
0070
0071 Q_CORE_EXPORT void setListData(const QList<int> &seg);
0072
0073 SegmentStorage(const SegmentStorage &other)
0074 {
0075 if (other.isUsingPointer())
0076 setListData(*other.pointer_segments);
0077 else
0078 dummy = other.dummy;
0079 }
0080
0081 SegmentStorage &operator=(const SegmentStorage &other)
0082 {
0083 if (isUsingPointer() && other.isUsingPointer()) {
0084 *pointer_segments = *other.pointer_segments;
0085 } else if (other.isUsingPointer()) {
0086 setListData(*other.pointer_segments);
0087 } else {
0088 if (isUsingPointer())
0089 delete pointer_segments;
0090 dummy = other.dummy;
0091 }
0092 return *this;
0093 }
0094
0095 SegmentStorage(SegmentStorage &&other) noexcept
0096 : dummy(other.dummy)
0097 {
0098 other.dummy = 1;
0099 }
0100
0101 QT_MOVE_ASSIGNMENT_OPERATOR_IMPL_VIA_PURE_SWAP(SegmentStorage)
0102
0103 void swap(SegmentStorage &other) noexcept
0104 {
0105 std::swap(dummy, other.dummy);
0106 }
0107
0108 explicit SegmentStorage(QList<int> &&seg)
0109 {
0110 if (dataFitsInline(std::as_const(seg).data(), seg.size()))
0111 setInlineData(std::as_const(seg).data(), seg.size());
0112 else
0113 setListData(std::move(seg));
0114 }
0115
0116 Q_CORE_EXPORT void setListData(QList<int> &&seg);
0117
0118 explicit SegmentStorage(QSpan<const int> args)
0119 : SegmentStorage(args.begin(), args.end()) {}
0120
0121 explicit SegmentStorage(const int *first, const int *last)
0122 {
0123 if (dataFitsInline(first, last - first)) {
0124 setInlineData(first, last - first);
0125 } else {
0126 setListData(first, last);
0127 }
0128 }
0129
0130 Q_CORE_EXPORT void setListData(const int *first, const int *last);
0131
0132 ~SegmentStorage() { if (isUsingPointer()) delete pointer_segments; }
0133
0134 bool isUsingPointer() const noexcept
0135 { return (inline_segments[InlineSegmentMarker] & 1) == 0; }
0136
0137 qsizetype size() const noexcept
0138 { return isUsingPointer() ? pointer_segments->size() : (inline_segments[InlineSegmentMarker] >> 1); }
0139
0140 void setInlineSize(qsizetype len)
0141 {
0142 Q_ASSERT(len <= InlineSegmentCount);
0143 inline_segments[InlineSegmentMarker] = qint8(1 + 2 * len);
0144 }
0145
0146 Q_CORE_EXPORT void resize(qsizetype len);
0147
0148 int at(qsizetype index) const
0149 {
0150 return isUsingPointer() ?
0151 pointer_segments->at(index) :
0152 inline_segments[InlineSegmentStartIdx + index];
0153 }
0154
0155 void setSegments(int len, int maj, int min = 0, int mic = 0)
0156 {
0157 if (maj == qint8(maj) && min == qint8(min) && mic == qint8(mic)) {
0158 int data[] = { maj, min, mic };
0159 setInlineData(data, len);
0160 } else {
0161 setVector(len, maj, min, mic);
0162 }
0163 }
0164
0165 private:
0166 static bool dataFitsInline(const int *data, qsizetype len)
0167 {
0168 if (len > InlineSegmentCount)
0169 return false;
0170 for (qsizetype i = 0; i < len; ++i)
0171 if (data[i] != qint8(data[i]))
0172 return false;
0173 return true;
0174 }
0175 void setInlineData(const int *data, qsizetype len)
0176 {
0177 Q_ASSERT(len <= InlineSegmentCount);
0178 dummy = 1 + len * 2;
0179 #if Q_BYTE_ORDER == Q_LITTLE_ENDIAN
0180 for (qsizetype i = 0; i < len; ++i)
0181 dummy |= quintptr(data[i] & 0xFF) << (8 * (i + 1));
0182 #elif Q_BYTE_ORDER == Q_BIG_ENDIAN
0183 for (qsizetype i = 0; i < len; ++i)
0184 dummy |= quintptr(data[i] & 0xFF) << (8 * (sizeof(void *) - i - 1));
0185 #else
0186
0187 setInlineSize(len);
0188 for (qsizetype i = 0; i < len; ++i)
0189 inline_segments[InlineSegmentStartIdx + i] = data[i] & 0xFF;
0190 #endif
0191 }
0192
0193 Q_CORE_EXPORT void setVector(int len, int maj, int min, int mic);
0194 } m_segments;
0195
0196 class It
0197 {
0198 const QVersionNumber *v;
0199 qsizetype i;
0200
0201 friend class QVersionNumber;
0202 explicit constexpr It(const QVersionNumber *vn, qsizetype idx) noexcept : v(vn), i(idx) {}
0203
0204 friend constexpr bool comparesEqual(const It &lhs, const It &rhs)
0205 { Q_ASSERT(lhs.v == rhs.v); return lhs.i == rhs.i; }
0206 friend constexpr Qt::strong_ordering compareThreeWay(const It &lhs, const It &rhs)
0207 { Q_ASSERT(lhs.v == rhs.v); return Qt::compareThreeWay(lhs.i, rhs.i); }
0208
0209 friend constexpr bool operator==(It lhs, It rhs) {
0210 return comparesEqual(lhs, rhs);
0211 }
0212 #ifdef __cpp_lib_three_way_comparison
0213 friend constexpr std::strong_ordering operator<=>(It lhs, It rhs) {
0214 return compareThreeWay(lhs, rhs);
0215 }
0216 #else
0217 friend constexpr bool operator!=(It lhs, It rhs) {
0218 return !operator==(lhs, rhs);
0219 }
0220 friend constexpr bool operator<(It lhs, It rhs) {
0221 return is_lt(compareThreeWay(lhs, rhs));
0222 }
0223 friend constexpr bool operator<=(It lhs, It rhs) {
0224 return is_lteq(compareThreeWay(lhs, rhs));
0225 }
0226 friend constexpr bool operator>(It lhs, It rhs) {
0227 return is_gt(compareThreeWay(lhs, rhs));
0228 }
0229 friend constexpr bool operator>=(It lhs, It rhs) {
0230 return is_gteq(compareThreeWay(lhs, rhs));
0231 }
0232 #endif
0233
0234 public:
0235
0236 It() = default;
0237
0238 using iterator_category = std::random_access_iterator_tag;
0239 using value_type = int;
0240 #ifdef QT_COMPILER_HAS_LWG3346
0241 using element_type = const int;
0242 #endif
0243 using difference_type = qptrdiff;
0244 using size_type = qsizetype;
0245 using reference = value_type;
0246 using pointer = QtPrivate::ArrowProxy<reference>;
0247
0248 reference operator*() const { return v->segmentAt(i); }
0249 pointer operator->() const { return {**this}; }
0250
0251 It &operator++() { ++i; return *this; }
0252 It operator++(int) { auto copy = *this; ++*this; return copy; }
0253
0254 It &operator--() { --i; return *this; }
0255 It operator--(int) { auto copy = *this; --*this; return copy; }
0256
0257 It &operator+=(difference_type n) { i += n; return *this; }
0258 friend It operator+(It it, difference_type n) { it += n; return it; }
0259 friend It operator+(difference_type n, It it) { return it + n; }
0260
0261 It &operator-=(difference_type n) { i -= n; return *this; }
0262 friend It operator-(It it, difference_type n) { it -= n; return it; }
0263
0264 friend difference_type operator-(It lhs, It rhs)
0265 { Q_ASSERT(lhs.v == rhs.v); return lhs.i - rhs.i; }
0266
0267 reference operator[](difference_type n) const { return *(*this + n); }
0268 };
0269
0270 public:
0271 using const_iterator = It;
0272 using const_reverse_iterator = std::reverse_iterator<const_iterator>;
0273
0274 using value_type = It::value_type;
0275 using difference_type = It::difference_type;
0276 using size_type = It::size_type;
0277 using reference = It::reference;
0278 using const_reference = reference;
0279 using pointer = It::pointer;
0280 using const_pointer = pointer;
0281
0282 inline QVersionNumber() noexcept
0283 : m_segments()
0284 {}
0285 Q_WEAK_OVERLOAD
0286 inline explicit QVersionNumber(const QList<int> &seg) : m_segments(seg) { }
0287
0288
0289
0290 Q_WEAK_OVERLOAD
0291 explicit QVersionNumber(QList<int> &&seg) : m_segments(std::move(seg)) { }
0292
0293 inline QVersionNumber(std::initializer_list<int> args)
0294 : m_segments(QSpan{args})
0295 {}
0296
0297 explicit QVersionNumber(QSpan<const int> args)
0298 : m_segments(args)
0299 {}
0300
0301 inline explicit QVersionNumber(int maj)
0302 { m_segments.setSegments(1, maj); }
0303
0304 inline explicit QVersionNumber(int maj, int min)
0305 { m_segments.setSegments(2, maj, min); }
0306
0307 inline explicit QVersionNumber(int maj, int min, int mic)
0308 { m_segments.setSegments(3, maj, min, mic); }
0309
0310 [[nodiscard]] inline bool isNull() const noexcept
0311 { return segmentCount() == 0; }
0312
0313 [[nodiscard]] inline bool isNormalized() const noexcept
0314 { return isNull() || segmentAt(segmentCount() - 1) != 0; }
0315
0316 [[nodiscard]] inline int majorVersion() const noexcept
0317 { return segmentAt(0); }
0318
0319 [[nodiscard]] inline int minorVersion() const noexcept
0320 { return segmentAt(1); }
0321
0322 [[nodiscard]] inline int microVersion() const noexcept
0323 { return segmentAt(2); }
0324
0325 [[nodiscard]] Q_CORE_EXPORT QVersionNumber normalized() const;
0326
0327 [[nodiscard]] Q_CORE_EXPORT QList<int> segments() const;
0328
0329 [[nodiscard]] inline int segmentAt(qsizetype index) const noexcept
0330 { return (m_segments.size() > index) ? m_segments.at(index) : 0; }
0331
0332 [[nodiscard]] inline qsizetype segmentCount() const noexcept
0333 { return m_segments.size(); }
0334
0335 [[nodiscard]] const_iterator begin() const noexcept { return const_iterator{this, 0}; }
0336 [[nodiscard]] const_iterator end() const noexcept { return begin() + segmentCount(); }
0337 [[nodiscard]] const_iterator cbegin() const noexcept { return begin(); }
0338 [[nodiscard]] const_iterator cend() const noexcept { return end(); }
0339
0340 [[nodiscard]] const_reverse_iterator rbegin() const noexcept { return const_reverse_iterator{end()}; }
0341 [[nodiscard]] const_reverse_iterator rend() const noexcept { return const_reverse_iterator{begin()}; }
0342 [[nodiscard]] const_reverse_iterator crbegin() const noexcept { return rbegin(); }
0343 [[nodiscard]] const_reverse_iterator crend() const noexcept { return rend(); }
0344
0345 [[nodiscard]] const_iterator constBegin() const noexcept { return begin(); }
0346 [[nodiscard]] const_iterator constEnd() const noexcept { return end(); }
0347
0348 [[nodiscard]] Q_CORE_EXPORT bool isPrefixOf(const QVersionNumber &other) const noexcept;
0349
0350 [[nodiscard]] Q_CORE_EXPORT static int compare(const QVersionNumber &v1, const QVersionNumber &v2) noexcept;
0351
0352 [[nodiscard]] Q_CORE_EXPORT static QVersionNumber commonPrefix(const QVersionNumber &v1, const QVersionNumber &v2);
0353
0354 [[nodiscard]] Q_CORE_EXPORT QString toString() const;
0355 [[nodiscard]] Q_CORE_EXPORT static QVersionNumber fromString(QAnyStringView string, qsizetype *suffixIndex = nullptr);
0356
0357 #if QT_DEPRECATED_SINCE(6, 4) && QT_POINTER_SIZE != 4
0358 Q_WEAK_OVERLOAD
0359 QT_DEPRECATED_VERSION_X_6_4("Use the 'qsizetype *suffixIndex' overload.")
0360 [[nodiscard]] static QVersionNumber fromString(QAnyStringView string, int *suffixIndex)
0361 {
0362 QT_WARNING_PUSH
0363
0364 QT_WARNING_DISABLE_GCC("-Wmaybe-uninitialized")
0365 qsizetype n;
0366 auto r = fromString(string, &n);
0367 if (suffixIndex) {
0368 Q_ASSERT(int(n) == n);
0369 *suffixIndex = int(n);
0370 }
0371 return r;
0372 QT_WARNING_POP
0373 }
0374 #endif
0375
0376
0377 #if QT_CORE_REMOVED_SINCE(6, 4)
0378 [[nodiscard]] Q_CORE_EXPORT static QVersionNumber fromString(const QString &string, int *suffixIndex);
0379 [[nodiscard]] Q_CORE_EXPORT static QVersionNumber fromString(QLatin1StringView string, int *suffixIndex);
0380 [[nodiscard]] Q_CORE_EXPORT static QVersionNumber fromString(QStringView string, int *suffixIndex);
0381 #endif
0382
0383 private:
0384 [[nodiscard]] friend bool comparesEqual(const QVersionNumber &lhs,
0385 const QVersionNumber &rhs) noexcept
0386 {
0387 return lhs.segmentCount() == rhs.segmentCount() && compare(lhs, rhs) == 0;
0388 }
0389 [[nodiscard]] friend Qt::strong_ordering compareThreeWay(const QVersionNumber &lhs,
0390 const QVersionNumber &rhs) noexcept
0391 {
0392 int c = compare(lhs, rhs);
0393 return Qt::compareThreeWay(c, 0);
0394 }
0395 Q_DECLARE_STRONGLY_ORDERED(QVersionNumber)
0396
0397 #ifndef QT_NO_DATASTREAM
0398 friend Q_CORE_EXPORT QDataStream& operator>>(QDataStream &in, QVersionNumber &version);
0399 #endif
0400 friend Q_CORE_EXPORT size_t qHash(const QVersionNumber &key, size_t seed);
0401 };
0402
0403 Q_DECLARE_TYPEINFO(QVersionNumber, Q_RELOCATABLE_TYPE);
0404
0405 #ifndef QT_NO_DEBUG_STREAM
0406 Q_CORE_EXPORT QDebug operator<<(QDebug, const QVersionNumber &version);
0407 #endif
0408
0409 QT_END_NAMESPACE
0410
0411 QT_DECL_METATYPE_EXTERN(QVersionNumber, Q_CORE_EXPORT)
0412
0413 #endif