Merge remote branch 'main/master' into radeon-rewrite
[mesa.git] / src / mesa / glapi / gl_enums.py
1 #!/usr/bin/python2
2 # -*- Mode: Python; py-indent-offset: 8 -*-
3
4 # (C) Copyright Zack Rusin 2005
5 # All Rights Reserved.
6 #
7 # Permission is hereby granted, free of charge, to any person obtaining a
8 # copy of this software and associated documentation files (the "Software"),
9 # to deal in the Software without restriction, including without limitation
10 # on the rights to use, copy, modify, merge, publish, distribute, sub
11 # license, and/or sell copies of the Software, and to permit persons to whom
12 # the Software is furnished to do so, subject to the following conditions:
13 #
14 # The above copyright notice and this permission notice (including the next
15 # paragraph) shall be included in all copies or substantial portions of the
16 # Software.
17 #
18 # THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
19 # IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
20 # FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
21 # IBM AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
22 # LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
23 # FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
24 # IN THE SOFTWARE.
25 #
26 # Authors:
27 # Zack Rusin <zack@kde.org>
28
29 import license
30 import gl_XML
31 import sys, getopt
32
33 class PrintGlEnums(gl_XML.gl_print_base):
34
35 def __init__(self):
36 gl_XML.gl_print_base.__init__(self)
37
38 self.name = "gl_enums.py (from Mesa)"
39 self.license = license.bsd_license_template % ( \
40 """Copyright (C) 1999-2005 Brian Paul All Rights Reserved.""", "BRIAN PAUL")
41 self.enum_table = {}
42
43
44 def printRealHeader(self):
45 print '#include "glheader.h"'
46 print '#include "mfeatures.h"'
47 print '#include "enums.h"'
48 print '#include "imports.h"'
49 print ''
50 print 'typedef struct {'
51 print ' size_t offset;'
52 print ' int n;'
53 print '} enum_elt;'
54 print ''
55 return
56
57 def print_code(self):
58 print """
59 typedef int (*cfunc)(const void *, const void *);
60
61 /**
62 * Compare a key name to an element in the \c all_enums array.
63 *
64 * \c bsearch always passes the key as the first parameter and the pointer
65 * to the array element as the second parameter. We can elimiate some
66 * extra work by taking advantage of that fact.
67 *
68 * \param a Pointer to the desired enum name.
69 * \param b Pointer to an element of the \c all_enums array.
70 */
71 static int compar_name( const char *a, const enum_elt *b )
72 {
73 return _mesa_strcmp( a, & enum_string_table[ b->offset ] );
74 }
75
76 /**
77 * Compare a key enum value to an element in the \c all_enums array.
78 *
79 * \c bsearch always passes the key as the first parameter and the pointer
80 * to the array element as the second parameter. We can elimiate some
81 * extra work by taking advantage of that fact.
82 *
83 * \param a Pointer to the desired enum name.
84 * \param b Pointer to an index into the \c all_enums array.
85 */
86 static int compar_nr( const int *a, const unsigned *b )
87 {
88 return a[0] - all_enums[*b].n;
89 }
90
91
92 static char token_tmp[20];
93
94 const char *_mesa_lookup_enum_by_nr( int nr )
95 {
96 unsigned * i;
97
98 i = (unsigned *) _mesa_bsearch(& nr, reduced_enums,
99 Elements(reduced_enums),
100 sizeof(reduced_enums[0]),
101 (cfunc) compar_nr);
102
103 if ( i != NULL ) {
104 return & enum_string_table[ all_enums[ *i ].offset ];
105 }
106 else {
107 /* this is not re-entrant safe, no big deal here */
108 _mesa_sprintf(token_tmp, "0x%x", nr);
109 return token_tmp;
110 }
111 }
112
113 int _mesa_lookup_enum_by_name( const char *symbol )
114 {
115 enum_elt * f = NULL;
116
117 if ( symbol != NULL ) {
118 f = (enum_elt *) _mesa_bsearch(symbol, all_enums,
119 Elements(all_enums),
120 sizeof( enum_elt ),
121 (cfunc) compar_name);
122 }
123
124 return (f != NULL) ? f->n : -1;
125 }
126
127 """
128 return
129
130
131 def printBody(self, api):
132 self.process_enums( api )
133
134 keys = self.enum_table.keys()
135 keys.sort()
136
137 name_table = []
138 enum_table = {}
139
140 for enum in keys:
141 low_pri = 9
142 for [name, pri] in self.enum_table[ enum ]:
143 name_table.append( [name, enum] )
144
145 if pri < low_pri:
146 low_pri = pri
147 enum_table[enum] = name
148
149
150 name_table.sort()
151
152 string_offsets = {}
153 i = 0;
154 print 'LONGSTRING static const char enum_string_table[] = '
155 for [name, enum] in name_table:
156 print ' "%s\\0"' % (name)
157 string_offsets[ name ] = i
158 i += len(name) + 1
159
160 print ' ;'
161 print ''
162
163
164 print 'static const enum_elt all_enums[%u] =' % (len(name_table))
165 print '{'
166 for [name, enum] in name_table:
167 print ' { %5u, 0x%08X }, /* %s */' % (string_offsets[name], enum, name)
168 print '};'
169 print ''
170
171 print 'static const unsigned reduced_enums[%u] =' % (len(keys))
172 print '{'
173 for enum in keys:
174 name = enum_table[ enum ]
175 if [name, enum] not in name_table:
176 print ' /* Error! %s, 0x%04x */ 0,' % (name, enum)
177 else:
178 i = name_table.index( [name, enum] )
179
180 print ' %4u, /* %s */' % (i, name)
181 print '};'
182
183
184 self.print_code()
185 return
186
187
188 def process_enums(self, api):
189 self.enum_table = {}
190
191 for obj in api.enumIterateByName():
192 if obj.value not in self.enum_table:
193 self.enum_table[ obj.value ] = []
194
195
196 name = "GL_" + obj.name
197 priority = obj.priority()
198 self.enum_table[ obj.value ].append( [name, priority] )
199
200
201 def show_usage():
202 print "Usage: %s [-f input_file_name]" % sys.argv[0]
203 sys.exit(1)
204
205 if __name__ == '__main__':
206 file_name = "gl_API.xml"
207
208 try:
209 (args, trail) = getopt.getopt(sys.argv[1:], "f:")
210 except Exception,e:
211 show_usage()
212
213 for (arg,val) in args:
214 if arg == "-f":
215 file_name = val
216
217 api = gl_XML.parse_GL_API( file_name )
218
219 printer = PrintGlEnums()
220 printer.Print( api )