1*706d0b42SXin Li /*
2*706d0b42SXin Li * Copyright © 2013 Intel Corporation
3*706d0b42SXin Li *
4*706d0b42SXin Li * Permission is hereby granted, free of charge, to any person obtaining a
5*706d0b42SXin Li * copy of this software and associated documentation files (the "Software"),
6*706d0b42SXin Li * to deal in the Software without restriction, including without limitation
7*706d0b42SXin Li * the rights to use, copy, modify, merge, publish, distribute, sublicense,
8*706d0b42SXin Li * and/or sell copies of the Software, and to permit persons to whom the
9*706d0b42SXin Li * Software is furnished to do so, subject to the following conditions:
10*706d0b42SXin Li *
11*706d0b42SXin Li * The above copyright notice and this permission notice (including the next
12*706d0b42SXin Li * paragraph) shall be included in all copies or substantial portions of the
13*706d0b42SXin Li * Software.
14*706d0b42SXin Li *
15*706d0b42SXin Li * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16*706d0b42SXin Li * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
17*706d0b42SXin Li * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
18*706d0b42SXin Li * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
19*706d0b42SXin Li * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
20*706d0b42SXin Li * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS
21*706d0b42SXin Li * IN THE SOFTWARE.
22*706d0b42SXin Li */
23*706d0b42SXin Li
24*706d0b42SXin Li #include <assert.h>
25*706d0b42SXin Li #include <string.h>
26*706d0b42SXin Li #include <stdio.h>
27*706d0b42SXin Li
28*706d0b42SXin Li #include "dispatch_common.h"
29*706d0b42SXin Li
30*706d0b42SXin Li /**
31*706d0b42SXin Li * If we can determine the GLX version from the current context, then
32*706d0b42SXin Li * return that, otherwise return a version that will just send us on
33*706d0b42SXin Li * to dlsym() or get_proc_address().
34*706d0b42SXin Li */
35*706d0b42SXin Li int
epoxy_conservative_glx_version(void)36*706d0b42SXin Li epoxy_conservative_glx_version(void)
37*706d0b42SXin Li {
38*706d0b42SXin Li Display *dpy = glXGetCurrentDisplay();
39*706d0b42SXin Li GLXContext ctx = glXGetCurrentContext();
40*706d0b42SXin Li int screen;
41*706d0b42SXin Li
42*706d0b42SXin Li if (!dpy || !ctx)
43*706d0b42SXin Li return 14;
44*706d0b42SXin Li
45*706d0b42SXin Li glXQueryContext(dpy, ctx, GLX_SCREEN, &screen);
46*706d0b42SXin Li
47*706d0b42SXin Li return epoxy_glx_version(dpy, screen);
48*706d0b42SXin Li }
49*706d0b42SXin Li
50*706d0b42SXin Li
51*706d0b42SXin Li /**
52*706d0b42SXin Li * @brief Returns the version of GLX we are using
53*706d0b42SXin Li *
54*706d0b42SXin Li * The version is encoded as:
55*706d0b42SXin Li *
56*706d0b42SXin Li * ```
57*706d0b42SXin Li *
58*706d0b42SXin Li * version = major * 10 + minor
59*706d0b42SXin Li *
60*706d0b42SXin Li * ```
61*706d0b42SXin Li *
62*706d0b42SXin Li * So it can be easily used for version comparisons.
63*706d0b42SXin Li *
64*706d0b42SXin Li * @param dpy The X11 display
65*706d0b42SXin Li * @param screen The X11 screen
66*706d0b42SXin Li *
67*706d0b42SXin Li * @return The encoded version of GLX we are using
68*706d0b42SXin Li *
69*706d0b42SXin Li * @see epoxy_gl_version()
70*706d0b42SXin Li */
71*706d0b42SXin Li int
epoxy_glx_version(Display * dpy,int screen)72*706d0b42SXin Li epoxy_glx_version(Display *dpy, int screen)
73*706d0b42SXin Li {
74*706d0b42SXin Li int server_major, server_minor;
75*706d0b42SXin Li int client_major, client_minor;
76*706d0b42SXin Li int server, client;
77*706d0b42SXin Li const char *version_string;
78*706d0b42SXin Li int ret;
79*706d0b42SXin Li
80*706d0b42SXin Li version_string = glXQueryServerString(dpy, screen, GLX_VERSION);
81*706d0b42SXin Li if (!version_string)
82*706d0b42SXin Li return 0;
83*706d0b42SXin Li
84*706d0b42SXin Li ret = sscanf(version_string, "%d.%d", &server_major, &server_minor);
85*706d0b42SXin Li assert(ret == 2);
86*706d0b42SXin Li server = server_major * 10 + server_minor;
87*706d0b42SXin Li
88*706d0b42SXin Li version_string = glXGetClientString(dpy, GLX_VERSION);
89*706d0b42SXin Li if (!version_string)
90*706d0b42SXin Li return 0;
91*706d0b42SXin Li
92*706d0b42SXin Li ret = sscanf(version_string, "%d.%d", &client_major, &client_minor);
93*706d0b42SXin Li assert(ret == 2);
94*706d0b42SXin Li client = client_major * 10 + client_minor;
95*706d0b42SXin Li
96*706d0b42SXin Li if (client < server)
97*706d0b42SXin Li return client;
98*706d0b42SXin Li else
99*706d0b42SXin Li return server;
100*706d0b42SXin Li }
101*706d0b42SXin Li
102*706d0b42SXin Li /**
103*706d0b42SXin Li * If we can determine the GLX extension support from the current
104*706d0b42SXin Li * context, then return that, otherwise give the answer that will just
105*706d0b42SXin Li * send us on to get_proc_address().
106*706d0b42SXin Li */
107*706d0b42SXin Li bool
epoxy_conservative_has_glx_extension(const char * ext)108*706d0b42SXin Li epoxy_conservative_has_glx_extension(const char *ext)
109*706d0b42SXin Li {
110*706d0b42SXin Li Display *dpy = glXGetCurrentDisplay();
111*706d0b42SXin Li GLXContext ctx = glXGetCurrentContext();
112*706d0b42SXin Li int screen;
113*706d0b42SXin Li
114*706d0b42SXin Li if (!dpy || !ctx)
115*706d0b42SXin Li return true;
116*706d0b42SXin Li
117*706d0b42SXin Li glXQueryContext(dpy, ctx, GLX_SCREEN, &screen);
118*706d0b42SXin Li
119*706d0b42SXin Li return epoxy_has_glx_extension(dpy, screen, ext);
120*706d0b42SXin Li }
121*706d0b42SXin Li
122*706d0b42SXin Li /**
123*706d0b42SXin Li * @brief Returns true if the given GLX extension is supported in the current context.
124*706d0b42SXin Li *
125*706d0b42SXin Li * @param dpy The X11 display
126*706d0b42SXin Li * @param screen The X11 screen
127*706d0b42SXin Li * @param extension The name of the GLX extension
128*706d0b42SXin Li *
129*706d0b42SXin Li * @return `true` if the extension is available
130*706d0b42SXin Li *
131*706d0b42SXin Li * @see epoxy_has_gl_extension()
132*706d0b42SXin Li * @see epoxy_has_egl_extension()
133*706d0b42SXin Li */
134*706d0b42SXin Li bool
epoxy_has_glx_extension(Display * dpy,int screen,const char * ext)135*706d0b42SXin Li epoxy_has_glx_extension(Display *dpy, int screen, const char *ext)
136*706d0b42SXin Li {
137*706d0b42SXin Li /* No, you can't just use glXGetClientString or
138*706d0b42SXin Li * glXGetServerString() here. Those each tell you about one half
139*706d0b42SXin Li * of what's needed for an extension to be supported, and
140*706d0b42SXin Li * glXQueryExtensionsString() is what gives you the intersection
141*706d0b42SXin Li * of the two.
142*706d0b42SXin Li */
143*706d0b42SXin Li return epoxy_extension_in_string(glXQueryExtensionsString(dpy, screen), ext);
144*706d0b42SXin Li }
145*706d0b42SXin Li
146*706d0b42SXin Li /**
147*706d0b42SXin Li * @brief Checks whether GLX is available.
148*706d0b42SXin Li *
149*706d0b42SXin Li * @param dpy The X11 display
150*706d0b42SXin Li *
151*706d0b42SXin Li * @return `true` if GLX is available
152*706d0b42SXin Li *
153*706d0b42SXin Li * @newin{1,4}
154*706d0b42SXin Li */
155*706d0b42SXin Li bool
epoxy_has_glx(Display * dpy)156*706d0b42SXin Li epoxy_has_glx(Display *dpy)
157*706d0b42SXin Li {
158*706d0b42SXin Li #if !PLATFORM_HAS_GLX
159*706d0b42SXin Li return false;
160*706d0b42SXin Li #else
161*706d0b42SXin Li if (epoxy_load_glx(false, true)) {
162*706d0b42SXin Li Bool (* pf_glXQueryExtension) (Display *, int *, int *);
163*706d0b42SXin Li int error_base, event_base;
164*706d0b42SXin Li
165*706d0b42SXin Li pf_glXQueryExtension = epoxy_conservative_glx_dlsym("glXQueryExtension", false);
166*706d0b42SXin Li if (pf_glXQueryExtension && pf_glXQueryExtension(dpy, &error_base, &event_base))
167*706d0b42SXin Li return true;
168*706d0b42SXin Li }
169*706d0b42SXin Li
170*706d0b42SXin Li return false;
171*706d0b42SXin Li #endif /* !PLATFORM_HAS_GLX */
172*706d0b42SXin Li }
173