| /* |
| * Copyright © 2011 Intel Corporation |
| * |
| * Permission is hereby granted, free of charge, to any person obtaining a |
| * copy of this software and associated documentation files (the "Software"), |
| * to deal in the Software without restriction, including without limitation |
| * the rights to use, copy, modify, merge, publish, distribute, sublicense, |
| * and/or sell copies of the Software, and to permit persons to whom the |
| * Software is furnished to do so, subject to the following conditions: |
| * |
| * The above copyright notice and this permission notice (including the next |
| * paragraph) shall be included in all copies or substantial portions of the |
| * Software. |
| * |
| * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
| * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
| * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL |
| * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
| * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING |
| * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER |
| * DEALINGS IN THE SOFTWARE. |
| */ |
| |
| #include <string.h> |
| #include <ctype.h> |
| |
| #include "glxclient.h" |
| #include <xcb/glx.h> |
| #include <X11/Xlib-xcb.h> |
| |
| _X_HIDDEN void |
| __glX_send_client_info(struct glx_display *glx_dpy) |
| { |
| const unsigned ext_length = strlen("GLX_ARB_create_context"); |
| const unsigned prof_length = strlen("_profile"); |
| char *gl_extension_string; |
| int gl_extension_length; |
| xcb_connection_t *c; |
| Bool any_screen_has_ARB_create_context = False; |
| Bool any_screen_has_ARB_create_context_profile = False; |
| unsigned i; |
| /* You need GLX_ARB_create_context_profile to get beyond 3.1 anyway */ |
| static const uint32_t gl_versions[] = { |
| 2, 1, |
| 3, 0, |
| 3, 1, |
| }; |
| /* |
| * This is weird, but it matches what NVIDIA does/expects. For big-GL |
| * below 3.2 there is no such thing as a "profile", so we name them all |
| * with no profile bits. Except we don't name anything lower than 2.1, |
| * since GLX_ARB_create_context_profile says: |
| * |
| * "Only the highest supported version below 3.0 should be sent, since |
| * OpenGL 2.1 is backwards compatible with all earlier versions." |
| * |
| * In order to also support GLES below 3.2, we name every possible GLES |
| * version with the ES2 bit set, which happens to just mean GLES generally |
| * and not a particular major version. 3.2 happens to be a legal version |
| * number for both big-GL and GLES, so it gets all three bits set. |
| * Everything 3.3 and above is big-GL only so gets the core and compat |
| * bits set. |
| */ |
| static const uint32_t gl_versions_profiles[] = { |
| 1, 0, GLX_CONTEXT_ES2_PROFILE_BIT_EXT, |
| 1, 1, GLX_CONTEXT_ES2_PROFILE_BIT_EXT, |
| 2, 0, GLX_CONTEXT_ES2_PROFILE_BIT_EXT, |
| 2, 1, 0x0, |
| 3, 0, 0x0, |
| 3, 0, GLX_CONTEXT_ES2_PROFILE_BIT_EXT, |
| 3, 1, 0x0, |
| 3, 1, GLX_CONTEXT_ES2_PROFILE_BIT_EXT, |
| 3, 2, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB | |
| GLX_CONTEXT_ES2_PROFILE_BIT_EXT, |
| 3, 3, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, |
| 4, 0, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, |
| 4, 1, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, |
| 4, 2, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, |
| 4, 3, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, |
| 4, 4, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, |
| 4, 5, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, |
| 4, 6, GLX_CONTEXT_CORE_PROFILE_BIT_ARB | |
| GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB, |
| }; |
| static const char glx_extensions[] = |
| "GLX_ARB_create_context GLX_ARB_create_context_profile"; |
| |
| /* There are three possible flavors of the client info structure that the |
| * client could send to the server. The version sent depends on the |
| * combination of GLX versions and extensions supported by the client and |
| * the server. |
| * |
| * Server supports Client sends |
| * ---------------------------------------------------------------------- |
| * GLX version = 1.0 Nothing. |
| * |
| * GLX version >= 1.1 struct GLXClientInfo |
| * |
| * GLX version >= 1.4 and |
| * GLX_ARB_create_context struct glXSetClientInfoARB |
| * |
| * GLX version >= 1.4 and |
| * GLX_ARB_create_context_profile struct glXSetClientInfo2ARB |
| * |
| * GLX_ARB_create_context and GLX_ARB_create_context_profile use FBConfigs, |
| * and these only exist in GLX 1.4 or with GLX_SGIX_fbconfig. I can't |
| * imagine an implementation that supports GLX_SGIX_fbconfig and |
| * GLX_ARB_create_context but not GLX 1.4. Making GLX 1.4 a hard |
| * requirement in this case does not seem like a limitation. |
| */ |
| |
| if (glx_dpy->majorVersion == 1 && glx_dpy->minorVersion == 0) |
| return; |
| |
| /* Determine whether any screen on the server supports either of the |
| * create-context extensions. |
| */ |
| for (i = 0; i < ScreenCount(glx_dpy->dpy); i++) { |
| struct glx_screen *src = glx_dpy->screens[i]; |
| |
| const char *haystack = src->serverGLXexts; |
| while (haystack != NULL) { |
| char *match = strstr(haystack, "GLX_ARB_create_context"); |
| |
| if (match == NULL) |
| break; |
| |
| match += ext_length; |
| |
| switch (match[0]) { |
| case '\0': |
| case ' ': |
| any_screen_has_ARB_create_context = True; |
| break; |
| |
| case '_': |
| if (strncmp(match, "_profile", prof_length) == 0 |
| && (match[prof_length] == '\0' |
| || match[prof_length] == ' ')) { |
| any_screen_has_ARB_create_context_profile = True; |
| match += prof_length; |
| } |
| break; |
| } |
| |
| haystack = match; |
| } |
| } |
| |
| gl_extension_string = __glXGetClientGLExtensionString(); |
| if (gl_extension_string == NULL) { |
| return; |
| } |
| |
| gl_extension_length = strlen(gl_extension_string) + 1; |
| |
| c = XGetXCBConnection(glx_dpy->dpy); |
| |
| /* Depending on the GLX verion and the available extensions on the server, |
| * send the correct "flavor" of protocol to the server. |
| * |
| * THE ORDER IS IMPORTANT. We want to send the most recent version of the |
| * protocol that the server can support. |
| */ |
| if (glx_dpy->majorVersion == 1 && glx_dpy->minorVersion == 4 |
| && any_screen_has_ARB_create_context_profile) { |
| xcb_glx_set_client_info_2arb(c, |
| GLX_MAJOR_VERSION, GLX_MINOR_VERSION, |
| sizeof(gl_versions_profiles) |
| / (3 * sizeof(gl_versions_profiles[0])), |
| gl_extension_length, |
| strlen(glx_extensions) + 1, |
| gl_versions_profiles, |
| gl_extension_string, |
| glx_extensions); |
| } else if (glx_dpy->majorVersion == 1 && glx_dpy->minorVersion == 4 |
| && any_screen_has_ARB_create_context) { |
| xcb_glx_set_client_info_arb(c, |
| GLX_MAJOR_VERSION, GLX_MINOR_VERSION, |
| sizeof(gl_versions) |
| / (2 * sizeof(gl_versions[0])), |
| gl_extension_length, |
| strlen(glx_extensions) + 1, |
| gl_versions, |
| gl_extension_string, |
| glx_extensions); |
| } else { |
| xcb_glx_client_info(c, |
| GLX_MAJOR_VERSION, GLX_MINOR_VERSION, |
| gl_extension_length, |
| gl_extension_string); |
| } |
| |
| free(gl_extension_string); |
| } |