blob: 81c94a5223000031265aa748476d0a824a6d809a [file] [log] [blame]
Ian Romanick74764062004-12-03 20:31:59 +00001#!/usr/bin/python2
2
Ian Romanick5f1f2292005-01-07 02:39:09 +00003# (C) Copyright IBM Corporation 2004, 2005
Ian Romanick74764062004-12-03 20:31:59 +00004# All Rights Reserved.
5#
6# Permission is hereby granted, free of charge, to any person obtaining a
7# copy of this software and associated documentation files (the "Software"),
8# to deal in the Software without restriction, including without limitation
9# on the rights to use, copy, modify, merge, publish, distribute, sub
10# license, and/or sell copies of the Software, and to permit persons to whom
11# the Software is furnished to do so, subject to the following conditions:
12#
13# The above copyright notice and this permission notice (including the next
14# paragraph) shall be included in all copies or substantial portions of the
15# Software.
16#
17# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
18# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19# FITNESS FOR A PARTICULAR PURPOSE AND NON-INFRINGEMENT. IN NO EVENT SHALL
20# IBM AND/OR ITS SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
21# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
22# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
23# IN THE SOFTWARE.
24#
25# Authors:
26# Ian Romanick <idr@us.ibm.com>
27
28from xml.sax import saxutils
29from xml.sax import make_parser
30from xml.sax.handler import feature_namespaces
31
32import gl_XML
33import license
Ian Romanick3fec8c22005-02-02 00:54:45 +000034import sys, getopt, string
Ian Romanick74764062004-12-03 20:31:59 +000035
36
Ian Romanick74764062004-12-03 20:31:59 +000037class glXItemFactory(gl_XML.glItemFactory):
38 """Factory to create GLX protocol oriented objects derived from glItem."""
39
40 def create(self, context, name, attrs):
41 if name == "function":
42 return glXFunction(context, name, attrs)
43 elif name == "enum":
44 return glXEnum(context, name, attrs)
45 elif name == "param":
46 return glXParameter(context, name, attrs)
47 else:
48 return gl_XML.glItemFactory.create(self, context, name, attrs)
49
50class glXEnumFunction:
Ian Romanickba09c192005-02-01 00:13:04 +000051 def __init__(self, name, context):
Ian Romanick74764062004-12-03 20:31:59 +000052 self.name = name
Ian Romanickba09c192005-02-01 00:13:04 +000053 self.context = context
Ian Romanick5aa6dc322005-01-27 01:08:48 +000054 self.mode = 0
55 self.sig = None
56
Ian Romanick74764062004-12-03 20:31:59 +000057 # "enums" is a set of lists. The element in the set is the
58 # value of the enum. The list is the list of names for that
59 # value. For example, [0x8126] = {"POINT_SIZE_MIN",
60 # "POINT_SIZE_MIN_ARB", "POINT_SIZE_MIN_EXT",
61 # "POINT_SIZE_MIN_SGIS"}.
62
63 self.enums = {}
64
65 # "count" is indexed by count values. Each element of count
66 # is a list of index to "enums" that have that number of
67 # associated data elements. For example, [4] =
68 # {GL_AMBIENT, GL_DIFFUSE, GL_SPECULAR, GL_EMISSION,
69 # GL_AMBIENT_AND_DIFFUSE} (the enum names are used here,
70 # but the actual hexadecimal values would be in the array).
71
72 self.count = {}
73
74
75 def append(self, count, value, name):
76 if self.enums.has_key( value ):
77 self.enums[value].append(name)
78 else:
79 if not self.count.has_key(count):
80 self.count[count] = []
81
82 self.enums[value] = []
83 self.enums[value].append(name)
84 self.count[count].append(value)
85
86
87 def signature( self ):
Ian Romanick5aa6dc322005-01-27 01:08:48 +000088 if self.sig == None:
89 self.sig = ""
90 for i in self.count:
Ian Romanick82e22f52005-01-27 19:39:16 +000091 self.count[i].sort()
Ian Romanick5aa6dc322005-01-27 01:08:48 +000092 for e in self.count[i]:
93 self.sig += "%04x,%u," % (e, i)
Ian Romanick74764062004-12-03 20:31:59 +000094
Ian Romanick5aa6dc322005-01-27 01:08:48 +000095 return self.sig
96
97
98 def set_mode( self, mode ):
99 """Mark an enum-function as a 'set' function."""
100
101 self.mode = mode
102
103
104 def is_set( self ):
105 return self.mode
Ian Romanick74764062004-12-03 20:31:59 +0000106
107
108 def PrintUsingTable(self):
109 """Emit the body of the __gl*_size function using a pair
110 of look-up tables and a mask. The mask is calculated such
111 that (e & mask) is unique for all the valid values of e for
112 this function. The result of (e & mask) is used as an index
113 into the first look-up table. If it matches e, then the
114 same entry of the second table is returned. Otherwise zero
115 is returned.
116
117 It seems like this should cause better code to be generated.
118 However, on x86 at least, the resulting .o file is about 20%
119 larger then the switch-statment version. I am leaving this
120 code in because the results may be different on other
121 platforms (e.g., PowerPC or x86-64)."""
122
123 return 0
124 count = 0
125 for a in self.enums:
126 count += 1
127
Ian Romanick80a939c2005-03-17 21:48:37 +0000128 if self.count.has_key(-1):
129 return 0
130
Ian Romanick74764062004-12-03 20:31:59 +0000131 # Determine if there is some mask M, such that M = (2^N) - 1,
132 # that will generate unique values for all of the enums.
133
134 mask = 0
135 for i in [1, 2, 3, 4, 5, 6, 7, 8]:
136 mask = (1 << i) - 1
137
138 fail = 0;
139 for a in self.enums:
140 for b in self.enums:
141 if a != b:
142 if (a & mask) == (b & mask):
143 fail = 1;
144
145 if not fail:
146 break;
147 else:
148 mask = 0
149
150 if (mask != 0) and (mask < (2 * count)):
151 masked_enums = {}
152 masked_count = {}
153
154 for i in range(0, mask + 1):
155 masked_enums[i] = "0";
156 masked_count[i] = 0;
157
158 for c in self.count:
159 for e in self.count[c]:
160 i = e & mask
161 masked_enums[i] = '0x%04x /* %s */' % (e, self.enums[e][0])
162 masked_count[i] = c
163
164
165 print ' static const GLushort a[%u] = {' % (mask + 1)
166 for e in masked_enums:
167 print ' %s, ' % (masked_enums[e])
168 print ' };'
169
170 print ' static const GLubyte b[%u] = {' % (mask + 1)
171 for c in masked_count:
172 print ' %u, ' % (masked_count[c])
173 print ' };'
174
175 print ' const unsigned idx = (e & 0x%02xU);' % (mask)
176 print ''
177 print ' return (e == a[idx]) ? (GLint) b[idx] : 0;'
178 return 1;
179 else:
180 return 0;
181
Ian Romanick80a939c2005-03-17 21:48:37 +0000182 def PrintUsingSwitch(self, name):
Ian Romanick74764062004-12-03 20:31:59 +0000183 """Emit the body of the __gl*_size function using a
184 switch-statement."""
185
186 print ' switch( e ) {'
187
188 for c in self.count:
189 for e in self.count[c]:
190 first = 1
191
192 # There may be multiple enums with the same
193 # value. This happens has extensions are
194 # promoted from vendor-specific or EXT to
195 # ARB and to the core. Emit the first one as
196 # a case label, and emit the others as
197 # commented-out case labels.
198
199 for j in self.enums[e]:
200 if first:
201 print ' case %s:' % (j)
202 first = 0
203 else:
204 print '/* case %s:*/' % (j)
205
Ian Romanick80a939c2005-03-17 21:48:37 +0000206 if c == -1:
207 print ' return __gl%s_variable_size( e );' % (name)
208 else:
209 print ' return %u;' % (c)
Ian Romanick74764062004-12-03 20:31:59 +0000210
211 print ' default: return 0;'
212 print ' }'
213
214
215 def Print(self, name):
216 print 'INTERNAL PURE FASTCALL GLint'
217 print '__gl%s_size( GLenum e )' % (name)
218 print '{'
219
220 if not self.PrintUsingTable():
Ian Romanick80a939c2005-03-17 21:48:37 +0000221 self.PrintUsingSwitch(name)
Ian Romanick74764062004-12-03 20:31:59 +0000222
223 print '}'
224 print ''
225
226
227
228class glXEnum(gl_XML.glEnum):
229 def __init__(self, context, name, attrs):
230 gl_XML.glEnum.__init__(self, context, name, attrs)
Ian Romanick0246b2a2005-01-24 20:59:32 +0000231
Ian Romanick74764062004-12-03 20:31:59 +0000232
Ian Romanick2510ba62005-04-18 19:16:07 +0000233 def startElementNS(self, name, qname, attrs):
234 [uri, true_name] = name
235 if true_name == "size":
Ian Romanick80a939c2005-03-17 21:48:37 +0000236 [temp_n, c, mode] = self.process_attributes(attrs)
Ian Romanick5ff2b942005-01-24 21:29:13 +0000237
Ian Romanick80a939c2005-03-17 21:48:37 +0000238 if temp_n == "Get":
239 names = ["GetIntegerv", "GetBooleanv", "GetFloatv", "GetDoublev" ]
240 else:
241 names = [ temp_n ]
Ian Romanick74764062004-12-03 20:31:59 +0000242
Ian Romanick80a939c2005-03-17 21:48:37 +0000243 for n in names:
244 if not self.context.glx_enum_functions.has_key( n ):
245 f = self.context.createEnumFunction( n )
246 f.set_mode( mode )
247 self.context.glx_enum_functions[ f.name ] = f
248
249 self.context.glx_enum_functions[ n ].append( c, self.value, self.name )
Ian Romanick74764062004-12-03 20:31:59 +0000250 else:
Ian Romanick2510ba62005-04-18 19:16:07 +0000251 gl_XML.glEnum.startElementNS(self, name, qname, attrs)
Ian Romanick74764062004-12-03 20:31:59 +0000252 return
253
254
255class glXParameter(gl_XML.glParameter):
256 def __init__(self, context, name, attrs):
257 self.order = 1;
258 gl_XML.glParameter.__init__(self, context, name, attrs);
259
260
Ian Romanick1d270842004-12-21 21:26:36 +0000261class glXParameterIterator:
262 """Class to iterate over a list of glXParameters.
263
264 Objects of this class are returned by the parameterIterator method of
265 the glXFunction class. They are used to iterate over the list of
266 parameters to the function."""
267
268 def __init__(self, data, skip_output, max_order):
269 self.data = data
270 self.index = 0
271 self.order = 0
272 self.skip_output = skip_output
273 self.max_order = max_order
274
275 def __iter__(self):
276 return self
277
278 def next(self):
279 if len( self.data ) == 0:
280 raise StopIteration
281
282 while 1:
283 if self.index == len( self.data ):
284 if self.order == self.max_order:
285 raise StopIteration
286 else:
287 self.order += 1
288 self.index = 0
289
290 i = self.index
291 self.index += 1
292
293 if self.data[i].order == self.order and not (self.data[i].is_output and self.skip_output):
294 return self.data[i]
295
296
Ian Romanick74764062004-12-03 20:31:59 +0000297class glXFunction(gl_XML.glFunction):
298 glx_rop = 0
299 glx_sop = 0
300 glx_vendorpriv = 0
301
302 # If this is set to true, it means that GLdouble parameters should be
303 # written to the GLX protocol packet in the order they appear in the
304 # prototype. This is different from the "classic" ordering. In the
305 # classic ordering GLdoubles are written to the protocol packet first,
306 # followed by non-doubles. NV_vertex_program was the first extension
307 # to break with this tradition.
308
309 glx_doubles_in_order = 0
310
311 vectorequiv = None
Ian Romanick74764062004-12-03 20:31:59 +0000312 can_be_large = 0
313
314 def __init__(self, context, name, attrs):
Ian Romanick2510ba62005-04-18 19:16:07 +0000315 self.vectorequiv = attrs.get((None, 'vectorequiv'), None)
Ian Romanick74764062004-12-03 20:31:59 +0000316 self.counter = None
317 self.output = None
318 self.can_be_large = 0
319 self.reply_always_array = 0
Ian Romanickd8634242005-02-09 03:11:23 +0000320 self.dimensions_in_reply = 0
321 self.img_reset = None
Ian Romanick74764062004-12-03 20:31:59 +0000322
Ian Romanickfdb05272005-01-28 17:30:25 +0000323 self.server_handcode = 0
324 self.client_handcode = 0
325 self.ignore = 0
326
Ian Romanick74764062004-12-03 20:31:59 +0000327 gl_XML.glFunction.__init__(self, context, name, attrs)
328 return
329
Ian Romanick1d270842004-12-21 21:26:36 +0000330
331 def parameterIterator(self, skip_output, max_order):
332 return glXParameterIterator(self.fn_parameters, skip_output, max_order)
333
334
Ian Romanick2510ba62005-04-18 19:16:07 +0000335 def startElementNS(self, name, qname, attrs):
Ian Romanick74764062004-12-03 20:31:59 +0000336 """Process elements within a function that are specific to GLX."""
337
Ian Romanick2510ba62005-04-18 19:16:07 +0000338 [uri, true_name] = name
339 if true_name == "glx":
340 self.glx_rop = int(attrs.get((None, 'rop'), "0"))
341 self.glx_sop = int(attrs.get((None, 'sop'), "0"))
342 self.glx_vendorpriv = int(attrs.get((None, 'vendorpriv'), "0"))
343 self.img_reset = attrs.get((None, 'img_reset'), None)
Ian Romanick74764062004-12-03 20:31:59 +0000344
Ian Romanickfdb05272005-01-28 17:30:25 +0000345 # The 'handcode' attribute can be one of 'true',
346 # 'false', 'client', or 'server'.
347
Ian Romanick2510ba62005-04-18 19:16:07 +0000348 handcode = attrs.get((None, 'handcode'), "false")
Ian Romanickfdb05272005-01-28 17:30:25 +0000349 if handcode == "false":
350 self.server_handcode = 0
351 self.client_handcode = 0
352 elif handcode == "true":
353 self.server_handcode = 1
354 self.client_handcode = 1
355 elif handcode == "client":
356 self.server_handcode = 0
357 self.client_handcode = 1
358 elif handcode == "server":
359 self.server_handcode = 1
360 self.client_handcode = 0
Ian Romanick74764062004-12-03 20:31:59 +0000361 else:
Ian Romanickfdb05272005-01-28 17:30:25 +0000362 raise RuntimeError('Invalid handcode mode "%s" in function "%s".' % (handcode, self.name))
363
Ian Romanick73b4c1b2005-04-14 23:00:34 +0000364 self.ignore = gl_XML.is_attr_true( attrs, 'ignore' )
365 self.can_be_large = gl_XML.is_attr_true( attrs, 'large' )
366 self.glx_doubles_in_order = gl_XML.is_attr_true( attrs, 'doubles_in_order' )
367 self.reply_always_array = gl_XML.is_attr_true( attrs, 'always_array' )
368 self.dimensions_in_reply = gl_XML.is_attr_true( attrs, 'dimensions_in_reply' )
Ian Romanick74764062004-12-03 20:31:59 +0000369 else:
Ian Romanick2510ba62005-04-18 19:16:07 +0000370 gl_XML.glFunction.startElementNS(self, name, qname, attrs)
Ian Romanick74764062004-12-03 20:31:59 +0000371
372
Ian Romanick2510ba62005-04-18 19:16:07 +0000373 def endElementNS(self, name, qname):
374 [uri, true_name] = name
375 if true_name == "function":
Ian Romanick7f958e92005-01-24 20:08:28 +0000376 # Mark any function that does not have GLX protocol
377 # defined as "ignore". This prevents bad things from
378 # happening when people add new functions to the GL
379 # API XML without adding any GLX section.
380 #
381 # This will also mark functions that don't have a
382 # dispatch offset at ignored.
383
Ian Romanickfdb05272005-01-28 17:30:25 +0000384 if (self.fn_offset == -1 and not self.fn_alias) or not (self.client_handcode or self.server_handcode or self.glx_rop or self.glx_sop or self.glx_vendorpriv or self.vectorequiv or self.fn_alias):
Ian Romanick7f958e92005-01-24 20:08:28 +0000385 #if not self.ignore:
386 # if self.fn_offset == -1:
387 # print '/* %s ignored becuase no offset assigned. */' % (self.name)
388 # else:
389 # print '/* %s ignored becuase no GLX opcode assigned. */' % (self.name)
390
391 self.ignore = 1
392
Ian Romanick2510ba62005-04-18 19:16:07 +0000393 return gl_XML.glFunction.endElementNS(self, name, qname)
Ian Romanick7f958e92005-01-24 20:08:28 +0000394
395
Ian Romanick74764062004-12-03 20:31:59 +0000396 def append(self, tag_name, p):
397 gl_XML.glFunction.append(self, tag_name, p)
398
399 if p.is_variable_length_array():
400 p.order = 2;
401 elif not self.glx_doubles_in_order and p.p_type.size == 8:
402 p.order = 0;
403
Ian Romanick74764062004-12-03 20:31:59 +0000404 if p.is_counter:
405 self.counter = p.name
406
407 if p.is_output:
408 self.output = p
409
410 return
411
Ian Romanick0246b2a2005-01-24 20:59:32 +0000412
Ian Romanick74764062004-12-03 20:31:59 +0000413 def variable_length_parameter(self):
Ian Romanick6af6a692005-03-17 20:56:13 +0000414 if len(self.variable_length_parameters):
415 return self.variable_length_parameters[0]
416
Ian Romanick74764062004-12-03 20:31:59 +0000417 return None
418
419
Ian Romanick54584df2005-01-28 18:20:43 +0000420 def output_parameter(self):
421 for param in self.fn_parameters:
422 if param.is_output:
423 return param
424
425 return None
426
427
Ian Romanick0246b2a2005-01-24 20:59:32 +0000428 def offset_of_first_parameter(self):
429 """Get the offset of the first parameter in the command.
430
431 Gets the offset of the first function parameter in the GLX
432 command packet. This byte offset is measured from the end
433 of the Render / RenderLarge header. The offset for all non-
434 pixel commends is zero. The offset for pixel commands depends
435 on the number of dimensions of the pixel data."""
Ian Romanick5f1f2292005-01-07 02:39:09 +0000436
Ian Romanickd8634242005-02-09 03:11:23 +0000437 if self.image and not self.image.is_output:
Ian Romanick5f1f2292005-01-07 02:39:09 +0000438 [dim, junk, junk, junk, junk] = self.dimensions()
Ian Romanick0246b2a2005-01-24 20:59:32 +0000439
Ian Romanick5f1f2292005-01-07 02:39:09 +0000440 # The base size is the size of the pixel pack info
441 # header used by images with the specified number
442 # of dimensions.
443
444 if dim <= 2:
Ian Romanick0246b2a2005-01-24 20:59:32 +0000445 return 20
Ian Romanick5f1f2292005-01-07 02:39:09 +0000446 elif dim <= 4:
Ian Romanick0246b2a2005-01-24 20:59:32 +0000447 return 36
Ian Romanick5f1f2292005-01-07 02:39:09 +0000448 else:
449 raise RuntimeError('Invalid number of dimensions %u for parameter "%s" in function "%s".' % (dim, self.image.name, self.name))
Ian Romanick0246b2a2005-01-24 20:59:32 +0000450 else:
451 return 0
Ian Romanick5f1f2292005-01-07 02:39:09 +0000452
Ian Romanick5f1f2292005-01-07 02:39:09 +0000453
Ian Romanick0246b2a2005-01-24 20:59:32 +0000454 def command_fixed_length(self):
455 """Return the length, in bytes as an integer, of the
456 fixed-size portion of the command."""
Ian Romanick5f1f2292005-01-07 02:39:09 +0000457
Ian Romanick0246b2a2005-01-24 20:59:32 +0000458 size = self.offset_of_first_parameter()
Ian Romanick5f1f2292005-01-07 02:39:09 +0000459
Ian Romanick0246b2a2005-01-24 20:59:32 +0000460 for p in gl_XML.glFunction.parameterIterator(self):
Ian Romanickd8634242005-02-09 03:11:23 +0000461 if not p.is_output and p.name != self.img_reset:
Ian Romanick0246b2a2005-01-24 20:59:32 +0000462 size += p.size()
463 if self.pad_after(p):
464 size += 4
465
Ian Romanickd8634242005-02-09 03:11:23 +0000466 if self.image and (self.image.img_null_flag or self.image.is_output):
Ian Romanick0246b2a2005-01-24 20:59:32 +0000467 size += 4
468
469 return size
470
471
472 def command_variable_length(self):
473 """Return the length, as a string, of the variable-sized
474 portion of the command."""
475
Ian Romanick74764062004-12-03 20:31:59 +0000476 size_string = ""
Ian Romanick1d270842004-12-21 21:26:36 +0000477 for p in gl_XML.glFunction.parameterIterator(self):
Ian Romanick0246b2a2005-01-24 20:59:32 +0000478 if (not p.is_output) and (p.size() == 0):
479 size_string = size_string + " + __GLX_PAD(%s)" % (p.size_string())
Ian Romanick74764062004-12-03 20:31:59 +0000480
Ian Romanick0246b2a2005-01-24 20:59:32 +0000481 return size_string
482
Ian Romanick74764062004-12-03 20:31:59 +0000483
484 def command_length(self):
Ian Romanick0246b2a2005-01-24 20:59:32 +0000485 size = self.command_fixed_length()
Ian Romanick74764062004-12-03 20:31:59 +0000486
487 if self.glx_rop != 0:
488 size += 4
489
490 size = ((size + 3) & ~3)
Ian Romanick0246b2a2005-01-24 20:59:32 +0000491 return "%u%s" % (size, self.command_variable_length())
Ian Romanick74764062004-12-03 20:31:59 +0000492
493
494 def opcode_real_value(self):
Ian Romanick1d270842004-12-21 21:26:36 +0000495 """Get the true numeric value of the GLX opcode
496
497 Behaves similarly to opcode_value, except for
498 X_GLXVendorPrivate and X_GLXVendorPrivateWithReply commands.
499 In these cases the value for the GLX opcode field (i.e.,
500 16 for X_GLXVendorPrivate or 17 for
501 X_GLXVendorPrivateWithReply) is returned. For other 'single'
502 commands, the opcode for the command (e.g., 101 for
503 X_GLsop_NewList) is returned."""
504
Ian Romanick74764062004-12-03 20:31:59 +0000505 if self.glx_vendorpriv != 0:
506 if self.needs_reply():
507 return 17
508 else:
509 return 16
510 else:
511 return self.opcode_value()
512
513 def opcode_value(self):
Ian Romanick1d270842004-12-21 21:26:36 +0000514 """Get the unique protocol opcode for the glXFunction"""
515
Ian Romanick74764062004-12-03 20:31:59 +0000516 if self.glx_rop != 0:
517 return self.glx_rop
518 elif self.glx_sop != 0:
519 return self.glx_sop
520 elif self.glx_vendorpriv != 0:
521 return self.glx_vendorpriv
522 else:
523 return -1
524
525 def opcode_rop_basename(self):
Ian Romanick1d270842004-12-21 21:26:36 +0000526 """Return either the name to be used for GLX protocol enum.
527
528 Returns either the name of the function or the name of the
529 name of the equivalent vector (e.g., glVertex3fv for
530 glVertex3f) function."""
531
Ian Romanick74764062004-12-03 20:31:59 +0000532 if self.vectorequiv == None:
533 return self.name
534 else:
535 return self.vectorequiv
536
537 def opcode_name(self):
Ian Romanick1d270842004-12-21 21:26:36 +0000538 """Get the unique protocol enum name for the glXFunction"""
539
Ian Romanick74764062004-12-03 20:31:59 +0000540 if self.glx_rop != 0:
541 return "X_GLrop_%s" % (self.opcode_rop_basename())
542 elif self.glx_sop != 0:
543 return "X_GLsop_%s" % (self.name)
544 elif self.glx_vendorpriv != 0:
545 return "X_GLvop_%s" % (self.name)
546 else:
Ian Romanick3fec8c22005-02-02 00:54:45 +0000547 raise RuntimeError('Function "%s" has no opcode.' % (self.name))
548
Ian Romanick74764062004-12-03 20:31:59 +0000549
550 def opcode_real_name(self):
Ian Romanick1d270842004-12-21 21:26:36 +0000551 """Get the true protocol enum name for the GLX opcode
552
553 Behaves similarly to opcode_name, except for
554 X_GLXVendorPrivate and X_GLXVendorPrivateWithReply commands.
555 In these cases the string 'X_GLXVendorPrivate' or
556 'X_GLXVendorPrivateWithReply' is returned. For other
557 single or render commands 'X_GLsop' or 'X_GLrop' plus the
558 name of the function returned."""
559
Ian Romanick74764062004-12-03 20:31:59 +0000560 if self.glx_vendorpriv != 0:
561 if self.needs_reply():
562 return "X_GLXVendorPrivateWithReply"
563 else:
564 return "X_GLXVendorPrivate"
565 else:
566 return self.opcode_name()
567
568
569 def return_string(self):
570 if self.fn_return_type != 'void':
571 return "return retval;"
572 else:
573 return "return;"
574
575
576 def needs_reply(self):
577 return self.fn_return_type != 'void' or self.output != None
578
579
Ian Romanick5f1f2292005-01-07 02:39:09 +0000580 def dimensions(self):
581 """Determine the dimensions of an image.
582
583 Returns a tuple representing the number of dimensions and the
584 string name of each of the dimensions of an image, If the
585 function is not a pixel function, the number of dimensions
586 will be zero."""
587
588 if not self.image:
589 return [0, "0", "0", "0", "0"]
590 else:
591 dim = 1
592 w = self.image.width
593
594 if self.image.height:
595 dim = 2
596 h = self.image.height
597 else:
598 h = "1"
599
600 if self.image.depth:
601 dim = 3
602 d = self.image.depth
603 else:
604 d = "1"
605
606 if self.image.extent:
607 dim = 4
608 e = self.image.extent
609 else:
610 e = "1"
611
612 return [dim, w, h, d, e]
613
614
615 def pad_after(self, p):
616 """Returns the name of the field inserted after the
617 specified field to pad out the command header."""
618
619 if self.image and self.image.img_pad_dimensions:
620 if not self.image.height:
621 if p.name == self.image.width:
622 return "height"
623 elif p.name == self.image.img_xoff:
624 return "yoffset"
625 elif not self.image.extent:
626 if p.name == self.image.depth:
627 # Should this be "size4d"?
628 return "extent"
629 elif p.name == self.image.img_zoff:
630 return "woffset"
631 return None
632
Ian Romanick3fec8c22005-02-02 00:54:45 +0000633
634class glXFunctionIterator(gl_XML.glFunctionIterator):
635 """Class to iterate over a list of glXFunctions"""
636
637 def __init__(self, context):
638 self.context = context
639 self.keys = context.functions.keys()
640 self.keys.sort()
641
642 for self.index in range(0, len(self.keys)):
643 if self.keys[ self.index ] >= 0: break
644
645 return
646
647
648 def next(self):
649 if self.index == len(self.keys):
650 raise StopIteration
651
652 f = self.context.functions[ self.keys[ self.index ] ]
653 self.index += 1
654
655 if f.ignore:
656 return self.next()
657 else:
658 return f
659
660
Ian Romanick74764062004-12-03 20:31:59 +0000661class GlxProto(gl_XML.FilterGLAPISpecBase):
662 name = "glX_proto_send.py (from Mesa)"
663
664 def __init__(self):
665 gl_XML.FilterGLAPISpecBase.__init__(self)
666 self.factory = glXItemFactory()
667 self.glx_enum_functions = {}
668
669
Ian Romanick2510ba62005-04-18 19:16:07 +0000670 def endElementNS(self, name, qname):
671 [uri, true_name] = name
672 if true_name == 'OpenGLAPI':
Ian Romanick74764062004-12-03 20:31:59 +0000673 # Once all the parsing is done, we have to go back and
674 # fix-up some cross references between different
675 # functions.
676
677 for k in self.functions:
678 f = self.functions[k]
679 if f.vectorequiv != None:
680 equiv = self.find_function(f.vectorequiv)
681 if equiv != None:
682 f.glx_doubles_in_order = equiv.glx_doubles_in_order
683 f.glx_rop = equiv.glx_rop
684 else:
685 raise RuntimeError("Could not find the vector equiv. function %s for %s!" % (f.name, f.vectorequiv))
686 else:
Ian Romanick2510ba62005-04-18 19:16:07 +0000687 gl_XML.FilterGLAPISpecBase.endElementNS(self, name, qname)
Ian Romanick74764062004-12-03 20:31:59 +0000688 return
Ian Romanickba09c192005-02-01 00:13:04 +0000689
690
691 def createEnumFunction(self, n):
692 return glXEnumFunction(n, self)
Ian Romanick3fec8c22005-02-02 00:54:45 +0000693
694
695 def functionIterator(self):
696 return glXFunctionIterator(self)
697
698
699 def size_call(self, func):
700 """Create C code to calculate 'compsize'.
701
702 Creates code to calculate 'compsize'. If the function does
703 not need 'compsize' to be calculated, None will be
704 returned."""
705
706 if not func.image and not func.count_parameter_list:
707 return None
708
709 if not func.image:
710 parameters = string.join( func.count_parameter_list, "," )
711 compsize = "__gl%s_size(%s)" % (func.name, parameters)
712 else:
713 [dim, w, h, d, junk] = func.dimensions()
714
715 compsize = '__glImageSize(%s, %s, %s, %s, %s, %s)' % (w, h, d, func.image.img_format, func.image.img_type, func.image.img_target)
716 if not func.image.img_send_null:
717 compsize = '(%s != NULL) ? %s : 0' % (func.image.name, compsize)
718
719 return compsize