1// Copyright (C) 2013 Klarälvdalens Datakonsult AB, a KDAB Group company, info@kdab.com, author Giuseppe D'Angelo <giuseppe.dangelo@kdab.com>
2// SPDX-License-Identifier: LicenseRef-Qt-Commercial OR LGPL-3.0-only OR GPL-2.0-only OR GPL-3.0-only
3
4#include <QtCore/private/qobject_p.h>
5#include <QtCore/qglobal.h>
6#include <QtCore/qvarlengtharray.h>
7#include <QtGui/qopengl.h>
8#include <QtGui/qopenglfunctions.h>
9#include <QtGui/qoffscreensurface.h>
10
11#include "qopengldebug.h"
12
13QT_BEGIN_NAMESPACE
14
15QT_IMPL_METATYPE_EXTERN(QOpenGLDebugMessage)
16
17/*!
18 \class QOpenGLDebugMessage
19 \brief The QOpenGLDebugMessage class wraps an OpenGL debug message.
20 \inmodule QtOpenGL
21 \reentrant
22 \since 5.1
23 \ingroup shared
24 \ingroup painting-3D
25
26 Debug messages are usually created by the OpenGL server and then read by
27 OpenGL clients (either from the OpenGL internal debug log, or logged in real-time).
28 A debug message has a textual representation, a vendor-specific numeric id,
29 a source, a type and a severity.
30
31 It's also possible for applications or third-party libraries and toolkits
32 to create and insert messages in the debug log. In order to do so, you can use
33 the createApplicationMessage() or the createThirdPartyMessage() static functions.
34
35 \sa QOpenGLDebugLogger
36*/
37
38/*!
39 \class QOpenGLDebugLogger
40 \brief The QOpenGLDebugLogger enables logging of OpenGL debugging messages.
41 \inmodule QtOpenGL
42 \since 5.1
43 \ingroup painting-3D
44
45 \section1 Introduction
46
47 OpenGL programming can be very error prone. Most of the time, a single
48 failing call to OpenGL can cause an entire portion of an application to
49 stop working, with nothing being drawn on the screen.
50
51 The only way to be sure that no errors are being returned from the OpenGL
52 implementation is checking with \c{glGetError} after each and every API
53 call. Moreover, OpenGL errors stack up, therefore glGetError should always
54 be used in a loop like this:
55
56 \snippet code/src_gui_opengl_qopengldebug.cpp 0
57
58 If you try to clear the error stack, make sure not just keep going until
59 GL_NO_ERROR is returned but also break on GL_CONTEXT_LOST as that error
60 value will keep repeating.
61
62 There are also many other information we are interested in (as application
63 developers), for instance performance issues, or warnings about using
64 deprecated APIs. Those kind of messages are not reported through the
65 ordinary OpenGL error reporting mechanisms.
66
67 QOpenGLDebugLogger aims at addressing these issues by providing access to
68 the \e{OpenGL debug log}. If your OpenGL implementation supports it (by
69 exposing the \c{GL_KHR_debug} extension), messages from the OpenGL server
70 will be either logged in an internal OpenGL log, or passed in "real-time"
71 to listeners as they're generated from OpenGL.
72
73 QOpenGLDebugLogger supports both these modes of operation. Refer to the
74 following sections to find out the differences between them.
75
76 \section1 Creating an OpenGL Debug Context
77
78 For efficiency reasons, OpenGL implementations are allowed not to create
79 any debug output at all, unless the OpenGL context is a debug context. In order
80 to create a debug context from Qt, you must set the QSurfaceFormat::DebugContext
81 format option on the QSurfaceFormat used to create the QOpenGLContext object:
82
83 \snippet code/src_gui_opengl_qopengldebug.cpp 1
84
85 Note that requesting a 3.2 OpenGL Core Profile is just for the example's
86 purposes; this class is not tied to any specific OpenGL or OpenGL ES
87 version, as it relies on the availability of the \c{GL_KHR_debug} extension
88 (see below).
89
90 \section1 Creating and Initializing a QOpenGLDebugLogger
91
92 QOpenGLDebugLogger is a simple QObject-derived class. Just like all QObject
93 subclasses, you create an instance (and optionally specify a parent
94 object), and like the other OpenGL functions in Qt you \e{must} initialize
95 it before usage by calling initialize() whilst there is a current OpenGL context:
96
97 \snippet code/src_gui_opengl_qopengldebug.cpp 2
98
99 Note that the \c{GL_KHR_debug} extension \e{must} be available in the context
100 in order to access the messages logged by OpenGL. You can check the
101 presence of this extension by calling:
102
103 \snippet code/src_gui_opengl_qopengldebug.cpp 3
104
105 where \c{ctx} is a valid QOpenGLContext. If the extension is not available,
106 initialize() will return false.
107
108 \section1 Reading the Internal OpenGL Debug Log
109
110 OpenGL implementations keep an internal log of debug messages. Messages
111 stored in this log can be retrieved by using the loggedMessages() function:
112
113 \snippet code/src_gui_opengl_qopengldebug.cpp 4
114
115 The internal log has a limited size; when it fills up, older messages will
116 get discarded to make room for the new incoming messages. When you call
117 loggedMessages(), the internal log will be emptied as well.
118
119 If you want to be sure not to lose any debug message, you must use real-time
120 logging instead of calling this function. However, debug messages might
121 still be generated in the timespan between context creation and activation
122 of real-time logging (or, in general, when the real-time logging is disabled).
123
124 \section1 Real-time logging of messages
125
126 It is also possible to receive a stream of debug messages from the OpenGL
127 server \e{as they are generated} by the implementation. In order to do so,
128 you need to connect a suitable slot to the messageLogged() signal, and
129 start logging by calling startLogging():
130
131 \snippet code/src_gui_opengl_qopengldebug.cpp 5
132
133 Similarly, logging can be disabled at any time by calling the stopLogging()
134 function.
135
136 Real-time logging can be either asynchronous or synchronous, depending on
137 the parameter passed to startLogging(). When logging in asynchronous mode
138 (the default, as it has a very small overhead), the OpenGL implementation
139 can generate messages at any time, and/or in an order which is different from the
140 order of the OpenGL commands which caused those messages to be logged.
141 The messages could also be generated from a thread that it's different from
142 the thread the context is currently bound to. This is because OpenGL
143 implementations are usually highly threaded and asynchronous, and therefore
144 no warranties are made about the relative order and the timings of the
145 debug messages.
146
147 On the other hand, logging in synchronous mode has a high overhead, but
148 the OpenGL implementation guarantees that all the messages caused by a
149 certain command are received in order, before the command returns,
150 and from the same thread the OpenGL context is bound to.
151
152 This means that when logging in synchronous mode you will be able to run
153 your OpenGL application in a debugger, put a breakpoint on a slot connected
154 to the messageLogged() signal, and see in the backtrace the exact call
155 that caused the logged message. This can be extremely useful to debug
156 an OpenGL problem. Note that if OpenGL rendering is happening in another
157 thread, you must force the signal/slot connection type to Qt::DirectConnection
158 in order to be able to see the actual backtrace.
159
160 Refer to the LoggingMode enum documentation for more information about
161 logging modes.
162
163 \note When real-time logging is enabled, debug messages will \e{not} be
164 inserted in the internal OpenGL debug log any more; messages already
165 present in the internal log will not be deleted, nor they will be emitted
166 through the messageLogged() signal. Since some messages might be generated
167 before real-time logging is started (and therefore be kept in the internal
168 OpenGL log), it is important to always check if it contains any message
169 after calling startLogging().
170
171 \section1 Inserting Messages in the Debug Log
172
173 It is possible for applications and libraries to insert custom messages in
174 the debug log, for instance for marking a group of related OpenGL commands
175 and therefore being then able to identify eventual messages coming from them.
176
177 In order to do so, you can create a QOpenGLDebugMessage object by calling
178 \l{QOpenGLDebugMessage::}{createApplicationMessage()} or
179 \l{QOpenGLDebugMessage::}{createThirdPartyMessage()}, and then inserting it
180 into the log by calling logMessage():
181
182 \snippet code/src_gui_opengl_qopengldebug.cpp 6
183
184 Note that OpenGL implementations have a vendor-specific limit to the length
185 of the messages that can be inserted in the debug log. You can retrieve
186 this length by calling the maximumMessageLength() method; messages longer
187 than the limit will automatically get truncated.
188
189 \section1 Controlling the Debug Output
190
191 QOpenGLDebugMessage is also able to apply filters to the debug messages, and
192 therefore limit the amount of messages logged. You can enable or disable
193 logging of messages by calling enableMessages() and disableMessages()
194 respectively. By default, all messages are logged.
195
196 It is possible to enable or disable messages by selecting them by:
197
198 \list
199 \li source, type and severity (and including all ids in the selection);
200 \li id, source and type (and including all severities in the selection).
201 \endlist
202
203 Note that the "enabled" status for a given message is a property of the
204 (id, source, type, severity) tuple; the message attributes \e{do not} form
205 a hierarchy of any kind. You should be careful about the order of the calls
206 to enableMessages() and disableMessages(), as it will change which
207 messages will are enabled / disabled.
208
209 It's not possible to filter by the message text itself; applications
210 have to do that on their own (in slots connected to the messageLogged()
211 signal, or after fetching the messages in the internal debug log
212 through loggedMessages()).
213
214 In order to simplify the management of the enabled / disabled statuses,
215 QOpenGLDebugMessage also supports the concept of \c{debug groups}. A debug
216 group contains the group of enabled / disabled configurations of debug
217 messages. Moreover, debug groups are organized in a stack: it is possible
218 to push and pop groups by calling pushGroup() and popGroup() respectively.
219 (When an OpenGL context is created, there is already a group in the stack).
220
221 The enableMessages() and disableMessages() functions will modify the
222 configuration in the current debug group, that is, the one at the top of
223 the debug groups stack.
224
225 When a new group is pushed onto the debug groups stack, it will inherit
226 the configuration of the group that was previously on the top of the stack.
227 Vice versa, popping a debug group will restore the configuration of
228 the debug group that becomes the new top.
229
230 Pushing (respectively popping) debug groups will also automatically generate
231 a debug message of type QOpenGLDebugMessage::GroupPushType (respectively
232 \l{QOpenGLDebugMessage::}{GroupPopType}).
233
234 \sa QOpenGLDebugMessage
235*/
236
237/*!
238 \enum QOpenGLDebugMessage::Source
239
240 The Source enum defines the source of the debug message.
241
242 \value InvalidSource
243 The source of the message is invalid; this is the source of a
244 default-constructed QOpenGLDebugMessage object.
245
246 \value APISource
247 The message was generated in response to OpenGL API calls.
248
249 \value WindowSystemSource
250 The message was generated by the window system.
251
252 \value ShaderCompilerSource
253 The message was generated by the shader compiler.
254
255 \value ThirdPartySource
256 The message was generated by a third party, for instance an OpenGL
257 framework a or debugging toolkit.
258
259 \value ApplicationSource
260 The message was generated by the application itself.
261
262 \value OtherSource
263 The message was generated by a source not included in this
264 enumeration.
265
266 \omitvalue LastSource
267
268 \value AnySource
269 This value corresponds to a mask of all possible message sources.
270*/
271
272/*!
273 \enum QOpenGLDebugMessage::Type
274
275 The Type enum defines the type of the debug message.
276
277 \value InvalidType
278 The type of the message is invalid; this is the type of a
279 default-constructed QOpenGLDebugMessage object.
280
281 \value ErrorType
282 The message represents an error.
283
284 \value DeprecatedBehaviorType
285 The message represents an usage of deprecated behavior.
286
287 \value UndefinedBehaviorType
288 The message represents an usage of undefined behavior.
289
290 \value PortabilityType
291 The message represents an usage of vendor-specific behavior,
292 that might pose portability concerns.
293
294 \value PerformanceType
295 The message represents a performance issue.
296
297 \value OtherType
298 The message represents a type not included in this
299 enumeration.
300
301 \value MarkerType
302 The message represents a marker in the debug log.
303
304 \value GroupPushType
305 The message represents a debug group push operation.
306
307 \value GroupPopType
308 The message represents a debug group pop operation.
309
310 \omitvalue LastType
311
312 \value AnyType
313 This value corresponds to a mask of all possible message types.
314*/
315
316/*!
317 \enum QOpenGLDebugMessage::Severity
318
319 The Severity enum defines the severity of the debug message.
320
321 \value InvalidSeverity
322 The severity of the message is invalid; this is the severity of a
323 default-constructed QOpenGLDebugMessage object.
324
325 \value HighSeverity
326 The message has a high severity.
327
328 \value MediumSeverity
329 The message has a medium severity.
330
331 \value LowSeverity
332 The message has a low severity.
333
334 \value NotificationSeverity
335 The message is a notification.
336
337 \omitvalue LastSeverity
338
339 \value AnySeverity
340 This value corresponds to a mask of all possible message severities.
341*/
342
343/*!
344 \property QOpenGLDebugLogger::loggingMode
345
346 \brief the logging mode passed to startLogging().
347
348 Note that logging must have been started or the value of this property
349 will be meaningless.
350
351 \sa startLogging(), isLogging()
352*/
353/*!
354 \enum QOpenGLDebugLogger::LoggingMode
355
356 The LoggingMode enum defines the logging mode of the logger object.
357
358 \value AsynchronousLogging
359 Messages from the OpenGL server are logged asynchronously. This means
360 that messages can be logged some time after the corresponding OpenGL
361 actions that caused them, and even be received in an out-of-order
362 fashion, depending on the OpenGL implementation. This mode has a very low
363 performance penalty, as OpenGL implementations are heavily threaded
364 and asynchronous by nature.
365
366 \value SynchronousLogging
367 Messages from the OpenGL server are logged synchronously and
368 sequentially. This has a severe performance hit, as OpenGL
369 implementations are very asynchronous by nature; but it's very useful
370 to debug OpenGL problems, as OpenGL guarantees that the messages
371 generated by a OpenGL command will be logged before the corresponding
372 command execution has returned. Therefore, you can install a breakpoint
373 on the messageLogged() signal and see in the backtrace which OpenGL
374 command caused it; the only caveat is that if you are using OpenGL from
375 multiple threads you may need to force direct connection when
376 connecting to the messageLogged() signal.
377*/
378
379// When using OpenGL ES 2.0, all the necessary GL_KHR_debug constants are
380// provided in qopengles2ext.h. Unfortunately, newer versions of that file
381// suffix everything with _KHR which causes extra headache when the goal is
382// to have a single piece of code that builds in all our target
383// environments. Therefore, try to detect this and use our custom defines
384// instead, which we anyway need for OS X.
385
386#if defined(GL_KHR_debug) && defined(GL_DEBUG_SOURCE_API_KHR)
387#define USE_MANUAL_DEFS
388#endif
389
390// Under OSX (at least up to 10.8) we cannot include our copy of glext.h,
391// but we use the system-wide one, which unfortunately lacks all the needed
392// defines/typedefs. In order to make the code compile, we just add here
393// the GL_KHR_debug defines.
394
395#ifndef GL_KHR_debug
396#define GL_KHR_debug 1
397#define USE_MANUAL_DEFS
398#endif
399
400#ifdef USE_MANUAL_DEFS
401
402#ifndef GL_DEBUG_OUTPUT_SYNCHRONOUS
403#define GL_DEBUG_OUTPUT_SYNCHRONOUS 0x8242
404#endif
405#ifndef GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH
406#define GL_DEBUG_NEXT_LOGGED_MESSAGE_LENGTH 0x8243
407#endif
408#ifndef GL_DEBUG_CALLBACK_FUNCTION
409#define GL_DEBUG_CALLBACK_FUNCTION 0x8244
410#endif
411#ifndef GL_DEBUG_CALLBACK_USER_PARAM
412#define GL_DEBUG_CALLBACK_USER_PARAM 0x8245
413#endif
414#ifndef GL_DEBUG_SOURCE_API
415#define GL_DEBUG_SOURCE_API 0x8246
416#endif
417#ifndef GL_DEBUG_SOURCE_WINDOW_SYSTEM
418#define GL_DEBUG_SOURCE_WINDOW_SYSTEM 0x8247
419#endif
420#ifndef GL_DEBUG_SOURCE_SHADER_COMPILER
421#define GL_DEBUG_SOURCE_SHADER_COMPILER 0x8248
422#endif
423#ifndef GL_DEBUG_SOURCE_THIRD_PARTY
424#define GL_DEBUG_SOURCE_THIRD_PARTY 0x8249
425#endif
426#ifndef GL_DEBUG_SOURCE_APPLICATION
427#define GL_DEBUG_SOURCE_APPLICATION 0x824A
428#endif
429#ifndef GL_DEBUG_SOURCE_OTHER
430#define GL_DEBUG_SOURCE_OTHER 0x824B
431#endif
432#ifndef GL_DEBUG_TYPE_ERROR
433#define GL_DEBUG_TYPE_ERROR 0x824C
434#endif
435#ifndef GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR
436#define GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR 0x824D
437#endif
438#ifndef GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR
439#define GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR 0x824E
440#endif
441#ifndef GL_DEBUG_TYPE_PORTABILITY
442#define GL_DEBUG_TYPE_PORTABILITY 0x824F
443#endif
444#ifndef GL_DEBUG_TYPE_PERFORMANCE
445#define GL_DEBUG_TYPE_PERFORMANCE 0x8250
446#endif
447#ifndef GL_DEBUG_TYPE_OTHER
448#define GL_DEBUG_TYPE_OTHER 0x8251
449#endif
450#ifndef GL_DEBUG_TYPE_MARKER
451#define GL_DEBUG_TYPE_MARKER 0x8268
452#endif
453#ifndef GL_DEBUG_TYPE_PUSH_GROUP
454#define GL_DEBUG_TYPE_PUSH_GROUP 0x8269
455#endif
456#ifndef GL_DEBUG_TYPE_POP_GROUP
457#define GL_DEBUG_TYPE_POP_GROUP 0x826A
458#endif
459#ifndef GL_DEBUG_SEVERITY_NOTIFICATION
460#define GL_DEBUG_SEVERITY_NOTIFICATION 0x826B
461#endif
462#ifndef GL_MAX_DEBUG_GROUP_STACK_DEPTH
463#define GL_MAX_DEBUG_GROUP_STACK_DEPTH 0x826C
464#endif
465#ifndef GL_DEBUG_GROUP_STACK_DEPTH
466#define GL_DEBUG_GROUP_STACK_DEPTH 0x826D
467#endif
468#ifndef GL_BUFFER
469#define GL_BUFFER 0x82E0
470#endif
471#ifndef GL_SHADER
472#define GL_SHADER 0x82E1
473#endif
474#ifndef GL_PROGRAM
475#define GL_PROGRAM 0x82E2
476#endif
477#ifndef GL_QUERY
478#define GL_QUERY 0x82E3
479#endif
480#ifndef GL_PROGRAM_PIPELINE
481#define GL_PROGRAM_PIPELINE 0x82E4
482#endif
483#ifndef GL_SAMPLER
484#define GL_SAMPLER 0x82E6
485#endif
486#ifndef GL_DISPLAY_LIST
487#define GL_DISPLAY_LIST 0x82E7
488#endif
489#ifndef GL_MAX_LABEL_LENGTH
490#define GL_MAX_LABEL_LENGTH 0x82E8
491#endif
492#ifndef GL_MAX_DEBUG_MESSAGE_LENGTH
493#define GL_MAX_DEBUG_MESSAGE_LENGTH 0x9143
494#endif
495#ifndef GL_MAX_DEBUG_LOGGED_MESSAGES
496#define GL_MAX_DEBUG_LOGGED_MESSAGES 0x9144
497#endif
498#ifndef GL_DEBUG_LOGGED_MESSAGES
499#define GL_DEBUG_LOGGED_MESSAGES 0x9145
500#endif
501#ifndef GL_DEBUG_SEVERITY_HIGH
502#define GL_DEBUG_SEVERITY_HIGH 0x9146
503#endif
504#ifndef GL_DEBUG_SEVERITY_MEDIUM
505#define GL_DEBUG_SEVERITY_MEDIUM 0x9147
506#endif
507#ifndef GL_DEBUG_SEVERITY_LOW
508#define GL_DEBUG_SEVERITY_LOW 0x9148
509#endif
510#ifndef GL_DEBUG_OUTPUT
511#define GL_DEBUG_OUTPUT 0x92E0
512#endif
513#ifndef GL_CONTEXT_FLAG_DEBUG_BIT
514#define GL_CONTEXT_FLAG_DEBUG_BIT 0x00000002
515#endif
516#ifndef GL_STACK_OVERFLOW
517#define GL_STACK_OVERFLOW 0x0503
518#endif
519#ifndef GL_STACK_UNDERFLOW
520#define GL_STACK_UNDERFLOW 0x0504
521#endif
522
523typedef void (QOPENGLF_APIENTRY *GLDEBUGPROC)(GLenum source,GLenum type,GLuint id,GLenum severity,GLsizei length,const GLchar *message,const GLvoid *userParam);
524
525#endif /* USE_MANUAL_DEFS */
526
527
528/*!
529 \internal
530*/
531static QOpenGLDebugMessage::Source qt_messageSourceFromGL(GLenum source)
532{
533 switch (source) {
534 case GL_DEBUG_SOURCE_API:
535 return QOpenGLDebugMessage::APISource;
536 case GL_DEBUG_SOURCE_WINDOW_SYSTEM:
537 return QOpenGLDebugMessage::WindowSystemSource;
538 case GL_DEBUG_SOURCE_SHADER_COMPILER:
539 return QOpenGLDebugMessage::ShaderCompilerSource;
540 case GL_DEBUG_SOURCE_THIRD_PARTY:
541 return QOpenGLDebugMessage::ThirdPartySource;
542 case GL_DEBUG_SOURCE_APPLICATION:
543 return QOpenGLDebugMessage::ApplicationSource;
544 case GL_DEBUG_SOURCE_OTHER:
545 return QOpenGLDebugMessage::OtherSource;
546 }
547
548 Q_ASSERT_X(false, Q_FUNC_INFO, "Unknown message source from GL");
549 return QOpenGLDebugMessage::OtherSource;
550}
551
552/*!
553 \internal
554*/
555static GLenum qt_messageSourceToGL(QOpenGLDebugMessage::Source source)
556{
557 switch (source) {
558 case QOpenGLDebugMessage::InvalidSource:
559 break;
560 case QOpenGLDebugMessage::APISource:
561 return GL_DEBUG_SOURCE_API;
562 case QOpenGLDebugMessage::WindowSystemSource:
563 return GL_DEBUG_SOURCE_WINDOW_SYSTEM;
564 case QOpenGLDebugMessage::ShaderCompilerSource:
565 return GL_DEBUG_SOURCE_SHADER_COMPILER;
566 case QOpenGLDebugMessage::ThirdPartySource:
567 return GL_DEBUG_SOURCE_THIRD_PARTY;
568 case QOpenGLDebugMessage::ApplicationSource:
569 return GL_DEBUG_SOURCE_APPLICATION;
570 case QOpenGLDebugMessage::OtherSource:
571 return GL_DEBUG_SOURCE_OTHER;
572 case QOpenGLDebugMessage::AnySource:
573 break;
574 }
575
576 Q_ASSERT_X(false, Q_FUNC_INFO, "Invalid message source");
577 return GL_DEBUG_SOURCE_OTHER;
578}
579
580/*!
581 \internal
582*/
583static QString qt_messageSourceToString(QOpenGLDebugMessage::Source source)
584{
585 switch (source) {
586 case QOpenGLDebugMessage::InvalidSource:
587 return QStringLiteral("InvalidSource");
588 case QOpenGLDebugMessage::APISource:
589 return QStringLiteral("APISource");
590 case QOpenGLDebugMessage::WindowSystemSource:
591 return QStringLiteral("WindowSystemSource");
592 case QOpenGLDebugMessage::ShaderCompilerSource:
593 return QStringLiteral("ShaderCompilerSource");
594 case QOpenGLDebugMessage::ThirdPartySource:
595 return QStringLiteral("ThirdPartySource");
596 case QOpenGLDebugMessage::ApplicationSource:
597 return QStringLiteral("ApplicationSource");
598 case QOpenGLDebugMessage::OtherSource:
599 return QStringLiteral("OtherSource");
600 case QOpenGLDebugMessage::AnySource:
601 return QStringLiteral("AnySource");
602 }
603
604 Q_ASSERT_X(false, Q_FUNC_INFO, "Unknown message source");
605 return QString();
606}
607
608/*!
609 \internal
610*/
611static QOpenGLDebugMessage::Type qt_messageTypeFromGL(GLenum type)
612{
613 switch (type) {
614 case GL_DEBUG_TYPE_ERROR:
615 return QOpenGLDebugMessage::ErrorType;
616 case GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR:
617 return QOpenGLDebugMessage::DeprecatedBehaviorType;
618 case GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR:
619 return QOpenGLDebugMessage::UndefinedBehaviorType;
620 case GL_DEBUG_TYPE_PORTABILITY:
621 return QOpenGLDebugMessage::PortabilityType;
622 case GL_DEBUG_TYPE_PERFORMANCE:
623 return QOpenGLDebugMessage::PerformanceType;
624 case GL_DEBUG_TYPE_OTHER:
625 return QOpenGLDebugMessage::OtherType;
626 case GL_DEBUG_TYPE_MARKER:
627 return QOpenGLDebugMessage::MarkerType;
628 case GL_DEBUG_TYPE_PUSH_GROUP:
629 return QOpenGLDebugMessage::GroupPushType;
630 case GL_DEBUG_TYPE_POP_GROUP:
631 return QOpenGLDebugMessage::GroupPopType;
632 }
633
634 Q_ASSERT_X(false, Q_FUNC_INFO, "Unknown message type from GL");
635 return QOpenGLDebugMessage::OtherType;
636}
637
638/*!
639 \internal
640*/
641static GLenum qt_messageTypeToGL(QOpenGLDebugMessage::Type type)
642{
643 switch (type) {
644 case QOpenGLDebugMessage::InvalidType:
645 break;
646 case QOpenGLDebugMessage::ErrorType:
647 return GL_DEBUG_TYPE_ERROR;
648 case QOpenGLDebugMessage::DeprecatedBehaviorType:
649 return GL_DEBUG_TYPE_DEPRECATED_BEHAVIOR;
650 case QOpenGLDebugMessage::UndefinedBehaviorType:
651 return GL_DEBUG_TYPE_UNDEFINED_BEHAVIOR;
652 case QOpenGLDebugMessage::PortabilityType:
653 return GL_DEBUG_TYPE_PORTABILITY;
654 case QOpenGLDebugMessage::PerformanceType:
655 return GL_DEBUG_TYPE_PERFORMANCE;
656 case QOpenGLDebugMessage::OtherType:
657 return GL_DEBUG_TYPE_OTHER;
658 case QOpenGLDebugMessage::MarkerType:
659 return GL_DEBUG_TYPE_MARKER;
660 case QOpenGLDebugMessage::GroupPushType:
661 return GL_DEBUG_TYPE_PUSH_GROUP;
662 case QOpenGLDebugMessage::GroupPopType:
663 return GL_DEBUG_TYPE_POP_GROUP;
664 case QOpenGLDebugMessage::AnyType:
665 break;
666 }
667
668 Q_ASSERT_X(false, Q_FUNC_INFO, "Invalid message type");
669 return GL_DEBUG_TYPE_OTHER;
670}
671
672/*!
673 \internal
674*/
675static QString qt_messageTypeToString(QOpenGLDebugMessage::Type type)
676{
677 switch (type) {
678 case QOpenGLDebugMessage::InvalidType:
679 return QStringLiteral("InvalidType");
680 case QOpenGLDebugMessage::ErrorType:
681 return QStringLiteral("ErrorType");
682 case QOpenGLDebugMessage::DeprecatedBehaviorType:
683 return QStringLiteral("DeprecatedBehaviorType");
684 case QOpenGLDebugMessage::UndefinedBehaviorType:
685 return QStringLiteral("UndefinedBehaviorType");
686 case QOpenGLDebugMessage::PortabilityType:
687 return QStringLiteral("PortabilityType");
688 case QOpenGLDebugMessage::PerformanceType:
689 return QStringLiteral("PerformanceType");
690 case QOpenGLDebugMessage::OtherType:
691 return QStringLiteral("OtherType");
692 case QOpenGLDebugMessage::MarkerType:
693 return QStringLiteral("MarkerType");
694 case QOpenGLDebugMessage::GroupPushType:
695 return QStringLiteral("GroupPushType");
696 case QOpenGLDebugMessage::GroupPopType:
697 return QStringLiteral("GroupPopType");
698 case QOpenGLDebugMessage::AnyType:
699 return QStringLiteral("AnyType");
700 }
701
702 Q_ASSERT_X(false, Q_FUNC_INFO, "Unknown message type");
703 return QString();
704}
705
706/*!
707 \internal
708*/
709static QOpenGLDebugMessage::Severity qt_messageSeverityFromGL(GLenum severity)
710{
711 switch (severity) {
712 case GL_DEBUG_SEVERITY_HIGH:
713 return QOpenGLDebugMessage::HighSeverity;
714 case GL_DEBUG_SEVERITY_MEDIUM:
715 return QOpenGLDebugMessage::MediumSeverity;
716 case GL_DEBUG_SEVERITY_LOW:
717 return QOpenGLDebugMessage::LowSeverity;
718 case GL_DEBUG_SEVERITY_NOTIFICATION:
719 return QOpenGLDebugMessage::NotificationSeverity;
720 }
721
722 Q_ASSERT_X(false, Q_FUNC_INFO, "Unknown message severity from GL");
723 return QOpenGLDebugMessage::NotificationSeverity;
724}
725
726/*!
727 \internal
728*/
729static GLenum qt_messageSeverityToGL(QOpenGLDebugMessage::Severity severity)
730{
731 switch (severity) {
732 case QOpenGLDebugMessage::InvalidSeverity:
733 break;
734 case QOpenGLDebugMessage::HighSeverity:
735 return GL_DEBUG_SEVERITY_HIGH;
736 case QOpenGLDebugMessage::MediumSeverity:
737 return GL_DEBUG_SEVERITY_MEDIUM;
738 case QOpenGLDebugMessage::LowSeverity:
739 return GL_DEBUG_SEVERITY_LOW;
740 case QOpenGLDebugMessage::NotificationSeverity:
741 return GL_DEBUG_SEVERITY_NOTIFICATION;
742 case QOpenGLDebugMessage::AnySeverity:
743 break;
744 }
745
746 Q_ASSERT_X(false, Q_FUNC_INFO, "Invalid message severity");
747 return GL_DEBUG_SEVERITY_NOTIFICATION;
748}
749
750/*!
751 \internal
752*/
753static QString qt_messageSeverityToString(QOpenGLDebugMessage::Severity severity)
754{
755 switch (severity) {
756 case QOpenGLDebugMessage::InvalidSeverity:
757 return QStringLiteral("InvalidSeverity");
758 case QOpenGLDebugMessage::HighSeverity:
759 return QStringLiteral("HighSeverity");
760 case QOpenGLDebugMessage::MediumSeverity:
761 return QStringLiteral("MediumSeverity");
762 case QOpenGLDebugMessage::LowSeverity:
763 return QStringLiteral("LowSeverity");
764 case QOpenGLDebugMessage::NotificationSeverity:
765 return QStringLiteral("NotificationSeverity");
766 case QOpenGLDebugMessage::AnySeverity:
767 return QStringLiteral("AnySeverity");
768 }
769
770 Q_ASSERT_X(false, Q_FUNC_INFO, "Unknown message severity");
771 return QString();
772}
773
774class QOpenGLDebugMessagePrivate : public QSharedData
775{
776public:
777 QOpenGLDebugMessagePrivate();
778
779 QString message;
780 GLuint id;
781 QOpenGLDebugMessage::Source source;
782 QOpenGLDebugMessage::Type type;
783 QOpenGLDebugMessage::Severity severity;
784};
785
786/*!
787 \internal
788*/
789QOpenGLDebugMessagePrivate::QOpenGLDebugMessagePrivate()
790 : message(),
791 id(0),
792 source(QOpenGLDebugMessage::InvalidSource),
793 type(QOpenGLDebugMessage::InvalidType),
794 severity(QOpenGLDebugMessage::InvalidSeverity)
795{
796}
797
798
799/*!
800 Constructs a debug message with an empty message string, id set to 0,
801 source set to InvalidSource, type set to InvalidType, and severity set to
802 InvalidSeverity.
803
804 \note This constructor should not be used to create a debug message;
805 instead, use the createApplicationMessage() or the createThirdPartyMessage()
806 static functions.
807
808 \sa createApplicationMessage(), createThirdPartyMessage()
809*/
810QOpenGLDebugMessage::QOpenGLDebugMessage()
811 : d(new QOpenGLDebugMessagePrivate)
812{
813}
814
815/*!
816 Constructs a debug message as a copy of \a debugMessage.
817
818 \sa operator=()
819*/
820QOpenGLDebugMessage::QOpenGLDebugMessage(const QOpenGLDebugMessage &debugMessage)
821 : d(debugMessage.d)
822{
823}
824
825/*!
826 Destroys this debug message.
827*/
828QOpenGLDebugMessage::~QOpenGLDebugMessage()
829{
830}
831
832/*!
833 Assigns the message \a debugMessage to this object, and returns a reference
834 to the copy.
835*/
836QOpenGLDebugMessage &QOpenGLDebugMessage::operator=(const QOpenGLDebugMessage &debugMessage)
837{
838 d = debugMessage.d;
839 return *this;
840}
841
842/*!
843 \fn QOpenGLDebugMessage &QOpenGLDebugMessage::operator=(QOpenGLDebugMessage &&debugMessage)
844
845 Move-assigns \a debugMessage to this object.
846*/
847
848/*!
849 \fn void QOpenGLDebugMessage::swap(QOpenGLDebugMessage &debugMessage)
850
851 Swaps the message \a debugMessage with this message. This operation is very
852 fast and never fails.
853*/
854
855/*!
856 Returns the source of the debug message.
857*/
858QOpenGLDebugMessage::Source QOpenGLDebugMessage::source() const
859{
860 return d->source;
861}
862
863/*!
864 Returns the type of the debug message.
865*/
866QOpenGLDebugMessage::Type QOpenGLDebugMessage::type() const
867{
868 return d->type;
869}
870
871/*!
872 Returns the severity of the debug message.
873*/
874QOpenGLDebugMessage::Severity QOpenGLDebugMessage::severity() const
875{
876 return d->severity;
877}
878
879/*!
880 Returns the id of the debug message. Ids are generally vendor-specific.
881*/
882GLuint QOpenGLDebugMessage::id() const
883{
884 return d->id;
885}
886
887/*!
888 Returns the textual message contained by this debug message.
889*/
890QString QOpenGLDebugMessage::message() const
891{
892 return d->message;
893}
894
895/*!
896 Constructs and returns a debug message with \a text as its text, \a id
897 as id, \a severity as severity, and \a type as type. The message source
898 will be set to ApplicationSource.
899
900 \sa QOpenGLDebugLogger::logMessage(), createThirdPartyMessage()
901*/
902QOpenGLDebugMessage QOpenGLDebugMessage::createApplicationMessage(const QString &text,
903 GLuint id,
904 QOpenGLDebugMessage::Severity severity,
905 QOpenGLDebugMessage::Type type)
906{
907 QOpenGLDebugMessage m;
908 m.d->message = text;
909 m.d->id = id;
910 m.d->severity = severity;
911 m.d->type = type;
912 m.d->source = ApplicationSource;
913 return m;
914}
915
916/*!
917 Constructs and returns a debug message with \a text as its text, \a id
918 as id, \a severity as severity, and \a type as type. The message source
919 will be set to ThirdPartySource.
920
921 \sa QOpenGLDebugLogger::logMessage(), createApplicationMessage()
922*/
923QOpenGLDebugMessage QOpenGLDebugMessage::createThirdPartyMessage(const QString &text,
924 GLuint id,
925 QOpenGLDebugMessage::Severity severity,
926 QOpenGLDebugMessage::Type type)
927{
928 QOpenGLDebugMessage m;
929 m.d->message = text;
930 m.d->id = id;
931 m.d->severity = severity;
932 m.d->type = type;
933 m.d->source = ThirdPartySource;
934 return m;
935}
936
937/*!
938 Returns \c true if this debug message is equal to \a debugMessage, or false
939 otherwise. Two debugging messages are equal if they have the same textual
940 message, the same id, the same source, the same type and the same severity.
941
942 \sa operator!=()
943*/
944bool QOpenGLDebugMessage::operator==(const QOpenGLDebugMessage &debugMessage) const
945{
946 return (d == debugMessage.d)
947 || (d->id == debugMessage.d->id
948 && d->source == debugMessage.d->source
949 && d->type == debugMessage.d->type
950 && d->severity == debugMessage.d->severity
951 && d->message == debugMessage.d->message);
952}
953
954/*!
955 \fn bool QOpenGLDebugMessage::operator!=(const QOpenGLDebugMessage &debugMessage) const
956
957 Returns \c true if this message is different from \a debugMessage, or false
958 otherwise.
959
960 \sa operator==()
961*/
962
963#ifndef QT_NO_DEBUG_STREAM
964/*!
965 \relates QOpenGLDebugMessage
966
967 Writes the source \a source into the debug object \a debug for debugging
968 purposes.
969*/
970QDebug operator<<(QDebug debug, QOpenGLDebugMessage::Source source)
971{
972 QDebugStateSaver saver(debug);
973 debug.nospace() << "QOpenGLDebugMessage::Source("
974 << qt_messageSourceToString(source)
975 << ')';
976 return debug;
977}
978
979/*!
980 \relates QOpenGLDebugMessage
981
982 Writes the type \a type into the debug object \a debug for debugging
983 purposes.
984*/
985QDebug operator<<(QDebug debug, QOpenGLDebugMessage::Type type)
986{
987 QDebugStateSaver saver(debug);
988 debug.nospace() << "QOpenGLDebugMessage::Type("
989 << qt_messageTypeToString(type)
990 << ')';
991 return debug;
992}
993
994/*!
995 \relates QOpenGLDebugMessage
996
997 Writes the severity \a severity into the debug object \a debug for debugging
998 purposes.
999*/
1000QDebug operator<<(QDebug debug, QOpenGLDebugMessage::Severity severity)
1001{
1002 QDebugStateSaver saver(debug);
1003 debug.nospace() << "QOpenGLDebugMessage::Severity("
1004 << qt_messageSeverityToString(severity)
1005 << ')';
1006 return debug;
1007}
1008
1009/*!
1010 \relates QOpenGLDebugMessage
1011
1012 Writes the message \a message into the debug object \a debug for debugging
1013 purposes.
1014*/
1015QDebug operator<<(QDebug debug, const QOpenGLDebugMessage &message)
1016{
1017 QDebugStateSaver saver(debug);
1018 debug.nospace() << "QOpenGLDebugMessage("
1019 << qt_messageSourceToString(source: message.source()) << ", "
1020 << message.id() << ", "
1021 << message.message() << ", "
1022 << qt_messageSeverityToString(severity: message.severity()) << ", "
1023 << qt_messageTypeToString(type: message.type()) << ')';
1024 return debug;
1025
1026}
1027#endif // QT_NO_DEBUG_STREAM
1028
1029typedef void (QOPENGLF_APIENTRYP qt_glDebugMessageControl_t)(GLenum source, GLenum type, GLenum severity, GLsizei count, const GLuint *ids, GLboolean enabled);
1030typedef void (QOPENGLF_APIENTRYP qt_glDebugMessageInsert_t)(GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *buf);
1031typedef void (QOPENGLF_APIENTRYP qt_glDebugMessageCallback_t)(GLDEBUGPROC callback, const void *userParam);
1032typedef GLuint (QOPENGLF_APIENTRYP qt_glGetDebugMessageLog_t)(GLuint count, GLsizei bufsize, GLenum *sources, GLenum *types, GLuint *ids, GLenum *severities, GLsizei *lengths, GLchar *messageLog);
1033typedef void (QOPENGLF_APIENTRYP qt_glPushDebugGroup_t)(GLenum source, GLuint id, GLsizei length, const GLchar *message);
1034typedef void (QOPENGLF_APIENTRYP qt_glPopDebugGroup_t)();
1035typedef void (QOPENGLF_APIENTRYP qt_glGetPointerv_t)(GLenum pname, GLvoid **params);
1036
1037class QOpenGLDebugLoggerPrivate : public QObjectPrivate
1038{
1039 Q_DECLARE_PUBLIC(QOpenGLDebugLogger)
1040public:
1041 QOpenGLDebugLoggerPrivate();
1042
1043 void handleMessage(GLenum source, GLenum type, GLuint id, GLenum severity, GLsizei length, const GLchar *rawMessage);
1044 void controlDebugMessages(QOpenGLDebugMessage::Sources sources,
1045 QOpenGLDebugMessage::Types types,
1046 QOpenGLDebugMessage::Severities severities, const QList<GLuint> &ids,
1047 const QByteArray &callerName, bool enable);
1048 void _q_contextAboutToBeDestroyed();
1049
1050 qt_glDebugMessageControl_t glDebugMessageControl;
1051 qt_glDebugMessageInsert_t glDebugMessageInsert;
1052 qt_glDebugMessageCallback_t glDebugMessageCallback;
1053 qt_glGetDebugMessageLog_t glGetDebugMessageLog;
1054 qt_glPushDebugGroup_t glPushDebugGroup;
1055 qt_glPopDebugGroup_t glPopDebugGroup;
1056 qt_glGetPointerv_t glGetPointerv;
1057
1058 GLDEBUGPROC oldDebugCallbackFunction;
1059 void *oldDebugCallbackParameter;
1060 QOpenGLContext *context;
1061 GLint maxMessageLength;
1062 QOpenGLDebugLogger::LoggingMode loggingMode;
1063 bool initialized : 1;
1064 bool isLogging : 1;
1065 bool debugWasEnabled : 1;
1066 bool syncDebugWasEnabled : 1;
1067};
1068
1069/*!
1070 \internal
1071*/
1072QOpenGLDebugLoggerPrivate::QOpenGLDebugLoggerPrivate()
1073 : glDebugMessageControl(nullptr),
1074 glDebugMessageInsert(nullptr),
1075 glDebugMessageCallback(nullptr),
1076 glGetDebugMessageLog(nullptr),
1077 glPushDebugGroup(nullptr),
1078 glPopDebugGroup(nullptr),
1079 oldDebugCallbackFunction(nullptr),
1080 context(nullptr),
1081 maxMessageLength(0),
1082 loggingMode(QOpenGLDebugLogger::AsynchronousLogging),
1083 initialized(false),
1084 isLogging(false),
1085 debugWasEnabled(false),
1086 syncDebugWasEnabled(false)
1087{
1088}
1089
1090/*!
1091 \internal
1092*/
1093void QOpenGLDebugLoggerPrivate::handleMessage(GLenum source,
1094 GLenum type,
1095 GLuint id,
1096 GLenum severity,
1097 GLsizei length,
1098 const GLchar *rawMessage)
1099{
1100 if (oldDebugCallbackFunction)
1101 oldDebugCallbackFunction(source, type, id, severity, length, rawMessage, oldDebugCallbackParameter);
1102
1103 QOpenGLDebugMessage message;
1104
1105 QOpenGLDebugMessagePrivate *messagePrivate = message.d.data();
1106 messagePrivate->source = qt_messageSourceFromGL(source);
1107 messagePrivate->type = qt_messageTypeFromGL(type);
1108 messagePrivate->id = id;
1109 messagePrivate->severity = qt_messageSeverityFromGL(severity);
1110 // not passing the length to fromUtf8, as some bugged OpenGL drivers
1111 // do not handle the length correctly. Just rely on the message to be NUL terminated.
1112 messagePrivate->message = QString::fromUtf8(utf8: rawMessage);
1113
1114 Q_Q(QOpenGLDebugLogger);
1115 emit q->messageLogged(debugMessage: message);
1116}
1117
1118/*!
1119 \internal
1120*/
1121void QOpenGLDebugLoggerPrivate::controlDebugMessages(QOpenGLDebugMessage::Sources sources,
1122 QOpenGLDebugMessage::Types types,
1123 QOpenGLDebugMessage::Severities severities,
1124 const QList<GLuint> &ids,
1125 const QByteArray &callerName, bool enable)
1126{
1127 if (!initialized) {
1128 qWarning(msg: "QOpenGLDebugLogger::%s(): object must be initialized before enabling/disabling messages", callerName.constData());
1129 return;
1130 }
1131 if (sources == QOpenGLDebugMessage::InvalidSource) {
1132 qWarning(msg: "QOpenGLDebugLogger::%s(): invalid source specified", callerName.constData());
1133 return;
1134 }
1135 if (types == QOpenGLDebugMessage::InvalidType) {
1136 qWarning(msg: "QOpenGLDebugLogger::%s(): invalid type specified", callerName.constData());
1137 return;
1138 }
1139 if (severities == QOpenGLDebugMessage::InvalidSeverity) {
1140 qWarning(msg: "QOpenGLDebugLogger::%s(): invalid severity specified", callerName.constData());
1141 return;
1142 }
1143
1144 QVarLengthArray<GLenum, 8> glSources;
1145 QVarLengthArray<GLenum, 8> glTypes;
1146 QVarLengthArray<GLenum, 8> glSeverities;
1147
1148 if (ids.size() > 0) {
1149 Q_ASSERT(severities == QOpenGLDebugMessage::AnySeverity);
1150
1151 // The GL_KHR_debug extension says:
1152 //
1153 // - If <count> is greater than zero, then <ids> is an array of <count>
1154 // message IDs for the specified combination of <source> and <type>. In
1155 // this case, if <source> or <type> is DONT_CARE, or <severity> is not
1156 // DONT_CARE, the error INVALID_OPERATION is generated. If <count> is
1157 // zero, the value if <ids> is ignored.
1158 //
1159 // This means we can't convert AnySource or AnyType into DONT_CARE, but we have to roll
1160 // them into individual sources/types.
1161
1162 if (sources == QOpenGLDebugMessage::AnySource) {
1163 sources = QOpenGLDebugMessage::InvalidSource;
1164 for (uint i = 1; i <= QOpenGLDebugMessage::LastSource; i = i << 1)
1165 sources |= QOpenGLDebugMessage::Source(i);
1166 }
1167
1168 if (types == QOpenGLDebugMessage::AnyType) {
1169 types = QOpenGLDebugMessage::InvalidType;
1170 for (uint i = 1; i <= QOpenGLDebugMessage::LastType; i = i << 1)
1171 types |= QOpenGLDebugMessage::Type(i);
1172 }
1173 }
1174
1175#define CONVERT_TO_GL_DEBUG_MESSAGE_CONTROL_PARAMETERS(type, source, target) \
1176 if (source == QOpenGLDebugMessage::Any ## type) { \
1177 target << GL_DONT_CARE; \
1178 } else { \
1179 for (uint i = 1; i <= QOpenGLDebugMessage::Last ## type; i = i << 1) \
1180 if (source.testFlag(QOpenGLDebugMessage:: type (i))) \
1181 target << qt_message ## type ## ToGL (QOpenGLDebugMessage:: type (i)); \
1182 }
1183
1184 CONVERT_TO_GL_DEBUG_MESSAGE_CONTROL_PARAMETERS(Source, sources, glSources)
1185 CONVERT_TO_GL_DEBUG_MESSAGE_CONTROL_PARAMETERS(Type, types, glTypes)
1186 CONVERT_TO_GL_DEBUG_MESSAGE_CONTROL_PARAMETERS(Severity, severities, glSeverities)
1187#undef CONVERT_TO_GL_DEBUG_MESSAGE_CONTROL_PARAMETERS
1188
1189 const GLsizei idCount = ids.size();
1190 // The GL_KHR_debug extension says that if idCount is 0, idPtr must be ignored.
1191 // Unfortunately, some bugged drivers do NOT ignore it, so pass NULL in case.
1192 const GLuint * const idPtr = idCount ? ids.constData() : nullptr;
1193
1194 for (GLenum source : glSources)
1195 for (GLenum type : glTypes)
1196 for (GLenum severity : glSeverities)
1197 glDebugMessageControl(source, type, severity, idCount, idPtr, GLboolean(enable));
1198}
1199
1200/*!
1201 \internal
1202*/
1203void QOpenGLDebugLoggerPrivate::_q_contextAboutToBeDestroyed()
1204{
1205 Q_ASSERT(context);
1206
1207 // Re-make our context current somehow, otherwise stopLogging will fail.
1208
1209 // Save the current context and its surface in case we need to set them back
1210 QOpenGLContext *currentContext = QOpenGLContext::currentContext();
1211 QSurface *currentSurface = nullptr;
1212
1213 QScopedPointer<QOffscreenSurface> offscreenSurface;
1214
1215 if (context != currentContext) {
1216 // Make our old context current on a temporary surface
1217 if (currentContext)
1218 currentSurface = currentContext->surface();
1219
1220 offscreenSurface.reset(other: new QOffscreenSurface);
1221 offscreenSurface->setFormat(context->format());
1222 offscreenSurface->create();
1223 if (!context->makeCurrent(surface: offscreenSurface.data()))
1224 qWarning(msg: "QOpenGLDebugLoggerPrivate::_q_contextAboutToBeDestroyed(): could not make the owning GL context current for cleanup");
1225 }
1226
1227 Q_Q(QOpenGLDebugLogger);
1228 q->stopLogging();
1229
1230 if (offscreenSurface) {
1231 // We did change the current context: set it back
1232 if (currentContext)
1233 currentContext->makeCurrent(surface: currentSurface);
1234 else
1235 context->doneCurrent();
1236 }
1237
1238 QObject::disconnect(sender: context, SIGNAL(aboutToBeDestroyed()), receiver: q, SLOT(_q_contextAboutToBeDestroyed()));
1239 context = nullptr;
1240 initialized = false;
1241}
1242
1243extern "C" {
1244static void QOPENGLF_APIENTRY qt_opengl_debug_callback(GLenum source,
1245 GLenum type,
1246 GLuint id,
1247 GLenum severity,
1248 GLsizei length,
1249 const GLchar *rawMessage,
1250 const GLvoid *userParam)
1251{
1252 QOpenGLDebugLoggerPrivate *loggerPrivate = static_cast<QOpenGLDebugLoggerPrivate *>(const_cast<GLvoid *>(userParam));
1253 loggerPrivate->handleMessage(source, type, id, severity, length, rawMessage);
1254}
1255}
1256
1257/*!
1258 Constructs a new logger object with the given \a parent.
1259
1260 \note The object must be initialized before logging can happen.
1261
1262 \sa initialize()
1263*/
1264QOpenGLDebugLogger::QOpenGLDebugLogger(QObject *parent)
1265 : QObject(*new QOpenGLDebugLoggerPrivate, parent)
1266{
1267 // QOpenGLDebugMessage is going to be mostly used as an argument
1268 // of a cross thread connection, therefore let's ease the life for the users
1269 // and register the type for them.
1270 qRegisterMetaType<QOpenGLDebugMessage>();
1271}
1272
1273/*!
1274 Destroys the logger object.
1275*/
1276QOpenGLDebugLogger::~QOpenGLDebugLogger()
1277{
1278 stopLogging();
1279}
1280
1281/*!
1282 Initializes the object in the current OpenGL context. The context must
1283 support the \c{GL_KHR_debug} extension for the initialization to succeed.
1284 The object must be initialized before any logging can happen.
1285
1286 It is safe to call this function multiple times from the same context.
1287
1288 This function can also be used to change the context of a previously
1289 initialized object; note that in this case the object must not be logging
1290 when you call this function.
1291
1292 Returns \c true if the logger is successfully initialized; false otherwise.
1293
1294 \sa QOpenGLContext
1295*/
1296bool QOpenGLDebugLogger::initialize()
1297{
1298 QOpenGLContext *context = QOpenGLContext::currentContext();
1299 if (!context) {
1300 qWarning(msg: "QOpenGLDebugLogger::initialize(): no current OpenGL context found.");
1301 return false;
1302 }
1303
1304 Q_D(QOpenGLDebugLogger);
1305 if (d->context == context) {
1306 // context is non-NULL, d->context is non NULL only on successful initialization.
1307 Q_ASSERT(d->initialized);
1308 return true;
1309 }
1310
1311 if (d->isLogging) {
1312 qWarning(msg: "QOpenGLDebugLogger::initialize(): cannot initialize the object while logging. Please stop the logging first.");
1313 return false;
1314 }
1315
1316 if (d->context)
1317 disconnect(sender: d->context, SIGNAL(aboutToBeDestroyed()), receiver: this, SLOT(_q_contextAboutToBeDestroyed()));
1318
1319 d->initialized = false;
1320 d->context = nullptr;
1321
1322 if (!context->hasExtension(QByteArrayLiteral("GL_KHR_debug")))
1323 return false;
1324
1325 d->context = context;
1326 connect(sender: d->context, SIGNAL(aboutToBeDestroyed()), receiver: this, SLOT(_q_contextAboutToBeDestroyed()));
1327
1328#define GET_DEBUG_PROC_ADDRESS(procName) \
1329 d->procName = reinterpret_cast< qt_ ## procName ## _t >( \
1330 d->context->getProcAddress(d->context->isOpenGLES() ? (#procName "KHR") : (#procName)) \
1331 );
1332
1333 GET_DEBUG_PROC_ADDRESS(glDebugMessageControl);
1334 GET_DEBUG_PROC_ADDRESS(glDebugMessageInsert);
1335 GET_DEBUG_PROC_ADDRESS(glDebugMessageCallback);
1336 GET_DEBUG_PROC_ADDRESS(glGetDebugMessageLog);
1337 GET_DEBUG_PROC_ADDRESS(glPushDebugGroup);
1338 GET_DEBUG_PROC_ADDRESS(glPopDebugGroup);
1339 GET_DEBUG_PROC_ADDRESS(glGetPointerv)
1340
1341#undef GET_DEBUG_PROC_ADDRESS
1342
1343 QOpenGLContext::currentContext()->functions()->glGetIntegerv(GL_MAX_DEBUG_MESSAGE_LENGTH, params: &d->maxMessageLength);
1344
1345#ifndef QT_NO_DEBUG
1346 if (!d->context->format().testOption(option: QSurfaceFormat::DebugContext)) {
1347 qWarning(msg: "QOpenGLDebugLogger::initialize(): the current context is not a debug context:\n"
1348 " this means that the GL may not generate any debug output at all.\n"
1349 " To avoid this warning, try creating the context with the\n"
1350 " QSurfaceFormat::DebugContext surface format option.");
1351 }
1352#endif // QT_NO_DEBUG
1353
1354 d->initialized = true;
1355 return true;
1356}
1357
1358/*!
1359 Returns \c true if this object is currently logging, false otherwise.
1360
1361 \sa startLogging()
1362*/
1363bool QOpenGLDebugLogger::isLogging() const
1364{
1365 Q_D(const QOpenGLDebugLogger);
1366 return d->isLogging;
1367}
1368
1369/*!
1370 Starts logging messages coming from the OpenGL server. When a new message
1371 is received, the signal messageLogged() is emitted, carrying the logged
1372 message as argument.
1373
1374 \a loggingMode specifies whether the logging must be asynchronous (the default)
1375 or synchronous.
1376
1377 QOpenGLDebugLogger will record the values of \c{GL_DEBUG_OUTPUT} and
1378 \c{GL_DEBUG_OUTPUT_SYNCHRONOUS} when logging is started, and set them back
1379 when logging is stopped. Moreover, any user-defined OpenGL debug callback
1380 installed when this function is invoked will be restored when logging is
1381 stopped; QOpenGLDebugLogger will ensure that the pre-existing callback will
1382 still be invoked when logging.
1383
1384 \note It's not possible to change the logging mode without stopping and
1385 starting logging again. This might change in a future version of Qt.
1386
1387 \note The object must be initialized before logging can happen.
1388
1389 \sa stopLogging(), initialize()
1390*/
1391void QOpenGLDebugLogger::startLogging(QOpenGLDebugLogger::LoggingMode loggingMode)
1392{
1393 Q_D(QOpenGLDebugLogger);
1394 if (!d->initialized) {
1395 qWarning(msg: "QOpenGLDebugLogger::startLogging(): object must be initialized before logging can start");
1396 return;
1397 }
1398 if (d->isLogging) {
1399 qWarning(msg: "QOpenGLDebugLogger::startLogging(): this object is already logging");
1400 return;
1401 }
1402
1403 d->isLogging = true;
1404 d->loggingMode = loggingMode;
1405
1406 d->glGetPointerv(GL_DEBUG_CALLBACK_FUNCTION, reinterpret_cast<void **>(&d->oldDebugCallbackFunction));
1407 d->glGetPointerv(GL_DEBUG_CALLBACK_USER_PARAM, &d->oldDebugCallbackParameter);
1408
1409 d->glDebugMessageCallback(&qt_opengl_debug_callback, d);
1410
1411 QOpenGLFunctions *funcs = QOpenGLContext::currentContext()->functions();
1412 d->debugWasEnabled = funcs->glIsEnabled(GL_DEBUG_OUTPUT);
1413 d->syncDebugWasEnabled = funcs->glIsEnabled(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1414
1415 if (d->loggingMode == SynchronousLogging)
1416 funcs->glEnable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1417 else
1418 funcs->glDisable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1419
1420 funcs->glEnable(GL_DEBUG_OUTPUT);
1421}
1422
1423/*!
1424 Returns the logging mode of the object.
1425
1426 \sa startLogging()
1427*/
1428QOpenGLDebugLogger::LoggingMode QOpenGLDebugLogger::loggingMode() const
1429{
1430 Q_D(const QOpenGLDebugLogger);
1431 return d->loggingMode;
1432}
1433
1434/*!
1435 Stops logging messages from the OpenGL server.
1436
1437 \sa startLogging()
1438*/
1439void QOpenGLDebugLogger::stopLogging()
1440{
1441 Q_D(QOpenGLDebugLogger);
1442 if (!d->isLogging)
1443 return;
1444
1445 QOpenGLContext *currentContext = QOpenGLContext::currentContext();
1446 if (!currentContext || currentContext != d->context) {
1447 qWarning(msg: "QOpenGLDebugLogger::stopLogging(): attempting to stop logging with the wrong OpenGL context current");
1448 return;
1449 }
1450
1451 d->isLogging = false;
1452
1453 d->glDebugMessageCallback(d->oldDebugCallbackFunction, d->oldDebugCallbackParameter);
1454
1455 QOpenGLFunctions *funcs = QOpenGLContext::currentContext()->functions();
1456 if (!d->debugWasEnabled)
1457 funcs->glDisable(GL_DEBUG_OUTPUT);
1458
1459 if (d->syncDebugWasEnabled)
1460 funcs->glEnable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1461 else
1462 funcs->glDisable(GL_DEBUG_OUTPUT_SYNCHRONOUS);
1463}
1464
1465/*!
1466 Inserts the message \a debugMessage into the OpenGL debug log. This provides
1467 a way for applications or libraries to insert custom messages that can
1468 ease the debugging of OpenGL applications.
1469
1470 \note \a debugMessage must have QOpenGLDebugMessage::ApplicationSource or
1471 QOpenGLDebugMessage::ThirdPartySource as its source, and a valid
1472 type and severity, otherwise it will not be inserted into the log.
1473
1474 \note The object must be initialized before logging can happen.
1475
1476 \sa initialize()
1477*/
1478void QOpenGLDebugLogger::logMessage(const QOpenGLDebugMessage &debugMessage)
1479{
1480 Q_D(QOpenGLDebugLogger);
1481 if (!d->initialized) {
1482 qWarning(msg: "QOpenGLDebugLogger::logMessage(): object must be initialized before logging messages");
1483 return;
1484 }
1485 if (debugMessage.source() != QOpenGLDebugMessage::ApplicationSource
1486 && debugMessage.source() != QOpenGLDebugMessage::ThirdPartySource) {
1487 qWarning(msg: "QOpenGLDebugLogger::logMessage(): using a message source different from ApplicationSource\n"
1488 " or ThirdPartySource is not supported by GL_KHR_debug. The message will not be logged.");
1489 return;
1490 }
1491 if (debugMessage.type() == QOpenGLDebugMessage::InvalidType
1492 || debugMessage.type() == QOpenGLDebugMessage::AnyType
1493 || debugMessage.severity() == QOpenGLDebugMessage::InvalidSeverity
1494 || debugMessage.severity() == QOpenGLDebugMessage::AnySeverity) {
1495 qWarning(msg: "QOpenGLDebugLogger::logMessage(): the message has a non-valid type and/or severity. The message will not be logged.");
1496 return;
1497 }
1498
1499 const GLenum source = qt_messageSourceToGL(source: debugMessage.source());
1500 const GLenum type = qt_messageTypeToGL(type: debugMessage.type());
1501 const GLenum severity = qt_messageSeverityToGL(severity: debugMessage.severity());
1502 QByteArray rawMessage = debugMessage.message().toUtf8();
1503 rawMessage.append(c: '\0');
1504
1505 if (rawMessage.size() > d->maxMessageLength) {
1506 qWarning(msg: "QOpenGLDebugLogger::logMessage(): message too long, truncating it\n"
1507 " (%d bytes long, but the GL accepts up to %d bytes)", int(rawMessage.size()), d->maxMessageLength);
1508 rawMessage.resize(size: d->maxMessageLength - 1);
1509 rawMessage.append(c: '\0');
1510 }
1511
1512 // Don't pass rawMessage.length(), as unfortunately bugged
1513 // OpenGL drivers will eat the trailing NUL in the message. Just rely
1514 // on the message being NUL terminated.
1515 d->glDebugMessageInsert(source,
1516 type,
1517 debugMessage.id(),
1518 severity,
1519 -1,
1520 rawMessage.constData());
1521}
1522
1523/*!
1524 Pushes a debug group with name \a name, id \a id, and source \a source onto
1525 the debug groups stack. If the group is successfully pushed, OpenGL will
1526 automatically log a message with message \a name, id \a id, source \a
1527 source, type QOpenGLDebugMessage::GroupPushType and severity
1528 QOpenGLDebugMessage::NotificationSeverity.
1529
1530 The newly pushed group will inherit the same filtering settings of the
1531 group that was on the top of the stack; that is, the filtering will not be
1532 changed by pushing a new group.
1533
1534 \note The \a source must either be QOpenGLDebugMessage::ApplicationSource or
1535 QOpenGLDebugMessage::ThirdPartySource, otherwise the group will not be pushed.
1536
1537 \note The object must be initialized before managing debug groups.
1538
1539 \sa popGroup(), enableMessages(), disableMessages()
1540*/
1541void QOpenGLDebugLogger::pushGroup(const QString &name, GLuint id, QOpenGLDebugMessage::Source source)
1542{
1543 Q_D(QOpenGLDebugLogger);
1544 if (!d->initialized) {
1545 qWarning(msg: "QOpenGLDebugLogger::pushGroup(): object must be initialized before pushing a debug group");
1546 return;
1547 }
1548 if (source != QOpenGLDebugMessage::ApplicationSource
1549 && source != QOpenGLDebugMessage::ThirdPartySource) {
1550 qWarning(msg: "QOpenGLDebugLogger::pushGroup(): using a source different from ApplicationSource\n"
1551 " or ThirdPartySource is not supported by GL_KHR_debug. The group will not be pushed.");
1552 return;
1553 }
1554
1555 QByteArray rawName = name.toUtf8();
1556 rawName.append(c: '\0');
1557 if (rawName.size() > d->maxMessageLength) {
1558 qWarning(msg: "QOpenGLDebugLogger::pushGroup(): group name too long, truncating it\n"
1559 " (%d bytes long, but the GL accepts up to %d bytes)", int(rawName.size()), d->maxMessageLength);
1560 rawName.resize(size: d->maxMessageLength - 1);
1561 rawName.append(c: '\0');
1562 }
1563
1564 // Don't pass rawMessage.length(), as unfortunately bugged
1565 // OpenGL drivers will eat the trailing NUL in the name. Just rely
1566 // on the name being NUL terminated.
1567 d->glPushDebugGroup(qt_messageSourceToGL(source), id, -1, rawName.constData());
1568}
1569
1570/*!
1571 Pops the topmost debug group from the debug groups stack. If the group is
1572 successfully popped, OpenGL will automatically log a message with message,
1573 id and source matching those of the popped group, type
1574 QOpenGLDebugMessage::GroupPopType and severity
1575 QOpenGLDebugMessage::NotificationSeverity.
1576
1577 Popping a debug group will restore the message filtering settings of the
1578 group that becomes the top of the debug groups stack.
1579
1580 \note The object must be initialized before managing debug groups.
1581
1582 \sa pushGroup()
1583*/
1584void QOpenGLDebugLogger::popGroup()
1585{
1586 Q_D(QOpenGLDebugLogger);
1587 if (!d->initialized) {
1588 qWarning(msg: "QOpenGLDebugLogger::pushGroup(): object must be initialized before popping a debug group");
1589 return;
1590 }
1591
1592 d->glPopDebugGroup();
1593}
1594
1595/*!
1596 Enables the logging of messages from the given \a sources, of the given \a
1597 types and with the given \a severities and any message id.
1598
1599 The logging will be enabled in the current control group.
1600
1601 \sa disableMessages(), pushGroup(), popGroup()
1602*/
1603void QOpenGLDebugLogger::enableMessages(QOpenGLDebugMessage::Sources sources,
1604 QOpenGLDebugMessage::Types types,
1605 QOpenGLDebugMessage::Severities severities)
1606{
1607 Q_D(QOpenGLDebugLogger);
1608 d->controlDebugMessages(sources, types, severities, ids: QList<GLuint>(),
1609 QByteArrayLiteral("enableMessages"), enable: true);
1610}
1611
1612/*!
1613 Enables the logging of messages with the given \a ids, from the given \a
1614 sources and of the given \a types and any severity.
1615
1616 The logging will be enabled in the current control group.
1617
1618 \sa disableMessages(), pushGroup(), popGroup()
1619*/
1620void QOpenGLDebugLogger::enableMessages(const QList<GLuint> &ids,
1621 QOpenGLDebugMessage::Sources sources,
1622 QOpenGLDebugMessage::Types types)
1623{
1624 Q_D(QOpenGLDebugLogger);
1625 d->controlDebugMessages(sources,
1626 types,
1627 severities: QOpenGLDebugMessage::AnySeverity,
1628 ids,
1629 QByteArrayLiteral("enableMessages"),
1630 enable: true);
1631}
1632
1633/*!
1634 Disables the logging of messages with the given \a sources, of the given \a
1635 types and with the given \a severities and any message id.
1636
1637 The logging will be disabled in the current control group.
1638
1639 \sa enableMessages(), pushGroup(), popGroup()
1640*/
1641void QOpenGLDebugLogger::disableMessages(QOpenGLDebugMessage::Sources sources,
1642 QOpenGLDebugMessage::Types types,
1643 QOpenGLDebugMessage::Severities severities)
1644{
1645 Q_D(QOpenGLDebugLogger);
1646 d->controlDebugMessages(sources, types, severities, ids: QList<GLuint>(),
1647 QByteArrayLiteral("disableMessages"), enable: false);
1648}
1649
1650/*!
1651 Disables the logging of messages with the given \a ids, from the given \a
1652 sources and of the given \a types and any severity.
1653
1654 The logging will be disabled in the current control group.
1655
1656 \sa enableMessages(), pushGroup(), popGroup()
1657*/
1658void QOpenGLDebugLogger::disableMessages(const QList<GLuint> &ids,
1659 QOpenGLDebugMessage::Sources sources,
1660 QOpenGLDebugMessage::Types types)
1661{
1662 Q_D(QOpenGLDebugLogger);
1663 d->controlDebugMessages(sources,
1664 types,
1665 severities: QOpenGLDebugMessage::AnySeverity,
1666 ids,
1667 QByteArrayLiteral("disableMessages"),
1668 enable: false);
1669}
1670
1671/*!
1672 Reads all the available messages in the OpenGL internal debug log and
1673 returns them. Moreover, this function will clear the internal debug log,
1674 so that subsequent invocations will not return messages that were
1675 already returned.
1676
1677 \sa startLogging()
1678*/
1679QList<QOpenGLDebugMessage> QOpenGLDebugLogger::loggedMessages() const
1680{
1681 Q_D(const QOpenGLDebugLogger);
1682 if (!d->initialized) {
1683 qWarning(msg: "QOpenGLDebugLogger::loggedMessages(): object must be initialized before reading logged messages");
1684 return QList<QOpenGLDebugMessage>();
1685 }
1686
1687 static const GLuint maxMessageCount = 128;
1688 GLuint messagesRead;
1689 GLenum messageSources[maxMessageCount];
1690 GLenum messageTypes[maxMessageCount];
1691 GLuint messageIds[maxMessageCount];
1692 GLenum messageSeverities[maxMessageCount];
1693 GLsizei messageLengths[maxMessageCount];
1694
1695 QByteArray messagesBuffer;
1696 messagesBuffer.resize(size: maxMessageCount * d->maxMessageLength);
1697
1698 QList<QOpenGLDebugMessage> messages;
1699 do {
1700 messagesRead = d->glGetDebugMessageLog(maxMessageCount,
1701 GLsizei(messagesBuffer.size()),
1702 messageSources,
1703 messageTypes,
1704 messageIds,
1705 messageSeverities,
1706 messageLengths,
1707 messagesBuffer.data());
1708
1709 const char *messagesBufferPtr = messagesBuffer.constData();
1710 for (GLuint i = 0; i < messagesRead; ++i) {
1711 QOpenGLDebugMessage message;
1712
1713 QOpenGLDebugMessagePrivate *messagePrivate = message.d.data();
1714 messagePrivate->source = qt_messageSourceFromGL(source: messageSources[i]);
1715 messagePrivate->type = qt_messageTypeFromGL(type: messageTypes[i]);
1716 messagePrivate->id = messageIds[i];
1717 messagePrivate->severity = qt_messageSeverityFromGL(severity: messageSeverities[i]);
1718 messagePrivate->message = QString::fromUtf8(utf8: messagesBufferPtr, size: messageLengths[i] - 1);
1719
1720 messagesBufferPtr += messageLengths[i];
1721 messages << message;
1722 }
1723 } while (messagesRead == maxMessageCount);
1724
1725 return messages;
1726}
1727
1728/*!
1729 \fn void QOpenGLDebugLogger::messageLogged(const QOpenGLDebugMessage &debugMessage)
1730
1731 This signal is emitted when a debug message (wrapped by the \a debugMessage
1732 argument) is logged from the OpenGL server.
1733
1734 Depending on the OpenGL implementation, this signal can be emitted
1735 from other threads than the one(s) the receiver(s) lives in, and even
1736 different from the thread the QOpenGLContext in which this object has
1737 been initialized lives in. Moreover, the signal could be emitted from
1738 multiple threads at the same time. This is normally not a problem,
1739 as Qt will utilize a queued connection for cross-thread signal emissions,
1740 but if you force the connection type to Direct then you must be aware of
1741 the potential races in the slots connected to this signal.
1742
1743 If logging have been started in SynchronousLogging mode, OpenGL guarantees
1744 that this signal will be emitted from the same thread the QOpenGLContext
1745 has been bound to, and no concurrent invocations will ever happen.
1746
1747 \note Logging must have been started, or this signal will not be emitted.
1748
1749 \sa startLogging()
1750*/
1751
1752/*!
1753 Returns the maximum supported length, in bytes, for the text of the messages
1754 passed to logMessage(). This is also the maximum length of a debug group
1755 name, as pushing or popping groups will automatically log a message with
1756 the debug group name as the message text.
1757
1758 If a message text is too long, it will be automatically truncated by
1759 QOpenGLDebugLogger.
1760
1761 \note Message texts are encoded in UTF-8 when they get passed to OpenGL, so
1762 their size in bytes does not usually match the amount of UTF-16 code units,
1763 as returned, for instance, by QString::length(). (It does if the message contains
1764 7-bit ASCII only data, which is typical for debug messages.)
1765*/
1766qint64 QOpenGLDebugLogger::maximumMessageLength() const
1767{
1768 Q_D(const QOpenGLDebugLogger);
1769 if (!d->initialized) {
1770 qWarning(msg: "QOpenGLDebugLogger::maximumMessageLength(): object must be initialized before reading the maximum message length");
1771 return -1;
1772 }
1773 return d->maxMessageLength;
1774}
1775
1776
1777QT_END_NAMESPACE
1778
1779#include "moc_qopengldebug.cpp"
1780

Provided by KDAB

Privacy Policy
Learn Advanced QML with KDAB
Find out more

source code of qtbase/src/opengl/qopengldebug.cpp