Qt
Internal/Contributor docs for the Qt SDK. Note: These are NOT official API docs; those are found at https://doc.qt.io/
Loading...
Searching...
No Matches
qbytearray.h
Go to the documentation of this file.
1// Copyright (C) 2022 The Qt Company Ltd.
2// Copyright (C) 2016 Intel Corporation.
3// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
4// Qt-Security score:critical reason:data-parser
5
6#ifndef QBYTEARRAY_H
7#define QBYTEARRAY_H
8
9#include <QtCore/qrefcount.h>
10#include <QtCore/qnamespace.h>
11#include <QtCore/qarraydata.h>
12#include <QtCore/qarraydatapointer.h>
13#include <QtCore/qcompare.h>
14#include <QtCore/qcontainerfwd.h>
15#include <QtCore/qbytearrayalgorithms.h>
16#include <QtCore/qbytearrayview.h>
17
18#include <stdlib.h>
19#include <string.h>
20
21#include <string>
22#include <iterator>
23
24#ifndef QT5_NULL_STRINGS
25// Would ideally be off, but in practice breaks too much (Qt 6.0).
26#define QT5_NULL_STRINGS 1
27#endif
28
29#ifdef truncate
30#error qbytearray.h must be included before any header file that defines truncate
31#endif
32
33#if defined(Q_OS_DARWIN) || defined(Q_QDOC)
34Q_FORWARD_DECLARE_CF_TYPE(CFData);
35Q_FORWARD_DECLARE_OBJC_CLASS(NSData);
36#endif
37
38#if defined(Q_OS_WASM) || defined(Q_QDOC)
39namespace emscripten {
40 class val;
41}
42#endif
43
44class tst_QByteArray;
45
46QT_BEGIN_NAMESPACE
47
48class QString;
49class QDataStream;
50
51using QByteArrayData = QArrayDataPointer<char>;
52
53# define QByteArrayLiteral(str)
54 (QByteArray(QByteArrayData(nullptr, const_cast<char *>(str), sizeof(str) - 1)))
55 /**/
56
57class Q_CORE_EXPORT QByteArray
58{
59public:
60 using DataPointer = QByteArrayData;
61private:
62 typedef QTypedArrayData<char> Data;
63
64 DataPointer d;
65 static const char _empty;
66
67 friend class ::tst_QByteArray;
68
69 template <typename InputIterator>
70 using if_input_iterator = QtPrivate::IfIsInputIterator<InputIterator>;
71public:
72 enum Base64Option {
73 Base64Encoding = 0,
74 Base64UrlEncoding = 1,
75
76 KeepTrailingEquals = 0,
77 OmitTrailingEquals = 2,
78
79 IgnoreBase64DecodingErrors = 0,
80 AbortOnBase64DecodingErrors = 4,
81 };
82 Q_DECLARE_FLAGS(Base64Options, Base64Option)
83
84 enum class Base64DecodingStatus {
85 Ok,
86 IllegalInputLength,
87 IllegalCharacter,
88 IllegalPadding,
89 };
90
91 inline constexpr QByteArray() noexcept;
92 QByteArray(const char *, qsizetype size = -1);
93 QByteArray(qsizetype size, char c);
94 QByteArray(qsizetype size, Qt::Initialization);
95 explicit QByteArray(QByteArrayView v) : QByteArray(v.data(), v.size()) {}
96 inline QByteArray(const QByteArray &) noexcept;
97 inline ~QByteArray();
98
99 QByteArray &operator=(const QByteArray &) noexcept;
100 QByteArray &operator=(const char *str);
101 inline QByteArray(QByteArray && other) noexcept
102 = default;
103 QT_MOVE_ASSIGNMENT_OPERATOR_IMPL_VIA_PURE_SWAP(QByteArray)
104 inline void swap(QByteArray &other) noexcept
105 { d.swap(other.d); }
106
107 constexpr bool isEmpty() const noexcept { return size() == 0; }
108 void resize(qsizetype size);
109 void resize(qsizetype size, char c);
110 void resizeForOverwrite(qsizetype size);
111
112 QByteArray &fill(char c, qsizetype size = -1);
113
114 inline qsizetype capacity() const;
115 inline void reserve(qsizetype size);
116 inline void squeeze();
117
118#ifndef QT_NO_CAST_FROM_BYTEARRAY
119 inline operator const char *() const;
120 inline operator const void *() const;
121#endif
122
123 // Some compilers consider this conversion ambiguous, so
124 // we're not offering it there:
125 // * QCC 8.3 on QNX
126 // * GHS 2022.1.4 on INTEGRITY
127#if (!defined(Q_OS_QNX) || !defined(Q_CC_GNU_ONLY) || Q_CC_GNU_ONLY > 803) &&
128 (!defined(Q_CC_GHS) || !defined(__GHS_VERSION_NUMBER) || __GHS_VERSION_NUMBER > 202214)
129# define QT_BYTEARRAY_CONVERTS_TO_STD_STRING_VIEW
130 Q_IMPLICIT operator std::string_view() const noexcept
131 { return std::string_view(data(), std::size_t(size())); }
132#endif
133
134 inline char *data();
135 inline const char *data() const noexcept;
136 const char *constData() const noexcept { return data(); }
137 inline void detach();
138 inline bool isDetached() const;
139 inline bool isSharedWith(const QByteArray &other) const noexcept
140 { return data() == other.data() && size() == other.size(); }
141 void clear();
142
143 inline char at(qsizetype i) const;
144 inline char operator[](qsizetype i) const;
145 [[nodiscard]] inline char &operator[](qsizetype i);
146 [[nodiscard]] char front() const { return at(0); }
147 [[nodiscard]] inline char &front();
148 [[nodiscard]] char back() const { return at(size() - 1); }
149 [[nodiscard]] inline char &back();
150
151 QT_CORE_INLINE_SINCE(6, 8)
152 qsizetype indexOf(char c, qsizetype from = 0) const;
153 qsizetype indexOf(QByteArrayView bv, qsizetype from = 0) const
154 { return QByteArrayView(begin(), size()).indexOf(bv, from); }
155
156 QT_CORE_INLINE_SINCE(6, 8)
157 qsizetype lastIndexOf(char c, qsizetype from = -1) const;
158 qsizetype lastIndexOf(QByteArrayView bv) const
159 { return lastIndexOf(bv, size()); }
160 qsizetype lastIndexOf(QByteArrayView bv, qsizetype from) const
161 { return QByteArrayView(begin(), size()).lastIndexOf(bv, from); }
162
163 inline bool contains(char c) const;
164 inline bool contains(QByteArrayView bv) const;
165 qsizetype count(char c) const;
166 qsizetype count(QByteArrayView bv) const
167 { return QByteArrayView(begin(), size()).count(bv); }
168
169 inline int compare(QByteArrayView a) const noexcept;
170 inline int compare(QByteArrayView a, Qt::CaseSensitivity cs) const noexcept;
171
172#if QT_CORE_REMOVED_SINCE(6, 7)
173 QByteArray left(qsizetype len) const;
174 QByteArray right(qsizetype len) const;
175 QByteArray mid(qsizetype index, qsizetype len = -1) const;
176 QByteArray first(qsizetype n) const;
177 QByteArray last(qsizetype n) const;
178 QByteArray sliced(qsizetype pos) const;
179 QByteArray sliced(qsizetype pos, qsizetype n) const;
180 QByteArray chopped(qsizetype len) const;
181#else
182 [[nodiscard]] QByteArray left(qsizetype n) const &
183 {
184 if (n >= size())
185 return *this;
186 return first(qMax(n, 0));
187 }
188 [[nodiscard]] QByteArray left(qsizetype n) &&
189 {
190 if (n >= size())
191 return std::move(*this);
192 return std::move(*this).first(qMax(n, 0));
193 }
194 [[nodiscard]] QByteArray right(qsizetype n) const &
195 {
196 if (n >= size())
197 return *this;
198 return last(qMax(n, 0));
199 }
200 [[nodiscard]] QByteArray right(qsizetype n) &&
201 {
202 if (n >= size())
203 return std::move(*this);
204 return std::move(*this).last(qMax(n, 0));
205 }
206 [[nodiscard]] QByteArray mid(qsizetype index, qsizetype len = -1) const &;
207 [[nodiscard]] QByteArray mid(qsizetype index, qsizetype len = -1) &&;
208
209 [[nodiscard]] QByteArray first(qsizetype n) const &
210 { verify(0, n); return sliced(0, n); }
211 [[nodiscard]] QByteArray last(qsizetype n) const &
212 { verify(0, n); return sliced(size() - n, n); }
213 [[nodiscard]] QByteArray sliced(qsizetype pos) const &
214 { verify(pos, 0); return sliced(pos, size() - pos); }
215 [[nodiscard]] QByteArray sliced(qsizetype pos, qsizetype n) const &
216 { verify(pos, n); return QByteArray(d.data() + pos, n); }
217 [[nodiscard]] QByteArray chopped(qsizetype len) const &
218 { verify(0, len); return sliced(0, size() - len); }
219
220 [[nodiscard]] QByteArray first(qsizetype n) &&
221 {
222 verify(0, n);
223 resize(n); // may detach and allocate memory
224 return std::move(*this);
225 }
226 [[nodiscard]] QByteArray last(qsizetype n) &&
227 { verify(0, n); return sliced_helper(*this, size() - n, n); }
228 [[nodiscard]] QByteArray sliced(qsizetype pos) &&
229 { verify(pos, 0); return sliced_helper(*this, pos, size() - pos); }
230 [[nodiscard]] QByteArray sliced(qsizetype pos, qsizetype n) &&
231 { verify(pos, n); return sliced_helper(*this, pos, n); }
232 [[nodiscard]] QByteArray chopped(qsizetype len) &&
233 { verify(0, len); chop(len); return std::move(*this); }
234#endif
235
236 bool startsWith(QByteArrayView bv) const
237 { return QByteArrayView(begin(), size()).startsWith(bv); }
238 bool startsWith(char c) const { return size() > 0 && front() == c; }
239
240 bool endsWith(char c) const { return size() > 0 && back() == c; }
241 bool endsWith(QByteArrayView bv) const
242 { return QByteArrayView(begin(), size()).endsWith(bv); }
243
244 bool isUpper() const;
245 bool isLower() const;
246
247 [[nodiscard]] bool isValidUtf8() const noexcept
248 {
249 return QtPrivate::isValidUtf8(qToByteArrayViewIgnoringNull(*this));
250 }
251
252 void truncate(qsizetype pos);
253 void chop(qsizetype n);
254
255 QByteArray &slice(qsizetype pos)
256 { verify(pos, 0); return remove(0, pos); }
257 QByteArray &slice(qsizetype pos, qsizetype n)
258 {
259 verify(pos, n);
260 if (isNull())
261 return *this;
262 resize(pos + n);
263 return remove(0, pos);
264 }
265
266#if !defined(Q_QDOC)
267 [[nodiscard]] QByteArray toLower() const &
268 { return toLower_helper(*this); }
269 [[nodiscard]] QByteArray toLower() &&
270 { return toLower_helper(*this); }
271 [[nodiscard]] QByteArray toUpper() const &
272 { return toUpper_helper(*this); }
273 [[nodiscard]] QByteArray toUpper() &&
274 { return toUpper_helper(*this); }
275 [[nodiscard]] QByteArray trimmed() const &
276 { return trimmed_helper(*this); }
277 [[nodiscard]] QByteArray trimmed() &&
278 { return trimmed_helper(*this); }
279 [[nodiscard]] QByteArray simplified() const &
280 { return simplified_helper(*this); }
281 [[nodiscard]] QByteArray simplified() &&
282 { return simplified_helper(*this); }
283#else
284 [[nodiscard]] QByteArray toLower() const;
285 [[nodiscard]] QByteArray toUpper() const;
286 [[nodiscard]] QByteArray trimmed() const;
287 [[nodiscard]] QByteArray simplified() const;
288#endif
289
290 [[nodiscard]] QByteArray leftJustified(qsizetype width, char fill = ' ', bool truncate = false) const;
291 [[nodiscard]] QByteArray rightJustified(qsizetype width, char fill = ' ', bool truncate = false) const;
292
293 QByteArray &prepend(char c)
294 { return insert(0, QByteArrayView(&c, 1)); }
295 inline QByteArray &prepend(qsizetype count, char c);
296 QByteArray &prepend(const char *s)
297 { return insert(0, QByteArrayView(s, qsizetype(qstrlen(s)))); }
298 QByteArray &prepend(const char *s, qsizetype len)
299 { return insert(0, QByteArrayView(s, len)); }
300 QByteArray &prepend(const QByteArray &a);
301 QByteArray &prepend(QByteArrayView a)
302 { return insert(0, a); }
303
304 QByteArray &append(char c);
305 inline QByteArray &append(qsizetype count, char c);
306 QByteArray &append(const char *s)
307 { return append(s, -1); }
308 QByteArray &append(const char *s, qsizetype len)
309 { return append(QByteArrayView(s, len < 0 ? qsizetype(qstrlen(s)) : len)); }
310 QByteArray &append(const QByteArray &a);
311 QByteArray &append(QByteArrayView a)
312 { return insert(size(), a); }
313
314 QByteArray &assign(QByteArrayView v);
315 QByteArray &assign(qsizetype n, char c)
316 {
317 Q_ASSERT(n >= 0);
318 return fill(c, n);
319 }
320 template <typename InputIterator, if_input_iterator<InputIterator> = true>
321 QByteArray &assign(InputIterator first, InputIterator last)
322 {
323 if constexpr (std::is_same_v<InputIterator, iterator> || std::is_same_v<InputIterator, const_iterator>)
324 return assign(QByteArrayView(first, last));
325 d->assign(first, last);
326 if (d.data())
327 d.data()[d.size] = '\0';
328 return *this;
329 }
330
331 QByteArray &insert(qsizetype i, QByteArrayView data);
332 inline QByteArray &insert(qsizetype i, const char *s)
333 { return insert(i, QByteArrayView(s)); }
334 inline QByteArray &insert(qsizetype i, const QByteArray &data)
335 { return insert(i, QByteArrayView(data)); }
336 QByteArray &insert(qsizetype i, qsizetype count, char c);
337 QByteArray &insert(qsizetype i, char c)
338 { return insert(i, QByteArrayView(&c, 1)); }
339 QByteArray &insert(qsizetype i, const char *s, qsizetype len)
340 { return insert(i, QByteArrayView(s, len)); }
341
342 QByteArray &remove(qsizetype index, qsizetype len);
343 QByteArray &removeAt(qsizetype pos)
344 { return size_t(pos) < size_t(size()) ? remove(pos, 1) : *this; }
345 QByteArray &removeFirst() { return !isEmpty() ? remove(0, 1) : *this; }
346 QByteArray &removeLast() { return !isEmpty() ? remove(size() - 1, 1) : *this; }
347
348 template <typename Predicate>
349 QByteArray &removeIf(Predicate pred)
350 {
351 removeIf_helper(pred);
352 return *this;
353 }
354
355 QByteArray &replace(qsizetype index, qsizetype len, const char *s, qsizetype alen)
356 { return replace(index, len, QByteArrayView(s, alen)); }
357 QByteArray &replace(qsizetype index, qsizetype len, QByteArrayView s);
358 QByteArray &replace(char before, QByteArrayView after)
359 { return replace(QByteArrayView(&before, 1), after); }
360 QByteArray &replace(const char *before, qsizetype bsize, const char *after, qsizetype asize)
361 { return replace(QByteArrayView(before, bsize), QByteArrayView(after, asize)); }
362 QByteArray &replace(QByteArrayView before, QByteArrayView after);
363 QByteArray &replace(char before, char after);
364
365 QByteArray &operator+=(char c)
366 { return append(c); }
367 QByteArray &operator+=(const char *s)
368 { return append(s); }
369 QByteArray &operator+=(const QByteArray &a)
370 { return append(a); }
371 QByteArray &operator+=(QByteArrayView a)
372 { return append(a); }
373
374 QList<QByteArray> split(char sep) const;
375
376 [[nodiscard]] QByteArray repeated(qsizetype times) const;
377
378#if !defined(QT_NO_CAST_FROM_ASCII) && !defined(QT_RESTRICTED_CAST_FROM_ASCII)
379#if QT_CORE_REMOVED_SINCE(6, 8)
380 QT_ASCII_CAST_WARN inline bool operator==(const QString &s2) const;
381 QT_ASCII_CAST_WARN inline bool operator!=(const QString &s2) const;
382 QT_ASCII_CAST_WARN inline bool operator<(const QString &s2) const;
383 QT_ASCII_CAST_WARN inline bool operator>(const QString &s2) const;
384 QT_ASCII_CAST_WARN inline bool operator<=(const QString &s2) const;
385 QT_ASCII_CAST_WARN inline bool operator>=(const QString &s2) const;
386#endif // QT_CORE_REMOVED_SINCE(6, 8)
387#endif // !defined(QT_NO_CAST_FROM_ASCII) && !defined(QT_RESTRICTED_CAST_FROM_ASCII)
388
389 short toShort(bool *ok = nullptr, int base = 10) const;
390 ushort toUShort(bool *ok = nullptr, int base = 10) const;
391 int toInt(bool *ok = nullptr, int base = 10) const;
392 uint toUInt(bool *ok = nullptr, int base = 10) const;
393 long toLong(bool *ok = nullptr, int base = 10) const;
394 ulong toULong(bool *ok = nullptr, int base = 10) const;
395 qlonglong toLongLong(bool *ok = nullptr, int base = 10) const;
396 qulonglong toULongLong(bool *ok = nullptr, int base = 10) const;
397 float toFloat(bool *ok = nullptr) const;
398 double toDouble(bool *ok = nullptr) const;
399 QByteArray toBase64(Base64Options options = Base64Encoding) const;
400 QByteArray toHex(char separator = '\0') const;
401 QByteArray toPercentEncoding(const QByteArray &exclude = QByteArray(),
402 const QByteArray &include = QByteArray(),
403 char percent = '%') const;
404#if QT_CORE_REMOVED_SINCE(6, 11)
405 [[nodiscard]] QByteArray percentDecoded(char percent = '%') const;
406#else
407 [[nodiscard]] QByteArray percentDecoded(char percent = '%') const &
408 { return fromPercentEncoding(*this, percent); }
409 [[nodiscard]] QByteArray percentDecoded(char percent = '%') &&
410 { return fromPercentEncoding(std::move(*this), percent); }
411#endif
412
413 inline QByteArray &setNum(short, int base = 10);
414 inline QByteArray &setNum(ushort, int base = 10);
415 inline QByteArray &setNum(int, int base = 10);
416 inline QByteArray &setNum(uint, int base = 10);
417 inline QByteArray &setNum(long, int base = 10);
418 inline QByteArray &setNum(ulong, int base = 10);
419 QByteArray &setNum(qlonglong, int base = 10);
420 QByteArray &setNum(qulonglong, int base = 10);
421 inline QByteArray &setNum(float, char format = 'g', int precision = 6);
422 QByteArray &setNum(double, char format = 'g', int precision = 6);
423 QByteArray &setRawData(const char *a, qsizetype n);
424
425 [[nodiscard]] static QByteArray number(int, int base = 10);
426 [[nodiscard]] static QByteArray number(uint, int base = 10);
427 [[nodiscard]] static QByteArray number(long, int base = 10);
428 [[nodiscard]] static QByteArray number(ulong, int base = 10);
429 [[nodiscard]] static QByteArray number(qlonglong, int base = 10);
430 [[nodiscard]] static QByteArray number(qulonglong, int base = 10);
431 [[nodiscard]] static QByteArray number(double, char format = 'g', int precision = 6);
432 [[nodiscard]] static QByteArray fromRawData(const char *data, qsizetype size)
433 {
434 return QByteArray(DataPointer::fromRawData(data, size));
435 }
436
437 class FromBase64Result;
438 [[nodiscard]] static FromBase64Result fromBase64Encoding(QByteArray &&base64, Base64Options options = Base64Encoding);
439 [[nodiscard]] static FromBase64Result fromBase64Encoding(const QByteArray &base64, Base64Options options = Base64Encoding);
440 [[nodiscard]] static QByteArray fromBase64(const QByteArray &base64, Base64Options options = Base64Encoding);
441 [[nodiscard]] static QByteArray fromHex(const QByteArray &hexEncoded);
442 [[nodiscard]] static QByteArray fromPercentEncoding(const QByteArray &pctEncoded, char percent = '%');
443 [[nodiscard]] static QByteArray fromPercentEncoding(QByteArray &&pctEncoded, char percent = '%');
444
445#if defined(Q_OS_DARWIN) || defined(Q_QDOC)
446 static QByteArray fromCFData(CFDataRef data);
447 static QByteArray fromRawCFData(CFDataRef data);
448 CFDataRef toCFData() const Q_DECL_CF_RETURNS_RETAINED;
449 CFDataRef toRawCFData() const Q_DECL_CF_RETURNS_RETAINED;
450 static QByteArray fromNSData(const NSData *data);
451 static QByteArray fromRawNSData(const NSData *data);
452 NSData *toNSData() const Q_DECL_NS_RETURNS_AUTORELEASED;
453 NSData *toRawNSData() const Q_DECL_NS_RETURNS_AUTORELEASED;
454#endif
455
456#if defined(Q_OS_WASM) || defined(Q_QDOC)
457 static QByteArray fromEcmaUint8Array(emscripten::val uint8array);
458 emscripten::val toEcmaUint8Array();
459#endif
460
461 typedef char *iterator;
462 typedef const char *const_iterator;
463 typedef iterator Iterator;
464 typedef const_iterator ConstIterator;
465 typedef std::reverse_iterator<iterator> reverse_iterator;
466 typedef std::reverse_iterator<const_iterator> const_reverse_iterator;
467 iterator begin() { return data(); }
468 const_iterator begin() const noexcept { return d.data(); }
469 const_iterator cbegin() const noexcept { return begin(); }
470 const_iterator constBegin() const noexcept { return begin(); }
471 iterator end() { return begin() + size(); }
472 const_iterator end() const noexcept { return begin() + size(); }
473 const_iterator cend() const noexcept { return end(); }
474 const_iterator constEnd() const noexcept { return end(); }
475 reverse_iterator rbegin() { return reverse_iterator(end()); }
476 reverse_iterator rend() { return reverse_iterator(begin()); }
477 const_reverse_iterator rbegin() const noexcept { return const_reverse_iterator(end()); }
478 const_reverse_iterator rend() const noexcept { return const_reverse_iterator(begin()); }
479 const_reverse_iterator crbegin() const noexcept { return rbegin(); }
480 const_reverse_iterator crend() const noexcept { return rend(); }
481
482 // stl compatibility
483 typedef qsizetype size_type;
484 typedef qptrdiff difference_type;
485 typedef const char & const_reference;
486 typedef char & reference;
487 typedef char *pointer;
488 typedef const char *const_pointer;
489 typedef char value_type;
490 void push_back(char c)
491 { append(c); }
492 void push_back(const char *s)
493 { append(s); }
494 void push_back(const QByteArray &a)
495 { append(a); }
496 void push_back(QByteArrayView a)
497 { append(a); }
498 void push_front(char c)
499 { prepend(c); }
500 void push_front(const char *c)
501 { prepend(c); }
502 void push_front(const QByteArray &a)
503 { prepend(a); }
504 void push_front(QByteArrayView a)
505 { prepend(a); }
506 void shrink_to_fit() { squeeze(); }
507 iterator erase(const_iterator first, const_iterator last);
508 inline iterator erase(const_iterator it) { return erase(it, it + 1); }
509 constexpr qsizetype max_size() const noexcept
510 {
511 return maxSize();
512 }
513
514 static QByteArray fromStdString(const std::string &s);
515 std::string toStdString() const;
516
517 static constexpr qsizetype maxSize() noexcept
518 {
519 // -1 to deal with the NUL terminator
520 return Data::maxSize() - 1;
521 }
522 constexpr qsizetype size() const noexcept
523 {
524 constexpr size_t MaxSize = maxSize();
525 Q_PRESUME(size_t(d.size) <= MaxSize);
526 return d.size;
527 }
528#if QT_DEPRECATED_SINCE(6, 4)
529 QT_DEPRECATED_VERSION_X_6_4("Use size() or length() instead.")
530 constexpr qsizetype count() const noexcept { return size(); }
531#endif
532 constexpr qsizetype length() const noexcept { return size(); }
533 QT_CORE_CONSTEXPR_INLINE_SINCE(6, 4)
534 bool isNull() const noexcept;
535
536 inline const DataPointer &data_ptr() const { return d; }
537 inline DataPointer &data_ptr() { return d; }
538#if QT_VERSION < QT_VERSION_CHECK(7, 0, 0)
539 explicit inline QByteArray(const DataPointer &dd) : d(dd) {}
540#endif
541 explicit inline QByteArray(DataPointer &&dd) : d(std::move(dd)) {}
542
543 [[nodiscard]] QByteArray nullTerminated() const &;
544 [[nodiscard]] QByteArray nullTerminated() &&;
545 QByteArray &nullTerminate();
546
547private:
548 friend bool comparesEqual(const QByteArray &lhs, char rhs) noexcept
549 { return QByteArrayView(lhs) == rhs; }
550 friend bool comparesEqual(const QByteArray &lhs, const QByteArrayView &rhs) noexcept
551 { return QByteArrayView(lhs) == rhs; }
552 friend Qt::strong_ordering
553 compareThreeWay(const QByteArray &lhs, char rhs) noexcept
554 {
555 return compareThreeWay(QByteArrayView(lhs), rhs);
556 }
557 friend Qt::strong_ordering
558 compareThreeWay(const QByteArray &lhs, const QByteArrayView &rhs) noexcept
559 {
560 const int res = QtPrivate::compareMemory(QByteArrayView(lhs), rhs);
561 return Qt::compareThreeWay(res, 0);
562 }
563 Q_DECLARE_STRONGLY_ORDERED(QByteArray)
564 Q_DECLARE_STRONGLY_ORDERED(QByteArray, char)
565 Q_DECLARE_STRONGLY_ORDERED(QByteArray, const char *)
566#if defined(__GLIBCXX__) && defined(__cpp_lib_three_way_comparison)
567 // libstdc++ has a bug [0] when `operator const void *()` is preferred over
568 // `operator<=>()` when calling std::less<> and other similar methods.
569 // Fix it by explicitly providing relational operators in such case.
570 // [0]: https://gcc.gnu.org/bugzilla/show_bug.cgi?id=114153
571 friend bool operator<(const QByteArray &lhs, const QByteArray &rhs) noexcept
572 { return is_lt(compareThreeWay(lhs, rhs)); }
573 friend bool operator<=(const QByteArray &lhs, const QByteArray &rhs) noexcept
574 { return is_lteq(compareThreeWay(lhs, rhs)); }
575 friend bool operator>(const QByteArray &lhs, const QByteArray &rhs) noexcept
576 { return is_gt(compareThreeWay(lhs, rhs)); }
577 friend bool operator>=(const QByteArray &lhs, const QByteArray &rhs) noexcept
578 { return is_gteq(compareThreeWay(lhs, rhs)); }
579#endif // defined(__GLIBCXX__) && defined(__cpp_lib_three_way_comparison)
580
581 // Check isEmpty() instead of isNull() for backwards compatibility.
582 friend bool comparesEqual(const QByteArray &lhs, std::nullptr_t) noexcept
583 { return lhs.isEmpty(); }
584 friend Qt::strong_ordering compareThreeWay(const QByteArray &lhs, std::nullptr_t) noexcept
585 { return lhs.isEmpty() ? Qt::strong_ordering::equivalent : Qt::strong_ordering::greater; }
586 Q_DECLARE_STRONGLY_ORDERED(QByteArray, std::nullptr_t)
587
588 // defined in qstring.cpp
589 friend Q_CORE_EXPORT bool comparesEqual(const QByteArray &lhs, const QChar &rhs) noexcept;
590 friend Q_CORE_EXPORT Qt::strong_ordering
591 compareThreeWay(const QByteArray &lhs, const QChar &rhs) noexcept;
592 friend Q_CORE_EXPORT bool comparesEqual(const QByteArray &lhs, char16_t rhs) noexcept;
593 friend Q_CORE_EXPORT Qt::strong_ordering
594 compareThreeWay(const QByteArray &lhs, char16_t rhs) noexcept;
595#if !defined(QT_NO_CAST_FROM_ASCII) && !defined(QT_RESTRICTED_CAST_FROM_ASCII)
596 Q_DECLARE_STRONGLY_ORDERED(QByteArray, QChar, QT_ASCII_CAST_WARN)
597 Q_DECLARE_STRONGLY_ORDERED(QByteArray, char16_t, QT_ASCII_CAST_WARN)
598#endif // !defined(QT_NO_CAST_FROM_ASCII) && !defined(QT_RESTRICTED_CAST_FROM_ASCII)
599
600
601 void reallocData(qsizetype alloc, QArrayData::AllocationOption option);
602 void reallocGrowData(qsizetype n);
603 void expand(qsizetype i);
604
605 Q_ALWAYS_INLINE constexpr void verify([[maybe_unused]] qsizetype pos = 0,
606 [[maybe_unused]] qsizetype n = 1) const
607 {
608 Q_ASSERT(pos >= 0);
609 Q_ASSERT(pos <= d.size);
610 Q_ASSERT(n >= 0);
611 Q_ASSERT(n <= d.size - pos);
612 }
613
614 static QByteArray sliced_helper(QByteArray &a, qsizetype pos, qsizetype n);
615 static QByteArray toLower_helper(const QByteArray &a);
616 static QByteArray toLower_helper(QByteArray &a);
617 static QByteArray toUpper_helper(const QByteArray &a);
618 static QByteArray toUpper_helper(QByteArray &a);
619 static QByteArray trimmed_helper(const QByteArray &a);
620 static QByteArray trimmed_helper(QByteArray &a);
621 static QByteArray simplified_helper(const QByteArray &a);
622 static QByteArray simplified_helper(QByteArray &a);
623 template <typename Predicate>
624 qsizetype removeIf_helper(Predicate pred)
625 {
626 const qsizetype result = d->eraseIf(pred);
627 if (result > 0)
628 d.data()[d.size] = '\0';
629 return result;
630 }
631
632 friend class QString;
633 friend Q_CORE_EXPORT QByteArray qUncompress(const uchar *data, qsizetype nbytes);
634
635 template <typename T> friend qsizetype erase(QByteArray &ba, const T &t);
636 template <typename Predicate> friend qsizetype erase_if(QByteArray &ba, Predicate pred);
637};
638
639Q_DECLARE_OPERATORS_FOR_FLAGS(QByteArray::Base64Options)
640
641inline constexpr QByteArray::QByteArray() noexcept {}
642inline QByteArray::~QByteArray() {}
643
644inline char QByteArray::at(qsizetype i) const
645{ verify(i, 1); return d.data()[i]; }
646inline char QByteArray::operator[](qsizetype i) const
647{ verify(i, 1); return d.data()[i]; }
648
649#ifndef QT_NO_CAST_FROM_BYTEARRAY
650inline QByteArray::operator const char *() const
651{ return data(); }
652inline QByteArray::operator const void *() const
653{ return data(); }
654#endif
655inline char *QByteArray::data()
656{
657 detach();
658 Q_ASSERT(d.data());
659 return d.data();
660}
661inline const char *QByteArray::data() const noexcept
662{
663#if QT5_NULL_STRINGS == 1
664 return d.data() ? d.data() : &_empty;
665#else
666 return d.data();
667#endif
668}
669inline void QByteArray::detach()
670{ if (d.needsDetach()) reallocData(size(), QArrayData::KeepSize); }
671inline bool QByteArray::isDetached() const
672{ return !d.isShared(); }
673inline QByteArray::QByteArray(const QByteArray &a) noexcept : d(a.d)
674{}
675
676inline qsizetype QByteArray::capacity() const { return qsizetype(d.constAllocatedCapacity()); }
677
678inline void QByteArray::reserve(qsizetype asize)
679{
680 if (d.needsDetach() || asize > capacity() - d.freeSpaceAtBegin())
681 reallocData(qMax(size(), asize), QArrayData::KeepSize);
682 if (d.constAllocatedCapacity())
683 d.setFlag(Data::CapacityReserved);
684}
685
686inline void QByteArray::squeeze()
687{
688 if (!d.isMutable())
689 return;
690 if (d.needsDetach() || size() < capacity())
691 reallocData(size(), QArrayData::KeepSize);
692 if (d.constAllocatedCapacity())
693 d.clearFlag(Data::CapacityReserved);
694}
695
696inline char &QByteArray::operator[](qsizetype i)
697{ verify(i, 1); return data()[i]; }
698inline char &QByteArray::front() { return operator[](0); }
699inline char &QByteArray::back() { return operator[](size() - 1); }
700inline QByteArray &QByteArray::append(qsizetype n, char ch)
701{ return insert(size(), n, ch); }
702inline QByteArray &QByteArray::prepend(qsizetype n, char ch)
703{ return insert(0, n, ch); }
704inline bool QByteArray::contains(char c) const
705{ return indexOf(c) != -1; }
706inline bool QByteArray::contains(QByteArrayView bv) const
707{ return indexOf(bv) != -1; }
708inline int QByteArray::compare(QByteArrayView a) const noexcept
709{
710 return QByteArrayView(*this).compare(a);
711}
712inline int QByteArray::compare(QByteArrayView a, Qt::CaseSensitivity cs) const noexcept
713{
714 return QByteArrayView(*this).compare(a, cs);
715}
716#if !defined(QT_USE_QSTRINGBUILDER)
717inline QByteArray operator+(const QByteArray &a1, const QByteArray &a2)
718{ return QByteArray(a1) += a2; }
719inline QByteArray operator+(QByteArray &&lhs, const QByteArray &rhs)
720{ return std::move(lhs += rhs); }
721inline QByteArray operator+(const QByteArray &a1, const char *a2)
722{ return QByteArray(a1) += a2; }
723inline QByteArray operator+(QByteArray &&lhs, const char *rhs)
724{ return std::move(lhs += rhs); }
725inline QByteArray operator+(const QByteArray &a1, char a2)
726{ return QByteArray(a1) += a2; }
727inline QByteArray operator+(QByteArray &&lhs, char rhs)
728{ return std::move(lhs += rhs); }
729inline QByteArray operator+(const char *a1, const QByteArray &a2)
730{ return QByteArray(a1) += a2; }
731inline QByteArray operator+(char a1, const QByteArray &a2)
732{ return QByteArray(&a1, 1) += a2; }
734inline QByteArray operator+(const QByteArray &lhs, QByteArrayView rhs)
735{
736 QByteArray tmp{lhs.size() + rhs.size(), Qt::Uninitialized};
737 return tmp.assign(lhs).append(rhs);
738}
740inline QByteArray operator+(QByteArrayView lhs, const QByteArray &rhs)
741{
742 QByteArray tmp{lhs.size() + rhs.size(), Qt::Uninitialized};
743 return tmp.assign(lhs).append(rhs);
744}
745#endif // QT_USE_QSTRINGBUILDER
746
747inline QByteArray &QByteArray::setNum(short n, int base)
748{ return setNum(qlonglong(n), base); }
749inline QByteArray &QByteArray::setNum(ushort n, int base)
750{ return setNum(qulonglong(n), base); }
751inline QByteArray &QByteArray::setNum(int n, int base)
752{ return setNum(qlonglong(n), base); }
753inline QByteArray &QByteArray::setNum(uint n, int base)
754{ return setNum(qulonglong(n), base); }
755inline QByteArray &QByteArray::setNum(long n, int base)
756{ return setNum(qlonglong(n), base); }
757inline QByteArray &QByteArray::setNum(ulong n, int base)
758{ return setNum(qulonglong(n), base); }
759inline QByteArray &QByteArray::setNum(float n, char format, int precision)
760{ return setNum(double(n), format, precision); }
761
762#if QT_CORE_INLINE_IMPL_SINCE(6, 4)
763QT_CORE_CONSTEXPR_INLINE_SINCE(6, 4)
764bool QByteArray::isNull() const noexcept
765{
766 return d.isNull();
767}
768#endif
769#if QT_CORE_INLINE_IMPL_SINCE(6, 8)
770qsizetype QByteArray::indexOf(char ch, qsizetype from) const
771{
772 return qToByteArrayViewIgnoringNull(*this).indexOf(ch, from);
773}
774qsizetype QByteArray::lastIndexOf(char ch, qsizetype from) const
775{
776 return qToByteArrayViewIgnoringNull(*this).lastIndexOf(ch, from);
777}
778#endif
779
780#if !defined(QT_NO_DATASTREAM) || defined(QT_BOOTSTRAPPED)
781Q_CORE_EXPORT QDataStream &operator<<(QDataStream &, const QByteArray &);
782Q_CORE_EXPORT QDataStream &operator>>(QDataStream &, QByteArray &);
783#endif
784
785#ifndef QT_NO_COMPRESS
786Q_CORE_EXPORT QByteArray qCompress(const uchar* data, qsizetype nbytes, int compressionLevel = -1);
787Q_CORE_EXPORT QByteArray qUncompress(const uchar* data, qsizetype nbytes);
788inline QByteArray qCompress(const QByteArray& data, int compressionLevel = -1)
789{ return qCompress(reinterpret_cast<const uchar *>(data.constData()), data.size(), compressionLevel); }
790inline QByteArray qUncompress(const QByteArray& data)
791{ return qUncompress(reinterpret_cast<const uchar*>(data.constData()), data.size()); }
792#endif
793
795
797{
798public:
801
802 void swap(QByteArray::FromBase64Result &other) noexcept
803 {
804 decoded.swap(other.decoded);
805 std::swap(decodingStatus, other.decodingStatus);
806 }
807
808 explicit operator bool() const noexcept { return decodingStatus == QByteArray::Base64DecodingStatus::Ok; }
809
810 QByteArray &operator*() & noexcept { return decoded; }
811 const QByteArray &operator*() const & noexcept { return decoded; }
812 QByteArray &&operator*() && noexcept { return std::move(decoded); }
813 const QByteArray &&operator*() const && noexcept { return std::move(decoded); }
814
815 friend inline bool operator==(const QByteArray::FromBase64Result &lhs, const QByteArray::FromBase64Result &rhs) noexcept
816 {
817 if (lhs.decodingStatus != rhs.decodingStatus)
818 return false;
819
820 if (lhs.decodingStatus == QByteArray::Base64DecodingStatus::Ok && lhs.decoded != rhs.decoded)
821 return false;
822
823 return true;
824 }
825
826 friend inline bool operator!=(const QByteArray::FromBase64Result &lhs, const QByteArray::FromBase64Result &rhs) noexcept
827 {
828 return !(lhs == rhs);
829 }
830};
831
832Q_DECLARE_SHARED(QByteArray::FromBase64Result)
833
834
835Q_CORE_EXPORT Q_DECL_PURE_FUNCTION size_t qHash(const QByteArray::FromBase64Result &key, size_t seed = 0) noexcept;
836
837template <typename T>
838qsizetype erase(QByteArray &ba, const T &t)
839{
840 return ba.removeIf_helper([&t](const auto &e) { return t == e; });
841}
842
843template <typename Predicate>
844qsizetype erase_if(QByteArray &ba, Predicate pred)
845{
846 return ba.removeIf_helper(pred);
847}
848
849//
850// QByteArrayView members that require QByteArray:
851//
852QByteArray QByteArrayView::toByteArray() const
853{
854 return QByteArray(*this);
855}
856
857namespace Qt {
858inline namespace Literals {
859inline namespace StringLiterals {
860
861inline QByteArray operator""_ba(const char *str, size_t size) noexcept
862{
863 return QByteArray(QByteArrayData(nullptr, const_cast<char *>(str), qsizetype(size)));
864}
865
866} // StringLiterals
867} // Literals
868} // Qt
869
870inline namespace QtLiterals {
871#if QT_DEPRECATED_SINCE(6, 8)
872
873QT_DEPRECATED_VERSION_X_6_8("Use _ba from Qt::StringLiterals namespace instead.")
874inline QByteArray operator""_qba(const char *str, size_t size) noexcept
875{
876 return Qt::StringLiterals::operator""_ba(str, size);
877}
878
879#endif // QT_DEPRECATED_SINCE(6, 8)
880} // QtLiterals
881
882QT_END_NAMESPACE
883
884#endif // QBYTEARRAY_H
QByteArray::Base64DecodingStatus decodingStatus
Definition qbytearray.h:800
friend bool operator==(const QByteArray::FromBase64Result &lhs, const QByteArray::FromBase64Result &rhs) noexcept
Returns true if lhs and rhs are equal, otherwise returns false.
Definition qbytearray.h:815
QByteArray & operator*() &noexcept
Definition qbytearray.h:810
const QByteArray & operator*() const &noexcept
Definition qbytearray.h:811
void swap(QByteArray::FromBase64Result &other) noexcept
Definition qbytearray.h:802
operator bool() const noexcept
\variable QByteArray::FromBase64Result::decoded
Definition qbytearray.h:808
QByteArray && operator*() &&noexcept
Definition qbytearray.h:812
const QByteArray && operator*() const &&noexcept
Definition qbytearray.h:813
friend bool operator!=(const QByteArray::FromBase64Result &lhs, const QByteArray::FromBase64Result &rhs) noexcept
Returns true if lhs and rhs are different, otherwise returns false.
Definition qbytearray.h:826
\inmodule QtCore
Definition qbytearray.h:58
\inmodule QtCore\reentrant
Definition qdatastream.h:50
int initFrom(const QMessageLogContext &logContext)
void populateBacktrace(int frameCount)
QInternalMessageLogContext(const QMessageLogContext &logContext, const QLoggingCategory &categoryOverride)
Definition qlogging_p.h:67
static constexpr int MaxBacktraceDepth
Definition qlogging_p.h:49
std::optional< BacktraceStorage > backtrace
Definition qlogging_p.h:59
static constexpr int DefaultBacktraceDepth
Definition qlogging_p.h:48
Definition qlist.h:81
\inmodule QtCore
Definition qlogging.h:44
constexpr QMessageLogContext(const char *fileName, int lineNumber, const char *functionName, const char *categoryName) noexcept
Definition qlogging.h:49
const char * category
Definition qlogging.h:56
constexpr QMessageLogContext() noexcept=default
const char * function
Definition qlogging.h:55
const char * file
Definition qlogging.h:54
\inmodule QtCore
Definition qlogging.h:74
QDebug debug(CategoryFunction catFunc) const
QDebug debug(const QLoggingCategory &cat) const
Logs a debug message into category cat using a QDebug stream.
Definition qlogging.cpp:534
void void void void Q_DECL_COLD_FUNCTION void Q_DECL_COLD_FUNCTION void Q_DECL_COLD_FUNCTION void Q_DECL_COLD_FUNCTION void QT_MESSAGE_LOGGER_NORETURN Q_DECL_COLD_FUNCTION void QT_MESSAGE_LOGGER_NORETURN Q_DECL_COLD_FUNCTION void QDebug debug() const
Logs a debug message using a QDebug stream.
Definition qlogging.cpp:520
QDebug info(const QLoggingCategory &cat) const
Logs an informational message into the category cat using a QDebug stream.
Definition qlogging.cpp:623
QDebug info() const
Logs an informational message using a QDebug stream.
Definition qlogging.cpp:609
QNoDebug noDebug(...) const noexcept
QDebug info(CategoryFunction catFunc) const
\macro QT_RESTRICTED_CAST_FROM_ASCII
Definition qstring.h:177
static const char ifCriticalTokenC[]
static bool grabMessageHandler()
void qt_message_output(QtMsgType msgType, const QMessageLogContext &context, const QString &message)
static const char emptyTokenC[]
static Q_NEVER_INLINE void qt_message(QtMsgType msgType, const QMessageLogContext &context, const char *msg, va_list ap)
Definition qlogging.cpp:417
static void preformattedMessageHandler(QtMsgType type, const QMessageLogContext &context, const QString &formattedMessage)
static bool systemHasStderr()
Returns true if writing to stderr is supported.
Definition qlogging.cpp:267
static const char endifTokenC[]
static bool isDefaultCategory(const char *category)
Definition qlogging.cpp:963
static const char messageTokenC[]
static bool qt_append_thread_name_to(QString &message)
Definition qlogging.cpp:252
static constexpr SystemMessageSink systemMessageSink
static void qt_maybe_message_fatal(QtMsgType, const QMessageLogContext &context, String &&message)
\inmodule QtCore \title Qt Logging Types
#define HANDLE_IF_TOKEN(LEVEL)
Q_DECLARE_TYPEINFO(QMessagePattern::BacktraceParams, Q_RELOCATABLE_TYPE)
static void qDefaultMessageHandler(QtMsgType type, const QMessageLogContext &context, const QString &buf)
static const char timeTokenC[]
static bool isFatalCountDown(const char *varname, QBasicAtomicInt &n)
Definition qlogging.cpp:157
void qErrnoWarning(int code, const char *msg,...)
static const char qthreadptrTokenC[]
static const char fileTokenC[]
static const char ifDebugTokenC[]
static const char ifFatalTokenC[]
static const char categoryTokenC[]
static void stderr_message_handler(QtMsgType type, const QMessageLogContext &context, const QString &formattedMessage)
static const char lineTokenC[]
static const char typeTokenC[]
static void ungrabMessageHandler()
static void copyInternalContext(QInternalMessageLogContext *self, const QMessageLogContext &logContext) noexcept
static const char ifCategoryTokenC[]
static int checked_var_value(const char *varname)
Definition qlogging.cpp:143
static const char threadnameTokenC[]
static const char pidTokenC[]
Q_TRACE_POINT(qtcore, qt_message_print, int type, const char *category, const char *function, const char *file, int line, const QString &message)
static const char threadidTokenC[]
static QString formatLogMessage(QtMsgType type, const QMessageLogContext &context, const QString &str)
static Q_CONSTINIT bool msgHandlerGrabbed
static const char backtraceTokenC[]
void qErrnoWarning(const char *msg,...)
static const char functionTokenC[]
#define IF_TOKEN(LEVEL)
static const char ifWarningTokenC[]
static const char appnameTokenC[]
static bool isFatal(QtMsgType msgType)
Definition qlogging.cpp:191
static const char ifInfoTokenC[]
QtMessageHandler qInstallMessageHandler(QtMessageHandler h)
static void qt_message_print(QtMsgType, const QMessageLogContext &context, const QString &message)
static bool stderrHasConsoleAttached()
Returns true if writing to stderr will end up in a console/terminal visible to the user.
Definition qlogging.cpp:296
void qSetMessagePattern(const QString &pattern)
Combined button and popup list for selecting options.
bool shouldLogToStderr()
Returns true if logging stderr should be ensured.
Definition qlogging.cpp:349
QByteArray operator""_ba(const char *str, size_t size) noexcept
Definition qbytearray.h:861
Definition qcompare.h:111
QByteArray operator+(const QByteArray &a1, const char *a2)
Definition qbytearray.h:721
QByteArray qUncompress(const QByteArray &data)
Definition qbytearray.h:790
QByteArray operator+(char a1, const QByteArray &a2)
Definition qbytearray.h:731
QByteArray operator+(QByteArray &&lhs, char rhs)
Definition qbytearray.h:727
QByteArray operator+(const QByteArray &a1, char a2)
Definition qbytearray.h:725
QByteArray operator+(const char *a1, const QByteArray &a2)
Definition qbytearray.h:729
QByteArray operator+(QByteArray &&lhs, const QByteArray &rhs)
Definition qbytearray.h:719
qsizetype erase_if(QByteArray &ba, Predicate pred)
Definition qbytearray.h:844
QByteArray operator+(const QByteArray &a1, const QByteArray &a2)
Definition qbytearray.h:717
QByteArray qCompress(const QByteArray &data, int compressionLevel=-1)
Definition qbytearray.h:788
#define QT5_NULL_STRINGS
Definition qbytearray.h:26
qsizetype erase(QByteArray &ba, const T &t)
Definition qbytearray.h:838
QByteArray operator+(QByteArray &&lhs, const char *rhs)
Definition qbytearray.h:723
#define __has_include(x)
Q_CORE_EXPORT QDebug operator<<(QDebug debug, QDir::Filters filters)
Definition qdir.cpp:2598
#define QT_MESSAGELOG_FUNC
Definition qlogging.h:162
#define QT_MESSAGELOG_FILE
Definition qlogging.h:160
#define QT_MESSAGE_LOGGER_NORETURN
Definition qlogging.h:70
#define QT_MESSAGELOG_LINE
Definition qlogging.h:161
Q_CORE_EXPORT void qSetMessagePattern(const QString &messagePattern)
#define QT_MESSAGELOGCONTEXT
Definition qlogging.h:155
QtMsgType
Definition qlogging.h:30
@ QtCriticalMsg
Definition qlogging.h:34
@ QtFatalMsg
Definition qlogging.h:35
@ QtDebugMsg
Definition qlogging.h:31
Q_CORE_EXPORT void qt_message_output(QtMsgType, const QMessageLogContext &context, const QString &message)
void(* QtMessageHandler)(QtMsgType, const QMessageLogContext &, const QString &)
Definition qlogging.h:197
QMutex QBasicMutex
Definition qmutex.h:360
void setPattern(const QString &pattern)
std::unique_ptr< std::unique_ptr< const char[]>[]> literals
std::chrono::steady_clock::time_point appStartTime
std::unique_ptr< const char *[]> tokens
QList< QString > timeArgs
static QBasicMutex mutex
void setDefaultPattern()