Merge branch 'lp-offset-twoside'
[mesa.git] / src / glx / single2.c
1 /*
2 * SGI FREE SOFTWARE LICENSE B (Version 2.0, Sept. 18, 2008)
3 * Copyright (C) 1991-2000 Silicon Graphics, Inc. All Rights Reserved.
4 *
5 * Permission is hereby granted, free of charge, to any person obtaining a
6 * copy of this software and associated documentation files (the "Software"),
7 * to deal in the Software without restriction, including without limitation
8 * the rights to use, copy, modify, merge, publish, distribute, sublicense,
9 * and/or sell copies of the Software, and to permit persons to whom the
10 * Software is furnished to do so, subject to the following conditions:
11 *
12 * The above copyright notice including the dates of first publication and
13 * either this permission notice or a reference to
14 * http://oss.sgi.com/projects/FreeB/
15 * shall be included in all copies or substantial portions of the Software.
16 *
17 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
18 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
19 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
20 * SILICON GRAPHICS, INC. BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
21 * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF
22 * OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
23 * SOFTWARE.
24 *
25 * Except as contained in this notice, the name of Silicon Graphics, Inc.
26 * shall not be used in advertising or otherwise to promote the sale, use or
27 * other dealings in this Software without prior written authorization from
28 * Silicon Graphics, Inc.
29 */
30
31 #include <stdio.h>
32 #include <assert.h>
33 #include "glxclient.h"
34 #include "packsingle.h"
35 #include "glxextensions.h"
36 #include "indirect.h"
37 #include "indirect_vertex_array.h"
38 #include "glapitable.h"
39 #include "glapi.h"
40 #ifdef USE_XCB
41 #include <xcb/xcb.h>
42 #include <xcb/glx.h>
43 #include <X11/Xlib-xcb.h>
44 #endif /* USE_XCB */
45
46 #if !defined(__GNUC__)
47 # define __builtin_expect(x, y) x
48 #endif
49
50 /* Used for GL_ARB_transpose_matrix */
51 static void
52 TransposeMatrixf(GLfloat m[16])
53 {
54 int i, j;
55 for (i = 0; i < 4; i++) {
56 for (j = 0; j < i; j++) {
57 GLfloat tmp = m[i * 4 + j];
58 m[i * 4 + j] = m[j * 4 + i];
59 m[j * 4 + i] = tmp;
60 }
61 }
62 }
63
64 /* Used for GL_ARB_transpose_matrix */
65 static void
66 TransposeMatrixb(GLboolean m[16])
67 {
68 int i, j;
69 for (i = 0; i < 4; i++) {
70 for (j = 0; j < i; j++) {
71 GLboolean tmp = m[i * 4 + j];
72 m[i * 4 + j] = m[j * 4 + i];
73 m[j * 4 + i] = tmp;
74 }
75 }
76 }
77
78 /* Used for GL_ARB_transpose_matrix */
79 static void
80 TransposeMatrixd(GLdouble m[16])
81 {
82 int i, j;
83 for (i = 0; i < 4; i++) {
84 for (j = 0; j < i; j++) {
85 GLdouble tmp = m[i * 4 + j];
86 m[i * 4 + j] = m[j * 4 + i];
87 m[j * 4 + i] = tmp;
88 }
89 }
90 }
91
92 /* Used for GL_ARB_transpose_matrix */
93 static void
94 TransposeMatrixi(GLint m[16])
95 {
96 int i, j;
97 for (i = 0; i < 4; i++) {
98 for (j = 0; j < i; j++) {
99 GLint tmp = m[i * 4 + j];
100 m[i * 4 + j] = m[j * 4 + i];
101 m[j * 4 + i] = tmp;
102 }
103 }
104 }
105
106
107 /**
108 * Remap a transpose-matrix enum to a non-transpose-matrix enum. Enums
109 * that are not transpose-matrix enums are unaffected.
110 */
111 static GLenum
112 RemapTransposeEnum(GLenum e)
113 {
114 switch (e) {
115 case GL_TRANSPOSE_MODELVIEW_MATRIX:
116 case GL_TRANSPOSE_PROJECTION_MATRIX:
117 case GL_TRANSPOSE_TEXTURE_MATRIX:
118 return e - (GL_TRANSPOSE_MODELVIEW_MATRIX - GL_MODELVIEW_MATRIX);
119 case GL_TRANSPOSE_COLOR_MATRIX:
120 return GL_COLOR_MATRIX;
121 default:
122 return e;
123 };
124 }
125
126
127 GLenum
128 __indirect_glGetError(void)
129 {
130 __GLX_SINGLE_DECLARE_VARIABLES();
131 GLuint retval = GL_NO_ERROR;
132 xGLXGetErrorReply reply;
133
134 if (gc->error) {
135 /* Use internal error first */
136 retval = gc->error;
137 gc->error = GL_NO_ERROR;
138 return retval;
139 }
140
141 __GLX_SINGLE_LOAD_VARIABLES();
142 __GLX_SINGLE_BEGIN(X_GLsop_GetError, 0);
143 __GLX_SINGLE_READ_XREPLY();
144 retval = reply.error;
145 __GLX_SINGLE_END();
146
147 return retval;
148 }
149
150
151 /**
152 * Get the selected attribute from the client state.
153 *
154 * \returns
155 * On success \c GL_TRUE is returned. Otherwise, \c GL_FALSE is returned.
156 */
157 static GLboolean
158 get_client_data(struct glx_context * gc, GLenum cap, GLintptr * data)
159 {
160 GLboolean retval = GL_TRUE;
161 __GLXattribute *state = (__GLXattribute *) (gc->client_state_private);
162 const GLint tex_unit = __glXGetActiveTextureUnit(state);
163
164
165 switch (cap) {
166 case GL_VERTEX_ARRAY:
167 case GL_NORMAL_ARRAY:
168 case GL_COLOR_ARRAY:
169 case GL_INDEX_ARRAY:
170 case GL_EDGE_FLAG_ARRAY:
171 case GL_SECONDARY_COLOR_ARRAY:
172 case GL_FOG_COORD_ARRAY:
173 retval = __glXGetArrayEnable(state, cap, 0, data);
174 break;
175
176 case GL_VERTEX_ARRAY_SIZE:
177 retval = __glXGetArraySize(state, GL_VERTEX_ARRAY, 0, data);
178 break;
179 case GL_COLOR_ARRAY_SIZE:
180 retval = __glXGetArraySize(state, GL_COLOR_ARRAY, 0, data);
181 break;
182 case GL_SECONDARY_COLOR_ARRAY_SIZE:
183 retval = __glXGetArraySize(state, GL_SECONDARY_COLOR_ARRAY, 0, data);
184 break;
185
186 case GL_VERTEX_ARRAY_TYPE:
187 retval = __glXGetArrayType(state, GL_VERTEX_ARRAY, 0, data);
188 break;
189 case GL_NORMAL_ARRAY_TYPE:
190 retval = __glXGetArrayType(state, GL_NORMAL_ARRAY, 0, data);
191 break;
192 case GL_INDEX_ARRAY_TYPE:
193 retval = __glXGetArrayType(state, GL_INDEX_ARRAY, 0, data);
194 break;
195 case GL_COLOR_ARRAY_TYPE:
196 retval = __glXGetArrayType(state, GL_COLOR_ARRAY, 0, data);
197 break;
198 case GL_SECONDARY_COLOR_ARRAY_TYPE:
199 retval = __glXGetArrayType(state, GL_SECONDARY_COLOR_ARRAY, 0, data);
200 break;
201 case GL_FOG_COORD_ARRAY_TYPE:
202 retval = __glXGetArrayType(state, GL_FOG_COORD_ARRAY, 0, data);
203 break;
204
205 case GL_VERTEX_ARRAY_STRIDE:
206 retval = __glXGetArrayStride(state, GL_VERTEX_ARRAY, 0, data);
207 break;
208 case GL_NORMAL_ARRAY_STRIDE:
209 retval = __glXGetArrayStride(state, GL_NORMAL_ARRAY, 0, data);
210 break;
211 case GL_INDEX_ARRAY_STRIDE:
212 retval = __glXGetArrayStride(state, GL_INDEX_ARRAY, 0, data);
213 break;
214 case GL_EDGE_FLAG_ARRAY_STRIDE:
215 retval = __glXGetArrayStride(state, GL_EDGE_FLAG_ARRAY, 0, data);
216 break;
217 case GL_COLOR_ARRAY_STRIDE:
218 retval = __glXGetArrayStride(state, GL_COLOR_ARRAY, 0, data);
219 break;
220 case GL_SECONDARY_COLOR_ARRAY_STRIDE:
221 retval = __glXGetArrayStride(state, GL_SECONDARY_COLOR_ARRAY, 0, data);
222 break;
223 case GL_FOG_COORD_ARRAY_STRIDE:
224 retval = __glXGetArrayStride(state, GL_FOG_COORD_ARRAY, 0, data);
225 break;
226
227 case GL_TEXTURE_COORD_ARRAY:
228 retval =
229 __glXGetArrayEnable(state, GL_TEXTURE_COORD_ARRAY, tex_unit, data);
230 break;
231 case GL_TEXTURE_COORD_ARRAY_SIZE:
232 retval =
233 __glXGetArraySize(state, GL_TEXTURE_COORD_ARRAY, tex_unit, data);
234 break;
235 case GL_TEXTURE_COORD_ARRAY_TYPE:
236 retval =
237 __glXGetArrayType(state, GL_TEXTURE_COORD_ARRAY, tex_unit, data);
238 break;
239 case GL_TEXTURE_COORD_ARRAY_STRIDE:
240 retval =
241 __glXGetArrayStride(state, GL_TEXTURE_COORD_ARRAY, tex_unit, data);
242 break;
243
244 case GL_MAX_ELEMENTS_VERTICES:
245 case GL_MAX_ELEMENTS_INDICES:
246 retval = GL_TRUE;
247 *data = ~0UL;
248 break;
249
250
251 case GL_PACK_ROW_LENGTH:
252 *data = (GLintptr) state->storePack.rowLength;
253 break;
254 case GL_PACK_IMAGE_HEIGHT:
255 *data = (GLintptr) state->storePack.imageHeight;
256 break;
257 case GL_PACK_SKIP_ROWS:
258 *data = (GLintptr) state->storePack.skipRows;
259 break;
260 case GL_PACK_SKIP_PIXELS:
261 *data = (GLintptr) state->storePack.skipPixels;
262 break;
263 case GL_PACK_SKIP_IMAGES:
264 *data = (GLintptr) state->storePack.skipImages;
265 break;
266 case GL_PACK_ALIGNMENT:
267 *data = (GLintptr) state->storePack.alignment;
268 break;
269 case GL_PACK_SWAP_BYTES:
270 *data = (GLintptr) state->storePack.swapEndian;
271 break;
272 case GL_PACK_LSB_FIRST:
273 *data = (GLintptr) state->storePack.lsbFirst;
274 break;
275 case GL_UNPACK_ROW_LENGTH:
276 *data = (GLintptr) state->storeUnpack.rowLength;
277 break;
278 case GL_UNPACK_IMAGE_HEIGHT:
279 *data = (GLintptr) state->storeUnpack.imageHeight;
280 break;
281 case GL_UNPACK_SKIP_ROWS:
282 *data = (GLintptr) state->storeUnpack.skipRows;
283 break;
284 case GL_UNPACK_SKIP_PIXELS:
285 *data = (GLintptr) state->storeUnpack.skipPixels;
286 break;
287 case GL_UNPACK_SKIP_IMAGES:
288 *data = (GLintptr) state->storeUnpack.skipImages;
289 break;
290 case GL_UNPACK_ALIGNMENT:
291 *data = (GLintptr) state->storeUnpack.alignment;
292 break;
293 case GL_UNPACK_SWAP_BYTES:
294 *data = (GLintptr) state->storeUnpack.swapEndian;
295 break;
296 case GL_UNPACK_LSB_FIRST:
297 *data = (GLintptr) state->storeUnpack.lsbFirst;
298 break;
299 case GL_CLIENT_ATTRIB_STACK_DEPTH:
300 *data = (GLintptr) (gc->attributes.stackPointer - gc->attributes.stack);
301 break;
302 case GL_MAX_CLIENT_ATTRIB_STACK_DEPTH:
303 *data = (GLintptr) __GL_CLIENT_ATTRIB_STACK_DEPTH;
304 break;
305 case GL_CLIENT_ACTIVE_TEXTURE:
306 *data = (GLintptr) (tex_unit + GL_TEXTURE0);
307 break;
308
309 default:
310 retval = GL_FALSE;
311 break;
312 }
313
314
315 return retval;
316 }
317
318
319 void
320 __indirect_glGetBooleanv(GLenum val, GLboolean * b)
321 {
322 const GLenum origVal = val;
323 __GLX_SINGLE_DECLARE_VARIABLES();
324 xGLXSingleReply reply;
325
326 val = RemapTransposeEnum(val);
327
328 __GLX_SINGLE_LOAD_VARIABLES();
329 __GLX_SINGLE_BEGIN(X_GLsop_GetBooleanv, 4);
330 __GLX_SINGLE_PUT_LONG(0, val);
331 __GLX_SINGLE_READ_XREPLY();
332 __GLX_SINGLE_GET_SIZE(compsize);
333
334 if (compsize == 0) {
335 /*
336 ** Error occured; don't modify user's buffer.
337 */
338 }
339 else {
340 GLintptr data;
341
342 /*
343 ** We still needed to send the request to the server in order to
344 ** find out whether it was legal to make a query (it's illegal,
345 ** for example, to call a query between glBegin() and glEnd()).
346 */
347
348 if (get_client_data(gc, val, &data)) {
349 *b = (GLboolean) data;
350 }
351 else {
352 /*
353 ** Not a local value, so use what we got from the server.
354 */
355 if (compsize == 1) {
356 __GLX_SINGLE_GET_CHAR(b);
357 }
358 else {
359 __GLX_SINGLE_GET_CHAR_ARRAY(b, compsize);
360 if (val != origVal) {
361 /* matrix transpose */
362 TransposeMatrixb(b);
363 }
364 }
365 }
366 }
367 __GLX_SINGLE_END();
368 }
369
370 void
371 __indirect_glGetDoublev(GLenum val, GLdouble * d)
372 {
373 const GLenum origVal = val;
374 __GLX_SINGLE_DECLARE_VARIABLES();
375 xGLXSingleReply reply;
376
377 val = RemapTransposeEnum(val);
378
379 __GLX_SINGLE_LOAD_VARIABLES();
380 __GLX_SINGLE_BEGIN(X_GLsop_GetDoublev, 4);
381 __GLX_SINGLE_PUT_LONG(0, val);
382 __GLX_SINGLE_READ_XREPLY();
383 __GLX_SINGLE_GET_SIZE(compsize);
384
385 if (compsize == 0) {
386 /*
387 ** Error occured; don't modify user's buffer.
388 */
389 }
390 else {
391 GLintptr data;
392
393 /*
394 ** We still needed to send the request to the server in order to
395 ** find out whether it was legal to make a query (it's illegal,
396 ** for example, to call a query between glBegin() and glEnd()).
397 */
398
399 if (get_client_data(gc, val, &data)) {
400 *d = (GLdouble) data;
401 }
402 else {
403 /*
404 ** Not a local value, so use what we got from the server.
405 */
406 if (compsize == 1) {
407 __GLX_SINGLE_GET_DOUBLE(d);
408 }
409 else {
410 __GLX_SINGLE_GET_DOUBLE_ARRAY(d, compsize);
411 if (val != origVal) {
412 /* matrix transpose */
413 TransposeMatrixd(d);
414 }
415 }
416 }
417 }
418 __GLX_SINGLE_END();
419 }
420
421 void
422 __indirect_glGetFloatv(GLenum val, GLfloat * f)
423 {
424 const GLenum origVal = val;
425 __GLX_SINGLE_DECLARE_VARIABLES();
426 xGLXSingleReply reply;
427
428 val = RemapTransposeEnum(val);
429
430 __GLX_SINGLE_LOAD_VARIABLES();
431 __GLX_SINGLE_BEGIN(X_GLsop_GetFloatv, 4);
432 __GLX_SINGLE_PUT_LONG(0, val);
433 __GLX_SINGLE_READ_XREPLY();
434 __GLX_SINGLE_GET_SIZE(compsize);
435
436 if (compsize == 0) {
437 /*
438 ** Error occured; don't modify user's buffer.
439 */
440 }
441 else {
442 GLintptr data;
443
444 /*
445 ** We still needed to send the request to the server in order to
446 ** find out whether it was legal to make a query (it's illegal,
447 ** for example, to call a query between glBegin() and glEnd()).
448 */
449
450 if (get_client_data(gc, val, &data)) {
451 *f = (GLfloat) data;
452 }
453 else {
454 /*
455 ** Not a local value, so use what we got from the server.
456 */
457 if (compsize == 1) {
458 __GLX_SINGLE_GET_FLOAT(f);
459 }
460 else {
461 __GLX_SINGLE_GET_FLOAT_ARRAY(f, compsize);
462 if (val != origVal) {
463 /* matrix transpose */
464 TransposeMatrixf(f);
465 }
466 }
467 }
468 }
469 __GLX_SINGLE_END();
470 }
471
472 void
473 __indirect_glGetIntegerv(GLenum val, GLint * i)
474 {
475 const GLenum origVal = val;
476 __GLX_SINGLE_DECLARE_VARIABLES();
477 xGLXSingleReply reply;
478
479 val = RemapTransposeEnum(val);
480
481 __GLX_SINGLE_LOAD_VARIABLES();
482 __GLX_SINGLE_BEGIN(X_GLsop_GetIntegerv, 4);
483 __GLX_SINGLE_PUT_LONG(0, val);
484 __GLX_SINGLE_READ_XREPLY();
485 __GLX_SINGLE_GET_SIZE(compsize);
486
487 if (compsize == 0) {
488 /*
489 ** Error occured; don't modify user's buffer.
490 */
491 }
492 else {
493 GLintptr data;
494
495 /*
496 ** We still needed to send the request to the server in order to
497 ** find out whether it was legal to make a query (it's illegal,
498 ** for example, to call a query between glBegin() and glEnd()).
499 */
500
501 if (get_client_data(gc, val, &data)) {
502 *i = (GLint) data;
503 }
504 else {
505 /*
506 ** Not a local value, so use what we got from the server.
507 */
508 if (compsize == 1) {
509 __GLX_SINGLE_GET_LONG(i);
510 }
511 else {
512 __GLX_SINGLE_GET_LONG_ARRAY(i, compsize);
513 if (val != origVal) {
514 /* matrix transpose */
515 TransposeMatrixi(i);
516 }
517 }
518 }
519 }
520 __GLX_SINGLE_END();
521 }
522
523 /*
524 ** Send all pending commands to server.
525 */
526 void
527 __indirect_glFlush(void)
528 {
529 __GLX_SINGLE_DECLARE_VARIABLES();
530
531 if (!dpy)
532 return;
533
534 __GLX_SINGLE_LOAD_VARIABLES();
535 __GLX_SINGLE_BEGIN(X_GLsop_Flush, 0);
536 __GLX_SINGLE_END();
537
538 /* And finally flush the X protocol data */
539 XFlush(dpy);
540 }
541
542 void
543 __indirect_glFeedbackBuffer(GLsizei size, GLenum type, GLfloat * buffer)
544 {
545 __GLX_SINGLE_DECLARE_VARIABLES();
546
547 if (!dpy)
548 return;
549
550 __GLX_SINGLE_LOAD_VARIABLES();
551 __GLX_SINGLE_BEGIN(X_GLsop_FeedbackBuffer, 8);
552 __GLX_SINGLE_PUT_LONG(0, size);
553 __GLX_SINGLE_PUT_LONG(4, type);
554 __GLX_SINGLE_END();
555
556 gc->feedbackBuf = buffer;
557 }
558
559 void
560 __indirect_glSelectBuffer(GLsizei numnames, GLuint * buffer)
561 {
562 __GLX_SINGLE_DECLARE_VARIABLES();
563
564 if (!dpy)
565 return;
566
567 __GLX_SINGLE_LOAD_VARIABLES();
568 __GLX_SINGLE_BEGIN(X_GLsop_SelectBuffer, 4);
569 __GLX_SINGLE_PUT_LONG(0, numnames);
570 __GLX_SINGLE_END();
571
572 gc->selectBuf = buffer;
573 }
574
575 GLint
576 __indirect_glRenderMode(GLenum mode)
577 {
578 __GLX_SINGLE_DECLARE_VARIABLES();
579 GLint retval = 0;
580 xGLXRenderModeReply reply;
581
582 if (!dpy)
583 return -1;
584
585 __GLX_SINGLE_LOAD_VARIABLES();
586 __GLX_SINGLE_BEGIN(X_GLsop_RenderMode, 4);
587 __GLX_SINGLE_PUT_LONG(0, mode);
588 __GLX_SINGLE_READ_XREPLY();
589 __GLX_SINGLE_GET_RETVAL(retval, GLint);
590
591 if (reply.newMode != mode) {
592 /*
593 ** Switch to new mode did not take effect, therefore an error
594 ** occured. When an error happens the server won't send us any
595 ** other data.
596 */
597 }
598 else {
599 /* Read the feedback or selection data */
600 if (gc->renderMode == GL_FEEDBACK) {
601 __GLX_SINGLE_GET_SIZE(compsize);
602 __GLX_SINGLE_GET_FLOAT_ARRAY(gc->feedbackBuf, compsize);
603 }
604 else if (gc->renderMode == GL_SELECT) {
605 __GLX_SINGLE_GET_SIZE(compsize);
606 __GLX_SINGLE_GET_LONG_ARRAY(gc->selectBuf, compsize);
607 }
608 gc->renderMode = mode;
609 }
610 __GLX_SINGLE_END();
611
612 return retval;
613 }
614
615 void
616 __indirect_glFinish(void)
617 {
618 __GLX_SINGLE_DECLARE_VARIABLES();
619 xGLXSingleReply reply;
620
621 __GLX_SINGLE_LOAD_VARIABLES();
622 __GLX_SINGLE_BEGIN(X_GLsop_Finish, 0);
623 __GLX_SINGLE_READ_XREPLY();
624 __GLX_SINGLE_END();
625 }
626
627
628 /**
629 * Extract the major and minor version numbers from a version string.
630 */
631 static void
632 version_from_string(const char *ver, int *major_version, int *minor_version)
633 {
634 const char *end;
635 long major;
636 long minor;
637
638 major = strtol(ver, (char **) &end, 10);
639 minor = strtol(end + 1, NULL, 10);
640 *major_version = major;
641 *minor_version = minor;
642 }
643
644
645 const GLubyte *
646 __indirect_glGetString(GLenum name)
647 {
648 struct glx_context *gc = __glXGetCurrentContext();
649 Display *dpy = gc->currentDpy;
650 GLubyte *s = NULL;
651
652 if (!dpy)
653 return 0;
654
655 /*
656 ** Return the cached copy if the string has already been fetched
657 */
658 switch (name) {
659 case GL_VENDOR:
660 if (gc->vendor)
661 return gc->vendor;
662 break;
663 case GL_RENDERER:
664 if (gc->renderer)
665 return gc->renderer;
666 break;
667 case GL_VERSION:
668 if (gc->version)
669 return gc->version;
670 break;
671 case GL_EXTENSIONS:
672 if (gc->extensions)
673 return gc->extensions;
674 break;
675 default:
676 __glXSetError(gc, GL_INVALID_ENUM);
677 return 0;
678 }
679
680 /*
681 ** Get requested string from server
682 */
683
684 (void) __glXFlushRenderBuffer(gc, gc->pc);
685 s = (GLubyte *) __glXGetString(dpy, gc->majorOpcode, gc->currentContextTag,
686 name);
687 if (!s) {
688 /* Throw data on the floor */
689 __glXSetError(gc, GL_OUT_OF_MEMORY);
690 }
691 else {
692 /*
693 ** Update local cache
694 */
695 switch (name) {
696 case GL_VENDOR:
697 gc->vendor = s;
698 break;
699
700 case GL_RENDERER:
701 gc->renderer = s;
702 break;
703
704 case GL_VERSION:{
705 int client_major;
706 int client_minor;
707
708 version_from_string((char *) s,
709 &gc->server_major, &gc->server_minor);
710 __glXGetGLVersion(&client_major, &client_minor);
711
712 if ((gc->server_major < client_major)
713 || ((gc->server_major == client_major)
714 && (gc->server_minor <= client_minor))) {
715 gc->version = s;
716 }
717 else {
718 /* Allow 7 bytes for the client-side GL version. This allows
719 * for upto version 999.999. I'm not holding my breath for
720 * that one! The extra 4 is for the ' ()\0' that will be
721 * added.
722 */
723 const size_t size = 7 + strlen((char *) s) + 4;
724
725 gc->version = Xmalloc(size);
726 if (gc->version == NULL) {
727 /* If we couldn't allocate memory for the new string,
728 * make a best-effort and just copy the client-side version
729 * to the string and use that. It probably doesn't
730 * matter what is done here. If there not memory available
731 * for a short string, the system is probably going to die
732 * soon anyway.
733 */
734 snprintf((char *) s, strlen((char *) s) + 1, "%u.%u",
735 client_major, client_minor);
736 gc->version = s;
737 }
738 else {
739 snprintf((char *) gc->version, size, "%u.%u (%s)",
740 client_major, client_minor, s);
741 Xfree(s);
742 s = gc->version;
743 }
744 }
745 break;
746 }
747
748 case GL_EXTENSIONS:{
749 int major = 1;
750 int minor = 0;
751
752 /* This code is currently disabled. I was reminded that some
753 * vendors intentionally exclude some extensions from their
754 * extension string that are part of the core version they
755 * advertise. In particular, on Nvidia drivers this means that
756 * the functionality is supported by the driver, but is not
757 * hardware accelerated. For example, a TNT will show core
758 * version 1.5, but most of the post-1.2 functionality is a
759 * software fallback.
760 *
761 * I don't want to break applications that rely on this odd
762 * behavior. At the same time, the code is written and tested,
763 * so I didn't want to throw it away. Therefore, the code is here
764 * but disabled. In the future, we may wish to and an environment
765 * variable to enable it.
766 */
767
768 #if 0
769 /* Call glGetString just to make sure that gc->server_major and
770 * gc->server_minor are set. This version may be higher than we
771 * can completely support, but it may imply support for some
772 * extensions that we can support.
773 *
774 * For example, at the time of this writing, the client-side
775 * library only supports upto core GL version 1.2. However, cubic
776 * textures, multitexture, multisampling, and some other 1.3
777 * features are supported. If the server reports back version
778 * 1.3, but does not report all of those extensions, we will
779 * enable them.
780 */
781 (void *) glGetString(GL_VERSION);
782 major = gc->server_major, minor = gc->server_minor;
783 #endif
784
785 __glXCalculateUsableGLExtensions(gc, (char *) s, major, minor);
786 XFree(s);
787 s = gc->extensions;
788 break;
789 }
790 }
791 }
792 return s;
793 }
794
795 GLboolean
796 __indirect_glIsEnabled(GLenum cap)
797 {
798 __GLX_SINGLE_DECLARE_VARIABLES();
799 __GLXattribute *state = (__GLXattribute *) (gc->client_state_private);
800 xGLXSingleReply reply;
801 GLboolean retval = 0;
802 GLintptr enable;
803
804 if (!dpy)
805 return 0;
806
807 switch (cap) {
808 case GL_VERTEX_ARRAY:
809 case GL_NORMAL_ARRAY:
810 case GL_COLOR_ARRAY:
811 case GL_INDEX_ARRAY:
812 case GL_EDGE_FLAG_ARRAY:
813 case GL_SECONDARY_COLOR_ARRAY:
814 case GL_FOG_COORD_ARRAY:
815 retval = __glXGetArrayEnable(state, cap, 0, &enable);
816 assert(retval);
817 return (GLboolean) enable;
818 break;
819 case GL_TEXTURE_COORD_ARRAY:
820 retval = __glXGetArrayEnable(state, GL_TEXTURE_COORD_ARRAY,
821 __glXGetActiveTextureUnit(state), &enable);
822 assert(retval);
823 return (GLboolean) enable;
824 break;
825 }
826
827 __GLX_SINGLE_LOAD_VARIABLES();
828 __GLX_SINGLE_BEGIN(X_GLsop_IsEnabled, 4);
829 __GLX_SINGLE_PUT_LONG(0, cap);
830 __GLX_SINGLE_READ_XREPLY();
831 __GLX_SINGLE_GET_RETVAL(retval, GLboolean);
832 __GLX_SINGLE_END();
833 return retval;
834 }
835
836 void
837 __indirect_glGetPointerv(GLenum pname, void **params)
838 {
839 struct glx_context *gc = __glXGetCurrentContext();
840 __GLXattribute *state = (__GLXattribute *) (gc->client_state_private);
841 Display *dpy = gc->currentDpy;
842
843 if (!dpy)
844 return;
845
846 switch (pname) {
847 case GL_VERTEX_ARRAY_POINTER:
848 case GL_NORMAL_ARRAY_POINTER:
849 case GL_COLOR_ARRAY_POINTER:
850 case GL_INDEX_ARRAY_POINTER:
851 case GL_EDGE_FLAG_ARRAY_POINTER:
852 __glXGetArrayPointer(state, pname - GL_VERTEX_ARRAY_POINTER
853 + GL_VERTEX_ARRAY, 0, params);
854 return;
855 case GL_TEXTURE_COORD_ARRAY_POINTER:
856 __glXGetArrayPointer(state, GL_TEXTURE_COORD_ARRAY,
857 __glXGetActiveTextureUnit(state), params);
858 return;
859 case GL_SECONDARY_COLOR_ARRAY_POINTER:
860 case GL_FOG_COORD_ARRAY_POINTER:
861 __glXGetArrayPointer(state, pname - GL_FOG_COORD_ARRAY_POINTER
862 + GL_FOG_COORD_ARRAY, 0, params);
863 return;
864 case GL_FEEDBACK_BUFFER_POINTER:
865 *params = (void *) gc->feedbackBuf;
866 return;
867 case GL_SELECTION_BUFFER_POINTER:
868 *params = (void *) gc->selectBuf;
869 return;
870 default:
871 __glXSetError(gc, GL_INVALID_ENUM);
872 return;
873 }
874 }
875
876
877
878 /**
879 * This was previously auto-generated, but we need to special-case
880 * how we handle writing into the 'residences' buffer when n%4!=0.
881 */
882 #define X_GLsop_AreTexturesResident 143
883 GLboolean
884 __indirect_glAreTexturesResident(GLsizei n, const GLuint * textures,
885 GLboolean * residences)
886 {
887 struct glx_context *const gc = __glXGetCurrentContext();
888 Display *const dpy = gc->currentDpy;
889 GLboolean retval = (GLboolean) 0;
890 const GLuint cmdlen = 4 + __GLX_PAD((n * 4));
891 if (__builtin_expect((n >= 0) && (dpy != NULL), 1)) {
892 #ifdef USE_XCB
893 xcb_connection_t *c = XGetXCBConnection(dpy);
894 (void) __glXFlushRenderBuffer(gc, gc->pc);
895 xcb_glx_are_textures_resident_reply_t *reply =
896 xcb_glx_are_textures_resident_reply(c,
897 xcb_glx_are_textures_resident
898 (c, gc->currentContextTag, n,
899 textures), NULL);
900 (void) memcpy(residences, xcb_glx_are_textures_resident_data(reply),
901 xcb_glx_are_textures_resident_data_length(reply) *
902 sizeof(GLboolean));
903 retval = reply->ret_val;
904 free(reply);
905 #else
906 GLubyte const *pc =
907 __glXSetupSingleRequest(gc, X_GLsop_AreTexturesResident, cmdlen);
908 (void) memcpy((void *) (pc + 0), (void *) (&n), 4);
909 (void) memcpy((void *) (pc + 4), (void *) (textures), (n * 4));
910 if (n & 3) {
911 /* n is not a multiple of four.
912 * When reply_is_always_array is TRUE, __glXReadReply() will
913 * put a multiple of four bytes into the dest buffer. If the
914 * caller's buffer is not a multiple of four in size, we'll write
915 * out of bounds. So use a temporary buffer that's a few bytes
916 * larger.
917 */
918 GLboolean *res4 = malloc((n + 3) & ~3);
919 retval = (GLboolean) __glXReadReply(dpy, 1, res4, GL_TRUE);
920 memcpy(residences, res4, n);
921 free(res4);
922 }
923 else {
924 retval = (GLboolean) __glXReadReply(dpy, 1, residences, GL_TRUE);
925 }
926 UnlockDisplay(dpy);
927 SyncHandle();
928 #endif /* USE_XCB */
929 }
930 return retval;
931 }
932
933
934 /**
935 * This was previously auto-generated, but we need to special-case
936 * how we handle writing into the 'residences' buffer when n%4!=0.
937 */
938 #define X_GLvop_AreTexturesResidentEXT 11
939 GLboolean
940 glAreTexturesResidentEXT(GLsizei n, const GLuint * textures,
941 GLboolean * residences)
942 {
943 struct glx_context *const gc = __glXGetCurrentContext();
944
945 if (gc->isDirect) {
946 return GET_DISPATCH()->AreTexturesResident(n, textures, residences);
947 }
948 else {
949 struct glx_context *const gc = __glXGetCurrentContext();
950 Display *const dpy = gc->currentDpy;
951 GLboolean retval = (GLboolean) 0;
952 const GLuint cmdlen = 4 + __GLX_PAD((n * 4));
953 if (__builtin_expect((n >= 0) && (dpy != NULL), 1)) {
954 GLubyte const *pc =
955 __glXSetupVendorRequest(gc, X_GLXVendorPrivateWithReply,
956 X_GLvop_AreTexturesResidentEXT,
957 cmdlen);
958 (void) memcpy((void *) (pc + 0), (void *) (&n), 4);
959 (void) memcpy((void *) (pc + 4), (void *) (textures), (n * 4));
960 if (n & 3) {
961 /* see comments in __indirect_glAreTexturesResident() */
962 GLboolean *res4 = malloc((n + 3) & ~3);
963 retval = (GLboolean) __glXReadReply(dpy, 1, res4, GL_TRUE);
964 memcpy(residences, res4, n);
965 free(res4);
966 }
967 else {
968 retval = (GLboolean) __glXReadReply(dpy, 1, residences, GL_TRUE);
969 }
970 UnlockDisplay(dpy);
971 SyncHandle();
972 }
973 return retval;
974 }
975 }