1// Copyright (C) 2020 The Qt Company Ltd.
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
4#ifndef QUUID_H
5#define QUUID_H
6
7#include <QtCore/qcompare.h>
8#include <QtCore/qendian.h>
9#include <QtCore/qstring.h>
10#include <QtCore/qsystemdetection.h>
11
12#if defined(Q_OS_WIN) || defined(Q_QDOC)
13#ifndef GUID_DEFINED
14#define GUID_DEFINED
15typedef struct _GUID
16{
17 ulong Data1;
18 ushort Data2;
19 ushort Data3;
20 uchar Data4[8];
21} GUID, *REFGUID, *LPGUID;
22#endif
23#endif
24
25#if defined(Q_OS_DARWIN) || defined(Q_QDOC)
26Q_FORWARD_DECLARE_CF_TYPE(CFUUID);
27Q_FORWARD_DECLARE_OBJC_CLASS(NSUUID);
28#endif
29
30QT_BEGIN_NAMESPACE
31
32class Q_CORE_EXPORT QUuid
33{
34 QUuid(Qt::Initialization) {}
35public:
36 enum Variant {
37 VarUnknown =-1,
38 NCS = 0, // 0 - -
39 DCE = 2, // 1 0 -
40 Microsoft = 6, // 1 1 0
41 Reserved = 7 // 1 1 1
42 };
43
44 enum Version {
45 VerUnknown =-1,
46 Time = 1, // 0 0 0 1
47 EmbeddedPOSIX = 2, // 0 0 1 0
48 Md5 = 3, // 0 0 1 1
49 Name = Md5,
50 Random = 4, // 0 1 0 0
51 Sha1 = 5 // 0 1 0 1
52 };
53
54 enum StringFormat {
55 WithBraces = 0,
56 WithoutBraces = 1,
57 Id128 = 3
58 };
59
60 union alignas(16) Id128Bytes {
61 quint8 data[16];
62 quint16 data16[8];
63 quint32 data32[4];
64 quint64 data64[2];
65#if defined(QT_COMPILER_SUPPORTS_INT128)
66QT_WARNING_PUSH
67QT_WARNING_DISABLE_GCC("-Wpedantic") // ISO C++ does not support ‘__int128’ for ‘data128’
68 unsigned __int128 data128[1];
69QT_WARNING_POP
70#elif defined(QT_SUPPORTS_INT128)
71# error "struct QUuid::Id128Bytes should not depend on QT_SUPPORTS_INT128 for ABI reasons."
72# error "Adjust the declaration of the `data128` member above so it is always defined if it's " \
73 "supported by the current compiler/architecture in any configuration."
74#endif
75
76 constexpr explicit operator QByteArrayView() const noexcept
77 {
78 return QByteArrayView(data, sizeof(data));
79 }
80
81 friend constexpr Id128Bytes qbswap(Id128Bytes b) noexcept
82 {
83 // 128-bit byte swap
84 auto b0 = qbswap(source: b.data64[0]);
85 auto b1 = qbswap(source: b.data64[1]);
86 b.data64[0] = b1;
87 b.data64[1] = b0;
88 return b;
89 }
90 };
91
92 constexpr QUuid() noexcept : data1(0), data2(0), data3(0), data4{0,0,0,0,0,0,0,0} {}
93
94 constexpr QUuid(uint l, ushort w1, ushort w2, uchar b1, uchar b2, uchar b3,
95 uchar b4, uchar b5, uchar b6, uchar b7, uchar b8) noexcept
96 : data1(l), data2(w1), data3(w2), data4{b1, b2, b3, b4, b5, b6, b7, b8} {}
97 explicit inline QUuid(Id128Bytes id128, QSysInfo::Endian order = QSysInfo::BigEndian) noexcept;
98
99 explicit QUuid(QAnyStringView string) noexcept
100 : QUuid{fromString(string)} {}
101 static QUuid fromString(QAnyStringView string) noexcept;
102#if QT_CORE_REMOVED_SINCE(6, 3)
103 explicit QUuid(const QString &);
104 static QUuid fromString(QStringView string) noexcept;
105 static QUuid fromString(QLatin1StringView string) noexcept;
106 explicit QUuid(const char *);
107 explicit QUuid(const QByteArray &);
108#endif
109 QString toString(StringFormat mode = WithBraces) const;
110 QByteArray toByteArray(StringFormat mode = WithBraces) const;
111 inline Id128Bytes toBytes(QSysInfo::Endian order = QSysInfo::BigEndian) const noexcept;
112 QByteArray toRfc4122() const;
113
114 static inline QUuid fromBytes(const void *bytes, QSysInfo::Endian order = QSysInfo::BigEndian);
115#if QT_CORE_REMOVED_SINCE(6, 3)
116 static QUuid fromRfc4122(const QByteArray &);
117#endif
118 static QUuid fromRfc4122(QByteArrayView) noexcept;
119
120 bool isNull() const noexcept;
121
122#ifdef QT_SUPPORTS_INT128
123 static constexpr QUuid fromUInt128(quint128 uuid, QSysInfo::Endian order = QSysInfo::BigEndian) noexcept;
124 constexpr quint128 toUInt128(QSysInfo::Endian order = QSysInfo::BigEndian) const noexcept;
125#endif
126
127private:
128 friend constexpr bool comparesEqual(const QUuid &lhs, const QUuid &rhs) noexcept
129 {
130 return is_eq(o: compareThreeWay_helper(lhs, rhs));
131 }
132 static constexpr Qt::strong_ordering
133 compareThreeWay_helper(const QUuid &lhs, const QUuid &rhs) noexcept
134 {
135#if QT_VERSION < QT_VERSION_CHECK(7, 0, 0) && !defined(QT_BOOTSTRAPPED)
136 if (const auto c = Qt::compareThreeWay(lhs: lhs.data1, rhs: rhs.data1); !is_eq(o: c))
137 return c;
138 if (const auto c = Qt::compareThreeWay(lhs: lhs.data2, rhs: rhs.data2); !is_eq(o: c))
139 return c;
140 if (const auto c = Qt::compareThreeWay(lhs: lhs.data3, rhs: rhs.data3); !is_eq(o: c))
141 return c;
142#elif defined(__cpp_lib_bit_cast) && defined(QT_SUPPORTS_INT128)
143 quint128 lu = qFromBigEndian(std::bit_cast<quint128>(lhs));
144 quint128 ru = qFromBigEndian(std::bit_cast<quint128>(rhs));
145 return Qt::compareThreeWay(lu, ru);
146#else
147 auto make_int = [](const QUuid &u) {
148 quint64 result = quint64(u.data3) << 48;
149 result |= quint64(u.data2) << 32;
150 return qFromBigEndian(result | u.data1);
151 };
152 if (const auto c = Qt::compareThreeWay(make_int(lhs), make_int(rhs)); !is_eq(c))
153 return c;
154#endif
155 for (unsigned i = 0; i < sizeof(lhs.data4); ++i) {
156 if (const auto c = Qt::compareThreeWay(lhs: lhs.data4[i], rhs: rhs.data4[i]); !is_eq(o: c))
157 return c;
158 }
159 return Qt::strong_ordering::equal;
160 }
161 friend constexpr Qt::strong_ordering compareThreeWay(const QUuid &lhs, const QUuid &rhs) noexcept
162 {
163#if QT_VERSION < QT_VERSION_CHECK(7, 0, 0) && !defined(QT_BOOTSTRAPPED)
164 // Keep the old sorting order from before Qt 6.8, which sorted first on
165 // variant(). We don't need the exact algorithm to achieve same results.
166 auto fastVariant = [](const QUuid &uuid) {
167 quint8 v = uuid.data4[0];
168 // i.e.: return v >= Microsoft ? v : v >= DCE ? DCE : NCS;
169 return v >= 0xC0 ? v & 0xE0 : v >= 0x80 ? 0x80 : 0;
170 };
171 if (const auto c = Qt::compareThreeWay(lhs: fastVariant(lhs), rhs: fastVariant(rhs)); !is_eq(o: c))
172 return c;
173#endif
174 return compareThreeWay_helper(lhs, rhs);
175 }
176
177public:
178/* To prevent a meta-type creation ambiguity on Windows, we put comparison
179 macros under NOT QT_CORE_REMOVED_SINCE(6, 8) part. */
180#if QT_CORE_REMOVED_SINCE(6, 8)
181 constexpr bool operator==(const QUuid &orig) const noexcept
182 {
183 return comparesEqual(*this, orig);
184 }
185
186 constexpr bool operator!=(const QUuid &orig) const noexcept
187 {
188 return !operator==(orig);
189 }
190
191 bool operator<(const QUuid &other) const noexcept;
192 bool operator>(const QUuid &other) const noexcept;
193#else
194 Q_DECLARE_STRONGLY_ORDERED_LITERAL_TYPE(QUuid)
195#endif // QT_CORE_REMOVED_SINCE(6, 8)
196#if defined(Q_OS_WIN) || defined(Q_QDOC)
197 // On Windows we have a type GUID that is used by the platform API, so we
198 // provide convenience operators to cast from and to this type.
199 constexpr QUuid(const GUID &guid) noexcept
200 : data1(guid.Data1), data2(guid.Data2), data3(guid.Data3),
201 data4{guid.Data4[0], guid.Data4[1], guid.Data4[2], guid.Data4[3],
202 guid.Data4[4], guid.Data4[5], guid.Data4[6], guid.Data4[7]} {}
203
204 constexpr QUuid &operator=(const GUID &guid) noexcept
205 {
206 *this = QUuid(guid);
207 return *this;
208 }
209
210 constexpr operator GUID() const noexcept
211 {
212 GUID guid = { data1, data2, data3, { data4[0], data4[1], data4[2], data4[3], data4[4], data4[5], data4[6], data4[7] } };
213 return guid;
214 }
215private:
216 friend constexpr bool comparesEqual(const QUuid &lhs, const GUID &rhs) noexcept
217 {
218 return comparesEqual(lhs, QUuid(rhs));
219 }
220public:
221/* To prevent a meta-type creation ambiguity on Windows, we put comparison
222 macros under NOT QT_CORE_REMOVED_SINCE(6, 8) part. */
223#if QT_CORE_REMOVED_SINCE(6, 8)
224 constexpr bool operator==(const GUID &guid) const noexcept
225 {
226 return comparesEqual(*this, QUuid(guid));
227 }
228
229 constexpr bool operator!=(const GUID &guid) const noexcept
230 {
231 return !operator==(guid);
232 }
233#else
234 Q_DECLARE_EQUALITY_COMPARABLE_LITERAL_TYPE(QUuid, GUID)
235#endif // !QT_CORE_REMOVED_SINCE(6, 8)
236#endif
237public:
238 static QUuid createUuid();
239#if QT_CORE_REMOVED_SINCE(6, 8)
240 static QUuid createUuidV3(const QUuid &ns, const QByteArray &baseData) noexcept;
241 static QUuid createUuidV5(const QUuid &ns, const QByteArray &baseData) noexcept;
242#endif
243 static QUuid createUuidV5(QUuid ns, QByteArrayView baseData) noexcept;
244#ifndef QT_BOOTSTRAPPED
245 static QUuid createUuidV3(QUuid ns, QByteArrayView baseData) noexcept;
246#if !QT_CORE_REMOVED_SINCE(6, 8)
247 Q_WEAK_OVERLOAD
248#endif
249 static inline QUuid createUuidV3(const QUuid &ns, const QString &baseData)
250 {
251 return QUuid::createUuidV3(ns, baseData: qToByteArrayViewIgnoringNull(b: baseData.toUtf8()));
252 }
253#endif
254#if !QT_CORE_REMOVED_SINCE(6, 8)
255 Q_WEAK_OVERLOAD
256#endif
257 static inline QUuid createUuidV5(const QUuid &ns, const QString &baseData)
258 {
259 return QUuid::createUuidV5(ns, baseData: qToByteArrayViewIgnoringNull(b: baseData.toUtf8()));
260 }
261
262 QUuid::Variant variant() const noexcept;
263 QUuid::Version version() const noexcept;
264
265#if defined(Q_OS_DARWIN) || defined(Q_QDOC)
266 static QUuid fromCFUUID(CFUUIDRef uuid);
267 CFUUIDRef toCFUUID() const Q_DECL_CF_RETURNS_RETAINED;
268 static QUuid fromNSUUID(const NSUUID *uuid);
269 NSUUID *toNSUUID() const Q_DECL_NS_RETURNS_AUTORELEASED;
270#endif
271
272 uint data1;
273 ushort data2;
274 ushort data3;
275 uchar data4[8];
276};
277
278Q_DECLARE_TYPEINFO(QUuid, Q_PRIMITIVE_TYPE);
279
280#ifndef QT_NO_DATASTREAM
281Q_CORE_EXPORT QDataStream &operator<<(QDataStream &, const QUuid &);
282Q_CORE_EXPORT QDataStream &operator>>(QDataStream &, QUuid &);
283#endif
284
285#ifndef QT_NO_DEBUG_STREAM
286Q_CORE_EXPORT QDebug operator<<(QDebug, const QUuid &);
287#endif
288
289Q_CORE_EXPORT size_t qHash(const QUuid &uuid, size_t seed = 0) noexcept;
290
291QUuid::QUuid(Id128Bytes uuid, QSysInfo::Endian order) noexcept
292{
293 char bytes[sizeof uuid];
294 if (order == QSysInfo::LittleEndian)
295 qbswap(src: uuid, dest: bytes);
296 else
297 memcpy(dest: bytes, src: &uuid, n: sizeof bytes);
298 data1 = qFromBigEndian<quint32>(src: &bytes[0]);
299 data2 = qFromBigEndian<quint16>(src: &bytes[4]);
300 data3 = qFromBigEndian<quint16>(src: &bytes[6]);
301 memcpy(dest: data4, src: &bytes[8], n: sizeof(data4));
302}
303
304QUuid::Id128Bytes QUuid::toBytes(QSysInfo::Endian order) const noexcept
305{
306 Id128Bytes result = {};
307 qToBigEndian(src: data1, dest: &result.data[0]);
308 qToBigEndian(src: data2, dest: &result.data[4]);
309 qToBigEndian(src: data3, dest: &result.data[6]);
310 memcpy(dest: &result.data[8], src: data4, n: sizeof(data4));
311 if (order == QSysInfo::LittleEndian)
312 return qbswap(b: result);
313 return result;
314}
315
316QUuid QUuid::fromBytes(const void *bytes, QSysInfo::Endian order)
317{
318 Id128Bytes result = {};
319 memcpy(dest: result.data, src: bytes, n: sizeof(result));
320 return QUuid(result, order);
321}
322
323#ifdef QT_SUPPORTS_INT128
324constexpr QUuid QUuid::fromUInt128(quint128 uuid, QSysInfo::Endian order) noexcept
325{
326 QUuid result = {};
327 if (order == QSysInfo::BigEndian) {
328 result.data1 = qFromBigEndian<quint32>(source: int(uuid));
329 result.data2 = qFromBigEndian<quint16>(source: ushort(uuid >> 32));
330 result.data3 = qFromBigEndian<quint16>(source: ushort(uuid >> 48));
331 for (int i = 0; i < 8; ++i)
332 result.data4[i] = uchar(uuid >> (64 + i * 8));
333 } else {
334 result.data1 = qFromLittleEndian<quint32>(source: uint(uuid >> 96));
335 result.data2 = qFromLittleEndian<quint16>(source: ushort(uuid >> 80));
336 result.data3 = qFromLittleEndian<quint16>(source: ushort(uuid >> 64));
337 for (int i = 0; i < 8; ++i)
338 result.data4[i] = uchar(uuid >> (56 - i * 8));
339 }
340 return result;
341}
342
343constexpr quint128 QUuid::toUInt128(QSysInfo::Endian order) const noexcept
344{
345 quint128 result = {};
346 if (order == QSysInfo::BigEndian) {
347 for (int i = 0; i < 8; ++i)
348 result |= quint64(data4[i]) << (i * 8);
349 result = result << 64;
350 result |= quint64(qToBigEndian<quint16>(source: data3)) << 48;
351 result |= quint64(qToBigEndian<quint16>(source: data2)) << 32;
352 result |= qToBigEndian<quint32>(source: data1);
353 } else {
354 result = qToLittleEndian<quint32>(source: data1);
355 result = result << 32;
356 result |= quint64(qToLittleEndian<quint16>(source: data2)) << 16;
357 result |= quint64(qToLittleEndian<quint16>(source: data3));
358 result = result << 64;
359 for (int i = 0; i < 8; ++i)
360 result |= quint64(data4[i]) << (56 - i * 8);
361 }
362 return result;
363}
364#endif
365
366#if defined(Q_QDOC)
367// provide fake declarations of qXXXEndian() functions, so that qDoc could
368// distinguish them from the general template
369QUuid::Id128Bytes qFromBigEndian(QUuid::Id128Bytes src);
370QUuid::Id128Bytes qFromLittleEndian(QUuid::Id128Bytes src);
371QUuid::Id128Bytes qToBigEndian(QUuid::Id128Bytes src);
372QUuid::Id128Bytes qToLittleEndian(QUuid::Id128Bytes src);
373#endif
374
375QT_END_NAMESPACE
376
377#endif // QUUID_H
378

Provided by KDAB

Privacy Policy
Learn Advanced QML with KDAB
Find out more

source code of qtbase/src/corelib/plugin/quuid.h