src/video/win32/SDL_win32opengl.c
author Sam Lantinga <slouken@libsdl.org>
Thu, 12 Jul 2007 06:59:43 +0000
changeset 2180 5ed37b16c1a7
parent 2178 114a541cfae2
child 2698 e1da92da346c
permissions -rw-r--r--
Yes, you need to set the pixel format before creating a context.
slouken@1913
     1
/*
slouken@1913
     2
    SDL - Simple DirectMedia Layer
slouken@1913
     3
    Copyright (C) 1997-2006 Sam Lantinga
slouken@1913
     4
slouken@1913
     5
    This library is free software; you can redistribute it and/or
slouken@1913
     6
    modify it under the terms of the GNU Lesser General Public
slouken@1913
     7
    License as published by the Free Software Foundation; either
slouken@1913
     8
    version 2.1 of the License, or (at your option) any later version.
slouken@1913
     9
slouken@1913
    10
    This library is distributed in the hope that it will be useful,
slouken@1913
    11
    but WITHOUT ANY WARRANTY; without even the implied warranty of
slouken@1913
    12
    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
slouken@1913
    13
    Lesser General Public License for more details.
slouken@1913
    14
slouken@1913
    15
    You should have received a copy of the GNU Lesser General Public
slouken@1913
    16
    License along with this library; if not, write to the Free Software
slouken@1913
    17
    Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA
slouken@1913
    18
slouken@1913
    19
    Sam Lantinga
slouken@1913
    20
    slouken@libsdl.org
slouken@1913
    21
*/
slouken@1913
    22
#include "SDL_config.h"
slouken@1913
    23
slouken@1913
    24
#include "SDL_win32video.h"
slouken@1913
    25
slouken@1913
    26
/* WGL implementation of SDL OpenGL support */
slouken@1913
    27
slouken@1952
    28
#if SDL_VIDEO_OPENGL_WGL
slouken@1913
    29
#include "SDL_opengl.h"
slouken@1913
    30
slouken@1952
    31
#define DEFAULT_OPENGL "OPENGL32.DLL"
slouken@1913
    32
slouken@1913
    33
slouken@1913
    34
int
slouken@1913
    35
WIN_GL_LoadLibrary(_THIS, const char *path)
slouken@1913
    36
{
slouken@1913
    37
    LPTSTR wpath;
slouken@1913
    38
    HANDLE handle;
slouken@1913
    39
slouken@1913
    40
    if (_this->gl_config.driver_loaded) {
slouken@1913
    41
        if (path) {
slouken@1913
    42
            SDL_SetError("OpenGL library already loaded");
slouken@1913
    43
            return -1;
slouken@1913
    44
        } else {
slouken@1913
    45
            ++_this->gl_config.driver_loaded;
slouken@1913
    46
            return 0;
slouken@1913
    47
        }
slouken@1913
    48
    }
slouken@1913
    49
    if (path == NULL) {
slouken@1952
    50
        path = SDL_getenv("SDL_OPENGL_LIBRARY");
slouken@1952
    51
    }
slouken@1952
    52
    if (path == NULL) {
slouken@1952
    53
        path = DEFAULT_OPENGL;
slouken@1913
    54
    }
slouken@1913
    55
    wpath = WIN_UTF8ToString(path);
slouken@1913
    56
    handle = LoadLibrary(wpath);
slouken@1913
    57
    SDL_free(wpath);
slouken@1913
    58
    if (!handle) {
slouken@1913
    59
        char message[1024];
slouken@1913
    60
        SDL_snprintf(message, SDL_arraysize(message), "LoadLibrary(\"%s\")",
slouken@1913
    61
                     path);
slouken@1913
    62
        WIN_SetError(message);
slouken@1913
    63
        return -1;
slouken@1913
    64
    }
slouken@1913
    65
slouken@1913
    66
    /* Load function pointers */
slouken@1913
    67
    _this->gl_data->wglGetProcAddress = (void *(WINAPI *) (const char *))
slouken@1913
    68
        GetProcAddress(handle, "wglGetProcAddress");
slouken@1913
    69
    _this->gl_data->wglCreateContext = (HGLRC(WINAPI *) (HDC))
slouken@1913
    70
        GetProcAddress(handle, "wglCreateContext");
slouken@1913
    71
    _this->gl_data->wglDeleteContext = (BOOL(WINAPI *) (HGLRC))
slouken@1913
    72
        GetProcAddress(handle, "wglDeleteContext");
slouken@1913
    73
    _this->gl_data->wglMakeCurrent = (BOOL(WINAPI *) (HDC, HGLRC))
slouken@1913
    74
        GetProcAddress(handle, "wglMakeCurrent");
slouken@1913
    75
    _this->gl_data->wglSwapIntervalEXT = (void (WINAPI *) (int))
slouken@1913
    76
        GetProcAddress(handle, "wglSwapIntervalEXT");
slouken@1913
    77
    _this->gl_data->wglGetSwapIntervalEXT = (int (WINAPI *) (void))
slouken@1913
    78
        GetProcAddress(handle, "wglGetSwapIntervalEXT");
slouken@1913
    79
slouken@1913
    80
    if (!_this->gl_data->wglGetProcAddress ||
slouken@1913
    81
        !_this->gl_data->wglCreateContext ||
slouken@1913
    82
        !_this->gl_data->wglDeleteContext ||
slouken@1913
    83
        !_this->gl_data->wglMakeCurrent) {
slouken@1913
    84
        SDL_SetError("Could not retrieve OpenGL functions");
slouken@1913
    85
        FreeLibrary(handle);
slouken@1913
    86
        return -1;
slouken@1913
    87
    }
slouken@1913
    88
slouken@1913
    89
    _this->gl_config.dll_handle = handle;
slouken@1913
    90
    SDL_strlcpy(_this->gl_config.driver_path, path,
slouken@1913
    91
                SDL_arraysize(_this->gl_config.driver_path));
slouken@1913
    92
    _this->gl_config.driver_loaded = 1;
slouken@1913
    93
    return 0;
slouken@1913
    94
}
slouken@1913
    95
slouken@1913
    96
void *
slouken@1913
    97
WIN_GL_GetProcAddress(_THIS, const char *proc)
slouken@1913
    98
{
slouken@1913
    99
    void *func;
slouken@1913
   100
slouken@1913
   101
    /* This is to pick up extensions */
slouken@1913
   102
    func = _this->gl_data->wglGetProcAddress(proc);
slouken@1913
   103
    if (!func) {
slouken@1913
   104
        /* This is probably a normal GL function */
slouken@1913
   105
        func = GetProcAddress(_this->gl_config.dll_handle, proc);
slouken@1913
   106
    }
slouken@1913
   107
    return func;
slouken@1913
   108
}
slouken@1913
   109
slouken@1913
   110
static void
slouken@1913
   111
WIN_GL_UnloadLibrary(_THIS)
slouken@1913
   112
{
slouken@1913
   113
    if (_this->gl_config.driver_loaded > 0) {
slouken@1913
   114
        if (--_this->gl_config.driver_loaded > 0) {
slouken@1913
   115
            return;
slouken@1913
   116
        }
slouken@1913
   117
        FreeLibrary((HMODULE) _this->gl_config.dll_handle);
slouken@1913
   118
        _this->gl_config.dll_handle = NULL;
slouken@1913
   119
    }
slouken@1913
   120
}
slouken@1913
   121
slouken@1913
   122
static void
slouken@1913
   123
WIN_GL_SetupPixelFormat(_THIS, PIXELFORMATDESCRIPTOR * pfd)
slouken@1913
   124
{
slouken@1913
   125
    SDL_zerop(pfd);
slouken@1913
   126
    pfd->nSize = sizeof(*pfd);
slouken@1913
   127
    pfd->nVersion = 1;
slouken@1913
   128
    pfd->dwFlags = (PFD_DRAW_TO_WINDOW | PFD_SUPPORT_OPENGL);
slouken@1913
   129
    if (_this->gl_config.double_buffer) {
slouken@1913
   130
        pfd->dwFlags |= PFD_DOUBLEBUFFER;
slouken@1913
   131
    }
slouken@1913
   132
    if (_this->gl_config.stereo) {
slouken@1913
   133
        pfd->dwFlags |= PFD_STEREO;
slouken@1913
   134
    }
slouken@1913
   135
    pfd->iLayerType = PFD_MAIN_PLANE;
slouken@1913
   136
    pfd->iPixelType = PFD_TYPE_RGBA;
slouken@1913
   137
    pfd->cRedBits = _this->gl_config.red_size;
slouken@1913
   138
    pfd->cGreenBits = _this->gl_config.green_size;
slouken@1913
   139
    pfd->cBlueBits = _this->gl_config.blue_size;
slouken@1913
   140
    pfd->cAlphaBits = _this->gl_config.alpha_size;
slouken@1913
   141
    if (_this->gl_config.buffer_size) {
slouken@1913
   142
        pfd->cColorBits =
slouken@1913
   143
            _this->gl_config.buffer_size - _this->gl_config.alpha_size;
slouken@1913
   144
    } else {
slouken@1913
   145
        pfd->cColorBits = (pfd->cRedBits + pfd->cGreenBits + pfd->cBlueBits);
slouken@1913
   146
    }
slouken@1913
   147
    pfd->cAccumRedBits = _this->gl_config.accum_red_size;
slouken@1913
   148
    pfd->cAccumGreenBits = _this->gl_config.accum_green_size;
slouken@1913
   149
    pfd->cAccumBlueBits = _this->gl_config.accum_blue_size;
slouken@1913
   150
    pfd->cAccumAlphaBits = _this->gl_config.accum_alpha_size;
slouken@1913
   151
    pfd->cAccumBits =
slouken@1913
   152
        (pfd->cAccumRedBits + pfd->cAccumGreenBits + pfd->cAccumBlueBits +
slouken@1913
   153
         pfd->cAccumAlphaBits);
slouken@1913
   154
    pfd->cDepthBits = _this->gl_config.depth_size;
slouken@1913
   155
    pfd->cStencilBits = _this->gl_config.stencil_size;
slouken@1913
   156
}
slouken@1913
   157
slouken@2150
   158
/* Choose the closest pixel format that meets or exceeds the target.
slouken@2150
   159
   FIXME: Should we weight any particular attribute over any other?
slouken@2150
   160
*/
slouken@2150
   161
static int
slouken@2150
   162
WIN_GL_ChoosePixelFormat(HDC hdc, PIXELFORMATDESCRIPTOR * target)
slouken@2150
   163
{
slouken@2150
   164
    PIXELFORMATDESCRIPTOR pfd;
slouken@2150
   165
    int count, index, best = 0;
slouken@2150
   166
    unsigned int dist, best_dist = ~0U;
slouken@2150
   167
slouken@2150
   168
    count = DescribePixelFormat(hdc, 1, sizeof(pfd), NULL);
slouken@2150
   169
slouken@2150
   170
    for (index = 1; index <= count; index++) {
slouken@2150
   171
slouken@2150
   172
        if (!DescribePixelFormat(hdc, index, sizeof(pfd), &pfd)) {
slouken@2150
   173
            continue;
slouken@2150
   174
        }
slouken@2150
   175
slouken@2150
   176
        if ((pfd.dwFlags & target->dwFlags) != target->dwFlags) {
slouken@2150
   177
            continue;
slouken@2150
   178
        }
slouken@2150
   179
slouken@2150
   180
        if (pfd.iLayerType != target->iLayerType) {
slouken@2150
   181
            continue;
slouken@2150
   182
        }
slouken@2150
   183
        if (pfd.iPixelType != target->iPixelType) {
slouken@2150
   184
            continue;
slouken@2150
   185
        }
slouken@2150
   186
slouken@2150
   187
        dist = 0;
slouken@2150
   188
slouken@2150
   189
        if (pfd.cColorBits < target->cColorBits) {
slouken@2150
   190
            continue;
slouken@2150
   191
        } else {
slouken@2150
   192
            dist += (pfd.cColorBits - target->cColorBits);
slouken@2150
   193
        }
slouken@2150
   194
        if (pfd.cRedBits < target->cRedBits) {
slouken@2150
   195
            continue;
slouken@2150
   196
        } else {
slouken@2150
   197
            dist += (pfd.cRedBits - target->cRedBits);
slouken@2150
   198
        }
slouken@2150
   199
        if (pfd.cGreenBits < target->cGreenBits) {
slouken@2150
   200
            continue;
slouken@2150
   201
        } else {
slouken@2150
   202
            dist += (pfd.cGreenBits - target->cGreenBits);
slouken@2150
   203
        }
slouken@2150
   204
        if (pfd.cBlueBits < target->cBlueBits) {
slouken@2150
   205
            continue;
slouken@2150
   206
        } else {
slouken@2150
   207
            dist += (pfd.cBlueBits - target->cBlueBits);
slouken@2150
   208
        }
slouken@2150
   209
        if (pfd.cAlphaBits < target->cAlphaBits) {
slouken@2150
   210
            continue;
slouken@2150
   211
        } else {
slouken@2150
   212
            dist += (pfd.cAlphaBits - target->cAlphaBits);
slouken@2150
   213
        }
slouken@2150
   214
        if (pfd.cAccumBits < target->cAccumBits) {
slouken@2150
   215
            continue;
slouken@2150
   216
        } else {
slouken@2150
   217
            dist += (pfd.cAccumBits - target->cAccumBits);
slouken@2150
   218
        }
slouken@2150
   219
        if (pfd.cAccumRedBits < target->cAccumRedBits) {
slouken@2150
   220
            continue;
slouken@2150
   221
        } else {
slouken@2150
   222
            dist += (pfd.cAccumRedBits - target->cAccumRedBits);
slouken@2150
   223
        }
slouken@2150
   224
        if (pfd.cAccumGreenBits < target->cAccumGreenBits) {
slouken@2150
   225
            continue;
slouken@2150
   226
        } else {
slouken@2150
   227
            dist += (pfd.cAccumGreenBits - target->cAccumGreenBits);
slouken@2150
   228
        }
slouken@2150
   229
        if (pfd.cAccumBlueBits < target->cAccumBlueBits) {
slouken@2150
   230
            continue;
slouken@2150
   231
        } else {
slouken@2150
   232
            dist += (pfd.cAccumBlueBits - target->cAccumBlueBits);
slouken@2150
   233
        }
slouken@2150
   234
        if (pfd.cAccumAlphaBits < target->cAccumAlphaBits) {
slouken@2150
   235
            continue;
slouken@2150
   236
        } else {
slouken@2150
   237
            dist += (pfd.cAccumAlphaBits - target->cAccumAlphaBits);
slouken@2150
   238
        }
slouken@2150
   239
        if (pfd.cDepthBits < target->cDepthBits) {
slouken@2150
   240
            continue;
slouken@2150
   241
        } else {
slouken@2150
   242
            dist += (pfd.cDepthBits - target->cDepthBits);
slouken@2150
   243
        }
slouken@2150
   244
        if (pfd.cStencilBits < target->cStencilBits) {
slouken@2150
   245
            continue;
slouken@2150
   246
        } else {
slouken@2150
   247
            dist += (pfd.cStencilBits - target->cStencilBits);
slouken@2150
   248
        }
slouken@2150
   249
slouken@2150
   250
        if (dist < best_dist) {
slouken@2150
   251
            best = index;
slouken@2150
   252
            best_dist = dist;
slouken@2150
   253
        }
slouken@2150
   254
    }
slouken@2150
   255
slouken@2150
   256
    return best;
slouken@2150
   257
}
slouken@2150
   258
slouken@1913
   259
static SDL_bool
slouken@1913
   260
HasExtension(const char *extension, const char *extensions)
slouken@1913
   261
{
slouken@1913
   262
    const char *start;
slouken@1913
   263
    const char *where, *terminator;
slouken@1913
   264
slouken@1913
   265
    /* Extension names should not have spaces. */
slouken@1913
   266
    where = SDL_strchr(extension, ' ');
slouken@1913
   267
    if (where || *extension == '\0')
slouken@1913
   268
        return SDL_FALSE;
slouken@1913
   269
slouken@1913
   270
    if (!extensions)
slouken@1913
   271
        return SDL_FALSE;
slouken@1913
   272
slouken@1913
   273
    /* It takes a bit of care to be fool-proof about parsing the
slouken@1913
   274
     * OpenGL extensions string. Don't be fooled by sub-strings,
slouken@1913
   275
     * etc. */
slouken@1913
   276
slouken@1913
   277
    start = extensions;
slouken@1913
   278
slouken@1913
   279
    for (;;) {
slouken@1913
   280
        where = SDL_strstr(start, extension);
slouken@1913
   281
        if (!where)
slouken@1913
   282
            break;
slouken@1913
   283
slouken@1913
   284
        terminator = where + SDL_strlen(extension);
slouken@1913
   285
        if (where == start || *(where - 1) == ' ')
slouken@1913
   286
            if (*terminator == ' ' || *terminator == '\0')
slouken@1913
   287
                return SDL_TRUE;
slouken@1913
   288
slouken@1913
   289
        start = terminator;
slouken@1913
   290
    }
slouken@1913
   291
    return SDL_FALSE;
slouken@1913
   292
}
slouken@1913
   293
slouken@1913
   294
static void
slouken@2178
   295
WIN_GL_InitExtensions(_THIS, HDC hdc)
slouken@1913
   296
{
slouken@1913
   297
    const char *(WINAPI * wglGetExtensionsStringARB) (HDC) = 0;
slouken@1913
   298
    const char *extensions;
slouken@1913
   299
slouken@1913
   300
    wglGetExtensionsStringARB = (const char *(WINAPI *) (HDC))
slouken@1913
   301
        _this->gl_data->wglGetProcAddress("wglGetExtensionsStringARB");
slouken@1913
   302
    if (wglGetExtensionsStringARB) {
slouken@1913
   303
        extensions = wglGetExtensionsStringARB(hdc);
slouken@1913
   304
    } else {
slouken@1913
   305
        extensions = NULL;
slouken@1913
   306
    }
slouken@1913
   307
slouken@1913
   308
    /* Check for WGL_ARB_pixel_format */
slouken@1913
   309
    _this->gl_data->WGL_ARB_pixel_format = 0;
slouken@1913
   310
    if (HasExtension("WGL_ARB_pixel_format", extensions)) {
slouken@1913
   311
        _this->gl_data->wglChoosePixelFormatARB = (BOOL(WINAPI *)
slouken@1913
   312
                                                   (HDC, const int *,
slouken@1913
   313
                                                    const FLOAT *, UINT,
slouken@1913
   314
                                                    int *, UINT *))
slouken@1913
   315
            WIN_GL_GetProcAddress(_this, "wglChoosePixelFormatARB");
slouken@1913
   316
        _this->gl_data->wglGetPixelFormatAttribivARB =
slouken@1913
   317
            (BOOL(WINAPI *) (HDC, int, int, UINT, const int *, int *))
slouken@1913
   318
            WIN_GL_GetProcAddress(_this, "wglGetPixelFormatAttribivARB");
slouken@1913
   319
slouken@1913
   320
        if ((_this->gl_data->wglChoosePixelFormatARB != NULL) &&
slouken@1913
   321
            (_this->gl_data->wglGetPixelFormatAttribivARB != NULL)) {
slouken@1913
   322
            _this->gl_data->WGL_ARB_pixel_format = 1;
slouken@1913
   323
        }
slouken@1913
   324
    }
slouken@1913
   325
slouken@1913
   326
    /* Check for WGL_EXT_swap_control */
slouken@1913
   327
    if (HasExtension("WGL_EXT_swap_control", extensions)) {
slouken@1913
   328
        _this->gl_data->wglSwapIntervalEXT =
slouken@1913
   329
            WIN_GL_GetProcAddress(_this, "wglSwapIntervalEXT");
slouken@1913
   330
        _this->gl_data->wglGetSwapIntervalEXT =
slouken@1913
   331
            WIN_GL_GetProcAddress(_this, "wglGetSwapIntervalEXT");
slouken@2178
   332
    } else {
slouken@2178
   333
        _this->gl_data->wglSwapIntervalEXT = NULL;
slouken@2178
   334
        _this->gl_data->wglGetSwapIntervalEXT = NULL;
slouken@1913
   335
    }
slouken@2178
   336
}
slouken@1913
   337
slouken@2178
   338
static int
slouken@2178
   339
WIN_GL_ChoosePixelFormatARB(_THIS, int *iAttribs, float *fAttribs)
slouken@2178
   340
{
slouken@2178
   341
    HWND hwnd;
slouken@2178
   342
    HDC hdc;
slouken@2180
   343
    PIXELFORMATDESCRIPTOR pfd;
slouken@2178
   344
    HGLRC hglrc;
slouken@2178
   345
    int pixel_format = 0;
slouken@2178
   346
    unsigned int matching;
slouken@2178
   347
slouken@2178
   348
    hwnd =
slouken@2178
   349
        CreateWindow(SDL_Appname, SDL_Appname, (WS_POPUP | WS_DISABLED), 0, 0,
slouken@2178
   350
                     10, 10, NULL, NULL, SDL_Instance, NULL);
slouken@2178
   351
    WIN_PumpEvents(_this);
slouken@2178
   352
slouken@2178
   353
    hdc = GetDC(hwnd);
slouken@2178
   354
slouken@2180
   355
    WIN_GL_SetupPixelFormat(_this, &pfd);
slouken@2180
   356
slouken@2180
   357
    SetPixelFormat(hdc, ChoosePixelFormat(hdc, &pfd), &pfd);
slouken@2180
   358
slouken@2178
   359
    hglrc = _this->gl_data->wglCreateContext(hdc);
slouken@1913
   360
    if (hglrc) {
slouken@2178
   361
        _this->gl_data->wglMakeCurrent(hdc, hglrc);
slouken@2178
   362
slouken@2178
   363
        WIN_GL_InitExtensions(_this, hdc);
slouken@2178
   364
slouken@2178
   365
        if (_this->gl_data->WGL_ARB_pixel_format) {
slouken@2178
   366
            _this->gl_data->wglChoosePixelFormatARB(hdc, iAttribs, fAttribs,
slouken@2178
   367
                                                    1, &pixel_format,
slouken@2178
   368
                                                    &matching);
slouken@2178
   369
        }
slouken@2178
   370
slouken@1913
   371
        _this->gl_data->wglMakeCurrent(NULL, NULL);
slouken@1913
   372
        _this->gl_data->wglDeleteContext(hglrc);
slouken@1913
   373
    }
slouken@1913
   374
    ReleaseDC(hwnd, hdc);
slouken@1913
   375
    DestroyWindow(hwnd);
slouken@1913
   376
    WIN_PumpEvents(_this);
slouken@2178
   377
slouken@2178
   378
    return pixel_format;
slouken@1913
   379
}
slouken@1913
   380
slouken@1913
   381
static int
slouken@1913
   382
WIN_GL_Initialize(_THIS)
slouken@1913
   383
{
slouken@1913
   384
    if (_this->gl_data) {
slouken@1913
   385
        ++_this->gl_data->initialized;
slouken@1913
   386
        return 0;
slouken@1913
   387
    }
slouken@1913
   388
slouken@1913
   389
    _this->gl_data =
slouken@1913
   390
        (struct SDL_GLDriverData *) SDL_calloc(1,
slouken@1913
   391
                                               sizeof(struct
slouken@1913
   392
                                                      SDL_GLDriverData));
slouken@1913
   393
    if (!_this->gl_data) {
slouken@1913
   394
        SDL_OutOfMemory();
slouken@1913
   395
        return -1;
slouken@1913
   396
    }
slouken@1913
   397
    _this->gl_data->initialized = 1;
slouken@1913
   398
slouken@1913
   399
    if (WIN_GL_LoadLibrary(_this, NULL) < 0) {
slouken@1913
   400
        return -1;
slouken@1913
   401
    }
slouken@1913
   402
slouken@1913
   403
    return 0;
slouken@1913
   404
}
slouken@1913
   405
slouken@1952
   406
static void
slouken@1952
   407
WIN_GL_Shutdown(_THIS)
slouken@1952
   408
{
slouken@1952
   409
    if (!_this->gl_data || (--_this->gl_data->initialized > 0)) {
slouken@1952
   410
        return;
slouken@1952
   411
    }
slouken@1952
   412
slouken@1952
   413
    WIN_GL_UnloadLibrary(_this);
slouken@1952
   414
slouken@1952
   415
    SDL_free(_this->gl_data);
slouken@1952
   416
    _this->gl_data = NULL;
slouken@1952
   417
}
slouken@1952
   418
slouken@1913
   419
int
slouken@1913
   420
WIN_GL_SetupWindow(_THIS, SDL_Window * window)
slouken@1913
   421
{
slouken@1913
   422
    HDC hdc = ((SDL_WindowData *) window->driverdata)->hdc;
slouken@1913
   423
    PIXELFORMATDESCRIPTOR pfd;
slouken@1913
   424
    int pixel_format;
slouken@1913
   425
    int iAttribs[64];
slouken@1913
   426
    int *iAttr;
slouken@1913
   427
    float fAttribs[1] = { 0 };
slouken@1913
   428
slouken@1913
   429
    if (WIN_GL_Initialize(_this) < 0) {
slouken@1913
   430
        return -1;
slouken@1913
   431
    }
slouken@1913
   432
slouken@1913
   433
    WIN_GL_SetupPixelFormat(_this, &pfd);
slouken@1913
   434
slouken@1913
   435
    /* setup WGL_ARB_pixel_format attribs */
slouken@1913
   436
    iAttr = &iAttribs[0];
slouken@1913
   437
slouken@1913
   438
    *iAttr++ = WGL_DRAW_TO_WINDOW_ARB;
slouken@1913
   439
    *iAttr++ = GL_TRUE;
slouken@1913
   440
    *iAttr++ = WGL_ACCELERATION_ARB;
slouken@1913
   441
    *iAttr++ = WGL_FULL_ACCELERATION_ARB;
slouken@1913
   442
    *iAttr++ = WGL_RED_BITS_ARB;
slouken@1913
   443
    *iAttr++ = _this->gl_config.red_size;
slouken@1913
   444
    *iAttr++ = WGL_GREEN_BITS_ARB;
slouken@1913
   445
    *iAttr++ = _this->gl_config.green_size;
slouken@1913
   446
    *iAttr++ = WGL_BLUE_BITS_ARB;
slouken@1913
   447
    *iAttr++ = _this->gl_config.blue_size;
slouken@1913
   448
slouken@1913
   449
    if (_this->gl_config.alpha_size) {
slouken@1913
   450
        *iAttr++ = WGL_ALPHA_BITS_ARB;
slouken@1913
   451
        *iAttr++ = _this->gl_config.alpha_size;
slouken@1913
   452
    }
slouken@1913
   453
slouken@1913
   454
    *iAttr++ = WGL_DOUBLE_BUFFER_ARB;
slouken@1913
   455
    *iAttr++ = _this->gl_config.double_buffer;
slouken@1913
   456
slouken@1913
   457
    *iAttr++ = WGL_DEPTH_BITS_ARB;
slouken@1913
   458
    *iAttr++ = _this->gl_config.depth_size;
slouken@1913
   459
slouken@1913
   460
    if (_this->gl_config.stencil_size) {
slouken@1913
   461
        *iAttr++ = WGL_STENCIL_BITS_ARB;
slouken@1913
   462
        *iAttr++ = _this->gl_config.stencil_size;
slouken@1913
   463
    }
slouken@1913
   464
slouken@1913
   465
    if (_this->gl_config.accum_red_size) {
slouken@1913
   466
        *iAttr++ = WGL_ACCUM_RED_BITS_ARB;
slouken@1913
   467
        *iAttr++ = _this->gl_config.accum_red_size;
slouken@1913
   468
    }
slouken@1913
   469
slouken@1913
   470
    if (_this->gl_config.accum_green_size) {
slouken@1913
   471
        *iAttr++ = WGL_ACCUM_GREEN_BITS_ARB;
slouken@1913
   472
        *iAttr++ = _this->gl_config.accum_green_size;
slouken@1913
   473
    }
slouken@1913
   474
slouken@1913
   475
    if (_this->gl_config.accum_blue_size) {
slouken@1913
   476
        *iAttr++ = WGL_ACCUM_BLUE_BITS_ARB;
slouken@1913
   477
        *iAttr++ = _this->gl_config.accum_blue_size;
slouken@1913
   478
    }
slouken@1913
   479
slouken@1913
   480
    if (_this->gl_config.accum_alpha_size) {
slouken@1913
   481
        *iAttr++ = WGL_ACCUM_ALPHA_BITS_ARB;
slouken@1913
   482
        *iAttr++ = _this->gl_config.accum_alpha_size;
slouken@1913
   483
    }
slouken@1913
   484
slouken@1913
   485
    if (_this->gl_config.stereo) {
slouken@1913
   486
        *iAttr++ = WGL_STEREO_ARB;
slouken@1913
   487
        *iAttr++ = GL_TRUE;
slouken@1913
   488
    }
slouken@1913
   489
slouken@1913
   490
    if (_this->gl_config.multisamplebuffers) {
slouken@1913
   491
        *iAttr++ = WGL_SAMPLE_BUFFERS_ARB;
slouken@1913
   492
        *iAttr++ = _this->gl_config.multisamplebuffers;
slouken@1913
   493
    }
slouken@1913
   494
slouken@1913
   495
    if (_this->gl_config.multisamplesamples) {
slouken@1913
   496
        *iAttr++ = WGL_SAMPLES_ARB;
slouken@1913
   497
        *iAttr++ = _this->gl_config.multisamplesamples;
slouken@1913
   498
    }
slouken@1913
   499
slouken@1913
   500
    if (_this->gl_config.accelerated >= 0) {
slouken@1913
   501
        *iAttr++ = WGL_ACCELERATION_ARB;
slouken@1913
   502
        *iAttr++ =
slouken@1913
   503
            (_this->gl_config.
slouken@1913
   504
             accelerated ? WGL_GENERIC_ACCELERATION_ARB :
slouken@1913
   505
             WGL_NO_ACCELERATION_ARB);
slouken@1913
   506
    }
slouken@1913
   507
slouken@1913
   508
    *iAttr = 0;
slouken@1913
   509
slouken@1913
   510
    /* Choose and set the closest available pixel format */
slouken@2178
   511
    pixel_format = WIN_GL_ChoosePixelFormatARB(_this, iAttribs, fAttribs);
slouken@2178
   512
    if (!pixel_format) {
slouken@2150
   513
        pixel_format = WIN_GL_ChoosePixelFormat(hdc, &pfd);
slouken@1913
   514
    }
slouken@1913
   515
    if (!pixel_format) {
slouken@1913
   516
        SDL_SetError("No matching GL pixel format available");
slouken@1913
   517
        return -1;
slouken@1913
   518
    }
slouken@1913
   519
    if (!SetPixelFormat(hdc, pixel_format, &pfd)) {
slouken@1913
   520
        WIN_SetError("SetPixelFormat()");
slouken@1913
   521
        return (-1);
slouken@1913
   522
    }
slouken@1913
   523
    return 0;
slouken@1913
   524
}
slouken@1913
   525
slouken@1913
   526
void
slouken@1913
   527
WIN_GL_CleanupWindow(_THIS, SDL_Window * window)
slouken@1913
   528
{
slouken@1913
   529
    WIN_GL_Shutdown(_this);
slouken@1913
   530
}
slouken@1913
   531
slouken@1913
   532
SDL_GLContext
slouken@1913
   533
WIN_GL_CreateContext(_THIS, SDL_Window * window)
slouken@1913
   534
{
slouken@1913
   535
    HDC hdc = ((SDL_WindowData *) window->driverdata)->hdc;
slouken@2178
   536
    HGLRC context;
slouken@1913
   537
slouken@2178
   538
    context = _this->gl_data->wglCreateContext(hdc);
slouken@2178
   539
    if (!context) {
slouken@2178
   540
        SDL_SetError("Could not create GL context");
slouken@2178
   541
        return NULL;
slouken@2178
   542
    }
slouken@2178
   543
slouken@2178
   544
    if (WIN_GL_MakeCurrent(_this, window, context) < 0) {
slouken@2178
   545
        WIN_GL_DeleteContext(_this, context);
slouken@2178
   546
        return NULL;
slouken@2178
   547
    }
slouken@2178
   548
slouken@2178
   549
    WIN_GL_InitExtensions(_this, hdc);
slouken@2178
   550
slouken@2178
   551
    return context;
slouken@1913
   552
}
slouken@1913
   553
slouken@1913
   554
int
slouken@1913
   555
WIN_GL_MakeCurrent(_THIS, SDL_Window * window, SDL_GLContext context)
slouken@1913
   556
{
slouken@1913
   557
    HDC hdc;
slouken@1913
   558
    int status;
slouken@1913
   559
slouken@1913
   560
    if (window) {
slouken@1913
   561
        hdc = ((SDL_WindowData *) window->driverdata)->hdc;
slouken@1913
   562
    } else {
slouken@1913
   563
        hdc = NULL;
slouken@1913
   564
    }
slouken@1913
   565
    if (!_this->gl_data->wglMakeCurrent(hdc, (HGLRC) context)) {
slouken@1913
   566
        WIN_SetError("wglMakeCurrent()");
slouken@1913
   567
        status = -1;
slouken@1913
   568
    } else {
slouken@1913
   569
        status = 0;
slouken@1913
   570
    }
slouken@1913
   571
    return status;
slouken@1913
   572
}
slouken@1913
   573
slouken@1913
   574
int
slouken@1913
   575
WIN_GL_SetSwapInterval(_THIS, int interval)
slouken@1913
   576
{
slouken@1913
   577
    if (_this->gl_data->wglSwapIntervalEXT) {
slouken@1913
   578
        _this->gl_data->wglSwapIntervalEXT(interval);
slouken@1913
   579
        return 0;
slouken@1913
   580
    } else {
slouken@1913
   581
        SDL_Unsupported();
slouken@1913
   582
        return -1;
slouken@1913
   583
    }
slouken@1913
   584
}
slouken@1913
   585
slouken@1913
   586
int
slouken@1913
   587
WIN_GL_GetSwapInterval(_THIS)
slouken@1913
   588
{
slouken@1913
   589
    if (_this->gl_data->wglGetSwapIntervalEXT) {
slouken@1913
   590
        return _this->gl_data->wglGetSwapIntervalEXT();
slouken@1913
   591
    } else {
slouken@1913
   592
        SDL_Unsupported();
slouken@1913
   593
        return -1;
slouken@1913
   594
    }
slouken@1913
   595
}
slouken@1913
   596
slouken@1913
   597
void
slouken@1913
   598
WIN_GL_SwapWindow(_THIS, SDL_Window * window)
slouken@1913
   599
{
slouken@1913
   600
    HDC hdc = ((SDL_WindowData *) window->driverdata)->hdc;
slouken@1913
   601
slouken@1913
   602
    SwapBuffers(hdc);
slouken@1913
   603
}
slouken@1913
   604
slouken@1913
   605
void
slouken@1913
   606
WIN_GL_DeleteContext(_THIS, SDL_GLContext context)
slouken@1913
   607
{
slouken@1936
   608
    _this->gl_data->wglDeleteContext((HGLRC) context);
slouken@1913
   609
}
slouken@1913
   610
slouken@1952
   611
#endif /* SDL_VIDEO_OPENGL_WGL */
slouken@1913
   612
slouken@1913
   613
/* vi: set ts=4 sw=4 expandtab: */