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
qdebug.h
Go to the documentation of this file.
1// Copyright (C) 2016 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-parsing
5
6#ifndef QDEBUG_H
7#define QDEBUG_H
8
9#if 0
10#pragma qt_class(QtDebug)
11#endif
12
13#include <QtCore/qcompare.h>
14#include <QtCore/qcontainerfwd.h>
15#include <QtCore/qfloat16.h>
16#include <QtCore/qtextstream.h>
17#include <QtCore/qttypetraits.h>
18#include <QtCore/qtypes.h>
19#include <QtCore/qstring.h>
20#include <QtCore/qcontiguouscache.h>
21#include <QtCore/qsharedpointer.h>
22
23// all these have already been included by various headers above, but don't rely on indirect includes:
24#include <array>
25#include <chrono>
26#include <list>
27#include <map>
28#include <memory>
29#include <optional>
30#include <string>
31#include <string_view>
32#include <set>
33#include <tuple>
34#include <QtCore/q20type_traits.h>
35#include <utility>
36#include <unordered_map>
37#include <unordered_set>
38#include <vector>
39
40#if !defined(QT_LEAN_HEADERS) || QT_LEAN_HEADERS < 1
41# include <QtCore/qlist.h>
42# include <QtCore/qmap.h>
43# include <QtCore/qset.h>
44# include <QtCore/qvarlengtharray.h>
45#endif
46
47QT_BEGIN_NAMESPACE
48
49class QT6_ONLY(Q_CORE_EXPORT) QDebug : public QIODeviceBase
50{
51 friend class QMessageLogger;
52 friend class QDebugStateSaver;
53 friend class QDebugStateSaverPrivate;
54 struct Stream {
55 enum { VerbosityShift = 29, VerbosityMask = 0x7 };
56
57 explicit Stream(QIODevice *device)
58 : ts(device)
59 {}
60 explicit Stream(QString *string)
61 : ts(string, WriteOnly)
62 {}
63 explicit Stream(QByteArray *ba)
64 : ts(ba, WriteOnly)
65 {}
66 explicit Stream(QtMsgType t)
67 : ts(&buffer, WriteOnly),
68 type(t),
69 message_output(true)
70 {}
71 QTextStream ts;
72 QString buffer;
73 int ref = 1;
74 QtMsgType type = QtDebugMsg;
75 bool space = true;
76 bool noQuotes = false;
77 bool message_output = false;
78 int verbosity = DefaultVerbosity;
79 QMessageLogContext context;
80 } *stream;
81
82 enum Latin1Content { ContainsBinary = 0, ContainsLatin1 };
83
84 QT7_ONLY(Q_CORE_EXPORT) void putUcs4(uint ucs4);
85 QT7_ONLY(Q_CORE_EXPORT) void putString(const QChar *begin, size_t length);
86 QT7_ONLY(Q_CORE_EXPORT) void putByteArray(const char *begin, size_t length, Latin1Content content);
87 QT7_ONLY(Q_CORE_EXPORT) void putTimeUnit(qint64 num, qint64 den);
88 QT7_ONLY(Q_CORE_EXPORT) void putInt128(const void *i);
89 QT7_ONLY(Q_CORE_EXPORT) void putUInt128(const void *i);
90 QT7_ONLY(Q_CORE_EXPORT) void putQtOrdering(QtOrderingPrivate::QtOrderingTypeFlag flags,
91 Qt::partial_ordering order);
92
93 template <typename...Ts>
94 using if_streamable = std::enable_if_t<
95 std::conjunction_v<QTypeTraits::has_ostream_operator<QDebug, Ts>...>
96 , bool>;
97public:
98 explicit QDebug(QIODevice *device) : stream(new Stream(device)) {}
99 explicit QDebug(QString *string) : stream(new Stream(string)) {}
100 explicit QDebug(QByteArray *bytes) : stream(new Stream(bytes)) {}
101 explicit QDebug(QtMsgType t) : stream(new Stream(t)) {}
102 QDebug(const QDebug &o) : stream(o.stream) { ++stream->ref; }
103 QDebug(QDebug &&other) noexcept : stream{std::exchange(other.stream, nullptr)} {}
104 inline QDebug &operator=(const QDebug &other);
105 QT_MOVE_ASSIGNMENT_OPERATOR_IMPL_VIA_MOVE_AND_SWAP(QDebug)
106 ~QDebug();
107 void swap(QDebug &other) noexcept { qt_ptr_swap(stream, other.stream); }
108
109 QT7_ONLY(Q_CORE_EXPORT) QDebug &resetFormat();
110
111 inline QDebug &space() { stream->space = true; stream->ts << ' '; return *this; }
112 inline QDebug &nospace() { stream->space = false; return *this; }
113 inline QDebug &maybeSpace() { if (stream->space) stream->ts << ' '; return *this; }
114 inline QDebug &verbosity(int verbosityLevel) { stream->verbosity = verbosityLevel; return *this; }
115 int verbosity() const { return stream->verbosity; }
116 void setVerbosity(int verbosityLevel) { stream->verbosity = verbosityLevel; }
117 enum VerbosityLevel { MinimumVerbosity = 0, DefaultVerbosity = 2, MaximumVerbosity = 7 };
118
119 bool autoInsertSpaces() const { return stream->space; }
120 void setAutoInsertSpaces(bool b) { stream->space = b; }
121
122 [[nodiscard]] bool quoteStrings() const noexcept { return !stream->noQuotes; }
123 void setQuoteStrings(bool b) { stream->noQuotes = !b; }
124
125 inline QDebug &quote() { stream->noQuotes = false; return *this; }
126 inline QDebug &noquote() { stream->noQuotes = true; return *this; }
127 inline QDebug &maybeQuote(char c = '"') { if (!stream->noQuotes) stream->ts << c; return *this; }
128
129 inline QDebug &operator<<(QChar t) { putUcs4(t.unicode()); return maybeSpace(); }
130 inline QDebug &operator<<(bool t) { stream->ts << (t ? "true" : "false"); return maybeSpace(); }
131 inline QDebug &operator<<(char t) { stream->ts << t; return maybeSpace(); }
132 inline QDebug &operator<<(signed short t) { stream->ts << t; return maybeSpace(); }
133 inline QDebug &operator<<(unsigned short t) { stream->ts << t; return maybeSpace(); }
134 inline QDebug &operator<<(char16_t t) { return *this << QChar(t); }
135 inline QDebug &operator<<(char32_t t) { putUcs4(t); return maybeSpace(); }
136 inline QDebug &operator<<(signed int t) { stream->ts << t; return maybeSpace(); }
137 inline QDebug &operator<<(unsigned int t) { stream->ts << t; return maybeSpace(); }
138 inline QDebug &operator<<(signed long t) { stream->ts << t; return maybeSpace(); }
139 inline QDebug &operator<<(unsigned long t) { stream->ts << t; return maybeSpace(); }
140 inline QDebug &operator<<(qint64 t) { stream->ts << t; return maybeSpace(); }
141 inline QDebug &operator<<(quint64 t) { stream->ts << t; return maybeSpace(); }
142 inline QDebug &operator<<(qfloat16 t) { stream->ts << t; return maybeSpace(); }
143 inline QDebug &operator<<(float t) { stream->ts << t; return maybeSpace(); }
144 inline QDebug &operator<<(double t) { stream->ts << t; return maybeSpace(); }
145 inline QDebug &operator<<(const char* t) { stream->ts << QString::fromUtf8(t); return maybeSpace(); }
146 inline QDebug &operator<<(const char16_t *t) { stream->ts << QStringView(t); return maybeSpace(); }
147 inline QDebug &operator<<(const QString & t) { putString(t.constData(), size_t(t.size())); return maybeSpace(); }
148 inline QDebug &operator<<(QStringView s) { putString(s.data(), size_t(s.size())); return maybeSpace(); }
149 inline QDebug &operator<<(QUtf8StringView s) { putByteArray(reinterpret_cast<const char*>(s.data()), s.size(), ContainsBinary); return maybeSpace(); }
150 inline QDebug &operator<<(QLatin1StringView t) { putByteArray(t.latin1(), t.size(), ContainsLatin1); return maybeSpace(); }
151 inline QDebug &operator<<(const QByteArray & t) { putByteArray(t.constData(), t.size(), ContainsBinary); return maybeSpace(); }
152 inline QDebug &operator<<(QByteArrayView t) { putByteArray(t.constData(), t.size(), ContainsBinary); return maybeSpace(); }
153 inline QDebug &operator<<(const void * t) { stream->ts << t; return maybeSpace(); }
154 inline QDebug &operator<<(std::nullptr_t) { stream->ts << "(nullptr)"; return maybeSpace(); }
155 inline QDebug &operator<<(std::nullopt_t) { stream->ts << "nullopt"; return maybeSpace(); }
156 inline QDebug &operator<<(QTextStreamFunction f) {
157 stream->ts << f;
158 return *this;
159 }
160
161 inline QDebug &operator<<(QTextStreamManipulator m)
162 { stream->ts << m; return *this; }
163
164#ifdef Q_QDOC
165 template <typename Char, typename...Args>
166 QDebug &operator<<(const std::basic_string<Char, Args...> &s);
167
168 template <typename Char, typename...Args>
169 QDebug &operator<<(std::basic_string_view<Char, Args...> s);
170#else
171 template <typename...Args>
172 QDebug &operator<<(const std::basic_string<char, Args...> &s)
173 { return *this << QUtf8StringView(s); }
174
175 template <typename...Args>
176 QDebug &operator<<(std::basic_string_view<char, Args...> s)
177 { return *this << QUtf8StringView(s); }
178
179#ifdef __cpp_char8_t
180 template <typename...Args>
181 QDebug &operator<<(const std::basic_string<char8_t, Args...> &s)
182 { return *this << QUtf8StringView(s); }
183
184 template <typename...Args>
185 QDebug &operator<<(std::basic_string_view<char8_t, Args...> s)
186 { return *this << QUtf8StringView(s); }
187#endif // __cpp_char8_t
188
189 template <typename...Args>
190 QDebug &operator<<(const std::basic_string<char16_t, Args...> &s)
191 { return *this << QStringView(s); }
192
193 template <typename...Args>
194 QDebug &operator<<(std::basic_string_view<char16_t, Args...> s)
195 { return *this << QStringView(s); }
196
197 template <typename...Args>
198 QDebug &operator<<(const std::basic_string<wchar_t, Args...> &s)
199 {
200 if constexpr (sizeof(wchar_t) == 2)
201 return *this << QStringView(s);
202 else
203 return *this << QString::fromWCharArray(s.data(), s.size()); // ### optimize
204 }
205
206 template <typename...Args>
207 QDebug &operator<<(std::basic_string_view<wchar_t, Args...> s)
208 {
209 if constexpr (sizeof(wchar_t) == 2)
210 return *this << QStringView(s);
211 else
212 return *this << QString::fromWCharArray(s.data(), s.size()); // ### optimize
213 }
214
215 template <typename...Args>
216 QDebug &operator<<(const std::basic_string<char32_t, Args...> &s)
217 { return *this << QString::fromUcs4(s.data(), s.size()); }
218
219 template <typename...Args>
220 QDebug &operator<<(std::basic_string_view<char32_t, Args...> s)
221 { return *this << QString::fromUcs4(s.data(), s.size()); }
222#endif // !Q_QDOC
223
224 template <typename Rep, typename Period>
225 QDebug &operator<<(std::chrono::duration<Rep, Period> duration)
226 {
227 stream->ts << duration.count();
228 putTimeUnit(Period::num, Period::den);
229 return maybeSpace();
230 }
231
232#ifdef QT_SUPPORTS_INT128
233private:
234 // Constrained templates so they only match q(u)int128 without conversions.
235 // Also keeps these operators out of the ABI.
236 template <typename T>
237 using if_qint128 = std::enable_if_t<std::is_same_v<T, qint128>, bool>;
238 template <typename T>
239 using if_quint128 = std::enable_if_t<std::is_same_v<T, quint128>, bool>;
240public:
241 template <typename T, if_qint128<T> = true>
242 QDebug &operator<<(T i128) { putInt128(&i128); return maybeSpace(); }
243 template <typename T, if_quint128<T> = true>
244 QDebug &operator<<(T u128) { putUInt128(&u128); return maybeSpace(); }
245#endif // QT_SUPPORTS_INT128
246
247private:
248 template <typename T>
249 static void streamTypeErased(QDebug &d, const void *obj)
250 {
251 d << *static_cast<const T*>(obj);
252 }
253 using StreamTypeErased = void(*)(QDebug&, const void*);
254 QT7_ONLY(Q_CORE_EXPORT) static QString toStringImpl(StreamTypeErased s, const void *obj);
255 QT7_ONLY(Q_CORE_EXPORT) static QByteArray toBytesImpl(StreamTypeErased s, const void *obj);
256 QT7_ONLY(Q_CORE_EXPORT) QDebug &putTupleLikeImplImpl(const char *ns, const char *what, size_t n,
257 StreamTypeErased *ops, const void **data);
258
259 template <typename TupleLike, size_t...Is>
260 QDebug &putTupleLikeImpl(const char *ns, const char *what, const TupleLike &t,
261 std::index_sequence<Is...>)
262 {
263 if constexpr (sizeof...(Is)) {
264 StreamTypeErased ops[] = {
265 &streamTypeErased<q20::remove_cvref_t<std::tuple_element_t<Is, TupleLike>>>...
266 };
267 const void *data[] = {
268 std::addressof(std::get<Is>(t))...
269 };
270 return putTupleLikeImplImpl(ns, what, sizeof...(Is), ops, data);
271 } else {
272 return putTupleLikeImplImpl(ns, what, 0, nullptr, nullptr);
273 }
274 }
275
276 template <typename TupleLike>
277 QDebug &putTupleLike(const char *ns, const char *what, const TupleLike &t)
278 {
279 using Indexes = std::make_index_sequence<std::tuple_size_v<TupleLike>>;
280 return putTupleLikeImpl(ns, what, t, Indexes{});
281 }
282public:
283 template <typename T>
284 static QString toString(const T &object)
285 {
286 return toStringImpl(&streamTypeErased<T>, std::addressof(object));
287 }
288
289 template <typename T>
290 static QByteArray toBytes(const T &object)
291 {
292 return toBytesImpl(&streamTypeErased<T>, std::addressof(object));
293 }
294
295 template <typename...Ts, if_streamable<Ts...> = true>
296 QDebug &operator<<(const std::tuple<Ts...> &t)
297 {
298 return putTupleLike("std", "tuple", t);
299 }
300
301 template <typename T, if_streamable<T> = true>
302 QDebug &operator<<(const std::optional<T> &o)
303 {
304 if (!o)
305 return *this << std::nullopt;
306 StreamTypeErased s = &streamTypeErased<std::remove_cv_t<T>>;
307 const void *d = std::addressof(*o);
308 return putTupleLikeImplImpl("std", "optional", 1, &s, &d);
309 }
310
311private:
312 template <typename T>
313 using if_ordering_type = std::enable_if_t<QtOrderingPrivate::is_ordering_type_v<T>, bool>;
314
315 template <typename T, if_ordering_type<T> = true>
316 friend QDebug operator<<(QDebug debug, T t)
317 {
318 debug.putQtOrdering(QtOrderingPrivate::orderingFlagsFor(t), Qt::partial_ordering(t));
319 return debug;
320 }
321};
322
324
326class QDebugStateSaver
327{
328public:
329 Q_NODISCARD_CTOR Q_CORE_EXPORT
330 QDebugStateSaver(QDebug &dbg);
331 Q_CORE_EXPORT
332 ~QDebugStateSaver();
333private:
334 Q_DISABLE_COPY(QDebugStateSaver)
335 std::unique_ptr<QDebugStateSaverPrivate> d;
336};
337
338class QNoDebug
339{
340public:
341 inline QNoDebug &operator<<(QTextStreamFunction) { return *this; }
342 inline QNoDebug &operator<<(QTextStreamManipulator) { return *this; }
343 inline QNoDebug &space() { return *this; }
344 inline QNoDebug &nospace() { return *this; }
345 inline QNoDebug &maybeSpace() { return *this; }
346 inline QNoDebug &quote() { return *this; }
347 inline QNoDebug &noquote() { return *this; }
348 inline QNoDebug &maybeQuote(const char = '"') { return *this; }
349 inline QNoDebug &verbosity(int) { return *this; }
350
351 template<typename T>
352 inline QNoDebug &operator<<(const T &) { return *this; }
353};
354
355QNoDebug QMessageLogger::noDebug(...) const noexcept
356{ return {}; }
357
358inline QDebug &QDebug::operator=(const QDebug &other)
359{
360 QDebug{other}.swap(*this);
361 return *this;
362}
363
364namespace QtPrivate {
365
366template <typename SequentialContainer>
367inline QDebug printSequentialContainer(QDebug debug, const char *which, const SequentialContainer &c)
368{
369 const QDebugStateSaver saver(debug);
370 debug.nospace() << which << '(';
371 typename SequentialContainer::const_iterator it = c.begin(), end = c.end();
372 if (it != end) {
373 debug << *it;
374 ++it;
375 }
376 while (it != end) {
377 debug << ", " << *it;
378 ++it;
379 }
380 debug << ')';
381 return debug;
382}
383
384template <typename AssociativeContainer>
385inline QDebug printAssociativeContainer(QDebug debug, const char *which, const AssociativeContainer &c)
386{
387 const QDebugStateSaver saver(debug);
388 debug.nospace() << which << "(";
389 for (typename AssociativeContainer::const_iterator it = c.constBegin();
390 it != c.constEnd(); ++it) {
391 debug << '(' << it.key() << ", " << it.value() << ')';
392 }
393 debug << ')';
394 return debug;
395}
396
397} // namespace QtPrivate
398
399template<typename ...T>
400using QDebugIfHasDebugStream =
401 std::enable_if_t<std::conjunction_v<QTypeTraits::has_ostream_operator<QDebug, T>...>, QDebug>;
402
403template<typename Container, typename ...T>
404using QDebugIfHasDebugStreamContainer =
405 std::enable_if_t<std::conjunction_v<QTypeTraits::has_ostream_operator_container<QDebug, Container, T>...>, QDebug>;
406
407#ifndef Q_QDOC
408
409template<typename T>
410inline QDebugIfHasDebugStreamContainer<QList<T>, T> operator<<(QDebug debug, const QList<T> &vec)
411{
412 return QtPrivate::printSequentialContainer(std::move(debug), "QList", vec);
413}
414
415template<typename T, qsizetype P>
416inline QDebugIfHasDebugStream<T> operator<<(QDebug debug, const QVarLengthArray<T, P> &vec)
417{
418 return QtPrivate::printSequentialContainer(std::move(debug), "QVarLengthArray", vec);
419}
420
421template <typename T, typename Alloc>
422inline QDebugIfHasDebugStream<T> operator<<(QDebug debug, const std::vector<T, Alloc> &vec)
423{
424 return QtPrivate::printSequentialContainer(std::move(debug), "std::vector", vec);
425}
426
427template <typename T, std::size_t N>
428inline QDebugIfHasDebugStream<T> operator<<(QDebug debug, const std::array<T, N> &array)
429{
430 return QtPrivate::printSequentialContainer(std::move(debug), "std::array", array);
431}
432
433template <typename T, typename Alloc>
434inline QDebugIfHasDebugStream<T> operator<<(QDebug debug, const std::list<T, Alloc> &vec)
435{
436 return QtPrivate::printSequentialContainer(std::move(debug), "std::list", vec);
437}
438
439template <typename T>
440inline QDebugIfHasDebugStream<T> operator<<(QDebug debug, std::initializer_list<T> list)
441{
442 return QtPrivate::printSequentialContainer(std::move(debug), "std::initializer_list", list);
443}
444
445template <typename Key, typename T, typename Compare, typename Alloc>
446inline QDebugIfHasDebugStream<Key, T> operator<<(QDebug debug, const std::map<Key, T, Compare, Alloc> &map)
447{
448 return QtPrivate::printSequentialContainer(std::move(debug), "std::map", map); // yes, sequential: *it is std::pair
449}
450
451template <typename Key, typename T, typename Compare, typename Alloc>
452inline QDebugIfHasDebugStream<Key, T> operator<<(QDebug debug, const std::multimap<Key, T, Compare, Alloc> &map)
453{
454 return QtPrivate::printSequentialContainer(std::move(debug), "std::multimap", map); // yes, sequential: *it is std::pair
455}
456
457template <typename Key, typename Compare, typename Alloc>
458inline QDebugIfHasDebugStream<Key> operator<<(QDebug debug, const std::multiset<Key, Compare, Alloc> &multiset)
459{
460 return QtPrivate::printSequentialContainer(std::move(debug), "std::multiset", multiset);
461}
462
463template <typename Key, typename Compare, typename Alloc>
464inline QDebugIfHasDebugStream<Key> operator<<(QDebug debug, const std::set<Key, Compare, Alloc> &set)
465{
466 return QtPrivate::printSequentialContainer(std::move(debug), "std::set", set);
467}
468
469template <typename Key, typename T, typename Hash, typename KeyEqual, typename Alloc>
470inline QDebugIfHasDebugStream<Key, T> operator<<(QDebug debug, const std::unordered_map<Key, T, Hash, KeyEqual, Alloc> &unordered_map)
471{
472 return QtPrivate::printSequentialContainer(std::move(debug), "std::unordered_map", unordered_map); // yes, sequential: *it is std::pair
473}
474
475template <typename Key, typename Hash, typename KeyEqual, typename Alloc>
476inline QDebugIfHasDebugStream<Key> operator<<(QDebug debug, const std::unordered_set<Key, Hash, KeyEqual, Alloc> &unordered_set)
477{
478 return QtPrivate::printSequentialContainer(std::move(debug), "std::unordered_set", unordered_set);
479}
480
481template <class Key, class T>
482inline QDebugIfHasDebugStreamContainer<QMap<Key, T>, Key, T> operator<<(QDebug debug, const QMap<Key, T> &map)
483{
484 return QtPrivate::printAssociativeContainer(std::move(debug), "QMap", map);
485}
486
487template <class Key, class T>
488inline QDebugIfHasDebugStreamContainer<QMultiMap<Key, T>, Key, T> operator<<(QDebug debug, const QMultiMap<Key, T> &map)
489{
490 return QtPrivate::printAssociativeContainer(std::move(debug), "QMultiMap", map);
491}
492
493template <class Key, class T>
494inline QDebugIfHasDebugStreamContainer<QHash<Key, T>, Key, T> operator<<(QDebug debug, const QHash<Key, T> &hash)
495{
496 return QtPrivate::printAssociativeContainer(std::move(debug), "QHash", hash);
497}
498
499template <class Key, class T>
500inline QDebugIfHasDebugStreamContainer<QMultiHash<Key, T>, Key, T> operator<<(QDebug debug, const QMultiHash<Key, T> &hash)
501{
502 return QtPrivate::printAssociativeContainer(std::move(debug), "QMultiHash", hash);
503}
504
505template <class T1, class T2>
506inline QDebugIfHasDebugStream<T1, T2> operator<<(QDebug debug, const std::pair<T1, T2> &pair)
507{
508 const QDebugStateSaver saver(debug);
509 debug.nospace() << "std::pair(" << pair.first << ", " << pair.second << ')';
510 return debug;
511}
512
513template <typename T>
514inline QDebugIfHasDebugStreamContainer<QSet<T>, T> operator<<(QDebug debug, const QSet<T> &set)
515{
516 return QtPrivate::printSequentialContainer(std::move(debug), "QSet", set);
517}
518
519template <class T>
520inline QDebugIfHasDebugStream<T> operator<<(QDebug debug, const QContiguousCache<T> &cache)
521{
522 const QDebugStateSaver saver(debug);
523 debug.nospace() << "QContiguousCache(";
524 for (qsizetype i = cache.firstIndex(); i <= cache.lastIndex(); ++i) {
525 debug << cache[i];
526 if (i != cache.lastIndex())
527 debug << ", ";
528 }
529 debug << ')';
530 return debug;
531}
532
533#else
534template <class T>
535QDebug operator<<(QDebug debug, const QList<T> &list);
536
537template <class T, qsizetype P>
538QDebug operator<<(QDebug debug, const QVarLengthArray<T, P> &array);
539
540template <typename T, typename Alloc>
541QDebug operator<<(QDebug debug, const std::vector<T, Alloc> &vec);
542
543template <typename T, std::size_t N>
544QDebug operator<<(QDebug debug, const std::array<T, N> &array);
545
546template <typename T, typename Alloc>
547QDebug operator<<(QDebug debug, const std::list<T, Alloc> &vec);
548
549template <typename Key, typename T, typename Compare, typename Alloc>
550QDebug operator<<(QDebug debug, const std::map<Key, T, Compare, Alloc> &map);
551
552template <typename Key, typename T, typename Compare, typename Alloc>
553QDebug operator<<(QDebug debug, const std::multimap<Key, T, Compare, Alloc> &map);
554
555template <class Key, class T>
556QDebug operator<<(QDebug debug, const QMap<Key, T> &map);
557
558template <class Key, class T>
559QDebug operator<<(QDebug debug, const QMultiMap<Key, T> &map);
560
561template <class Key, class T>
562QDebug operator<<(QDebug debug, const QHash<Key, T> &hash);
563
564template <class Key, class T>
565QDebug operator<<(QDebug debug, const QMultiHash<Key, T> &hash);
566
567template <typename T>
568QDebug operator<<(QDebug debug, const QSet<T> &set);
569
570template <class T1, class T2>
571QDebug operator<<(QDebug debug, const std::pair<T1, T2> &pair);
572
573template <typename T>
574QDebug operator<<(QDebug debug, const QContiguousCache<T> &cache);
575
576template <typename Key, typename Compare, typename Alloc>
577QDebug operator<<(QDebug debug, const std::multiset<Key, Compare, Alloc> &multiset);
578
579template <typename Key, typename Compare, typename Alloc>
580QDebug operator<<(QDebug debug, const std::set<Key, Compare, Alloc> &set);
581
582template <typename Key, typename T, typename Hash, typename KeyEqual, typename Alloc>
583QDebug operator<<(QDebug debug, const std::unordered_map<Key, T, Hash, KeyEqual, Alloc> &unordered_map);
584
585template <typename Key, typename Hash, typename KeyEqual, typename Alloc>
586QDebug operator<<(QDebug debug, const std::unordered_set<Key, Hash, KeyEqual, Alloc> &unordered_set);
587
588#endif // Q_QDOC
589
590template <class T>
591inline QDebug operator<<(QDebug debug, const QSharedPointer<T> &ptr)
592{
593 QDebugStateSaver saver(debug);
594 debug.nospace() << "QSharedPointer(" << ptr.data() << ")";
595 return debug;
596}
597
598template <typename T, typename Tag> class QTaggedPointer;
599
600template <typename T, typename Tag>
601inline QDebug operator<<(QDebug debug, const QTaggedPointer<T, Tag> &ptr)
602{
603 QDebugStateSaver saver(debug);
604 debug.nospace() << "QTaggedPointer(" << ptr.pointer() << ", " << ptr.tag() << ")";
605 return debug;
606}
607
608Q_CORE_EXPORT QDebug qt_QMetaEnum_debugOperator(QDebug&, qint64 value, const QMetaObject *meta, const char *name);
609Q_CORE_EXPORT QDebug qt_QMetaEnum_flagDebugOperator(QDebug &dbg, quint64 value, const QMetaObject *meta, const char *name);
610Q_CORE_EXPORT void qt_QMetaEnum_flagDebugOperator(QDebug &debug, size_t sizeofT, uint value);
611Q_CORE_EXPORT void qt_QMetaEnum_flagDebugOperator(QDebug &debug, size_t sizeofT, quint64 value);
612
613template <typename Int>
614void qt_QMetaEnum_flagDebugOperator(QDebug &debug, size_t sizeofT, Int value)
615{
616 static_assert(std::is_unsigned_v<Int>,
617 "Cast value to an unsigned type before calling this function");
618 const QDebugStateSaver saver(debug);
619 debug.resetFormat();
620 debug.nospace() << "QFlags(" << Qt::hex << Qt::showbase;
621 bool needSeparator = false;
622 for (size_t i = 0; i < sizeofT * 8; ++i) {
623 if (value & (Int(1) << i)) {
624 if (needSeparator)
625 debug << '|';
626 else
627 needSeparator = true;
628 debug << (Int(1) << i);
629 }
630 }
631 debug << ')';
632}
633
634template <class Flags,
635 std::enable_if_t<QtPrivate::IsQFlags<Flags>::value, bool> = true>
636inline QDebug operator<<(QDebug debug, Flags flags)
637{
638 using T = typename Flags::enum_type;
639 using UInt = typename QIntegerForSizeof<T>::Unsigned;
640#if !defined(QT_NO_QOBJECT)
641 if constexpr (QtPrivate::IsQEnumHelper<T>::Value || QtPrivate::IsQEnumHelper<Flags>::Value) {
642 // if QFlags<T> is a Q_FLAG, we always zero-extend; if not, we need to
643 // allow it to sign-extend if it is signed (see QMetaEnum::valueToKeys)
644 using Int = std::conditional_t<QtPrivate::IsQEnumHelper<Flags>::Value, UInt,
645 std::underlying_type_t<T>>;
646 const QMetaObject *obj = qt_getEnumMetaObject(T());
647 const char *name = qt_getEnumName(T());
648 return qt_QMetaEnum_flagDebugOperator(debug, Int(flags.toInt()), obj, name);
649 } else
650#endif
651 {
652 qt_QMetaEnum_flagDebugOperator(debug, sizeof(T), UInt(flags.toInt()));
653 return debug;
654 }
655}
656
657#ifdef Q_QDOC
658template <typename T>
659QDebug operator<<(QDebug debug, const QFlags<T> &flags);
660#endif // Q_QDOC
661
662#if !defined(QT_NO_QOBJECT) && !defined(Q_QDOC)
663// Debugging of plain enums. There are three cases:
664// 1) the enum is part of a Q_DECLARE_FLAGS and there's a Q_FLAG for that
665// -> debugs as that QFlags (even if a Q_ENUM is present)
666// 2) the enum is declared a Q_ENUM but is not part of a Q_DECLARE_FLAGS
667// -> debugs via qt_QMetaEnum_debugOperator()
668// 3) the enum is not associated with a QMetaObject
669// -> no streaming
670// To avoid ambiguity in overload resolution, the template conditions are
671// mutually exclusive.
672
673namespace QtPrivate {
674template <typename T, bool IsEnum = std::is_enum_v<T>, bool = sizeof(T) <= sizeof(quint64)>
675struct EnumHasQFlag { static constexpr bool Value = false; };
676template <typename T> struct EnumHasQFlag<T, true, true> : QtPrivate::IsQEnumHelper<QFlags<T>> {};
677
678template <typename T, bool IsEnum = std::is_enum_v<T>, bool HasQFlag = EnumHasQFlag<T>::Value>
679struct EnumHasQEnum { static constexpr bool Value = false; };
680template <typename T> struct EnumHasQEnum<T, true, false> : QtPrivate::IsQEnumHelper<T> {};
681}
682
683template <typename T>
684std::enable_if_t<QtPrivate::EnumHasQFlag<T>::Value, QDebug> // case 1
685operator<<(QDebug debug, T flag)
686{
687 return debug << QFlags(flag);
688}
689
690template<typename T>
691std::enable_if_t<QtPrivate::EnumHasQEnum<T>::Value, QDebug> // case 2
692operator<<(QDebug dbg, T value)
693{
694 const QMetaObject *obj = qt_getEnumMetaObject(value);
695 const char *name = qt_getEnumName(value);
696 return qt_QMetaEnum_debugOperator(dbg, static_cast<typename std::underlying_type<T>::type>(value), obj, name);
697}
698#endif // !QT_NO_QOBJECT && !Q_QDOC
699
700inline QDebug operator<<(QDebug debug, QKeyCombination combination)
701{
702 QDebugStateSaver saver(debug);
703 debug.nospace() << "QKeyCombination("
704 << combination.keyboardModifiers()
705 << ", "
706 << combination.key()
707 << ")";
708 return debug;
709}
710
711#ifdef Q_OS_DARWIN
712
713// We provide QDebug stream operators for commonly used Core Foundation
714// and Core Graphics types, as well as NSObject. Additional CF/CG types
715// may be added by the user, using Q_DECLARE_QDEBUG_OPERATOR_FOR_CF_TYPE.
716
717#define QT_FOR_EACH_CORE_FOUNDATION_TYPE(F)
718 F(CFArray)
719 F(CFURL)
720 F(CFData)
721 F(CFNumber)
722 F(CFDictionary)
723 F(CFLocale)
724 F(CFDate)
725 F(CFBoolean)
726 F(CFTimeZone)
727
728#define QT_FOR_EACH_MUTABLE_CORE_FOUNDATION_TYPE(F)
729 F(CFError)
730 F(CFBundle)
731
732#define QT_FOR_EACH_CORE_GRAPHICS_TYPE(F)
733 F(CGPath)
734
735#define QT_FOR_EACH_MUTABLE_CORE_GRAPHICS_TYPE(F)
736 F(CGColorSpace)
737 F(CGImage)
738 F(CGFont)
739 F(CGColor)
740
741#define QT_FORWARD_DECLARE_CF_TYPE(type) Q_FORWARD_DECLARE_CF_TYPE(type);
742#define QT_FORWARD_DECLARE_MUTABLE_CF_TYPE(type) Q_FORWARD_DECLARE_MUTABLE_CF_TYPE(type);
743#define QT_FORWARD_DECLARE_CG_TYPE(type) Q_FORWARD_DECLARE_CG_TYPE(type);
744#define QT_FORWARD_DECLARE_MUTABLE_CG_TYPE(type) Q_FORWARD_DECLARE_MUTABLE_CG_TYPE(type);
745
746QT_END_NAMESPACE
747Q_FORWARD_DECLARE_CF_TYPE(CFString);
748struct objc_object;
749Q_FORWARD_DECLARE_OBJC_CLASS(NSObject);
750QT_FOR_EACH_CORE_FOUNDATION_TYPE(QT_FORWARD_DECLARE_CF_TYPE)
751QT_FOR_EACH_MUTABLE_CORE_FOUNDATION_TYPE(QT_FORWARD_DECLARE_MUTABLE_CF_TYPE)
752QT_FOR_EACH_CORE_GRAPHICS_TYPE(QT_FORWARD_DECLARE_CG_TYPE)
753QT_FOR_EACH_MUTABLE_CORE_GRAPHICS_TYPE(QT_FORWARD_DECLARE_MUTABLE_CG_TYPE)
754QT_BEGIN_NAMESPACE
755
756#define QT_FORWARD_DECLARE_QDEBUG_OPERATOR_FOR_CF_TYPE(CFType)
757 Q_CORE_EXPORT QDebug operator<<(QDebug, CFType##Ref);
758
759#define Q_DECLARE_QDEBUG_OPERATOR_FOR_CF_TYPE(CFType)
760 QDebug operator<<(QDebug debug, CFType##Ref ref)
761 {
762 if (!ref)
763 return debug << QT_STRINGIFY(CFType) "Ref(0x0)";
764 if (CFStringRef description = CFCopyDescription(ref)) {
765 QDebugStateSaver saver(debug);
766 debug.noquote() << description;
767 CFRelease(description);
768 }
769 return debug;
770 }
771
772// Defined in qcore_mac_objc.mm
773#if defined(__OBJC__)
774Q_CORE_EXPORT QDebug operator<<(QDebug, id);
775#endif
776Q_CORE_EXPORT QDebug operator<<(QDebug, objc_object *);
777Q_CORE_EXPORT QDebug operator<<(QDebug, const NSObject *);
778Q_CORE_EXPORT QDebug operator<<(QDebug, CFStringRef);
779
780QT_FOR_EACH_CORE_FOUNDATION_TYPE(QT_FORWARD_DECLARE_QDEBUG_OPERATOR_FOR_CF_TYPE)
781QT_FOR_EACH_MUTABLE_CORE_FOUNDATION_TYPE(QT_FORWARD_DECLARE_QDEBUG_OPERATOR_FOR_CF_TYPE)
782QT_FOR_EACH_CORE_GRAPHICS_TYPE(QT_FORWARD_DECLARE_QDEBUG_OPERATOR_FOR_CF_TYPE)
783QT_FOR_EACH_MUTABLE_CORE_GRAPHICS_TYPE(QT_FORWARD_DECLARE_QDEBUG_OPERATOR_FOR_CF_TYPE)
784
785#undef QT_FORWARD_DECLARE_CF_TYPE
786#undef QT_FORWARD_DECLARE_MUTABLE_CF_TYPE
787#undef QT_FORWARD_DECLARE_CG_TYPE
788#undef QT_FORWARD_DECLARE_MUTABLE_CG_TYPE
789
790#endif // Q_OS_DARWIN
791
792QT_END_NAMESPACE
793
794#endif // QDEBUG_H
QByteArray & operator*() noexcept
Definition qbytearray.h:797
QByteArray::Base64DecodingStatus decodingStatus
Definition qbytearray.h:782
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:801
void swap(QByteArray::FromBase64Result &other) noexcept
Definition qbytearray.h:784
operator bool() const noexcept
\variable QByteArray::FromBase64Result::decoded
Definition qbytearray.h:790
const QByteArray & operator*() const noexcept
Returns the decoded byte array.
Definition qbytearray.h:798
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:812
\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:65
std::optional< BacktraceStorage > backtrace
Definition qlogging_p.h:57
static constexpr int DefaultBacktraceDepth
Definition qlogging_p.h:47
Definition qlist.h:80
\inmodule QtCore
Definition qlogging.h:43
constexpr QMessageLogContext(const char *fileName, int lineNumber, const char *functionName, const char *categoryName) noexcept
Definition qlogging.h:48
const char * category
Definition qlogging.h:55
constexpr QMessageLogContext() noexcept=default
const char * function
Definition qlogging.h:54
const char * file
Definition qlogging.h:53
\inmodule QtCore
Definition qlogging.h:73
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:525
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:511
QDebug info(const QLoggingCategory &cat) const
Logs an informational message into the category cat using a QDebug stream.
Definition qlogging.cpp:614
QDebug info() const
Logs an informational message using a QDebug stream.
Definition qlogging.cpp:600
QNoDebug noDebug(...) const noexcept
QDebug info(CategoryFunction catFunc) const
static Q_CONSTINIT thread_local bool msgHandlerGrabbed
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:408
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:262
static const char endifTokenC[]
static bool isDefaultCategory(const char *category)
Definition qlogging.cpp:954
static const char messageTokenC[]
static bool qt_append_thread_name_to(QString &message)
Definition qlogging.cpp:247
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:152
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:138
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 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:186
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:287
void qSetMessagePattern(const QString &pattern)
QDebug printAssociativeContainer(QDebug debug, const char *which, const AssociativeContainer &c)
Definition qdebug.h:385
bool shouldLogToStderr()
Returns true if logging stderr should be ensured.
Definition qlogging.cpp:340
QDebug printSequentialContainer(QDebug debug, const char *which, const SequentialContainer &c)
Definition qdebug.h:367
QByteArray operator""_ba(const char *str, size_t size) noexcept
Definition qbytearray.h:847
Definition qcompare.h:76
QByteArray operator+(const QByteArray &a1, const char *a2)
Definition qbytearray.h:703
QByteArray qUncompress(const QByteArray &data)
Definition qbytearray.h:772
QByteArray operator+(char a1, const QByteArray &a2)
Definition qbytearray.h:713
QByteArray operator+(QByteArray &&lhs, char rhs)
Definition qbytearray.h:709
QByteArray operator+(const QByteArray &a1, char a2)
Definition qbytearray.h:707
QByteArray operator+(const char *a1, const QByteArray &a2)
Definition qbytearray.h:711
QByteArray operator+(QByteArray &&lhs, const QByteArray &rhs)
Definition qbytearray.h:701
qsizetype erase_if(QByteArray &ba, Predicate pred)
Definition qbytearray.h:830
QByteArray operator+(const QByteArray &a1, const QByteArray &a2)
Definition qbytearray.h:699
QByteArray qCompress(const QByteArray &data, int compressionLevel=-1)
Definition qbytearray.h:770
#define QT5_NULL_STRINGS
Definition qbytearray.h:26
qsizetype erase(QByteArray &ba, const T &t)
Definition qbytearray.h:824
QByteArray operator+(QByteArray &&lhs, const char *rhs)
Definition qbytearray.h:705
#define __has_include(x)
#define __has_cpp_attribute(x)
void qt_QMetaEnum_flagDebugOperator(QDebug &debug, size_t sizeofT, Int value)
Definition qdebug.h:614
Q_CORE_EXPORT void qt_QMetaEnum_flagDebugOperator(QDebug &debug, size_t sizeofT, quint64 value)
Definition qdebug.cpp:1428
Q_CORE_EXPORT void qt_QMetaEnum_flagDebugOperator(QDebug &debug, size_t sizeofT, uint value)
Definition qdebug.cpp:1419
Q_CORE_EXPORT QDebug operator<<(QDebug debug, QDir::Filters filters)
Definition qdir.cpp:2568
#define QT_MESSAGELOG_FUNC
Definition qlogging.h:161
#define QT_MESSAGELOG_FILE
Definition qlogging.h:159
#define QT_MESSAGE_LOGGER_NORETURN
Definition qlogging.h:69
#define QT_MESSAGELOG_LINE
Definition qlogging.h:160
Q_CORE_EXPORT void qSetMessagePattern(const QString &messagePattern)
#define QT_MESSAGELOGCONTEXT
Definition qlogging.h:154
QtMsgType
Definition qlogging.h:29
@ QtCriticalMsg
Definition qlogging.h:33
@ QtFatalMsg
Definition qlogging.h:34
@ QtDebugMsg
Definition qlogging.h:30
Q_CORE_EXPORT void qt_message_output(QtMsgType, const QMessageLogContext &context, const QString &message)
void(* QtMessageHandler)(QtMsgType, const QMessageLogContext &, const QString &)
Definition qlogging.h:196
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()
static constexpr bool Value
Definition qdebug.h:679
static constexpr bool Value
Definition qdebug.h:675