1Name 2 3 AMD_debug_output 4 5Name Strings 6 7 GL_AMD_debug_output 8 9Contact 10 11 Jaakko Konttinen (jaakko.konttinen 'at' amd.com) 12 13Contributors 14 15 Graham Sellers, AMD 16 Mark Young, AMD 17 Ahmet Oguz Akyuz, AMD 18 19Status 20 21 Experimental 22 23Version 24 25 Last Modified Date: July 21, 2022 26 Author Revision: 10 27 28Number 29 30 395 31 32Dependencies 33 34 OpenGL 1.1 is required. 35 36 WGL_ARB_create_context or GLX_ARB_create_context is required. 37 38 The extension is written against the OpenGL 3.0 specification. 39 40 This extension trivially interacts with ARB_vertex_program and 41 ARB_fragment_program. 42 43Overview 44 45 This extension allows the GL to notify applications when various 46 debug events occur in contexts that have been created with the debug 47 flag, as provided by WGL_ARB_create_context and GLX_ARB_create_context. 48 49 These events are represented in the form of enumerable messages with an 50 included human-readable translation. Examples of debug events include 51 incorrect use of the GL, warnings of undefined behavior, and performance 52 warnings. 53 54 A message is uniquely identified by a category and an implementation- 55 dependent ID within that category. Message categories are general and are 56 used to organize large groups of similar messages together. Examples of 57 categories include GL errors, performance warnings, and deprecated 58 functionality warnings. Each message is also assigned a severity level 59 that denotes roughly how "important" that message is in comparison to 60 other messages across all categories. For example, notification of a GL 61 error would have a higher severity than a performance warning due to 62 redundant state changes. 63 64 Messages are communicated to the application through an application-defined 65 callback function that is called by the GL implementation on each debug 66 message. The motivation for the callback routine is to free application 67 developers from actively having to query whether any GL error or other 68 debuggable event has happened after each call to a GL function. With a 69 callback, developers can keep their code free of debug checks, and only have 70 to react to messages as they occur. In order to support indirect rendering, 71 a message log is also provided that stores copies of recent messages until 72 they are actively queried. 73 74 To control the volume of debug output, messages can be disabled either 75 individually by ID, or entire groups of messages can be turned off based 76 on category or severity. 77 78 The only requirement on the minimum quantity and type of messages that 79 implementations of this extension must support is that a message must be 80 sent notifying the application whenever any GL error occurs. Any further 81 messages are left to the implementation. Implementations do not have 82 to output messages from all categories listed by this extension 83 in order to support this extension, and new categories can be added by 84 other extensions. 85 86 This extension places no restrictions or requirements on any additional 87 functionality provided by the debug context flag through other extensions. 88 89IP Status 90 91 No known IP claims. 92 93New Procedures and Functions 94 95 void DebugMessageEnableAMD(enum category, 96 enum severity, 97 sizei count, 98 const uint* ids, 99 boolean enabled); 100 101 void DebugMessageInsertAMD(enum category, 102 enum severity, 103 uint id, 104 sizei length, 105 const char* buf); 106 107 void DebugMessageCallbackAMD(DEBUGPROCAMD callback, 108 void* userParam); 109 110 uint GetDebugMessageLogAMD(uint count, 111 sizei bufsize, 112 enum* categories, 113 enum* severities, 114 uint* ids, 115 sizei* lengths, 116 char* message); 117 118New Types 119 120 The callback function that applications can define, and 121 is accepted by DebugMessageCallbackAMD, is defined as: 122 123 typedef void (APIENTRY *GLDEBUGPROCAMD)(GLuint id, 124 GLenum category, 125 GLenum severity, 126 GLsizei length, 127 const GLchar* message, 128 GLvoid* userParam); 129 130 Note that this function pointer is defined as having the 131 same calling convention as the GL functions. 132 133New Tokens 134 135 Tokens accepted by GetIntegerv: 136 137 MAX_DEBUG_MESSAGE_LENGTH_AMD 0x9143 138 MAX_DEBUG_LOGGED_MESSAGES_AMD 0x9144 139 DEBUG_LOGGED_MESSAGES_AMD 0x9145 140 141 Tokens accepted by DebugMessageEnableAMD, GetDebugMessageLogAMD, 142 DebugMessageInsertAMD, and DEBUGPROCAMD callback function 143 for <severity>: 144 145 DEBUG_SEVERITY_HIGH_AMD 0x9146 146 DEBUG_SEVERITY_MEDIUM_AMD 0x9147 147 DEBUG_SEVERITY_LOW_AMD 0x9148 148 149 Tokens accepted by DebugMessageEnableAMD, GetDebugMessageLogAMD, 150 and DEBUGPROCAMD callback function for <category>: 151 152 DEBUG_CATEGORY_API_ERROR_AMD 0x9149 153 DEBUG_CATEGORY_WINDOW_SYSTEM_AMD 0x914A 154 DEBUG_CATEGORY_DEPRECATION_AMD 0x914B 155 DEBUG_CATEGORY_UNDEFINED_BEHAVIOR_AMD 0x914C 156 DEBUG_CATEGORY_PERFORMANCE_AMD 0x914D 157 DEBUG_CATEGORY_SHADER_COMPILER_AMD 0x914E 158 DEBUG_CATEGORY_APPLICATION_AMD 0x914F 159 DEBUG_CATEGORY_OTHER_AMD 0x9150 160 161Additions to Chapter 2 of the OpenGL 3.0 Specification (OpenGL Operation) 162 163 In section 2.5 - GL Errors: 164 Add to the end of the section (pg 19): "If an error is generated by a debug 165 GL context (Section 2.20), the context will send a message to the application 166 that a GL error has occurred. This message may contain more information 167 about the nature of the error." 168 169 After section 2.19 - Primitive Clipping: 170 Add new section: 2.20 - Debug Contexts 171 172 Application developers can obtain more information from the GL runtime using 173 a debug-enabled context. This information can include details about 174 GL errors, undefined behavior, implementation-dependent performance warnings, 175 or other useful hints. This information is communicated through the 176 generation of debug messages when GL commands are executed. The application 177 can choose to either actively query for these messages, or allow the GL to 178 call back to the application on each message via a function pointer. 179 180 2.20.1 - Debug Messages 181 182 A debug message is uniquely identified by its category and an unsigned 183 integer message ID within that category. The category must be one of the 184 symbolic constants listed in Table 2.12. Although every distinct message 185 must have a unique number within a category, there is no enforcement by this 186 extension for how GL implementations assign numbers to specific messages. 187 188 Each message also has a severity level that roughly describes its 189 significance across all categories. The severity level of a message is 190 one of the symbolic constants defined in Table 2.13. Because messages 191 can be disabled across all categories by severity level, this feature can 192 be used to quickly control the volume of debug output by the application. 193 194 Category Token Informs about 195 -------------- ------------- 196 DEBUG_CATEGORY_API_ERROR_AMD GL errors caused by invalid API use 197 198 DEBUG_CATEGORY_WINDOW_SYSTEM_AMD Errors and notices from the 199 windowing layer 200 201 DEBUG_CATEGORY_DEPRECATION_AMD Use of functionality that is either 202 deprecated or marked for future 203 deprecation 204 205 DEBUG_CATEGORY_UNDEFINED_BEHAVIOR_AMD Behavior undefined according to 206 specification 207 208 DEBUG_CATEGORY_PERFORMANCE_AMD Implementation-dependent performance 209 warnings 210 211 DEBUG_CATEGORY_SHADER_COMPILER_AMD Information from the GLSL or ARB 212 shader compiler and linker 213 214 DEBUG_CATEGORY_APPLICATION_AMD Application-generated messages 215 216 DEBUG_CATEGORY_OTHER_AMD Messages that do not fit in any of 217 the other categories 218 ---------------------------------------------------------------------------- 219 Table 2.12: Categories of debug messages. Each message is associated with 220 one of these categories. 221 222 223 Severity Level Token Suggested examples of messages 224 -------------------- ------------------------------ 225 226 DEBUG_SEVERITY_HIGH_AMD Any GL error; any undefined behavior; 227 any GLSL or ARB shader compiler and 228 linker errors; 229 230 DEBUG_SEVERITY_MEDIUM_AMD Severe performance warnings; GLSL 231 or ARB shader compiler and linker 232 warnings; use of currently deprecated 233 behavior 234 235 DEBUG_SEVERITY_LOW_AMD Performance warnings from redundant 236 state changes 237 ---------------------------------------------------------------------------- 238 Table 2.13: Severity levels of messagse. Each debug message is associated 239 with one of these severity levels. 240 241 242 Every message also has a null-terminated string representation that is used 243 to describe the message. The contents of the string can change slightly 244 between different instances of the same message (e.g. which parameter value 245 caused a specific GL error to occur). The formatting of a message string is 246 left as implementation-dependent, although it should give a concise and 247 legible description of the message's purpose. Messages with different IDs 248 should also have sufficiently distinguishable string representations to 249 warrant their separation. 250 251 2.20.2 - Receiving Messages 252 253 Applications can listen for messages by providing the GL with a callback 254 function pointer by calling: 255 256 void DebugMessageCallbackAMD(DEBUGPROCAMD callback, 257 void* userParam); 258 259 With <callback> storing the address of the callback function. This 260 function's signature must follow the type definition of DEBUGPROCAMD, and 261 its calling convention must be the same as the calling convention of GL 262 functions. Anything else will result in undefined behavior. Only one 263 debug callback can be specified for the current context, and further calls 264 overwrite the previous callback. Specifying zero as the value of <callback> 265 clears the current callback and disables message output through callbacks. 266 Applications can specify user-specified data through the pointer 267 <userParam>. The context will store this pointer and will include it 268 as one of the parameters of each call to the callback function. The error 269 INVALID_OPERATION will be generated if this function is called for contexts 270 created without the debug flag. 271 272 If the application has specified a callback function in a debug context, 273 the implementation will call that function whenever any unfiltered message 274 is generated. The ID, category, and severity of the message are specified 275 by the callback parameters <id>, <category> and <severity>, respectively. 276 The string representation of the message is stored in <message> and its length 277 (excluding the null-terminator) is stored in <length>. The parameter 278 <userParam> is the user-specified value that was passed when calling 279 DebugMessageCallbackAMD. The memory for <message> is allocated, owned and 280 released by the implementation, and should only be considered valid for 281 the duration of the callback function call. While it is allowed to 282 concurrently use multiple debug contexts with the same debug callback function, 283 note that it is the application's responsibility to ensure that any work that 284 occurs inside the debug callback function is thread-safe. Furthermore, 285 calling any GL or window layer function from within the callback function 286 results in undefined behavior. 287 288 If no callback is set, then messages are instead stored in an internal 289 message log up to some maximum number of strings as defined by the 290 implementation-dependent constant MAX_DEBUG_LOGGED_MESSAGES_AMD. Each 291 context stores its own message log and will only store messages generated by 292 commands operating in that context. If the message log is full, then the 293 oldest messages will be removed from the log to make room for newer ones. 294 The application can query the number of messages currently in the log by 295 obtaining the value of DEBUG_LOGGED_MESSAGES_AMD, and can get the contents 296 of those messages using the command: 297 298 uint GetDebugMessageLogAMD(uint count, 299 sizei logSize, 300 enum* categories, 301 enum* severities, 302 uint* ids, 303 sizei* lengths, 304 char* messageLog); 305 306 This function will fetch as many messages as possible from the message 307 log up to <count> in order from oldest to newest, and will return the 308 number of messages fetched. Those messages that were fetched will be 309 removed from the log. The value of <count> must be greater than zero and 310 less than MAX_DEBUG_LOGGED_MESSAGES_AMD or otherwise the error 311 INVALID_VALUE will be generated. The value of <count> can be larger than 312 the actual number of messages currently in the log. If <messageLog> is not 313 a null pointer, then the string representations of all fetched messages will 314 be stored in the buffer <messageLog> and will be separated by null- 315 terminators. The maximum size of the buffer (including all null- 316 terminators) is denoted by <logSize>, and strings of messages within <count> 317 that do not fit in the buffer will not be fetched. If <logSize> is less 318 than zero, the error INVALID_VALUE will be generated. If <messageLog> 319 is a null pointer, then the value of <logSize> is ignored. The 320 categories, severity levels, IDs, and string representation lengths of all 321 (up to <count>) removed messages will be stored in the arrays <categories>, 322 <severities>, <ids>, and <lengths>, respectively. The counts stored in the 323 array <lengths> include the null-terminator of each string. Any and all of 324 the output arrays, including <messageLog>, are optional, and no data is returned 325 for those arrays that are specified with a null pointer. To simply delete up 326 to <count> messages from the message log and ignoring, the application can call 327 the function with null pointers for all output arrays. The error 328 INVALID_OPERATION will be generated by GetDebugMessageLogAMD if it is 329 called in a non-debug context. 330 331 2.20.3 - Controlling Debug Messages 332 333 Applications can control which messages are generated by calling 334 335 void DebugMessageEnableAMD(enum category, 336 enum severity, 337 sizei count, 338 const uint* ids, 339 boolean enabled); 340 341 This command allows disabling or enabling generation of subsets of 342 messages. If <enabled> is TRUE, the referenced subset of messages 343 is enabled. If FALSE, then those messages are disabled. This 344 command can reference different subsets of messages by 345 varying its parameter values in the following ways: 346 347 1. To reference all messages, let <category>, <severity>, 348 and <count> all be zero. The value of <ids> is ignored 349 in this case. 350 351 2. To reference all messages across all categories with 352 a specific severity level, let <category> and <count> 353 be zero and let <severity> identify the severity level. 354 The value of <ids> is ignored in this case. 355 356 3. To reference all messages within a single category, let 357 <category> identify the referenced category and let 358 <severity> and <count> be zero. The value of <ids> is 359 ignored in this case. 360 361 4. To reference all messages within a single category and 362 at a specific severity level, let <category> identify the 363 category and <severity> identify the severity level, 364 and let <count> be zero. The value of <ids> is ignored 365 in this case. 366 367 5. To reference specific messages by ID within a single 368 category, let <category> identify the category, let 369 <severity> be zero, let <count> be greater than zero 370 and let <ids> identify the IDs of <count> messages within 371 the identified category. Operations on message IDs that 372 are not valid within the category are silently ignored. 373 374 In all of the above cases, if <category> is non-zero and specifies 375 an invalid category, the error INVALID_ENUM is generated. 376 Similarly if <severity> is non-zero and is an invalid severity level, 377 the error INVALID_ENUM is generated. If <count> is less than 378 zero, the error INVALID_VALUE is generated. If the parameters do 379 not fall into one of the cases defined above, the error INVALID_VALUE 380 is generated. The error INVALID_OPERATION is generated if this 381 command is called in a non-debug context. 382 383 Although messages are grouped into categories and severities, 384 and entire groups of messages can be turned off with a single 385 call, there is no explicit per-category or per-severity enabled state. 386 Instead the enabled state is stored individually for each message. 387 There is no difference between disabling a category of messages with a 388 single call, and enumerating all messages of that category and individually 389 disabling each of them by their ID. 390 391 All messages of severity level DEBUG_SEVERITY_MEDIUM_AMD and 392 DEBUG_SEVERITY_HIGH_AMD in all categories are initially enabled, and 393 all messages at DEBUG_SEVERITY_LOW_AMD are initially disabled. 394 395 2.20.4 - Application Messages 396 397 To easily support custom application timestamps, applications can inject 398 their own messages to the debug message stream through the command 399 400 void DebugMessageInsertAMD(enum category, 401 enum severity, 402 uint id, 403 sizei length, 404 const char* buf); 405 406 The value of <id> specifies the ID for the message and <severity> indicates 407 its severity level as defined by the application. If <severity> is not a 408 valid severity level, the error INVALID_ENUM will be generated. The value 409 of <category> must be DEBUG_CATEGORY_APPLICATION_AMD, or the error 410 INVALID_ENUM will be generated. The string <buf> contains the string 411 representation of the message. The parameter <length> contains the size of 412 the message's string representation, excluding the null-terminator. If 413 <length> is zero, then its value is derived from the string-length of <buf> 414 and <buf> must contain a null-terminated string. The error INVALID_VALUE 415 will be generated if <length> is less than zero or its derived value is 416 larger than or equal to MAX_DEBUG_MESSAGE_LENGTH_AMD. The error 417 INVALID_OPERATION will be generated if this function is called in a 418 non-debug context. 419 420Additions to the OpenGL / GLX / GLX Protocol Specifications 421 422 None. 423 424Additions to the WGL Specification 425 426 None. 427 428Errors 429 430 The error INVALID_OPERATION will be generated by DebugMessageCallbackAMD if 431 the function is called in a non-debug context. 432 433 The error INVALID_ENUM will be generated by DebugMessageEnableAMD if 434 <category> is non-zero and specifies an invalid category. 435 436 The error INVALID_ENUM will be generated by DebugMessageEnableAMD if 437 <severity> is non-zero and an invalid severity level. 438 439 The error INVALID_VALUE will be generated by DebugMessageEnableAMD if 440 if <count> is less than zero. 441 442 The error INVALID_VALUE will be generated by DebugMessageEnableAMD if 443 if the combination of values for <category>, <severity> and <count> do not 444 fall within one of the accepted combinations for referencing a subset of 445 messages. 446 447 The error INVALID_OPERATION will be generated by GetDebugMessageLogAMD if 448 it is called in a non-debug context. 449 450 The error INVALID_VALUE will be generated by GetDebugMessageLogAMD if the 451 value of <count> is zero or greater than the value of 452 MAX_DEBUG_LOGGED_MESSAGES_AMD. 453 454 The error INVALID_VALUE will be generated by GetDebugMessageLogAMD if 455 <logSize> is less than zero. 456 457 The error INVALID_ENUM will be generated by DebugMessageInsertAMD if 458 <severity> is not a valid debug severity level. 459 460 The error INVALID_ENUM will be generated by DebugMessageInsertAMD if the 461 value of <category> is not DEBUG_CATEGORY_APPLICATION_AMD. 462 463 The error INVALID_VALUE will be generated by DebugMessageInsertAMD 464 if <length> is less than zero. 465 466 The error INVALID_VALUE will be generated by DebugMessageInsertAMD 467 if <length> or its derived value is larger than 468 MAX_DEBUG_MESSAGE_LENGTH_AMD. 469 470 The error INVALID_OPERATION will be generated by DebugMessageInsertAMD if 471 this function is called in a non-debug context. 472 473New State 474 Initial 475 Get Value Type Get Command Value Description Sec Attribute 476 -------------------------- ---- ----------- ----- ------------------------- ------ --------- 477 DEBUG_LOGGED_MESSAGES_AMD Z+ GetIntegerv 0 The number of messages 2.20.2 - 478 currently in the debug 479 message log 480 481New Implementation Dependent State 482 Minimum 483 Get Value Type Get Command Value Description Sec Attribute 484 -------------------------------- -- ----------- ----- ------------------------- ------ ---------- 485 MAX_DEBUG_MESSAGE_LENGTH_AMD Z+ GetIntegerv 80 The maximum length of a 2.20.2 - 486 debug message string 487 488 MAX_DEBUG_LOGGED_MESSAGES_AMD Z+ GetIntegerv 1 The maximum number of 2.20.2 - 489 messages stored in the 490 debug message log 491 492Issues 493 494 01) Should we reserve tokens for arbitrary vendor-specific categories (e.g. 495 DEBUG_CATEGORY_VENDOR0)? 496 497 RESOLVED: No. Since this is an AMD extension, there is no 498 reason to do this now in the current version. 499 500 02) Should we allow explicit controls for printing to stderr or through 501 OutputDebugString instead of the callback? 502 503 RESOLVED: No. It is up to the application to setup this 504 behavior itself using the provided functionality. 505 506 507 03) How do the different filtering rules interact? If a category is 508 filtered, but a message in that same category is set to be unfiltered by 509 ID or severity level, should that message still be filtered? If I 510 specifically filter a category and then unfilter all messages globally 511 with a later command, should that category still be filtered? 512 513 RESOLVED: Message enable state is stored individually for each 514 message. There is no explicit group-wise enable state 515 or different ordered levels of "enabledness" in the 516 implied hierarchy of messages. Operations on groups 517 of messages affect all messages within 518 that group individually, and overwrite the previous 519 and individual state of those messages. 520 521 04) Should applications be allowed to insert their own messages through 522 a custom category? How would this be done? 523 524 RESOLVED: Yes. A new category will be provided for application- 525 specific messages and an entry point will be provided to 526 write messages in that category. 527 528 529 530Revision History 531 532 (v10 2022-07-21, jhager) 533 - Changed type of argument "severities" from uint to enum in GetDebugMessageLogAMD. 534 535 (v9, 2010-05-07, jkontti) 536 - Removed inconsistent language from New Tokens category for tokens 537 that are no longer accepted by DebugMessageEnableAMD since 538 revision v5. 539 - Cleaned up some language and formatting issues. 540 541 (v8, 2010-04-09, jkontti) 542 - Renamed extension string from AMDX to AMD. 543 - Renamed new function, type and token suffixes from AMDX to AMD. 544 - No changes in functionality between AMDX and AMD versions. 545 546 (v7, 2010-01-21, jkontti) 547 - Added user-specifiable parameter to debug message callback function 548 549 (v6, 2010-01-15, jkontti) 550 - Updated contact section 551 - Updated contributor section 552 - Updated status section 553 - Updated enums 554 555 (v5, 2009-09-17, jkontti) 556 - Message ID namespaces are now contained within categories instead of 557 being in a single global namespace 558 - Reworked DebugMessageEnable to allow disabling/enabling more 559 combinations of messages 560 - Resolved issue 01 561 - Resolved issue 03 562 563 (v4, 2009-09-16, jkontti) 564 - Added category as a parameter to DebugMessageInsert for 565 future-proofing purposes 566 - Added missing errors to DebugMessageInsert and GetDebugMessageLog 567 - Added missing tokens to New Tokens 568 - Renamed DebugMessageFilter to DebugMessageEnable 569 570 (v3, 2009-09-15, myoung) 571 - Cleaned up some language 572 - Added values using AMD reserved ranges. Values do not overlap so can 573 be re-used. 574 575 (v2, 2009-09-15, jkontti) 576 - Application-generated messages 577 - More categories (window system, deprecation, profile, application) 578 579 (v1, 2009-09-09, jkontti) 580 - Initial revision 581