src/video/x11/SDL_x11opengl.c
author Sam Lantinga <slouken@libsdl.org>
Tue, 23 Jul 2013 19:20:03 -0700
changeset 7512 462839f2f408
parent 7462 66b536503733
child 7659 ac4ce59c40e7
permissions -rw-r--r--
Fix for recent GLX error bug

Lee Salzman

I messed up in the patch I sent you regarding gobbling up the GLX error codes signaled when trying to create a context. After reading the spec I realized those error codes are relative to a base error that needs to be queried when setting up the GLX extension...

So I have made a patch that fixes it for a user I had who was still getting the bug with my old patch.

Without this patch my previous one won't work, so it is recommended to merge this...
slouken@1952
     1
/*
slouken@5535
     2
  Simple DirectMedia Layer
slouken@6885
     3
  Copyright (C) 1997-2013 Sam Lantinga <slouken@libsdl.org>
slouken@1952
     4
slouken@5535
     5
  This software is provided 'as-is', without any express or implied
slouken@5535
     6
  warranty.  In no event will the authors be held liable for any damages
slouken@5535
     7
  arising from the use of this software.
slouken@1952
     8
slouken@5535
     9
  Permission is granted to anyone to use this software for any purpose,
slouken@5535
    10
  including commercial applications, and to alter it and redistribute it
slouken@5535
    11
  freely, subject to the following restrictions:
slouken@1952
    12
slouken@5535
    13
  1. The origin of this software must not be misrepresented; you must not
slouken@5535
    14
     claim that you wrote the original software. If you use this software
slouken@5535
    15
     in a product, an acknowledgment in the product documentation would be
slouken@5535
    16
     appreciated but is not required.
slouken@5535
    17
  2. Altered source versions must be plainly marked as such, and must not be
slouken@5535
    18
     misrepresented as being the original software.
slouken@5535
    19
  3. This notice may not be removed or altered from any source distribution.
slouken@1952
    20
*/
slouken@1952
    21
#include "SDL_config.h"
slouken@6044
    22
slouken@6044
    23
#if SDL_VIDEO_DRIVER_X11
slouken@6044
    24
slouken@1952
    25
#include "SDL_x11video.h"
icculus@5981
    26
#include "SDL_assert.h"
slouken@1952
    27
slouken@1952
    28
/* GLX implementation of SDL OpenGL support */
slouken@1952
    29
slouken@1952
    30
#if SDL_VIDEO_OPENGL_GLX
slouken@1952
    31
#include "SDL_loadso.h"
slouken@6370
    32
#include "SDL_x11opengles.h"
slouken@1952
    33
slouken@1952
    34
#if defined(__IRIX__)
slouken@1952
    35
/* IRIX doesn't have a GL library versioning system */
slouken@7191
    36
#define DEFAULT_OPENGL  "libGL.so"
slouken@1952
    37
#elif defined(__MACOSX__)
slouken@7191
    38
#define DEFAULT_OPENGL  "/usr/X11R6/lib/libGL.1.dylib"
slouken@1952
    39
#elif defined(__QNXNTO__)
slouken@7191
    40
#define DEFAULT_OPENGL  "libGL.so.3"
slouken@1952
    41
#else
slouken@7191
    42
#define DEFAULT_OPENGL  "libGL.so.1"
slouken@1952
    43
#endif
slouken@1952
    44
slouken@3241
    45
#ifndef GLX_NONE_EXT
slouken@3241
    46
#define GLX_NONE_EXT                       0x8000
slouken@3241
    47
#endif
slouken@3241
    48
slouken@1952
    49
#ifndef GLX_ARB_multisample
slouken@1952
    50
#define GLX_ARB_multisample
slouken@1952
    51
#define GLX_SAMPLE_BUFFERS_ARB             100000
slouken@1952
    52
#define GLX_SAMPLES_ARB                    100001
slouken@1952
    53
#endif
slouken@1952
    54
slouken@1952
    55
#ifndef GLX_EXT_visual_rating
slouken@1952
    56
#define GLX_EXT_visual_rating
slouken@1952
    57
#define GLX_VISUAL_CAVEAT_EXT              0x20
slouken@1952
    58
#define GLX_NONE_EXT                       0x8000
slouken@1952
    59
#define GLX_SLOW_VISUAL_EXT                0x8001
slouken@1952
    60
#define GLX_NON_CONFORMANT_VISUAL_EXT      0x800D
slouken@1952
    61
#endif
slouken@1952
    62
mikesart@6570
    63
#ifndef GLX_EXT_visual_info
mikesart@6570
    64
#define GLX_EXT_visual_info
mikesart@6570
    65
#define GLX_X_VISUAL_TYPE_EXT              0x22
mikesart@6570
    66
#define GLX_DIRECT_COLOR_EXT               0x8003
mikesart@6570
    67
#endif
mikesart@6570
    68
slouken@3100
    69
#ifndef GLX_ARB_create_context
slouken@3100
    70
#define GLX_ARB_create_context
slouken@3100
    71
#define GLX_CONTEXT_MAJOR_VERSION_ARB      0x2091
slouken@3100
    72
#define GLX_CONTEXT_MINOR_VERSION_ARB      0x2092
slouken@3100
    73
#define GLX_CONTEXT_FLAGS_ARB              0x2094
slouken@3100
    74
#define GLX_CONTEXT_DEBUG_BIT_ARB          0x0001
slouken@3100
    75
#define GLX_CONTEXT_FORWARD_COMPATIBLE_BIT_ARB 0x0002
slouken@5021
    76
slouken@5021
    77
/* Typedef for the GL 3.0 context creation function */
slouken@5021
    78
typedef GLXContext(*PFNGLXCREATECONTEXTATTRIBSARBPROC) (Display * dpy,
slouken@5021
    79
                                                        GLXFBConfig config,
slouken@5021
    80
                                                        GLXContext
slouken@5021
    81
                                                        share_context,
slouken@5021
    82
                                                        Bool direct,
slouken@5021
    83
                                                        const int
slouken@5021
    84
                                                        *attrib_list);
icculus@6360
    85
#endif
slouken@6296
    86
slouken@6296
    87
#ifndef GLX_ARB_create_context_profile
slouken@6296
    88
#define GLX_ARB_create_context_profile
slouken@6296
    89
#define GLX_CONTEXT_PROFILE_MASK_ARB       0x9126
slouken@6296
    90
#define GLX_CONTEXT_CORE_PROFILE_BIT_ARB   0x00000001
slouken@6296
    91
#define GLX_CONTEXT_COMPATIBILITY_PROFILE_BIT_ARB 0x00000002
slouken@6296
    92
#endif
slouken@6296
    93
slouken@6296
    94
#ifndef GLX_ARB_create_context_robustness
slouken@6296
    95
#define GLX_ARB_create_context_robustness
slouken@6296
    96
#define GLX_CONTEXT_ROBUST_ACCESS_BIT_ARB  0x00000004
slouken@6296
    97
#define GLX_CONTEXT_RESET_NOTIFICATION_STRATEGY_ARB     0x8256
slouken@6296
    98
#define GLX_NO_RESET_NOTIFICATION_ARB                   0x8261
slouken@6296
    99
#define GLX_LOSE_CONTEXT_ON_RESET_ARB                   0x8252
slouken@6296
   100
#endif
slouken@6296
   101
slouken@6296
   102
#ifndef GLX_EXT_create_context_es2_profile
slouken@6296
   103
#define GLX_EXT_create_context_es2_profile
icculus@6375
   104
#ifndef GLX_CONTEXT_ES2_PROFILE_BIT_EXT
slouken@6296
   105
#define GLX_CONTEXT_ES2_PROFILE_BIT_EXT    0x00000002
slouken@6296
   106
#endif
icculus@6375
   107
#endif
slouken@6296
   108
slouken@6296
   109
#ifndef GLX_EXT_swap_control
slouken@6296
   110
#define GLX_SWAP_INTERVAL_EXT              0x20F1
slouken@6296
   111
#define GLX_MAX_SWAP_INTERVAL_EXT          0x20F2
slouken@3100
   112
#endif
slouken@3100
   113
icculus@6382
   114
#ifndef GLX_EXT_swap_control_tear
icculus@6382
   115
#define GLX_LATE_SWAPS_TEAR_EXT 0x20F3
icculus@6382
   116
#endif
icculus@6382
   117
slouken@6183
   118
#define OPENGL_REQUIRES_DLOPEN
slouken@6183
   119
#if defined(OPENGL_REQUIRES_DLOPEN) && defined(SDL_LOADSO_DLOPEN)
slouken@1952
   120
#include <dlfcn.h>
slouken@7191
   121
#define GL_LoadObject(X)    dlopen(X, (RTLD_NOW|RTLD_GLOBAL))
slouken@7191
   122
#define GL_LoadFunction     dlsym
slouken@7191
   123
#define GL_UnloadObject     dlclose
slouken@1952
   124
#else
slouken@7191
   125
#define GL_LoadObject   SDL_LoadObject
slouken@7191
   126
#define GL_LoadFunction SDL_LoadFunction
slouken@7191
   127
#define GL_UnloadObject SDL_UnloadObject
slouken@1952
   128
#endif
slouken@1952
   129
slouken@3057
   130
static void X11_GL_InitExtensions(_THIS);
bob@2322
   131
slouken@3100
   132
slouken@1952
   133
int
slouken@1952
   134
X11_GL_LoadLibrary(_THIS, const char *path)
slouken@1952
   135
{
slouken@7512
   136
    Display *display;
slouken@1952
   137
    void *handle;
slouken@1952
   138
slouken@6370
   139
    if (_this->gl_data) {
icculus@7037
   140
        return SDL_SetError("OpenGL context already created");
slouken@6370
   141
    }
slouken@6370
   142
slouken@6370
   143
    /* If SDL_GL_CONTEXT_EGL has been changed to 1, switch over to X11_GLES functions  */
slouken@6370
   144
    if (_this->gl_config.use_egl == 1) {
slouken@6373
   145
#if SDL_VIDEO_OPENGL_ES || SDL_VIDEO_OPENGL_ES2
slouken@6370
   146
        _this->GL_LoadLibrary = X11_GLES_LoadLibrary;
slouken@6370
   147
        _this->GL_GetProcAddress = X11_GLES_GetProcAddress;
slouken@6370
   148
        _this->GL_UnloadLibrary = X11_GLES_UnloadLibrary;
slouken@6370
   149
        _this->GL_CreateContext = X11_GLES_CreateContext;
slouken@6370
   150
        _this->GL_MakeCurrent = X11_GLES_MakeCurrent;
slouken@6370
   151
        _this->GL_SetSwapInterval = X11_GLES_SetSwapInterval;
slouken@6370
   152
        _this->GL_GetSwapInterval = X11_GLES_GetSwapInterval;
slouken@6370
   153
        _this->GL_SwapWindow = X11_GLES_SwapWindow;
slouken@6370
   154
        _this->GL_DeleteContext = X11_GLES_DeleteContext;
slouken@6370
   155
        return X11_GLES_LoadLibrary(_this, path);
slouken@6373
   156
#else
icculus@7037
   157
        return SDL_SetError("SDL not configured with OpenGL ES/EGL support");
slouken@6373
   158
#endif
slouken@6370
   159
    }
slouken@6370
   160
slouken@6370
   161
slouken@3057
   162
    /* Load the OpenGL library */
slouken@3057
   163
    if (path == NULL) {
slouken@3057
   164
        path = SDL_getenv("SDL_OPENGL_LIBRARY");
slouken@1952
   165
    }
slouken@3057
   166
    if (path == NULL) {
slouken@3057
   167
        path = DEFAULT_OPENGL;
slouken@3057
   168
    }
slouken@6183
   169
    _this->gl_config.dll_handle = GL_LoadObject(path);
slouken@3057
   170
    if (!_this->gl_config.dll_handle) {
slouken@6183
   171
#if defined(OPENGL_REQUIRES_DLOPEN) && defined(SDL_LOADSO_DLOPEN)
slouken@6183
   172
        SDL_SetError("Failed loading %s: %s", path, dlerror());
slouken@6183
   173
#endif
slouken@3057
   174
        return -1;
slouken@3057
   175
    }
slouken@3057
   176
    SDL_strlcpy(_this->gl_config.driver_path, path,
slouken@3057
   177
                SDL_arraysize(_this->gl_config.driver_path));
slouken@2244
   178
slouken@3057
   179
    /* Allocate OpenGL memory */
slouken@3057
   180
    _this->gl_data =
slouken@3057
   181
        (struct SDL_GLDriverData *) SDL_calloc(1,
slouken@3057
   182
                                               sizeof(struct
slouken@3057
   183
                                                      SDL_GLDriverData));
slouken@3057
   184
    if (!_this->gl_data) {
icculus@7037
   185
        return SDL_OutOfMemory();
slouken@3057
   186
    }
slouken@3057
   187
slouken@3057
   188
    /* Load function pointers */
slouken@3057
   189
    handle = _this->gl_config.dll_handle;
slouken@7512
   190
    _this->gl_data->glXQueryExtension =
slouken@7512
   191
        (Bool (*)(Display *, int *, int *))
slouken@7512
   192
            GL_LoadFunction(handle, "glXQueryExtension");
slouken@1952
   193
    _this->gl_data->glXGetProcAddress =
slouken@4550
   194
        (void *(*)(const GLubyte *))
slouken@4550
   195
            GL_LoadFunction(handle, "glXGetProcAddressARB");
slouken@1952
   196
    _this->gl_data->glXChooseVisual =
slouken@4550
   197
        (XVisualInfo * (*)(Display *, int, int *))
slouken@4550
   198
            X11_GL_GetProcAddress(_this, "glXChooseVisual");
slouken@1952
   199
    _this->gl_data->glXCreateContext =
slouken@1952
   200
        (GLXContext(*)(Display *, XVisualInfo *, GLXContext, int))
slouken@4550
   201
            X11_GL_GetProcAddress(_this, "glXCreateContext");
slouken@1952
   202
    _this->gl_data->glXDestroyContext =
slouken@4550
   203
        (void (*)(Display *, GLXContext))
slouken@4550
   204
            X11_GL_GetProcAddress(_this, "glXDestroyContext");
slouken@1952
   205
    _this->gl_data->glXMakeCurrent =
slouken@4550
   206
        (int (*)(Display *, GLXDrawable, GLXContext))
slouken@4550
   207
            X11_GL_GetProcAddress(_this, "glXMakeCurrent");
slouken@1952
   208
    _this->gl_data->glXSwapBuffers =
slouken@4550
   209
        (void (*)(Display *, GLXDrawable))
slouken@4550
   210
            X11_GL_GetProcAddress(_this, "glXSwapBuffers");
icculus@5572
   211
    _this->gl_data->glXQueryDrawable =
icculus@5572
   212
        (void (*)(Display*,GLXDrawable,int,unsigned int*))
icculus@5572
   213
            X11_GL_GetProcAddress(_this, "glXQueryDrawable");
slouken@1952
   214
slouken@7512
   215
    if (!_this->gl_data->glXQueryExtension ||
slouken@7512
   216
        !_this->gl_data->glXChooseVisual ||
slouken@1952
   217
        !_this->gl_data->glXCreateContext ||
slouken@1952
   218
        !_this->gl_data->glXDestroyContext ||
slouken@4550
   219
        !_this->gl_data->glXMakeCurrent ||
slouken@4550
   220
        !_this->gl_data->glXSwapBuffers) {
icculus@7037
   221
        return SDL_SetError("Could not retrieve OpenGL functions");
slouken@1952
   222
    }
slouken@1952
   223
slouken@7512
   224
    display = ((SDL_VideoData *) _this->driverdata)->display;
slouken@7512
   225
    if (!_this->gl_data->glXQueryExtension(display, &_this->gl_data->errorBase, &_this->gl_data->eventBase)) {
slouken@7512
   226
        return SDL_SetError("GLX is not supported");
slouken@7512
   227
    }
slouken@7512
   228
slouken@3057
   229
    /* Initialize extensions */
slouken@3057
   230
    X11_GL_InitExtensions(_this);
slouken@3057
   231
slouken@1952
   232
    return 0;
slouken@1952
   233
}
slouken@1952
   234
slouken@1952
   235
void *
slouken@1952
   236
X11_GL_GetProcAddress(_THIS, const char *proc)
slouken@1952
   237
{
slouken@1952
   238
    if (_this->gl_data->glXGetProcAddress) {
slouken@1952
   239
        return _this->gl_data->glXGetProcAddress((const GLubyte *) proc);
slouken@1952
   240
    }
slouken@4550
   241
    return GL_LoadFunction(_this->gl_config.dll_handle, proc);
slouken@1952
   242
}
slouken@1952
   243
slouken@3057
   244
void
slouken@1952
   245
X11_GL_UnloadLibrary(_THIS)
slouken@1952
   246
{
slouken@3057
   247
    /* Don't actually unload the library, since it may have registered
slouken@3057
   248
     * X11 shutdown hooks, per the notes at:
slouken@3057
   249
     * http://dri.sourceforge.net/doc/DRIuserguide.html
slouken@3057
   250
     */
slouken@3057
   251
#if 0
slouken@3057
   252
    GL_UnloadObject(_this->gl_config.dll_handle);
slouken@3057
   253
    _this->gl_config.dll_handle = NULL;
slouken@3057
   254
#endif
slouken@3057
   255
slouken@3057
   256
    /* Free OpenGL memory */
slouken@6181
   257
    if (_this->gl_data) {
slouken@6181
   258
        SDL_free(_this->gl_data);
slouken@6181
   259
        _this->gl_data = NULL;
slouken@6181
   260
    }
slouken@1952
   261
}
slouken@1952
   262
slouken@1952
   263
static SDL_bool
slouken@1952
   264
HasExtension(const char *extension, const char *extensions)
slouken@1952
   265
{
slouken@1952
   266
    const char *start;
slouken@1952
   267
    const char *where, *terminator;
slouken@1952
   268
slouken@1952
   269
    /* Extension names should not have spaces. */
slouken@1952
   270
    where = SDL_strchr(extension, ' ');
slouken@1952
   271
    if (where || *extension == '\0')
slouken@1952
   272
        return SDL_FALSE;
slouken@1952
   273
slouken@1952
   274
    if (!extensions)
slouken@1952
   275
        return SDL_FALSE;
slouken@1952
   276
slouken@1952
   277
    /* It takes a bit of care to be fool-proof about parsing the
slouken@1952
   278
     * OpenGL extensions string. Don't be fooled by sub-strings,
slouken@1952
   279
     * etc. */
slouken@1952
   280
slouken@1952
   281
    start = extensions;
slouken@1952
   282
slouken@1952
   283
    for (;;) {
slouken@1952
   284
        where = SDL_strstr(start, extension);
slouken@1952
   285
        if (!where)
slouken@1952
   286
            break;
slouken@1952
   287
slouken@1952
   288
        terminator = where + SDL_strlen(extension);
slouken@1952
   289
        if (where == start || *(where - 1) == ' ')
slouken@1952
   290
            if (*terminator == ' ' || *terminator == '\0')
slouken@1952
   291
                return SDL_TRUE;
slouken@1952
   292
slouken@1952
   293
        start = terminator;
slouken@1952
   294
    }
slouken@1952
   295
    return SDL_FALSE;
slouken@1952
   296
}
slouken@1952
   297
slouken@1952
   298
static void
slouken@1952
   299
X11_GL_InitExtensions(_THIS)
slouken@1952
   300
{
slouken@1952
   301
    Display *display = ((SDL_VideoData *) _this->driverdata)->display;
slouken@5244
   302
    int screen = DefaultScreen(display);
slouken@1952
   303
    XVisualInfo *vinfo;
slouken@1952
   304
    XSetWindowAttributes xattr;
slouken@1952
   305
    Window w;
slouken@1952
   306
    GLXContext context;
slouken@1952
   307
    const char *(*glXQueryExtensionsStringFunc) (Display *, int);
slouken@1952
   308
    const char *extensions;
slouken@1952
   309
slouken@1952
   310
    vinfo = X11_GL_GetVisual(_this, display, screen);
slouken@1952
   311
    if (!vinfo) {
slouken@1952
   312
        return;
slouken@1952
   313
    }
slouken@1952
   314
    xattr.background_pixel = 0;
slouken@1952
   315
    xattr.border_pixel = 0;
slouken@1952
   316
    xattr.colormap =
slouken@1952
   317
        XCreateColormap(display, RootWindow(display, screen), vinfo->visual,
slouken@1952
   318
                        AllocNone);
slouken@1952
   319
    w = XCreateWindow(display, RootWindow(display, screen), 0, 0, 32, 32, 0,
slouken@1952
   320
                      vinfo->depth, InputOutput, vinfo->visual,
slouken@1952
   321
                      (CWBackPixel | CWBorderPixel | CWColormap), &xattr);
slouken@1952
   322
    context = _this->gl_data->glXCreateContext(display, vinfo, NULL, True);
slouken@1952
   323
    if (context) {
slouken@1952
   324
        _this->gl_data->glXMakeCurrent(display, w, context);
slouken@1952
   325
    }
slouken@1952
   326
    XFree(vinfo);
slouken@1952
   327
slouken@1952
   328
    glXQueryExtensionsStringFunc =
slouken@1952
   329
        (const char *(*)(Display *, int)) X11_GL_GetProcAddress(_this,
slouken@1952
   330
                                                                "glXQueryExtensionsString");
slouken@1952
   331
    if (glXQueryExtensionsStringFunc) {
slouken@1952
   332
        extensions = glXQueryExtensionsStringFunc(display, screen);
slouken@1952
   333
    } else {
slouken@1952
   334
        extensions = NULL;
slouken@1952
   335
    }
slouken@1952
   336
icculus@6382
   337
    /* Check for GLX_EXT_swap_control(_tear) */
icculus@6382
   338
    _this->gl_data->HAS_GLX_EXT_swap_control_tear = SDL_FALSE;
icculus@5572
   339
    if (HasExtension("GLX_EXT_swap_control", extensions)) {
icculus@5572
   340
        _this->gl_data->glXSwapIntervalEXT =
slouken@6846
   341
            (void (*)(Display*,GLXDrawable,int))
icculus@5572
   342
                X11_GL_GetProcAddress(_this, "glXSwapIntervalEXT");
icculus@6382
   343
        if (HasExtension("GLX_EXT_swap_control_tear", extensions)) {
icculus@6382
   344
            _this->gl_data->HAS_GLX_EXT_swap_control_tear = SDL_TRUE;
icculus@6382
   345
        }
slouken@1952
   346
    }
slouken@1952
   347
slouken@1952
   348
    /* Check for GLX_MESA_swap_control */
slouken@1952
   349
    if (HasExtension("GLX_MESA_swap_control", extensions)) {
slouken@1952
   350
        _this->gl_data->glXSwapIntervalMESA =
icculus@5572
   351
            (int(*)(int)) X11_GL_GetProcAddress(_this, "glXSwapIntervalMESA");
slouken@1952
   352
        _this->gl_data->glXGetSwapIntervalMESA =
icculus@5572
   353
            (int(*)(void)) X11_GL_GetProcAddress(_this,
slouken@1952
   354
                                                   "glXGetSwapIntervalMESA");
slouken@1952
   355
    }
slouken@1952
   356
icculus@5572
   357
    /* Check for GLX_SGI_swap_control */
icculus@5572
   358
    if (HasExtension("GLX_SGI_swap_control", extensions)) {
icculus@5572
   359
        _this->gl_data->glXSwapIntervalSGI =
icculus@5572
   360
            (int (*)(int)) X11_GL_GetProcAddress(_this, "glXSwapIntervalSGI");
icculus@5572
   361
    }
icculus@5572
   362
slouken@1952
   363
    /* Check for GLX_EXT_visual_rating */
slouken@1952
   364
    if (HasExtension("GLX_EXT_visual_rating", extensions)) {
slouken@1952
   365
        _this->gl_data->HAS_GLX_EXT_visual_rating = SDL_TRUE;
slouken@1952
   366
    }
slouken@1952
   367
mikesart@6570
   368
    /* Check for GLX_EXT_visual_info */
mikesart@6570
   369
    if (HasExtension("GLX_EXT_visual_info", extensions)) {
mikesart@6570
   370
        _this->gl_data->HAS_GLX_EXT_visual_info = SDL_TRUE;
mikesart@6570
   371
    }
mikesart@6570
   372
slouken@1952
   373
    if (context) {
slouken@1952
   374
        _this->gl_data->glXMakeCurrent(display, None, NULL);
slouken@1952
   375
        _this->gl_data->glXDestroyContext(display, context);
slouken@1952
   376
    }
slouken@1952
   377
    XDestroyWindow(display, w);
bob@2323
   378
    X11_PumpEvents(_this);
slouken@1952
   379
}
slouken@1952
   380
slouken@6296
   381
/* glXChooseVisual and glXChooseFBConfig have some small differences in
slouken@7191
   382
 * the attribute encoding, it can be chosen with the for_FBConfig parameter.
slouken@6296
   383
 */
slouken@7191
   384
int
slouken@6296
   385
X11_GL_GetAttributes(_THIS, Display * display, int screen, int * attribs, int size, Bool for_FBConfig)
slouken@1952
   386
{
slouken@5395
   387
    int i = 0;
slouken@7191
   388
    const int MAX_ATTRIBUTES = 64;
slouken@1952
   389
slouken@7191
   390
    /* assert buffer is large enough to hold all SDL attributes. */
mikesart@6570
   391
    SDL_assert(size >= MAX_ATTRIBUTES);
slouken@3139
   392
slouken@1952
   393
    /* Setup our GLX attributes according to the gl_config. */
slouken@6296
   394
    if( for_FBConfig ) {
slouken@6296
   395
        attribs[i++] = GLX_RENDER_TYPE;
icculus@7462
   396
        attribs[i++] = GLX_RGBA_BIT;
slouken@6296
   397
    } else {
slouken@6296
   398
        attribs[i++] = GLX_RGBA;
slouken@6296
   399
    }
slouken@1952
   400
    attribs[i++] = GLX_RED_SIZE;
slouken@1952
   401
    attribs[i++] = _this->gl_config.red_size;
slouken@1952
   402
    attribs[i++] = GLX_GREEN_SIZE;
slouken@1952
   403
    attribs[i++] = _this->gl_config.green_size;
slouken@1952
   404
    attribs[i++] = GLX_BLUE_SIZE;
slouken@1952
   405
    attribs[i++] = _this->gl_config.blue_size;
slouken@1952
   406
slouken@1952
   407
    if (_this->gl_config.alpha_size) {
slouken@1952
   408
        attribs[i++] = GLX_ALPHA_SIZE;
slouken@1952
   409
        attribs[i++] = _this->gl_config.alpha_size;
slouken@1952
   410
    }
slouken@1952
   411
slouken@1952
   412
    if (_this->gl_config.double_buffer) {
slouken@1952
   413
        attribs[i++] = GLX_DOUBLEBUFFER;
icculus@7462
   414
        if( for_FBConfig ) {
icculus@7462
   415
            attribs[i++] = True;
icculus@7462
   416
        }
slouken@1952
   417
    }
slouken@1952
   418
slouken@1952
   419
    attribs[i++] = GLX_DEPTH_SIZE;
slouken@1952
   420
    attribs[i++] = _this->gl_config.depth_size;
slouken@1952
   421
slouken@1952
   422
    if (_this->gl_config.stencil_size) {
slouken@1952
   423
        attribs[i++] = GLX_STENCIL_SIZE;
slouken@1952
   424
        attribs[i++] = _this->gl_config.stencil_size;
slouken@1952
   425
    }
slouken@1952
   426
slouken@1952
   427
    if (_this->gl_config.accum_red_size) {
slouken@1952
   428
        attribs[i++] = GLX_ACCUM_RED_SIZE;
slouken@1952
   429
        attribs[i++] = _this->gl_config.accum_red_size;
slouken@1952
   430
    }
slouken@1952
   431
slouken@1952
   432
    if (_this->gl_config.accum_green_size) {
slouken@1952
   433
        attribs[i++] = GLX_ACCUM_GREEN_SIZE;
slouken@1952
   434
        attribs[i++] = _this->gl_config.accum_green_size;
slouken@1952
   435
    }
slouken@1952
   436
slouken@1952
   437
    if (_this->gl_config.accum_blue_size) {
slouken@1952
   438
        attribs[i++] = GLX_ACCUM_BLUE_SIZE;
slouken@1952
   439
        attribs[i++] = _this->gl_config.accum_blue_size;
slouken@1952
   440
    }
slouken@1952
   441
slouken@1952
   442
    if (_this->gl_config.accum_alpha_size) {
slouken@1952
   443
        attribs[i++] = GLX_ACCUM_ALPHA_SIZE;
slouken@1952
   444
        attribs[i++] = _this->gl_config.accum_alpha_size;
slouken@1952
   445
    }
slouken@1952
   446
slouken@1952
   447
    if (_this->gl_config.stereo) {
slouken@1952
   448
        attribs[i++] = GLX_STEREO;
icculus@7462
   449
        if( for_FBConfig ) {
icculus@7462
   450
            attribs[i++] = True;
icculus@7462
   451
        }
slouken@1952
   452
    }
slouken@1952
   453
slouken@1952
   454
    if (_this->gl_config.multisamplebuffers) {
slouken@1952
   455
        attribs[i++] = GLX_SAMPLE_BUFFERS_ARB;
slouken@1952
   456
        attribs[i++] = _this->gl_config.multisamplebuffers;
slouken@1952
   457
    }
slouken@1952
   458
slouken@1952
   459
    if (_this->gl_config.multisamplesamples) {
slouken@1952
   460
        attribs[i++] = GLX_SAMPLES_ARB;
slouken@1952
   461
        attribs[i++] = _this->gl_config.multisamplesamples;
slouken@1952
   462
    }
slouken@1952
   463
slouken@3571
   464
    if (_this->gl_config.accelerated >= 0 &&
slouken@3571
   465
        _this->gl_data->HAS_GLX_EXT_visual_rating) {
slouken@1952
   466
        attribs[i++] = GLX_VISUAL_CAVEAT_EXT;
slouken@3570
   467
        attribs[i++] = _this->gl_config.accelerated ? GLX_NONE_EXT :
slouken@3570
   468
                                                      GLX_SLOW_VISUAL_EXT;
slouken@1952
   469
    }
slouken@3570
   470
icculus@6817
   471
    /* If we're supposed to use DirectColor visuals, and we've got the
icculus@6817
   472
       EXT_visual_info extension, then add GLX_X_VISUAL_TYPE_EXT. */
mikesart@6570
   473
    if (X11_UseDirectColorVisuals() &&
mikesart@6570
   474
        _this->gl_data->HAS_GLX_EXT_visual_info) {
mikesart@6570
   475
        attribs[i++] = GLX_X_VISUAL_TYPE_EXT;
mikesart@6570
   476
        attribs[i++] = GLX_DIRECT_COLOR_EXT;
mikesart@6570
   477
    }
mikesart@6570
   478
slouken@1952
   479
    attribs[i++] = None;
mikesart@6570
   480
mikesart@6570
   481
    SDL_assert(i <= MAX_ATTRIBUTES);
slouken@7191
   482
slouken@5395
   483
    return i;
slouken@5395
   484
}
slouken@5395
   485
slouken@5395
   486
XVisualInfo *
slouken@5395
   487
X11_GL_GetVisual(_THIS, Display * display, int screen)
slouken@5395
   488
{
slouken@5395
   489
    XVisualInfo *vinfo;
slouken@5395
   490
slouken@5395
   491
    /* 64 seems nice. */
slouken@6296
   492
    int attribs[64];
icculus@6359
   493
    X11_GL_GetAttributes(_this,display,screen,attribs,64,SDL_FALSE);
slouken@1952
   494
slouken@6181
   495
    if (!_this->gl_data) {
slouken@6181
   496
        /* The OpenGL library wasn't loaded, SDL_GetError() should have info */
slouken@6181
   497
        return NULL;
slouken@6181
   498
    }
slouken@6181
   499
slouken@1952
   500
    vinfo = _this->gl_data->glXChooseVisual(display, screen, attribs);
slouken@1952
   501
    if (!vinfo) {
slouken@1952
   502
        SDL_SetError("Couldn't find matching GLX visual");
slouken@1952
   503
    }
slouken@1952
   504
    return vinfo;
slouken@1952
   505
}
slouken@1952
   506
slouken@6988
   507
#ifndef GLXBadContext
slouken@6988
   508
#define GLXBadContext 0
slouken@6988
   509
#endif
slouken@6988
   510
#ifndef GLXBadFBConfig
slouken@6988
   511
#define GLXBadFBConfig 9
slouken@6988
   512
#endif
slouken@6988
   513
#ifndef GLXBadProfileARB
slouken@6988
   514
#define GLXBadProfileARB 13
slouken@6988
   515
#endif
slouken@6988
   516
static int (*handler) (Display *, XErrorEvent *) = NULL;
slouken@7512
   517
static int errorBase = 0;
slouken@6988
   518
static int
slouken@6988
   519
X11_GL_CreateContextErrorHandler(Display * d, XErrorEvent * e)
slouken@6988
   520
{
slouken@6988
   521
    switch (e->error_code) {
slouken@6988
   522
    case BadRequest:
slouken@6988
   523
    case BadMatch:
slouken@6988
   524
    case BadValue:
slouken@6988
   525
    case BadAlloc:
slouken@6988
   526
        return (0);
slouken@6988
   527
    default:
slouken@7512
   528
        if (errorBase && 
slouken@7512
   529
            (e->error_code == errorBase + GLXBadContext ||
slouken@7512
   530
             e->error_code == errorBase + GLXBadFBConfig ||
slouken@7512
   531
             e->error_code == errorBase + GLXBadProfileARB)) {
slouken@7512
   532
            return (0);
slouken@7512
   533
        }
slouken@6988
   534
        return (handler(d, e));
slouken@6988
   535
    }
slouken@6988
   536
}
slouken@6988
   537
slouken@1952
   538
SDL_GLContext
slouken@1952
   539
X11_GL_CreateContext(_THIS, SDL_Window * window)
slouken@1952
   540
{
slouken@1952
   541
    SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
slouken@1952
   542
    Display *display = data->videodata->display;
slouken@1952
   543
    int screen =
slouken@5246
   544
        ((SDL_DisplayData *) SDL_GetDisplayForWindow(window)->driverdata)->screen;
slouken@1952
   545
    XWindowAttributes xattr;
slouken@1952
   546
    XVisualInfo v, *vinfo;
slouken@1952
   547
    int n;
slouken@6393
   548
    GLXContext context = NULL, share_context;
slouken@6393
   549
slouken@6393
   550
    if (_this->gl_config.share_with_current_context) {
slouken@7413
   551
        share_context = (GLXContext)SDL_GL_GetCurrentContext();
slouken@6393
   552
    } else {
slouken@6393
   553
        share_context = NULL;
slouken@6393
   554
    }
slouken@1952
   555
slouken@1955
   556
    /* We do this to create a clean separation between X and GLX errors. */
slouken@1952
   557
    XSync(display, False);
slouken@7512
   558
    errorBase = _this->gl_data->errorBase;
slouken@6988
   559
    handler = XSetErrorHandler(X11_GL_CreateContextErrorHandler);
slouken@3685
   560
    XGetWindowAttributes(display, data->xwindow, &xattr);
slouken@1952
   561
    v.screen = screen;
slouken@1952
   562
    v.visualid = XVisualIDFromVisual(xattr.visual);
slouken@1952
   563
    vinfo = XGetVisualInfo(display, VisualScreenMask | VisualIDMask, &v, &n);
slouken@1952
   564
    if (vinfo) {
slouken@6393
   565
        if (_this->gl_config.major_version < 3 &&
slouken@6393
   566
            _this->gl_config.profile_mask == 0 &&
slouken@6393
   567
            _this->gl_config.flags == 0) {
slouken@6393
   568
            /* Create legacy context */
slouken@3100
   569
            context =
slouken@6393
   570
                _this->gl_data->glXCreateContext(display, vinfo, share_context, True);
slouken@3100
   571
        } else {
slouken@3100
   572
            /* If we want a GL 3.0 context or later we need to get a temporary
slouken@3100
   573
               context to grab the new context creation function */
slouken@3139
   574
            GLXContext temp_context =
slouken@3139
   575
                _this->gl_data->glXCreateContext(display, vinfo, NULL, True);
slouken@6988
   576
            if (temp_context) {
slouken@6393
   577
                /* max 8 attributes plus terminator */
slouken@6296
   578
                int attribs[9] = {
slouken@3139
   579
                    GLX_CONTEXT_MAJOR_VERSION_ARB,
slouken@3139
   580
                    _this->gl_config.major_version,
slouken@3139
   581
                    GLX_CONTEXT_MINOR_VERSION_ARB,
slouken@3139
   582
                    _this->gl_config.minor_version,
slouken@3139
   583
                    0
slouken@3100
   584
                };
slouken@6393
   585
                int iattr = 4;
slouken@6296
   586
slouken@6393
   587
                /* SDL profile bits match GLX profile bits */
slouken@6393
   588
                if( _this->gl_config.profile_mask != 0 ) {
slouken@6393
   589
                    attribs[iattr++] = GLX_CONTEXT_PROFILE_MASK_ARB;
slouken@6393
   590
                    attribs[iattr++] = _this->gl_config.profile_mask;
slouken@6393
   591
                }
slouken@6296
   592
slouken@6393
   593
                /* SDL flags match GLX flags */
slouken@6393
   594
                if( _this->gl_config.flags != 0 ) {
slouken@6393
   595
                    attribs[iattr++] = GLX_CONTEXT_FLAGS_ARB;
slouken@6393
   596
                    attribs[iattr++] = _this->gl_config.flags;
slouken@6393
   597
                }
slouken@6296
   598
slouken@6393
   599
                attribs[iattr++] = 0;
slouken@3139
   600
slouken@3100
   601
                /* Get a pointer to the context creation function for GL 3.0 */
slouken@3139
   602
                PFNGLXCREATECONTEXTATTRIBSARBPROC glXCreateContextAttribs =
slouken@3139
   603
                    (PFNGLXCREATECONTEXTATTRIBSARBPROC) _this->gl_data->
slouken@3139
   604
                    glXGetProcAddress((GLubyte *)
slouken@3139
   605
                                      "glXCreateContextAttribsARB");
slouken@3100
   606
                if (!glXCreateContextAttribs) {
slouken@3100
   607
                    SDL_SetError("GL 3.x is not supported");
slouken@3100
   608
                    context = temp_context;
slouken@3100
   609
                } else {
slouken@5395
   610
                    int glxAttribs[64];
slouken@5395
   611
slouken@3105
   612
                    /* Create a GL 3.x context */
slouken@3100
   613
                    GLXFBConfig *framebuffer_config = NULL;
slouken@3139
   614
                    int fbcount = 0;
slouken@3139
   615
                    GLXFBConfig *(*glXChooseFBConfig) (Display * disp,
slouken@3139
   616
                                                       int screen,
slouken@3139
   617
                                                       const int *attrib_list,
slouken@3139
   618
                                                       int *nelements);
slouken@3105
   619
slouken@3139
   620
                    glXChooseFBConfig =
slouken@3139
   621
                        (GLXFBConfig *
slouken@3139
   622
                         (*)(Display *, int, const int *,
slouken@3139
   623
                             int *)) _this->gl_data->
slouken@3139
   624
                        glXGetProcAddress((GLubyte *) "glXChooseFBConfig");
slouken@3105
   625
slouken@6296
   626
                    X11_GL_GetAttributes(_this,display,screen,glxAttribs,64,SDL_TRUE);
slouken@5395
   627
slouken@3139
   628
                    if (!glXChooseFBConfig
slouken@3139
   629
                        || !(framebuffer_config =
slouken@3139
   630
                             glXChooseFBConfig(display,
slouken@5395
   631
                                               DefaultScreen(display), glxAttribs,
slouken@3139
   632
                                               &fbcount))) {
slouken@3139
   633
                        SDL_SetError
slouken@3139
   634
                            ("No good framebuffers found. GL 3.x disabled");
slouken@3100
   635
                        context = temp_context;
slouken@3139
   636
                    } else {
slouken@3139
   637
                        context =
slouken@3139
   638
                            glXCreateContextAttribs(display,
slouken@3139
   639
                                                    framebuffer_config[0],
slouken@6393
   640
                                                    share_context, True, attribs);
slouken@3139
   641
                        _this->gl_data->glXDestroyContext(display,
slouken@3139
   642
                                                          temp_context);
slouken@3100
   643
                    }
slouken@3100
   644
                }
slouken@3100
   645
            }
slouken@3100
   646
        }
slouken@1952
   647
        XFree(vinfo);
slouken@1952
   648
    }
slouken@1952
   649
    XSync(display, False);
slouken@6988
   650
    XSetErrorHandler(handler);
slouken@7191
   651
slouken@1952
   652
    if (!context) {
slouken@1952
   653
        SDL_SetError("Could not create GL context");
slouken@2178
   654
        return NULL;
slouken@1952
   655
    }
slouken@2178
   656
slouken@2178
   657
    if (X11_GL_MakeCurrent(_this, window, context) < 0) {
slouken@2178
   658
        X11_GL_DeleteContext(_this, context);
slouken@2178
   659
        return NULL;
slouken@2178
   660
    }
slouken@2178
   661
slouken@2178
   662
    return context;
slouken@1952
   663
}
slouken@1952
   664
slouken@1952
   665
int
slouken@1952
   666
X11_GL_MakeCurrent(_THIS, SDL_Window * window, SDL_GLContext context)
slouken@1952
   667
{
slouken@1952
   668
    Display *display = ((SDL_VideoData *) _this->driverdata)->display;
slouken@1952
   669
    Window drawable =
urkle@6794
   670
        (context ? ((SDL_WindowData *) window->driverdata)->xwindow : None);
slouken@1952
   671
    GLXContext glx_context = (GLXContext) context;
slouken@1952
   672
slouken@6522
   673
    if (!_this->gl_data) {
icculus@7037
   674
        return SDL_SetError("OpenGL not initialized");
slouken@6522
   675
    }
slouken@6522
   676
slouken@1952
   677
    if (!_this->gl_data->glXMakeCurrent(display, drawable, glx_context)) {
icculus@7037
   678
        return SDL_SetError("Unable to make GL context current");
slouken@1952
   679
    }
slouken@1952
   680
icculus@7037
   681
    return 0;
slouken@1952
   682
}
slouken@1952
   683
slouken@7191
   684
/*
icculus@5572
   685
   0 is a valid argument to glxSwapInterval(MESA|EXT) and setting it to 0
icculus@5572
   686
   will undo the effect of a previous call with a value that is greater
icculus@5572
   687
   than zero (or at least that is what the docs say). OTOH, 0 is an invalid
icculus@5572
   688
   argument to glxSwapIntervalSGI and it returns an error if you call it
icculus@5572
   689
   with 0 as an argument.
bob@2327
   690
*/
bob@2327
   691
bob@2327
   692
static int swapinterval = -1;
slouken@1952
   693
int
slouken@1952
   694
X11_GL_SetSwapInterval(_THIS, int interval)
slouken@1952
   695
{
icculus@6382
   696
    int status = -1;
slouken@1952
   697
icculus@6382
   698
    if ((interval < 0) && (!_this->gl_data->HAS_GLX_EXT_swap_control_tear)) {
icculus@6382
   699
        SDL_SetError("Negative swap interval unsupported in this GL");
icculus@6382
   700
    } else if (_this->gl_data->glXSwapIntervalEXT) {
icculus@5572
   701
        Display *display = ((SDL_VideoData *) _this->driverdata)->display;
icculus@5572
   702
        const SDL_WindowData *windowdata = (SDL_WindowData *)
slouken@7412
   703
            SDL_GL_GetCurrentWindow()->driverdata;
slouken@6846
   704
icculus@5572
   705
        Window drawable = windowdata->xwindow;
slouken@6846
   706
slouken@6846
   707
        /*
slouken@6846
   708
         * This is a workaround for a bug in NVIDIA drivers. Bug has been reported
slouken@6846
   709
         * and will be fixed in a future release (probably 319.xx).
slouken@6846
   710
         *
slouken@6846
   711
         * There's a bug where glXSetSwapIntervalEXT ignores updates because
slouken@6846
   712
         * it has the wrong value cached. To work around it, we just run a no-op
slouken@6846
   713
         * update to the current value.
slouken@6846
   714
         */
slouken@6851
   715
        int currentInterval = X11_GL_GetSwapInterval(_this);
slouken@6846
   716
        _this->gl_data->glXSwapIntervalEXT(display, drawable, currentInterval);
slouken@6846
   717
        _this->gl_data->glXSwapIntervalEXT(display, drawable, interval);
slouken@6846
   718
slouken@6846
   719
        status = 0;
slouken@6846
   720
        swapinterval = interval;
icculus@5572
   721
    } else if (_this->gl_data->glXSwapIntervalMESA) {
slouken@1952
   722
        status = _this->gl_data->glXSwapIntervalMESA(interval);
slouken@1952
   723
        if (status != 0) {
slouken@1952
   724
            SDL_SetError("glxSwapIntervalMESA failed");
bob@2327
   725
        } else {
bob@2327
   726
            swapinterval = interval;
slouken@1952
   727
        }
slouken@1952
   728
    } else if (_this->gl_data->glXSwapIntervalSGI) {
slouken@1952
   729
        status = _this->gl_data->glXSwapIntervalSGI(interval);
slouken@1952
   730
        if (status != 0) {
slouken@1952
   731
            SDL_SetError("glxSwapIntervalSGI failed");
bob@2327
   732
        } else {
bob@2327
   733
            swapinterval = interval;
slouken@1952
   734
        }
slouken@1952
   735
    } else {
slouken@1952
   736
        SDL_Unsupported();
slouken@1952
   737
    }
slouken@1952
   738
    return status;
slouken@1952
   739
}
slouken@1952
   740
slouken@1952
   741
int
slouken@1952
   742
X11_GL_GetSwapInterval(_THIS)
slouken@1952
   743
{
icculus@5572
   744
    if (_this->gl_data->glXSwapIntervalEXT) {
icculus@5572
   745
        Display *display = ((SDL_VideoData *) _this->driverdata)->display;
icculus@5572
   746
        const SDL_WindowData *windowdata = (SDL_WindowData *)
slouken@7412
   747
            SDL_GL_GetCurrentWindow()->driverdata;
icculus@5572
   748
        Window drawable = windowdata->xwindow;
icculus@6382
   749
        unsigned int allow_late_swap_tearing = 0;
icculus@6382
   750
        unsigned int interval = 0;
icculus@6382
   751
icculus@6382
   752
        if (_this->gl_data->HAS_GLX_EXT_swap_control_tear) {
icculus@6382
   753
            _this->gl_data->glXQueryDrawable(display, drawable,
icculus@6382
   754
                                            GLX_LATE_SWAPS_TEAR_EXT,
icculus@6382
   755
                                            &allow_late_swap_tearing);
icculus@6382
   756
        }
icculus@6382
   757
icculus@5572
   758
        _this->gl_data->glXQueryDrawable(display, drawable,
icculus@6382
   759
                                         GLX_SWAP_INTERVAL_EXT, &interval);
icculus@6382
   760
icculus@6382
   761
        if ((allow_late_swap_tearing) && (interval > 0)) {
icculus@6382
   762
            return -((int) interval);
icculus@6382
   763
        }
icculus@6382
   764
icculus@6382
   765
        return (int) interval;
icculus@5572
   766
    } else if (_this->gl_data->glXGetSwapIntervalMESA) {
slouken@1952
   767
        return _this->gl_data->glXGetSwapIntervalMESA();
slouken@1952
   768
    } else {
bob@2327
   769
        return swapinterval;
slouken@1952
   770
    }
slouken@1952
   771
}
slouken@1952
   772
slouken@1952
   773
void
slouken@1952
   774
X11_GL_SwapWindow(_THIS, SDL_Window * window)
slouken@1952
   775
{
slouken@1952
   776
    SDL_WindowData *data = (SDL_WindowData *) window->driverdata;
slouken@1952
   777
    Display *display = data->videodata->display;
slouken@1952
   778
slouken@3685
   779
    _this->gl_data->glXSwapBuffers(display, data->xwindow);
slouken@1952
   780
}
slouken@1952
   781
slouken@1952
   782
void
slouken@1952
   783
X11_GL_DeleteContext(_THIS, SDL_GLContext context)
slouken@1952
   784
{
slouken@1952
   785
    Display *display = ((SDL_VideoData *) _this->driverdata)->display;
slouken@1952
   786
    GLXContext glx_context = (GLXContext) context;
slouken@1952
   787
slouken@6522
   788
    if (!_this->gl_data) {
slouken@6522
   789
        return;
slouken@6522
   790
    }
slouken@1952
   791
    _this->gl_data->glXDestroyContext(display, glx_context);
bob@2324
   792
    XSync(display, False);
slouken@1952
   793
}
slouken@1952
   794
slouken@1952
   795
#endif /* SDL_VIDEO_OPENGL_GLX */
slouken@1952
   796
slouken@6044
   797
#endif /* SDL_VIDEO_DRIVER_X11 */
slouken@6044
   798
slouken@1952
   799
/* vi: set ts=4 sw=4 expandtab: */