1 /*
2 * (C) Copyright IBM Corporation 2004
3 * 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 * on the rights to use, copy, modify, merge, publish, distribute, sub
9 * license, and/or sell copies of the Software, and to permit persons to whom
10 * the Software is furnished to do so, subject to the following conditions:
11 *
12 * The above copyright notice and this permission notice (including the next
13 * paragraph) shall be included in all copies or substantial portions of the
14 * 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 NON-INFRINGEMENT. IN NO EVENT SHALL
19 * IBM AND/OR THEIR SUPPLIERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
20 * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
21 * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
22 * DEALINGS IN THE SOFTWARE.
23 */
24
25 /**
26 * \file glx_pbuffer.c
27 * Implementation of pbuffer related functions.
28 *
29 * \author Ian Romanick <idr@us.ibm.com>
30 */
31
32 #include <inttypes.h>
33 #include "glxclient.h"
34 #include <X11/extensions/extutil.h>
35 #include <X11/extensions/Xext.h>
36 #include <assert.h>
37 #include <string.h>
38 #include "glxextensions.h"
39
40 #ifdef GLX_USE_APPLEGL
41 #include <pthread.h>
42 #include "apple/apple_glx_drawable.h"
43 #endif
44
45 #include "glx_error.h"
46
47 #define WARN_ONCE_GLX_1_3(a, b) { \
48 static int warned=1; \
49 if(warned) { \
50 warn_GLX_1_3((a), b ); \
51 warned=0; \
52 } \
53 }
54
55 /**
56 * Emit a warning when clients use GLX 1.3 functions on pre-1.3 systems.
57 */
58 static void
warn_GLX_1_3(Display * dpy,const char * function_name)59 warn_GLX_1_3(Display * dpy, const char *function_name)
60 {
61 struct glx_display *priv = __glXInitialize(dpy);
62
63 if (priv && priv->minorVersion < 3) {
64 fprintf(stderr,
65 "WARNING: Application calling GLX 1.3 function \"%s\" "
66 "when GLX 1.3 is not supported! This is an application bug!\n",
67 function_name);
68 }
69 }
70
71 #ifndef GLX_USE_APPLEGL
72 /**
73 * Change a drawable's attribute.
74 *
75 * This function is used to implement \c glXSelectEvent and
76 * \c glXSelectEventSGIX.
77 *
78 * \note
79 * This function dynamically determines whether to use the SGIX_pbuffer
80 * version of the protocol or the GLX 1.3 version of the protocol.
81 */
82 static void
ChangeDrawableAttribute(Display * dpy,GLXDrawable drawable,const CARD32 * attribs,size_t num_attribs)83 ChangeDrawableAttribute(Display * dpy, GLXDrawable drawable,
84 const CARD32 * attribs, size_t num_attribs)
85 {
86 struct glx_display *priv = __glXInitialize(dpy);
87 #ifdef GLX_DIRECT_RENDERING
88 __GLXDRIdrawable *pdraw;
89 #endif
90 CARD32 *output;
91 CARD8 opcode;
92 int i;
93
94 if ((priv == NULL) || (dpy == NULL) || (drawable == 0)) {
95 return;
96 }
97
98 opcode = __glXSetupForCommand(dpy);
99 if (!opcode)
100 return;
101
102 LockDisplay(dpy);
103
104 if ((priv->majorVersion > 1) || (priv->minorVersion >= 3)) {
105 xGLXChangeDrawableAttributesReq *req;
106
107 GetReqExtra(GLXChangeDrawableAttributes, 8 * num_attribs, req);
108 output = (CARD32 *) (req + 1);
109
110 req->reqType = opcode;
111 req->glxCode = X_GLXChangeDrawableAttributes;
112 req->drawable = drawable;
113 req->numAttribs = (CARD32) num_attribs;
114 }
115 else {
116 xGLXVendorPrivateWithReplyReq *vpreq;
117
118 GetReqExtra(GLXVendorPrivateWithReply, 8 + (8 * num_attribs), vpreq);
119 output = (CARD32 *) (vpreq + 1);
120
121 vpreq->reqType = opcode;
122 vpreq->glxCode = X_GLXVendorPrivateWithReply;
123 vpreq->vendorCode = X_GLXvop_ChangeDrawableAttributesSGIX;
124
125 output[0] = (CARD32) drawable;
126 output[1] = num_attribs;
127 output += 2;
128 }
129
130 (void) memcpy(output, attribs, sizeof(CARD32) * 2 * num_attribs);
131
132 UnlockDisplay(dpy);
133 SyncHandle();
134
135 #ifdef GLX_DIRECT_RENDERING
136 pdraw = GetGLXDRIDrawable(dpy, drawable);
137
138 if (!pdraw)
139 return;
140
141 for (i = 0; i < num_attribs; i++) {
142 switch(attribs[i * 2]) {
143 case GLX_EVENT_MASK:
144 /* Keep a local copy for masking out DRI2 proto events as needed */
145 pdraw->eventMask = attribs[i * 2 + 1];
146 break;
147 }
148 }
149 #endif
150
151 return;
152 }
153
154
155 #ifdef GLX_DIRECT_RENDERING
156 static GLenum
determineTextureTarget(const int * attribs,int numAttribs)157 determineTextureTarget(const int *attribs, int numAttribs)
158 {
159 GLenum target = 0;
160 int i;
161
162 for (i = 0; i < numAttribs; i++) {
163 if (attribs[2 * i] == GLX_TEXTURE_TARGET_EXT) {
164 switch (attribs[2 * i + 1]) {
165 case GLX_TEXTURE_2D_EXT:
166 target = GL_TEXTURE_2D;
167 break;
168 case GLX_TEXTURE_RECTANGLE_EXT:
169 target = GL_TEXTURE_RECTANGLE_ARB;
170 break;
171 }
172 }
173 }
174
175 return target;
176 }
177
178 static GLenum
determineTextureFormat(const int * attribs,int numAttribs)179 determineTextureFormat(const int *attribs, int numAttribs)
180 {
181 int i;
182
183 for (i = 0; i < numAttribs; i++) {
184 if (attribs[2 * i] == GLX_TEXTURE_FORMAT_EXT)
185 return attribs[2 * i + 1];
186 }
187
188 return 0;
189 }
190
191 static GLboolean
CreateDRIDrawable(Display * dpy,struct glx_config * config,XID drawable,XID glxdrawable,const int * attrib_list,size_t num_attribs)192 CreateDRIDrawable(Display *dpy, struct glx_config *config,
193 XID drawable, XID glxdrawable,
194 const int *attrib_list, size_t num_attribs)
195 {
196 struct glx_display *const priv = __glXInitialize(dpy);
197 __GLXDRIdrawable *pdraw;
198 struct glx_screen *psc;
199
200 if (priv == NULL) {
201 fprintf(stderr, "failed to create drawable\n");
202 return GL_FALSE;
203 }
204
205 psc = priv->screens[config->screen];
206 if (psc->driScreen == NULL)
207 return GL_TRUE;
208
209 pdraw = psc->driScreen->createDrawable(psc, drawable,
210 glxdrawable, config);
211 if (pdraw == NULL) {
212 fprintf(stderr, "failed to create drawable\n");
213 return GL_FALSE;
214 }
215
216 if (__glxHashInsert(priv->drawHash, glxdrawable, pdraw)) {
217 (*pdraw->destroyDrawable) (pdraw);
218 return GL_FALSE;
219 }
220
221 pdraw->textureTarget = determineTextureTarget(attrib_list, num_attribs);
222 pdraw->textureFormat = determineTextureFormat(attrib_list, num_attribs);
223
224 return GL_TRUE;
225 }
226
227 static void
DestroyDRIDrawable(Display * dpy,GLXDrawable drawable,int destroy_xdrawable)228 DestroyDRIDrawable(Display *dpy, GLXDrawable drawable, int destroy_xdrawable)
229 {
230 struct glx_display *const priv = __glXInitialize(dpy);
231 __GLXDRIdrawable *pdraw = GetGLXDRIDrawable(dpy, drawable);
232 XID xid;
233
234 if (priv != NULL && pdraw != NULL) {
235 xid = pdraw->xDrawable;
236 (*pdraw->destroyDrawable) (pdraw);
237 __glxHashDelete(priv->drawHash, drawable);
238 if (destroy_xdrawable)
239 XFreePixmap(priv->dpy, xid);
240 }
241 }
242
243 #else
244
245 static GLboolean
CreateDRIDrawable(Display * dpy,const struct glx_config * fbconfig,XID drawable,XID glxdrawable,const int * attrib_list,size_t num_attribs)246 CreateDRIDrawable(Display *dpy, const struct glx_config * fbconfig,
247 XID drawable, XID glxdrawable,
248 const int *attrib_list, size_t num_attribs)
249 {
250 return GL_TRUE;
251 }
252
253 static void
DestroyDRIDrawable(Display * dpy,GLXDrawable drawable,int destroy_xdrawable)254 DestroyDRIDrawable(Display *dpy, GLXDrawable drawable, int destroy_xdrawable)
255 {
256 }
257
258 #endif
259
260 /**
261 * Get a drawable's attribute.
262 *
263 * This function is used to implement \c glXGetSelectedEvent and
264 * \c glXGetSelectedEventSGIX.
265 *
266 * \note
267 * This function dynamically determines whether to use the SGIX_pbuffer
268 * version of the protocol or the GLX 1.3 version of the protocol.
269 *
270 * \todo
271 * The number of attributes returned is likely to be small, probably less than
272 * 10. Given that, this routine should try to use an array on the stack to
273 * capture the reply rather than always calling Xmalloc.
274 */
275 int
__glXGetDrawableAttribute(Display * dpy,GLXDrawable drawable,int attribute,unsigned int * value)276 __glXGetDrawableAttribute(Display * dpy, GLXDrawable drawable,
277 int attribute, unsigned int *value)
278 {
279 struct glx_display *priv;
280 xGLXGetDrawableAttributesReply reply;
281 CARD32 *data;
282 CARD8 opcode;
283 unsigned int length;
284 unsigned int i;
285 unsigned int num_attributes;
286 GLboolean use_glx_1_3;
287
288 #if defined(GLX_DIRECT_RENDERING) && !defined(GLX_USE_APPLEGL)
289 __GLXDRIdrawable *pdraw;
290 #endif
291
292 if (dpy == NULL)
293 return 0;
294
295 /* Page 38 (page 52 of the PDF) of glxencode1.3.pdf says:
296 *
297 * "If drawable is not a valid GLX drawable, a GLXBadDrawable error is
298 * generated."
299 */
300 if (drawable == 0) {
301 __glXSendError(dpy, GLXBadDrawable, 0, X_GLXGetDrawableAttributes, false);
302 return 0;
303 }
304
305 priv = __glXInitialize(dpy);
306 if (priv == NULL)
307 return 0;
308
309 use_glx_1_3 = ((priv->majorVersion > 1) || (priv->minorVersion >= 3));
310
311 *value = 0;
312
313
314 opcode = __glXSetupForCommand(dpy);
315 if (!opcode)
316 return 0;
317
318 #if defined(GLX_DIRECT_RENDERING) && !defined(GLX_USE_APPLEGL)
319 pdraw = GetGLXDRIDrawable(dpy, drawable);
320
321 if (attribute == GLX_BACK_BUFFER_AGE_EXT) {
322 struct glx_context *gc = __glXGetCurrentContext();
323 struct glx_screen *psc;
324
325 /* The GLX_EXT_buffer_age spec says:
326 *
327 * "If querying GLX_BACK_BUFFER_AGE_EXT and <draw> is not bound to
328 * the calling thread's current context a GLXBadDrawable error is
329 * generated."
330 */
331 if (pdraw == NULL || gc == &dummyContext || gc->currentDpy != dpy ||
332 (gc->currentDrawable != drawable &&
333 gc->currentReadable != drawable)) {
334 __glXSendError(dpy, GLXBadDrawable, drawable,
335 X_GLXGetDrawableAttributes, false);
336 return 0;
337 }
338
339 psc = pdraw->psc;
340
341 if (psc->driScreen->getBufferAge != NULL)
342 *value = psc->driScreen->getBufferAge(pdraw);
343
344 return 0;
345 }
346 #endif
347
348 LockDisplay(dpy);
349
350 if (use_glx_1_3) {
351 xGLXGetDrawableAttributesReq *req;
352
353 GetReq(GLXGetDrawableAttributes, req);
354 req->reqType = opcode;
355 req->glxCode = X_GLXGetDrawableAttributes;
356 req->drawable = drawable;
357 }
358 else {
359 xGLXVendorPrivateWithReplyReq *vpreq;
360
361 GetReqExtra(GLXVendorPrivateWithReply, 4, vpreq);
362 data = (CARD32 *) (vpreq + 1);
363 data[0] = (CARD32) drawable;
364
365 vpreq->reqType = opcode;
366 vpreq->glxCode = X_GLXVendorPrivateWithReply;
367 vpreq->vendorCode = X_GLXvop_GetDrawableAttributesSGIX;
368 }
369
370 _XReply(dpy, (xReply *) & reply, 0, False);
371
372 if (reply.type == X_Error) {
373 UnlockDisplay(dpy);
374 SyncHandle();
375 return 0;
376 }
377
378 length = reply.length;
379 if (length) {
380 num_attributes = (use_glx_1_3) ? reply.numAttribs : length / 2;
381 data = malloc(length * sizeof(CARD32));
382 if (data == NULL) {
383 /* Throw data on the floor */
384 _XEatData(dpy, length);
385 }
386 else {
387 _XRead(dpy, (char *) data, length * sizeof(CARD32));
388
389 /* Search the set of returned attributes for the attribute requested by
390 * the caller.
391 */
392 for (i = 0; i < num_attributes; i++) {
393 if (data[i * 2] == attribute) {
394 *value = data[(i * 2) + 1];
395 break;
396 }
397 }
398
399 #if defined(GLX_DIRECT_RENDERING) && !defined(GLX_USE_APPLEGL)
400 if (pdraw != NULL) {
401 if (!pdraw->textureTarget)
402 pdraw->textureTarget =
403 determineTextureTarget((const int *) data, num_attributes);
404 if (!pdraw->textureFormat)
405 pdraw->textureFormat =
406 determineTextureFormat((const int *) data, num_attributes);
407 }
408 #endif
409
410 free(data);
411 }
412 }
413
414 UnlockDisplay(dpy);
415 SyncHandle();
416
417 return 0;
418 }
419
420 static void
protocolDestroyDrawable(Display * dpy,GLXDrawable drawable,CARD32 glxCode)421 protocolDestroyDrawable(Display *dpy, GLXDrawable drawable, CARD32 glxCode)
422 {
423 xGLXDestroyPbufferReq *req;
424 CARD8 opcode;
425
426 opcode = __glXSetupForCommand(dpy);
427 if (!opcode)
428 return;
429
430 LockDisplay(dpy);
431
432 GetReq(GLXDestroyPbuffer, req);
433 req->reqType = opcode;
434 req->glxCode = glxCode;
435 req->pbuffer = (GLXPbuffer) drawable;
436
437 UnlockDisplay(dpy);
438 SyncHandle();
439 }
440
441 /**
442 * Create a non-pbuffer GLX drawable.
443 */
444 static GLXDrawable
CreateDrawable(Display * dpy,struct glx_config * config,Drawable drawable,const int * attrib_list,CARD8 glxCode)445 CreateDrawable(Display *dpy, struct glx_config *config,
446 Drawable drawable, const int *attrib_list, CARD8 glxCode)
447 {
448 xGLXCreateWindowReq *req;
449 struct glx_drawable *glxDraw;
450 CARD32 *data;
451 unsigned int i;
452 CARD8 opcode;
453 GLXDrawable xid;
454
455 i = 0;
456 if (attrib_list) {
457 while (attrib_list[i * 2] != None)
458 i++;
459 }
460
461 opcode = __glXSetupForCommand(dpy);
462 if (!opcode)
463 return None;
464
465 glxDraw = malloc(sizeof(*glxDraw));
466 if (!glxDraw)
467 return None;
468
469 LockDisplay(dpy);
470 GetReqExtra(GLXCreateWindow, 8 * i, req);
471 data = (CARD32 *) (req + 1);
472
473 req->reqType = opcode;
474 req->glxCode = glxCode;
475 req->screen = config->screen;
476 req->fbconfig = config->fbconfigID;
477 req->window = drawable;
478 req->glxwindow = xid = XAllocID(dpy);
479 req->numAttribs = i;
480
481 if (attrib_list)
482 memcpy(data, attrib_list, 8 * i);
483
484 UnlockDisplay(dpy);
485 SyncHandle();
486
487 if (InitGLXDrawable(dpy, glxDraw, drawable, xid)) {
488 free(glxDraw);
489 return None;
490 }
491
492 if (!CreateDRIDrawable(dpy, config, drawable, xid, attrib_list, i)) {
493 if (glxCode == X_GLXCreatePixmap)
494 glxCode = X_GLXDestroyPixmap;
495 else
496 glxCode = X_GLXDestroyWindow;
497 protocolDestroyDrawable(dpy, xid, glxCode);
498 xid = None;
499 }
500
501 return xid;
502 }
503
504
505 /**
506 * Destroy a non-pbuffer GLX drawable.
507 */
508 static void
DestroyDrawable(Display * dpy,GLXDrawable drawable,CARD32 glxCode)509 DestroyDrawable(Display * dpy, GLXDrawable drawable, CARD32 glxCode)
510 {
511 if ((dpy == NULL) || (drawable == 0)) {
512 return;
513 }
514
515 protocolDestroyDrawable(dpy, drawable, glxCode);
516
517 DestroyGLXDrawable(dpy, drawable);
518 DestroyDRIDrawable(dpy, drawable, GL_FALSE);
519
520 return;
521 }
522
523
524 /**
525 * Create a pbuffer.
526 *
527 * This function is used to implement \c glXCreatePbuffer and
528 * \c glXCreateGLXPbufferSGIX.
529 *
530 * \note
531 * This function dynamically determines whether to use the SGIX_pbuffer
532 * version of the protocol or the GLX 1.3 version of the protocol.
533 */
534 static GLXDrawable
CreatePbuffer(Display * dpy,struct glx_config * config,unsigned int width,unsigned int height,const int * attrib_list,GLboolean size_in_attribs)535 CreatePbuffer(Display * dpy, struct glx_config *config,
536 unsigned int width, unsigned int height,
537 const int *attrib_list, GLboolean size_in_attribs)
538 {
539 struct glx_display *priv = __glXInitialize(dpy);
540 GLXDrawable id = 0;
541 CARD32 *data;
542 CARD8 opcode;
543 unsigned int i;
544 Pixmap pixmap;
545 GLboolean glx_1_3 = GL_FALSE;
546
547 if (priv == NULL)
548 return None;
549
550 i = 0;
551 if (attrib_list) {
552 while (attrib_list[i * 2])
553 i++;
554 }
555
556 opcode = __glXSetupForCommand(dpy);
557 if (!opcode)
558 return None;
559
560 LockDisplay(dpy);
561 id = XAllocID(dpy);
562
563 if ((priv->majorVersion > 1) || (priv->minorVersion >= 3)) {
564 xGLXCreatePbufferReq *req;
565 unsigned int extra = (size_in_attribs) ? 0 : 2;
566
567 glx_1_3 = GL_TRUE;
568
569 GetReqExtra(GLXCreatePbuffer, (8 * (i + extra)), req);
570 data = (CARD32 *) (req + 1);
571
572 req->reqType = opcode;
573 req->glxCode = X_GLXCreatePbuffer;
574 req->screen = config->screen;
575 req->fbconfig = config->fbconfigID;
576 req->pbuffer = id;
577 req->numAttribs = i + extra;
578
579 if (!size_in_attribs) {
580 data[(2 * i) + 0] = GLX_PBUFFER_WIDTH;
581 data[(2 * i) + 1] = width;
582 data[(2 * i) + 2] = GLX_PBUFFER_HEIGHT;
583 data[(2 * i) + 3] = height;
584 data += 4;
585 }
586 }
587 else {
588 xGLXVendorPrivateReq *vpreq;
589
590 GetReqExtra(GLXVendorPrivate, 20 + (8 * i), vpreq);
591 data = (CARD32 *) (vpreq + 1);
592
593 vpreq->reqType = opcode;
594 vpreq->glxCode = X_GLXVendorPrivate;
595 vpreq->vendorCode = X_GLXvop_CreateGLXPbufferSGIX;
596
597 data[0] = config->screen;
598 data[1] = config->fbconfigID;
599 data[2] = id;
600 data[3] = width;
601 data[4] = height;
602 data += 5;
603 }
604
605 (void) memcpy(data, attrib_list, sizeof(CARD32) * 2 * i);
606
607 UnlockDisplay(dpy);
608 SyncHandle();
609
610 pixmap = XCreatePixmap(dpy, RootWindow(dpy, config->screen),
611 width, height, config->rgbBits);
612
613 if (!CreateDRIDrawable(dpy, config, pixmap, id, attrib_list, i)) {
614 CARD32 o = glx_1_3 ? X_GLXDestroyPbuffer : X_GLXvop_DestroyGLXPbufferSGIX;
615 XFreePixmap(dpy, pixmap);
616 protocolDestroyDrawable(dpy, id, o);
617 id = None;
618 }
619
620 return id;
621 }
622
623 /**
624 * Destroy a pbuffer.
625 *
626 * This function is used to implement \c glXDestroyPbuffer and
627 * \c glXDestroyGLXPbufferSGIX.
628 *
629 * \note
630 * This function dynamically determines whether to use the SGIX_pbuffer
631 * version of the protocol or the GLX 1.3 version of the protocol.
632 */
633 static void
DestroyPbuffer(Display * dpy,GLXDrawable drawable)634 DestroyPbuffer(Display * dpy, GLXDrawable drawable)
635 {
636 struct glx_display *priv = __glXInitialize(dpy);
637 CARD8 opcode;
638
639 if ((priv == NULL) || (dpy == NULL) || (drawable == 0)) {
640 return;
641 }
642
643 opcode = __glXSetupForCommand(dpy);
644 if (!opcode)
645 return;
646
647 LockDisplay(dpy);
648
649 if ((priv->majorVersion > 1) || (priv->minorVersion >= 3)) {
650 xGLXDestroyPbufferReq *req;
651
652 GetReq(GLXDestroyPbuffer, req);
653 req->reqType = opcode;
654 req->glxCode = X_GLXDestroyPbuffer;
655 req->pbuffer = (GLXPbuffer) drawable;
656 }
657 else {
658 xGLXVendorPrivateWithReplyReq *vpreq;
659 CARD32 *data;
660
661 GetReqExtra(GLXVendorPrivateWithReply, 4, vpreq);
662 data = (CARD32 *) (vpreq + 1);
663
664 data[0] = (CARD32) drawable;
665
666 vpreq->reqType = opcode;
667 vpreq->glxCode = X_GLXVendorPrivateWithReply;
668 vpreq->vendorCode = X_GLXvop_DestroyGLXPbufferSGIX;
669 }
670
671 UnlockDisplay(dpy);
672 SyncHandle();
673
674 DestroyDRIDrawable(dpy, drawable, GL_TRUE);
675
676 return;
677 }
678
679 /**
680 * Create a new pbuffer.
681 */
682 _GLX_PUBLIC GLXPbufferSGIX
glXCreateGLXPbufferSGIX(Display * dpy,GLXFBConfigSGIX config,unsigned int width,unsigned int height,int * attrib_list)683 glXCreateGLXPbufferSGIX(Display * dpy, GLXFBConfigSGIX config,
684 unsigned int width, unsigned int height,
685 int *attrib_list)
686 {
687 return (GLXPbufferSGIX) CreatePbuffer(dpy, (struct glx_config *) config,
688 width, height,
689 attrib_list, GL_FALSE);
690 }
691
692 #endif /* GLX_USE_APPLEGL */
693
694 /**
695 * Create a new pbuffer.
696 */
697 _GLX_PUBLIC GLXPbuffer
glXCreatePbuffer(Display * dpy,GLXFBConfig config,const int * attrib_list)698 glXCreatePbuffer(Display * dpy, GLXFBConfig config, const int *attrib_list)
699 {
700 int i, width, height;
701 #ifdef GLX_USE_APPLEGL
702 GLXPbuffer result;
703 int errorcode;
704 #endif
705
706 width = 0;
707 height = 0;
708
709 WARN_ONCE_GLX_1_3(dpy, __func__);
710
711 #ifdef GLX_USE_APPLEGL
712 for (i = 0; attrib_list[i]; ++i) {
713 switch (attrib_list[i]) {
714 case GLX_PBUFFER_WIDTH:
715 width = attrib_list[i + 1];
716 ++i;
717 break;
718
719 case GLX_PBUFFER_HEIGHT:
720 height = attrib_list[i + 1];
721 ++i;
722 break;
723
724 case GLX_LARGEST_PBUFFER:
725 /* This is a hint we should probably handle, but how? */
726 ++i;
727 break;
728
729 case GLX_PRESERVED_CONTENTS:
730 /* The contents are always preserved with AppleSGLX with CGL. */
731 ++i;
732 break;
733
734 default:
735 return None;
736 }
737 }
738
739 if (apple_glx_pbuffer_create(dpy, config, width, height, &errorcode,
740 &result)) {
741 /*
742 * apple_glx_pbuffer_create only sets the errorcode to core X11
743 * errors.
744 */
745 __glXSendError(dpy, errorcode, 0, X_GLXCreatePbuffer, true);
746
747 return None;
748 }
749
750 return result;
751 #else
752 for (i = 0; attrib_list[i * 2]; i++) {
753 switch (attrib_list[i * 2]) {
754 case GLX_PBUFFER_WIDTH:
755 width = attrib_list[i * 2 + 1];
756 break;
757 case GLX_PBUFFER_HEIGHT:
758 height = attrib_list[i * 2 + 1];
759 break;
760 }
761 }
762
763 return (GLXPbuffer) CreatePbuffer(dpy, (struct glx_config *) config,
764 width, height, attrib_list, GL_TRUE);
765 #endif
766 }
767
768
769 /**
770 * Destroy an existing pbuffer.
771 */
772 _GLX_PUBLIC void
glXDestroyPbuffer(Display * dpy,GLXPbuffer pbuf)773 glXDestroyPbuffer(Display * dpy, GLXPbuffer pbuf)
774 {
775 #ifdef GLX_USE_APPLEGL
776 if (apple_glx_pbuffer_destroy(dpy, pbuf)) {
777 __glXSendError(dpy, GLXBadPbuffer, pbuf, X_GLXDestroyPbuffer, false);
778 }
779 #else
780 DestroyPbuffer(dpy, pbuf);
781 #endif
782 }
783
784
785 /**
786 * Query an attribute of a drawable.
787 */
788 _GLX_PUBLIC void
glXQueryDrawable(Display * dpy,GLXDrawable drawable,int attribute,unsigned int * value)789 glXQueryDrawable(Display * dpy, GLXDrawable drawable,
790 int attribute, unsigned int *value)
791 {
792 WARN_ONCE_GLX_1_3(dpy, __func__);
793 #ifdef GLX_USE_APPLEGL
794 Window root;
795 int x, y;
796 unsigned int width, height, bd, depth;
797
798 if (apple_glx_pixmap_query(drawable, attribute, value))
799 return; /*done */
800
801 if (apple_glx_pbuffer_query(drawable, attribute, value))
802 return; /*done */
803
804 /*
805 * The OpenGL spec states that we should report GLXBadDrawable if
806 * the drawable is invalid, however doing so would require that we
807 * use XSetErrorHandler(), which is known to not be thread safe.
808 * If we use a round-trip call to validate the drawable, there could
809 * be a race, so instead we just opt in favor of letting the
810 * XGetGeometry request fail with a GetGeometry request X error
811 * rather than GLXBadDrawable, in what is hoped to be a rare
812 * case of an invalid drawable. In practice most and possibly all
813 * X11 apps using GLX shouldn't notice a difference.
814 */
815 if (XGetGeometry
816 (dpy, drawable, &root, &x, &y, &width, &height, &bd, &depth)) {
817 switch (attribute) {
818 case GLX_WIDTH:
819 *value = width;
820 break;
821
822 case GLX_HEIGHT:
823 *value = height;
824 break;
825 }
826 }
827 #else
828 __glXGetDrawableAttribute(dpy, drawable, attribute, value);
829 #endif
830 }
831
832
833 #ifndef GLX_USE_APPLEGL
834 /**
835 * Query an attribute of a pbuffer.
836 */
837 _GLX_PUBLIC int
glXQueryGLXPbufferSGIX(Display * dpy,GLXPbufferSGIX drawable,int attribute,unsigned int * value)838 glXQueryGLXPbufferSGIX(Display * dpy, GLXPbufferSGIX drawable,
839 int attribute, unsigned int *value)
840 {
841 return __glXGetDrawableAttribute(dpy, drawable, attribute, value);
842 }
843 #endif
844
845 /**
846 * Select the event mask for a drawable.
847 */
848 _GLX_PUBLIC void
glXSelectEvent(Display * dpy,GLXDrawable drawable,unsigned long mask)849 glXSelectEvent(Display * dpy, GLXDrawable drawable, unsigned long mask)
850 {
851 #ifdef GLX_USE_APPLEGL
852 XWindowAttributes xwattr;
853
854 if (apple_glx_pbuffer_set_event_mask(drawable, mask))
855 return; /*done */
856
857 /*
858 * The spec allows a window, but currently there are no valid
859 * events for a window, so do nothing.
860 */
861 if (XGetWindowAttributes(dpy, drawable, &xwattr))
862 return; /*done */
863 /* The drawable seems to be invalid. Report an error. */
864
865 __glXSendError(dpy, GLXBadDrawable, drawable,
866 X_GLXChangeDrawableAttributes, false);
867 #else
868 CARD32 attribs[2];
869
870 attribs[0] = (CARD32) GLX_EVENT_MASK;
871 attribs[1] = (CARD32) mask;
872
873 ChangeDrawableAttribute(dpy, drawable, attribs, 1);
874 #endif
875 }
876
877
878 /**
879 * Get the selected event mask for a drawable.
880 */
881 _GLX_PUBLIC void
glXGetSelectedEvent(Display * dpy,GLXDrawable drawable,unsigned long * mask)882 glXGetSelectedEvent(Display * dpy, GLXDrawable drawable, unsigned long *mask)
883 {
884 #ifdef GLX_USE_APPLEGL
885 XWindowAttributes xwattr;
886
887 if (apple_glx_pbuffer_get_event_mask(drawable, mask))
888 return; /*done */
889
890 /*
891 * The spec allows a window, but currently there are no valid
892 * events for a window, so do nothing, but set the mask to 0.
893 */
894 if (XGetWindowAttributes(dpy, drawable, &xwattr)) {
895 /* The window is valid, so set the mask to 0. */
896 *mask = 0;
897 return; /*done */
898 }
899 /* The drawable seems to be invalid. Report an error. */
900
901 __glXSendError(dpy, GLXBadDrawable, drawable, X_GLXGetDrawableAttributes,
902 true);
903 #else
904 unsigned int value = 0;
905
906
907 /* The non-sense with value is required because on LP64 platforms
908 * sizeof(unsigned int) != sizeof(unsigned long). On little-endian
909 * we could just type-cast the pointer, but why?
910 */
911
912 __glXGetDrawableAttribute(dpy, drawable, GLX_EVENT_MASK_SGIX, &value);
913 *mask = value;
914 #endif
915 }
916
917
918 _GLX_PUBLIC GLXPixmap
glXCreatePixmap(Display * dpy,GLXFBConfig config,Pixmap pixmap,const int * attrib_list)919 glXCreatePixmap(Display * dpy, GLXFBConfig config, Pixmap pixmap,
920 const int *attrib_list)
921 {
922 WARN_ONCE_GLX_1_3(dpy, __func__);
923
924 #ifdef GLX_USE_APPLEGL
925 const struct glx_config *modes = (const struct glx_config *) config;
926
927 if (apple_glx_pixmap_create(dpy, modes->screen, pixmap, modes))
928 return None;
929
930 return pixmap;
931 #else
932 return CreateDrawable(dpy, (struct glx_config *) config,
933 (Drawable) pixmap, attrib_list, X_GLXCreatePixmap);
934 #endif
935 }
936
937
938 _GLX_PUBLIC GLXWindow
glXCreateWindow(Display * dpy,GLXFBConfig config,Window win,const int * attrib_list)939 glXCreateWindow(Display * dpy, GLXFBConfig config, Window win,
940 const int *attrib_list)
941 {
942 WARN_ONCE_GLX_1_3(dpy, __func__);
943 #ifdef GLX_USE_APPLEGL
944 XWindowAttributes xwattr;
945 XVisualInfo *visinfo;
946
947 (void) attrib_list; /*unused according to GLX 1.4 */
948
949 XGetWindowAttributes(dpy, win, &xwattr);
950
951 visinfo = glXGetVisualFromFBConfig(dpy, config);
952
953 if (NULL == visinfo) {
954 __glXSendError(dpy, GLXBadFBConfig, 0, X_GLXCreateWindow, false);
955 return None;
956 }
957
958 if (visinfo->visualid != XVisualIDFromVisual(xwattr.visual)) {
959 __glXSendError(dpy, BadMatch, 0, X_GLXCreateWindow, true);
960 return None;
961 }
962
963 free(visinfo);
964
965 return win;
966 #else
967 return CreateDrawable(dpy, (struct glx_config *) config,
968 (Drawable) win, attrib_list, X_GLXCreateWindow);
969 #endif
970 }
971
972
973 _GLX_PUBLIC void
glXDestroyPixmap(Display * dpy,GLXPixmap pixmap)974 glXDestroyPixmap(Display * dpy, GLXPixmap pixmap)
975 {
976 WARN_ONCE_GLX_1_3(dpy, __func__);
977 #ifdef GLX_USE_APPLEGL
978 if (apple_glx_pixmap_destroy(dpy, pixmap))
979 __glXSendError(dpy, GLXBadPixmap, pixmap, X_GLXDestroyPixmap, false);
980 #else
981 DestroyDrawable(dpy, (GLXDrawable) pixmap, X_GLXDestroyPixmap);
982 #endif
983 }
984
985
986 _GLX_PUBLIC void
glXDestroyWindow(Display * dpy,GLXWindow win)987 glXDestroyWindow(Display * dpy, GLXWindow win)
988 {
989 WARN_ONCE_GLX_1_3(dpy, __func__);
990 #ifndef GLX_USE_APPLEGL
991 DestroyDrawable(dpy, (GLXDrawable) win, X_GLXDestroyWindow);
992 #endif
993 }
994
995 _GLX_PUBLIC
996 GLX_ALIAS_VOID(glXDestroyGLXPbufferSGIX,
997 (Display * dpy, GLXPbufferSGIX pbuf),
998 (dpy, pbuf), glXDestroyPbuffer)
999
1000 _GLX_PUBLIC
1001 GLX_ALIAS_VOID(glXSelectEventSGIX,
1002 (Display * dpy, GLXDrawable drawable,
1003 unsigned long mask), (dpy, drawable, mask), glXSelectEvent)
1004
1005 _GLX_PUBLIC
1006 GLX_ALIAS_VOID(glXGetSelectedEventSGIX,
1007 (Display * dpy, GLXDrawable drawable,
1008 unsigned long *mask), (dpy, drawable, mask),
1009 glXGetSelectedEvent)
1010