1*61046927SAndroid Build Coastguard Worker /*
2*61046927SAndroid Build Coastguard Worker * Copyright © 2011 Intel Corporation
3*61046927SAndroid Build Coastguard Worker *
4*61046927SAndroid Build Coastguard Worker * Permission is hereby granted, free of charge, to any person obtaining a
5*61046927SAndroid Build Coastguard Worker * copy of this software and associated documentation files (the "Software"),
6*61046927SAndroid Build Coastguard Worker * to deal in the Software without restriction, including without limitation
7*61046927SAndroid Build Coastguard Worker * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*61046927SAndroid Build Coastguard Worker * and/or sell copies of the Software, and to permit persons to whom the
9*61046927SAndroid Build Coastguard Worker * Software is furnished to do so, subject to the following conditions:
10*61046927SAndroid Build Coastguard Worker *
11*61046927SAndroid Build Coastguard Worker * The above copyright notice and this permission notice (including the next
12*61046927SAndroid Build Coastguard Worker * paragraph) shall be included in all copies or substantial portions of the
13*61046927SAndroid Build Coastguard Worker * Software.
14*61046927SAndroid Build Coastguard Worker *
15*61046927SAndroid Build Coastguard Worker * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*61046927SAndroid Build Coastguard Worker * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*61046927SAndroid Build Coastguard Worker * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18*61046927SAndroid Build Coastguard Worker * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*61046927SAndroid Build Coastguard Worker * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*61046927SAndroid Build Coastguard Worker * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
21*61046927SAndroid Build Coastguard Worker * DEALINGS IN THE SOFTWARE.
22*61046927SAndroid Build Coastguard Worker */
23*61046927SAndroid Build Coastguard Worker
24*61046927SAndroid Build Coastguard Worker #include <limits.h>
25*61046927SAndroid Build Coastguard Worker #include "glxclient.h"
26*61046927SAndroid Build Coastguard Worker #include "glx_error.h"
27*61046927SAndroid Build Coastguard Worker #include <xcb/glx.h>
28*61046927SAndroid Build Coastguard Worker #include <X11/Xlib-xcb.h>
29*61046927SAndroid Build Coastguard Worker
30*61046927SAndroid Build Coastguard Worker #include <assert.h>
31*61046927SAndroid Build Coastguard Worker
32*61046927SAndroid Build Coastguard Worker #if INT_MAX != 2147483647
33*61046927SAndroid Build Coastguard Worker #error This code requires sizeof(uint32_t) == sizeof(int).
34*61046927SAndroid Build Coastguard Worker #endif
35*61046927SAndroid Build Coastguard Worker
36*61046927SAndroid Build Coastguard Worker /* An "Atrribs/Attribs" typo was fixed in glxproto.h in Nov 2014.
37*61046927SAndroid Build Coastguard Worker * This is in case we don't have the updated header.
38*61046927SAndroid Build Coastguard Worker */
39*61046927SAndroid Build Coastguard Worker #if !defined(X_GLXCreateContextAttribsARB) && \
40*61046927SAndroid Build Coastguard Worker defined(X_GLXCreateContextAtrribsARB)
41*61046927SAndroid Build Coastguard Worker #define X_GLXCreateContextAttribsARB X_GLXCreateContextAtrribsARB
42*61046927SAndroid Build Coastguard Worker #endif
43*61046927SAndroid Build Coastguard Worker
44*61046927SAndroid Build Coastguard Worker _X_HIDDEN GLXContext
glXCreateContextAttribsARB(Display * dpy,GLXFBConfig config,GLXContext share_context,Bool direct,const int * attrib_list)45*61046927SAndroid Build Coastguard Worker glXCreateContextAttribsARB(Display *dpy, GLXFBConfig config,
46*61046927SAndroid Build Coastguard Worker GLXContext share_context, Bool direct,
47*61046927SAndroid Build Coastguard Worker const int *attrib_list)
48*61046927SAndroid Build Coastguard Worker {
49*61046927SAndroid Build Coastguard Worker xcb_connection_t *const c = XGetXCBConnection(dpy);
50*61046927SAndroid Build Coastguard Worker struct glx_config *const cfg = (struct glx_config *) config;
51*61046927SAndroid Build Coastguard Worker struct glx_context *const share = (struct glx_context *) share_context;
52*61046927SAndroid Build Coastguard Worker struct glx_context *gc = NULL;
53*61046927SAndroid Build Coastguard Worker unsigned num_attribs = 0;
54*61046927SAndroid Build Coastguard Worker struct glx_screen *psc;
55*61046927SAndroid Build Coastguard Worker xcb_generic_error_t *err;
56*61046927SAndroid Build Coastguard Worker xcb_void_cookie_t cookie;
57*61046927SAndroid Build Coastguard Worker unsigned error = BadImplementation;
58*61046927SAndroid Build Coastguard Worker uint32_t xid, share_xid;
59*61046927SAndroid Build Coastguard Worker int screen = -1;
60*61046927SAndroid Build Coastguard Worker
61*61046927SAndroid Build Coastguard Worker if (dpy == NULL)
62*61046927SAndroid Build Coastguard Worker return NULL;
63*61046927SAndroid Build Coastguard Worker
64*61046927SAndroid Build Coastguard Worker /* Count the number of attributes specified by the application. All
65*61046927SAndroid Build Coastguard Worker * attributes appear in pairs, except the terminating None.
66*61046927SAndroid Build Coastguard Worker */
67*61046927SAndroid Build Coastguard Worker if (attrib_list != NULL) {
68*61046927SAndroid Build Coastguard Worker for (/* empty */; attrib_list[num_attribs * 2] != 0; num_attribs++)
69*61046927SAndroid Build Coastguard Worker /* empty */ ;
70*61046927SAndroid Build Coastguard Worker }
71*61046927SAndroid Build Coastguard Worker
72*61046927SAndroid Build Coastguard Worker if (cfg) {
73*61046927SAndroid Build Coastguard Worker screen = cfg->screen;
74*61046927SAndroid Build Coastguard Worker } else {
75*61046927SAndroid Build Coastguard Worker for (unsigned int i = 0; i < num_attribs; i++) {
76*61046927SAndroid Build Coastguard Worker if (attrib_list[i * 2] == GLX_SCREEN)
77*61046927SAndroid Build Coastguard Worker screen = attrib_list[i * 2 + 1];
78*61046927SAndroid Build Coastguard Worker }
79*61046927SAndroid Build Coastguard Worker if (screen == -1) {
80*61046927SAndroid Build Coastguard Worker __glXSendError(dpy, BadValue, 0, X_GLXCreateContextAttribsARB, True);
81*61046927SAndroid Build Coastguard Worker return NULL;
82*61046927SAndroid Build Coastguard Worker }
83*61046927SAndroid Build Coastguard Worker }
84*61046927SAndroid Build Coastguard Worker
85*61046927SAndroid Build Coastguard Worker /* This means that either the caller passed the wrong display pointer or
86*61046927SAndroid Build Coastguard Worker * one of the internal GLX data structures (probably the fbconfig) has an
87*61046927SAndroid Build Coastguard Worker * error. There is nothing sensible to do, so return an error.
88*61046927SAndroid Build Coastguard Worker */
89*61046927SAndroid Build Coastguard Worker psc = GetGLXScreenConfigs(dpy, screen);
90*61046927SAndroid Build Coastguard Worker if (psc == NULL)
91*61046927SAndroid Build Coastguard Worker return NULL;
92*61046927SAndroid Build Coastguard Worker
93*61046927SAndroid Build Coastguard Worker assert(screen == psc->scr);
94*61046927SAndroid Build Coastguard Worker
95*61046927SAndroid Build Coastguard Worker /* Some application may request an indirect context but we may want to force a direct
96*61046927SAndroid Build Coastguard Worker * one because Xorg only allows indirect contexts if they were enabled.
97*61046927SAndroid Build Coastguard Worker */
98*61046927SAndroid Build Coastguard Worker if (!direct &&
99*61046927SAndroid Build Coastguard Worker psc->force_direct_context) {
100*61046927SAndroid Build Coastguard Worker direct = true;
101*61046927SAndroid Build Coastguard Worker }
102*61046927SAndroid Build Coastguard Worker
103*61046927SAndroid Build Coastguard Worker #ifdef GLX_USE_APPLEGL
104*61046927SAndroid Build Coastguard Worker gc = applegl_create_context(psc, cfg, share, 0);
105*61046927SAndroid Build Coastguard Worker #else
106*61046927SAndroid Build Coastguard Worker if (direct && psc->vtable->create_context_attribs) {
107*61046927SAndroid Build Coastguard Worker gc = psc->vtable->create_context_attribs(psc, cfg, share, num_attribs,
108*61046927SAndroid Build Coastguard Worker (const uint32_t *) attrib_list,
109*61046927SAndroid Build Coastguard Worker &error);
110*61046927SAndroid Build Coastguard Worker } else if (!direct) {
111*61046927SAndroid Build Coastguard Worker gc = indirect_create_context_attribs(psc, cfg, share, num_attribs,
112*61046927SAndroid Build Coastguard Worker (const uint32_t *) attrib_list,
113*61046927SAndroid Build Coastguard Worker &error);
114*61046927SAndroid Build Coastguard Worker }
115*61046927SAndroid Build Coastguard Worker #endif
116*61046927SAndroid Build Coastguard Worker
117*61046927SAndroid Build Coastguard Worker if (gc == NULL) {
118*61046927SAndroid Build Coastguard Worker /* Increment dpy->request in order to give a unique serial number to the error.
119*61046927SAndroid Build Coastguard Worker * This may break creating contexts on some video cards, if libx11 <1.7.4 is used.
120*61046927SAndroid Build Coastguard Worker * However, this fixes creating contexts (on some video cards) if libx11 >=1.7.4 is used.
121*61046927SAndroid Build Coastguard Worker */
122*61046927SAndroid Build Coastguard Worker XNoOp(dpy);
123*61046927SAndroid Build Coastguard Worker /* -1 isn't a legal XID, which is sort of the point, we've failed
124*61046927SAndroid Build Coastguard Worker * before we even got to XID allocation.
125*61046927SAndroid Build Coastguard Worker */
126*61046927SAndroid Build Coastguard Worker if (error == GLXBadContext || error == GLXBadFBConfig ||
127*61046927SAndroid Build Coastguard Worker error == GLXBadProfileARB)
128*61046927SAndroid Build Coastguard Worker __glXSendError(dpy, error, -1, 0, False);
129*61046927SAndroid Build Coastguard Worker else
130*61046927SAndroid Build Coastguard Worker __glXSendError(dpy, error, -1, 0, True);
131*61046927SAndroid Build Coastguard Worker return NULL;
132*61046927SAndroid Build Coastguard Worker }
133*61046927SAndroid Build Coastguard Worker
134*61046927SAndroid Build Coastguard Worker xid = xcb_generate_id(c);
135*61046927SAndroid Build Coastguard Worker share_xid = (share != NULL) ? share->xid : 0;
136*61046927SAndroid Build Coastguard Worker
137*61046927SAndroid Build Coastguard Worker /* The manual pages for glXCreateContext and glXCreateNewContext say:
138*61046927SAndroid Build Coastguard Worker *
139*61046927SAndroid Build Coastguard Worker * "NULL is returned if execution fails on the client side."
140*61046927SAndroid Build Coastguard Worker *
141*61046927SAndroid Build Coastguard Worker * If the server generates an error, the application is supposed to catch
142*61046927SAndroid Build Coastguard Worker * the protocol error and handle it. Part of handling the error is freeing
143*61046927SAndroid Build Coastguard Worker * the possibly non-NULL value returned by this function.
144*61046927SAndroid Build Coastguard Worker */
145*61046927SAndroid Build Coastguard Worker cookie =
146*61046927SAndroid Build Coastguard Worker xcb_glx_create_context_attribs_arb_checked(c,
147*61046927SAndroid Build Coastguard Worker xid,
148*61046927SAndroid Build Coastguard Worker cfg ? cfg->fbconfigID : 0,
149*61046927SAndroid Build Coastguard Worker screen,
150*61046927SAndroid Build Coastguard Worker share_xid,
151*61046927SAndroid Build Coastguard Worker gc->isDirect,
152*61046927SAndroid Build Coastguard Worker num_attribs,
153*61046927SAndroid Build Coastguard Worker (const uint32_t *)
154*61046927SAndroid Build Coastguard Worker attrib_list);
155*61046927SAndroid Build Coastguard Worker err = xcb_request_check(c, cookie);
156*61046927SAndroid Build Coastguard Worker if (err != NULL) {
157*61046927SAndroid Build Coastguard Worker if (gc)
158*61046927SAndroid Build Coastguard Worker gc->vtable->destroy(gc);
159*61046927SAndroid Build Coastguard Worker gc = NULL;
160*61046927SAndroid Build Coastguard Worker
161*61046927SAndroid Build Coastguard Worker __glXSendErrorForXcb(dpy, err);
162*61046927SAndroid Build Coastguard Worker free(err);
163*61046927SAndroid Build Coastguard Worker } else {
164*61046927SAndroid Build Coastguard Worker gc->xid = xid;
165*61046927SAndroid Build Coastguard Worker gc->share_xid = share_xid;
166*61046927SAndroid Build Coastguard Worker }
167*61046927SAndroid Build Coastguard Worker
168*61046927SAndroid Build Coastguard Worker return (GLXContext) gc;
169*61046927SAndroid Build Coastguard Worker }
170