]> git.cworth.org Git - apitrace/blob - retrace/glstate_params.py
glstate: Dump more object labels.
[apitrace] / retrace / glstate_params.py
1 ##########################################################################
2 #
3 # Copyright 2011 Jose Fonseca
4 # All Rights Reserved.
5 #
6 # Permission is hereby granted, free of charge, to any person obtaining a copy
7 # of this software and associated documentation files (the "Software"), to deal
8 # in the Software without restriction, including without limitation the rights
9 # to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
10 # copies of the Software, and to permit persons to whom the Software is
11 # furnished to do so, subject to the following conditions:
12 #
13 # The above copyright notice and this permission notice shall be included in
14 # all copies or substantial portions of the Software.
15 #
16 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
17 # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
18 # FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
19 # AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
21 # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
22 # THE SOFTWARE.
23 #
24 ##########################################################################/
25
26
27 '''Generate code to dump most GL state into JSON.'''
28
29
30 import retrace # to adjust sys.path
31
32 from specs.stdapi import *
33
34 from specs.gltypes import *
35 from specs.glparams import *
36
37
38 texture_targets = [
39     ('GL_TEXTURE_1D', 'GL_TEXTURE_BINDING_1D'),
40     ('GL_TEXTURE_1D_ARRAY', 'GL_TEXTURE_BINDING_1D_ARRAY'),
41     ('GL_TEXTURE_2D', 'GL_TEXTURE_BINDING_2D'),
42     ('GL_TEXTURE_2D_ARRAY', 'GL_TEXTURE_BINDING_2D_ARRAY'),
43     ('GL_TEXTURE_2D_MULTISAMPLE', 'GL_TEXTURE_BINDING_2D_MULTISAMPLE'),
44     ('GL_TEXTURE_2D_MULTISAMPLE_ARRAY', 'GL_TEXTURE_BINDING_2D_MULTISAMPLE_ARRAY'),
45     ('GL_TEXTURE_3D', 'GL_TEXTURE_BINDING_3D'),
46     ('GL_TEXTURE_RECTANGLE', 'GL_TEXTURE_BINDING_RECTANGLE'),
47     ('GL_TEXTURE_CUBE_MAP', 'GL_TEXTURE_BINDING_CUBE_MAP'),
48     ('GL_TEXTURE_CUBE_MAP_ARRAY', 'GL_TEXTURE_BINDING_CUBE_MAP_ARRAY'),
49 ]
50
51 framebuffer_targets = [
52     ('GL_DRAW_FRAMEBUFFER', 'GL_DRAW_FRAMEBUFFER_BINDING'),
53     ('GL_READ_FRAMEBUFFER', 'GL_READ_FRAMEBUFFER_BINDING'),
54 ]
55
56 class GetInflector:
57     '''Objects that describes how to inflect.'''
58
59     reduced_types = {
60         B: I,
61         E: I,
62         I: F,
63     }
64
65     def __init__(self, radical, inflections, suffix = ''):
66         self.radical = radical
67         self.inflections = inflections
68         self.suffix = suffix
69
70     def reduced_type(self, type):
71         if type in self.inflections:
72             return type
73         if type in self.reduced_types:
74             return self.reduced_type(self.reduced_types[type])
75         raise NotImplementedError
76
77     def inflect(self, type):
78         return self.radical + self.inflection(type) + self.suffix
79
80     def inflection(self, type):
81         type = self.reduced_type(type)
82         assert type in self.inflections
83         return self.inflections[type]
84
85     def __str__(self):
86         return self.radical + self.suffix
87
88
89 class StateGetter(Visitor):
90     '''Type visitor that is able to extract the state via one of the glGet*
91     functions.
92
93     It will declare any temporary variable
94     '''
95
96     def __init__(self, radical, inflections, suffix=''):
97         self.inflector = GetInflector(radical, inflections)
98         self.suffix = suffix
99
100     def iter(self):
101         for function, type, count, name in parameters:
102             inflection = self.inflector.radical + self.suffix
103             if inflection not in function.split(','):
104                 continue
105             if type is X:
106                 continue
107             yield type, count, name
108
109     def __call__(self, *args):
110         pname = args[-1]
111
112         for type, count, name in self.iter():
113             if name == pname:
114                 if count != 1:
115                     type = Array(type, str(count))
116
117                 return type, self.visit(type, args)
118
119         raise NotImplementedError
120
121     def temp_name(self, args):
122         '''Return the name of a temporary variable to hold the state.'''
123         pname = args[-1]
124
125         return pname[3:].lower()
126
127     def visitConst(self, const, args):
128         return self.visit(const.type, args)
129
130     def visitScalar(self, type, args):
131         temp_name = self.temp_name(args)
132         elem_type = self.inflector.reduced_type(type)
133         inflection = self.inflector.inflect(type)
134         if inflection.endswith('v'):
135             print '    %s %s = 0;' % (elem_type, temp_name)
136             print '    %s(%s, &%s);' % (inflection + self.suffix, ', '.join(args), temp_name)
137         else:
138             print '    %s %s = %s(%s);' % (elem_type, temp_name, inflection + self.suffix, ', '.join(args))
139         return temp_name
140
141     def visitString(self, string, args):
142         temp_name = self.temp_name(args)
143         inflection = self.inflector.inflect(string)
144         assert not inflection.endswith('v')
145         print '    %s %s = (%s)%s(%s);' % (string, temp_name, string, inflection + self.suffix, ', '.join(args))
146         return temp_name
147
148     def visitAlias(self, alias, args):
149         return self.visitScalar(alias, args)
150
151     def visitEnum(self, enum, args):
152         return self.visitScalar(enum, args)
153
154     def visitBitmask(self, bitmask, args):
155         return self.visit(GLint, args)
156
157     def visitArray(self, array, args):
158         temp_name = self.temp_name(args)
159         if array.length == '1':
160             return self.visit(array.type)
161         elem_type = self.inflector.reduced_type(array.type)
162         inflection = self.inflector.inflect(array.type)
163         assert inflection.endswith('v')
164         array_length = array.length
165         if array_length.isdigit():
166             # Static integer length
167             print '    %s %s[%s + 1];' % (elem_type, temp_name, array_length)
168         else:
169             # Put the length in a variable to avoid recomputing it every time
170             print '    size_t _%s_length = %s;' % (temp_name, array_length)
171             array_length = '_%s_length' % temp_name
172             # Allocate a dynamic sized array
173             print '    %s *%s = _allocator.alloc<%s>(%s + 1);' % (elem_type, temp_name, elem_type, array_length)
174         print '    memset(%s, 0, %s * sizeof *%s);' % (temp_name, array_length, temp_name)
175         print '    %s[%s] = (%s)0xdeadc0de;' % (temp_name, array_length, elem_type)
176         print '    if (%s) {' % array_length
177         print '        %s(%s, %s);' % (inflection + self.suffix, ', '.join(args), temp_name)
178         print '    }'
179         # Simple buffer overflow detection
180         print '    assert(%s[%s] == (%s)0xdeadc0de);' % (temp_name, array_length, elem_type)
181         return temp_name
182
183     def visitOpaque(self, pointer, args):
184         temp_name = self.temp_name(args)
185         inflection = self.inflector.inflect(pointer)
186         assert inflection.endswith('v')
187         print '    GLvoid *%s;' % temp_name
188         print '    %s(%s, &%s);' % (inflection + self.suffix, ', '.join(args), temp_name)
189         return temp_name
190
191
192 glGet = StateGetter('glGet', {
193     B: 'Booleanv',
194     I: 'Integerv',
195     F: 'Floatv',
196     D: 'Doublev',
197     S: 'String',
198     P: 'Pointerv',
199 })
200
201 glGetMaterial = StateGetter('glGetMaterial', {I: 'iv', F: 'fv'})
202 glGetLight = StateGetter('glGetLight', {I: 'iv', F: 'fv'})
203 glGetVertexAttrib = StateGetter('glGetVertexAttrib', {I: 'iv', F: 'fv', D: 'dv', P: 'Pointerv'})
204 glGetTexParameter = StateGetter('glGetTexParameter', {I: 'iv', F: 'fv'})
205 glGetTexEnv = StateGetter('glGetTexEnv', {I: 'iv', F: 'fv'})
206 glGetTexLevelParameter = StateGetter('glGetTexLevelParameter', {I: 'iv', F: 'fv'})
207 glGetShader = StateGetter('glGetShaderiv', {I: 'iv'})
208 glGetProgram = StateGetter('glGetProgram', {I: 'iv'})
209 glGetProgramARB = StateGetter('glGetProgram', {I: 'iv', F: 'fv', S: 'Stringv'}, 'ARB')
210 glGetFramebufferAttachmentParameter = StateGetter('glGetFramebufferAttachmentParameter', {I: 'iv'})
211 glGetSamplerParameter = StateGetter('glGetSamplerParameter', {I: 'iv', F: 'fv'})
212
213
214 class JsonWriter(Visitor):
215     '''Type visitor that will dump a value of the specified type through the
216     JSON writer.
217     
218     It expects a previously declared JSONWriter instance named "json".'''
219
220     def visitLiteral(self, literal, instance):
221         if literal.kind == 'Bool':
222             print '    json.writeBool(%s);' % instance
223         elif literal.kind in ('SInt', 'Uint'):
224             print '    json.writeInt(%s);' % instance
225         elif literal.kind in ('Float', 'Double'):
226             print '    json.writeFloat(%s);' % instance
227         else:
228             raise NotImplementedError
229
230     def visitString(self, string, instance):
231         assert string.length is None
232         print '    json.writeString((const char *)%s);' % instance
233
234     def visitEnum(self, enum, instance):
235         if enum is GLboolean:
236             print '    dumpBoolean(json, %s);' % instance
237         elif enum is GLenum:
238             print '    dumpEnum(json, %s);' % instance
239         else:
240             assert False
241             print '    json.writeInt(%s);' % instance
242
243     def visitBitmask(self, bitmask, instance):
244         raise NotImplementedError
245
246     def visitAlias(self, alias, instance):
247         self.visit(alias.type, instance)
248
249     def visitOpaque(self, opaque, instance):
250         print '    json.writeInt((size_t)%s);' % instance
251
252     __index = 0
253
254     def visitArray(self, array, instance):
255         index = '_i%u' % JsonWriter.__index
256         JsonWriter.__index += 1
257         print '    json.beginArray();'
258         print '    for (unsigned %s = 0; %s < %s; ++%s) {' % (index, index, array.length, index)
259         self.visit(array.type, '%s[%s]' % (instance, index))
260         print '    }'
261         print '    json.endArray();'
262
263
264
265 class StateDumper:
266     '''Class to generate code to dump all GL state in JSON format via
267     stdout.'''
268
269     def __init__(self):
270         pass
271
272     def dump(self):
273         print '#include <assert.h>'
274         print '#include <string.h>'
275         print
276         print '#include "json.hpp"'
277         print '#include "scoped_allocator.hpp"'
278         print '#include "glproc.hpp"'
279         print '#include "glsize.hpp"'
280         print '#include "glstate.hpp"'
281         print '#include "glstate_internal.hpp"'
282         print
283         print 'namespace glstate {'
284         print
285
286         print 'static void'
287         print 'flushErrors(void) {'
288         print '    while (glGetError() != GL_NO_ERROR) {}'
289         print '}'
290         print
291
292         print 'void'
293         print 'dumpBoolean(JSONWriter &json, GLboolean value)'
294         print '{'
295         print '    switch (value) {'
296         print '    case GL_FALSE:'
297         print '        json.writeString("GL_FALSE");'
298         print '        break;'
299         print '    case GL_TRUE:'
300         print '        json.writeString("GL_TRUE");'
301         print '        break;'
302         print '    default:'
303         print '        json.writeInt(static_cast<GLint>(value));'
304         print '        break;'
305         print '    }'
306         print '}'
307         print
308
309         print 'const char *'
310         print 'enumToString(GLenum pname)'
311         print '{'
312         print '    switch (pname) {'
313         for name in GLenum.values:
314             print '    case %s:' % name
315             print '        return "%s";' % name
316         print '    default:'
317         print '        return NULL;'
318         print '    }'
319         print '}'
320         print
321
322         print 'void'
323         print 'dumpEnum(JSONWriter &json, GLenum pname)'
324         print '{'
325         print '    const char *s = enumToString(pname);'
326         print '    if (s) {'
327         print '        json.writeString(s);'
328         print '    } else {'
329         print '        json.writeInt(pname);'
330         print '    }'
331         print '}'
332         print
333
334         print 'static void'
335         print 'dumpTextureTargetParameters(JSONWriter &json, Context &context, GLenum target, GLenum binding_param)'
336         print '{'
337         print '    GLboolean enabled = GL_FALSE;'
338         print '    GLint binding = 0;'
339         print '    glGetBooleanv(target, &enabled);'
340         print '    json.beginMember(enumToString(target));'
341         print '    dumpBoolean(json, enabled);'
342         print '    json.endMember();'
343         print '    glGetIntegerv(binding_param, &binding);'
344         print '    json.writeIntMember(enumToString(binding_param), binding);'
345         print '    if (enabled || binding) {'
346         print '        json.beginMember(enumToString(target));'
347         print '        json.beginObject();'
348         print '        dumpObjectLabel(json, context, GL_TEXTURE, binding, "GL_TEXTURE_LABEL");'
349         self.dump_atoms(glGetTexParameter, 'target')
350         print '        if (!context.ES) {'
351         print '            GLenum levelTarget;'
352         print '            if (target == GL_TEXTURE_CUBE_MAP ||'
353         print '                target == GL_TEXTURE_CUBE_MAP_ARRAY) {'
354         print '                // Must pick a face'
355         print '                levelTarget = GL_TEXTURE_CUBE_MAP_POSITIVE_X;'
356         print '            } else {'
357         print '                levelTarget = target;'
358         print '            }'
359         self.dump_atoms(glGetTexLevelParameter, 'levelTarget', '0')
360         print '        }'
361         print '        json.endObject();'
362         print '        json.endMember(); // target'
363         print '    }'
364         print '}'
365         print
366
367         print 'static void'
368         print 'dumpFramebufferAttachementParameters(JSONWriter &json, Context &context, GLenum target, GLenum attachment)'
369         print '{'
370         self.dump_attachment_parameters('target', 'attachment')
371         print '}'
372         print
373
374         print 'void dumpParameters(JSONWriter &json, Context &context)'
375         print '{'
376         print '    ScopedAllocator _allocator;'
377         print '    (void)_allocator;'
378         print
379         print '    json.beginMember("parameters");'
380         print '    json.beginObject();'
381         
382         self.dump_atoms(glGet)
383         
384         self.dump_material_params()
385         self.dump_light_params()
386         self.dump_vertex_attribs()
387         self.dump_program_params()
388         self.dump_texture_parameters()
389         self.dump_framebuffer_parameters()
390         self.dump_labels()
391
392         print '    json.endObject();'
393         print '    json.endMember(); // parameters'
394         print '}'
395         print
396         
397         print '} /*namespace glstate */'
398
399     def dump_material_params(self):
400         print '    if (!context.ES) {'
401         for face in ['GL_FRONT', 'GL_BACK']:
402             print '    json.beginMember("%s");' % face
403             print '    json.beginObject();'
404             self.dump_atoms(glGetMaterial, face)
405             print '    json.endObject();'
406         print '    }'
407         print
408
409     def dump_light_params(self):
410         print '    GLint max_lights = 0;'
411         print '    _glGetIntegerv(GL_MAX_LIGHTS, &max_lights);'
412         print '    for (GLint index = 0; index < max_lights; ++index) {'
413         print '        GLenum light = GL_LIGHT0 + index;'
414         print '        if (glIsEnabled(light)) {'
415         print '            char name[32];'
416         print '            snprintf(name, sizeof name, "GL_LIGHT%i", index);'
417         print '            json.beginMember(name);'
418         print '            json.beginObject();'
419         self.dump_atoms(glGetLight, '    GL_LIGHT0 + index')
420         print '            json.endObject();'
421         print '            json.endMember(); // GL_LIGHTi'
422         print '        }'
423         print '    }'
424         print
425
426     def dump_sampler_params(self):
427         print '    // GL_SAMPLER_BINDING'
428         print '    if (context.ARB_sampler_objects) {'
429         print '        GLint sampler_binding = 0;'
430         print '        glGetIntegerv(GL_SAMPLER_BINDING, &sampler_binding);'
431         print '        json.beginMember("GL_SAMPLER_BINDING");'
432         print '        json.writeInt(sampler_binding);'
433         print '        json.endMember();'
434         print '        if (sampler_binding) {'
435         print '            json.beginMember("GL_SAMPLER");'
436         print '            json.beginObject();'
437         print '            dumpObjectLabel(json, context, GL_SAMPLER, sampler_binding, "GL_SAMPLER_LABEL");'
438         for _, _, name in glGetSamplerParameter.iter():
439             self.dump_atom(glGetSamplerParameter, 'sampler_binding', name)
440         print '           json.endObject();'
441         print '           json.endMember(); // GL_SAMPLER'
442         print '       }'
443         print '    }'
444
445     def texenv_param_target(self, name):
446         if name == 'GL_TEXTURE_LOD_BIAS':
447            return 'GL_TEXTURE_FILTER_CONTROL'
448         elif name == 'GL_COORD_REPLACE':
449            return 'GL_POINT_SPRITE'
450         else:
451            return 'GL_TEXTURE_ENV'
452
453     def dump_texenv_params(self):
454         for target in ['GL_TEXTURE_ENV', 'GL_TEXTURE_FILTER_CONTROL', 'GL_POINT_SPRITE']:
455             print '    if (!context.ES) {'
456             print '        json.beginMember("%s");' % target
457             print '        json.beginObject();'
458             for _, _, name in glGetTexEnv.iter():
459                 if self.texenv_param_target(name) == target:
460                     self.dump_atom(glGetTexEnv, target, name) 
461             print '        json.endObject();'
462             print '    }'
463
464     def dump_vertex_attribs(self):
465         print '    GLint max_vertex_attribs = 0;'
466         print '    _glGetIntegerv(GL_MAX_VERTEX_ATTRIBS, &max_vertex_attribs);'
467         print '    for (GLint index = 0; index < max_vertex_attribs; ++index) {'
468         print '        char name[32];'
469         print '        snprintf(name, sizeof name, "GL_VERTEX_ATTRIB_ARRAY%i", index);'
470         print '        json.beginMember(name);'
471         print '        json.beginObject();'
472         self.dump_atoms(glGetVertexAttrib, 'index')
473         
474         # Dump vertex attrib buffer label
475         print '        GLint buffer_binding = 0;'
476         print '        glGetVertexAttribiv(index, GL_VERTEX_ATTRIB_ARRAY_BUFFER_BINDING, &buffer_binding);'
477         print '        dumpObjectLabel(json, context, GL_BUFFER, buffer_binding, "GL_VERTEX_ATTRIB_ARRAY_BUFFER_LABEL");'
478
479         print '        json.endObject();'
480         print '        json.endMember(); // GL_VERTEX_ATTRIB_ARRAYi'
481         print '    }'
482         print
483
484     object_bindings = [
485         ('GL_BUFFER', 'GL_ARRAY_BUFFER_BINDING'),
486         ('GL_BUFFER', 'GL_COLOR_ARRAY_BUFFER_BINDING'),
487         ('GL_BUFFER', 'GL_EDGE_FLAG_ARRAY_BUFFER_BINDING'),
488         ('GL_BUFFER', 'GL_ELEMENT_ARRAY_BUFFER_BINDING'),
489         ('GL_BUFFER', 'GL_FOG_COORDINATE_ARRAY_BUFFER_BINDING'),
490         ('GL_BUFFER', 'GL_INDEX_ARRAY_BUFFER_BINDING'),
491         ('GL_BUFFER', 'GL_NORMAL_ARRAY_BUFFER_BINDING'),
492         ('GL_BUFFER', 'GL_SECONDARY_COLOR_ARRAY_BUFFER_BINDING'),
493         ('GL_BUFFER', 'GL_TEXTURE_COORD_ARRAY_BUFFER_BINDING'),
494         ('GL_BUFFER', 'GL_TRANSFORM_FEEDBACK_BUFFER_BINDING'),
495         ('GL_BUFFER', 'GL_VERTEX_ARRAY_BUFFER_BINDING'),
496         ('GL_BUFFER', 'GL_WEIGHT_ARRAY_BUFFER_BINDING'),
497         ('GL_FRAMEBUFFER', 'GL_DRAW_FRAMEBUFFER_BINDING'),
498         ('GL_FRAMEBUFFER', 'GL_READ_FRAMEBUFFER_BINDING'),
499         ('GL_PROGRAM', 'GL_CURRENT_PROGRAM'),
500         ('GL_PROGRAM_PIPELINE', 'GL_PROGRAM_PIPELINE_BINDING'),
501         ('GL_RENDERBUFFER', 'GL_RENDERBUFFER_BINDING'),
502         ('GL_TRANSFORM_FEEDBACK', 'GL_TRANSFORM_FEEDBACK_BINDING'),
503         ('GL_VERTEX_ARRAY', 'GL_VERTEX_ARRAY_BINDING'),
504     ]
505
506     def dump_labels(self):
507         for object_type, object_binding in self.object_bindings:
508             member_name = object_binding.replace('BINDING', 'LABEL')
509             if member_name == object_binding:
510                 member_name += '_LABEL'
511             print '    {'
512             print '        GLint binding = 0;'
513             print '        glGetIntegerv(%s, &binding);' % object_binding
514             print '        dumpObjectLabel(json, context, %s, binding, "%s");' % (object_type, member_name)
515             print '    }'
516
517     program_targets = [
518         'GL_FRAGMENT_PROGRAM_ARB',
519         'GL_VERTEX_PROGRAM_ARB',
520     ]
521
522     def dump_program_params(self):
523         for target in self.program_targets:
524             print '    if (glIsEnabled(%s)) {' % target
525             print '        json.beginMember("%s");' % target
526             print '        json.beginObject();'
527             self.dump_atoms(glGetProgramARB, target)
528             print '        json.endObject();'
529             print '    }'
530
531     def dump_texture_parameters(self):
532         print '    {'
533         print '        GLint active_texture = GL_TEXTURE0;'
534         print '        glGetIntegerv(GL_ACTIVE_TEXTURE, &active_texture);'
535         print '        GLint max_texture_coords = 0;'
536         print '        glGetIntegerv(GL_MAX_TEXTURE_COORDS, &max_texture_coords);'
537         print '        GLint max_combined_texture_image_units = 0;'
538         print '        glGetIntegerv(GL_MAX_COMBINED_TEXTURE_IMAGE_UNITS, &max_combined_texture_image_units);'
539         print '        GLint max_units = std::max(std::max(max_combined_texture_image_units, max_texture_coords), 2);'
540         print '        for (GLint unit = 0; unit < max_units; ++unit) {'
541         print '            char name[32];'
542         print '            snprintf(name, sizeof name, "GL_TEXTURE%i", unit);'
543         print '            json.beginMember(name);'
544         print '            glActiveTexture(GL_TEXTURE0 + unit);'
545         print '            json.beginObject();'
546         print
547         for target, binding in texture_targets:
548             print '            dumpTextureTargetParameters(json, context, %s, %s);' % (target, binding)
549         print '            if (unit < max_texture_coords) {'
550         self.dump_sampler_params()
551         self.dump_texenv_params()
552         print '            }'
553         print '            json.endObject();'
554         print '            json.endMember(); // GL_TEXTUREi'
555         print '        }'
556         print '        glActiveTexture(active_texture);'
557         print '    }'
558         print
559
560     def dump_framebuffer_parameters(self):
561         print '    {'
562         print '        GLint max_color_attachments = 0;'
563         print '        glGetIntegerv(GL_MAX_COLOR_ATTACHMENTS, &max_color_attachments);'
564         print '        GLint framebuffer;'
565         for target, binding in framebuffer_targets:
566             print '            // %s' % target
567             print '            framebuffer = 0;'
568             print '            glGetIntegerv(%s, &framebuffer);' % binding
569             print '            if (framebuffer) {'
570             print '                json.beginMember("%s");' % target
571             print '                json.beginObject();'
572             print '                dumpObjectLabel(json, context, GL_FRAMEBUFFER, framebuffer, "GL_FRAMEBUFFER_LABEL");'
573             print '                for (GLint i = 0; i < max_color_attachments; ++i) {'
574             print '                    GLint color_attachment = GL_COLOR_ATTACHMENT0 + i;'
575             print '                    dumpFramebufferAttachementParameters(json, context, %s, color_attachment);' % target
576             print '                }'
577             print '                dumpFramebufferAttachementParameters(json, context, %s, GL_DEPTH_ATTACHMENT);' % target
578             print '                dumpFramebufferAttachementParameters(json, context, %s, GL_STENCIL_ATTACHMENT);' % target
579             print '                json.endObject();'
580             print '                json.endMember(); // %s' % target
581             print '            }'
582             print
583         print '    }'
584         print
585
586     def dump_attachment_parameters(self, target, attachment):
587         print '            {'
588         print '                GLint object_type = GL_NONE;'
589         print '                glGetFramebufferAttachmentParameteriv(%s, %s, GL_FRAMEBUFFER_ATTACHMENT_OBJECT_TYPE, &object_type);' % (target, attachment)
590         print '                if (object_type != GL_NONE) {'
591         print '                    json.beginMember(enumToString(%s));' % attachment
592         print '                    json.beginObject();'
593         self.dump_atoms(glGetFramebufferAttachmentParameter, target, attachment)
594         print '                    GLint object_name = 0;'
595         print '                    glGetFramebufferAttachmentParameteriv(%s, %s, GL_FRAMEBUFFER_ATTACHMENT_OBJECT_NAME, &object_name);' % (target, attachment)
596         print '                    dumpObjectLabel(json, context, object_type, object_name, "GL_FRAMEBUFFER_ATTACHMENT_OBJECT_LABEL");'
597         print '                    json.endObject();'
598         print '                    json.endMember(); // GL_x_ATTACHMENT'
599         print '                }'
600         print '            }'
601
602     def dump_atoms(self, getter, *args):
603         for _, _, name in getter.iter():
604             self.dump_atom(getter, *(args + (name,)))
605
606     def dump_atom(self, getter, *args):
607         name = args[-1]
608
609         print '        // %s' % name
610         print '        {'
611         print '            flushErrors();'
612         type, value = getter(*args)
613         print '            if (glGetError() != GL_NO_ERROR) {'
614         #print '                std::cerr << "warning: %s(%s) failed\\n";' % (inflection, name)
615         print '                flushErrors();'
616         print '            } else {'
617         print '                json.beginMember("%s");' % name
618         JsonWriter().visit(type, value)
619         print '                json.endMember();'
620         print '            }'
621         print '        }'
622         print
623
624
625 if __name__ == '__main__':
626     StateDumper().dump()