xref: /aosp_15_r20/external/mesa3d/src/glx/create_context.c (revision 6104692788411f58d303aa86923a9ff6ecaded22)
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