2010-09-07 15:34:51 +00:00
|
|
|
//========================================================================
|
|
|
|
// GLFW - An OpenGL framework
|
|
|
|
// Platform: X11/GLX
|
2010-09-07 15:41:26 +00:00
|
|
|
// API version: 3.0
|
2010-09-07 15:34:51 +00:00
|
|
|
// WWW: http://www.glfw.org/
|
|
|
|
//------------------------------------------------------------------------
|
|
|
|
// Copyright (c) 2002-2006 Marcus Geelnard
|
|
|
|
// Copyright (c) 2006-2010 Camilla Berglund <elmindreda@elmindreda.org>
|
|
|
|
//
|
|
|
|
// This software is provided 'as-is', without any express or implied
|
|
|
|
// warranty. In no event will the authors be held liable for any damages
|
|
|
|
// arising from the use of this software.
|
|
|
|
//
|
|
|
|
// Permission is granted to anyone to use this software for any purpose,
|
|
|
|
// including commercial applications, and to alter it and redistribute it
|
|
|
|
// freely, subject to the following restrictions:
|
|
|
|
//
|
|
|
|
// 1. The origin of this software must not be misrepresented; you must not
|
|
|
|
// claim that you wrote the original software. If you use this software
|
|
|
|
// in a product, an acknowledgment in the product documentation would
|
|
|
|
// be appreciated but is not required.
|
|
|
|
//
|
|
|
|
// 2. Altered source versions must be plainly marked as such, and must not
|
|
|
|
// be misrepresented as being the original software.
|
|
|
|
//
|
|
|
|
// 3. This notice may not be removed or altered from any source
|
|
|
|
// distribution.
|
|
|
|
//
|
|
|
|
//========================================================================
|
|
|
|
|
|
|
|
#include "internal.h"
|
|
|
|
|
2010-09-08 15:01:39 +00:00
|
|
|
#include <stdio.h>
|
2010-10-13 02:04:43 +00:00
|
|
|
#include <stdlib.h>
|
2010-09-08 15:01:39 +00:00
|
|
|
|
2010-09-07 15:34:51 +00:00
|
|
|
|
|
|
|
//========================================================================
|
|
|
|
// Dynamically load libraries
|
|
|
|
//========================================================================
|
|
|
|
|
2010-09-08 13:51:25 +00:00
|
|
|
static void initLibraries(void)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
|
|
|
#ifdef _GLFW_DLOPEN_LIBGL
|
|
|
|
int i;
|
2010-09-08 13:58:43 +00:00
|
|
|
char* libGL_names[ ] =
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
|
|
|
"libGL.so",
|
|
|
|
"libGL.so.1",
|
|
|
|
"/usr/lib/libGL.so",
|
|
|
|
"/usr/lib/libGL.so.1",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
2010-09-09 18:25:33 +00:00
|
|
|
_glfwLibrary.X11.libGL = NULL;
|
2010-09-08 13:51:25 +00:00
|
|
|
for (i = 0; libGL_names[i] != NULL; i++)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
2010-09-09 18:25:33 +00:00
|
|
|
_glfwLibrary.X11.libGL = dlopen(libGL_names[i], RTLD_LAZY | RTLD_GLOBAL);
|
|
|
|
if (_glfwLibrary.X11.libGL)
|
2010-09-07 15:34:51 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-01-02 10:18:14 +00:00
|
|
|
//========================================================================
|
|
|
|
// Update the key code LUT
|
|
|
|
//========================================================================
|
|
|
|
|
|
|
|
static void updateKeyCodeLUT(void)
|
|
|
|
{
|
|
|
|
int i, keyCode, keyCodeGLFW;
|
|
|
|
char name[XkbKeyNameLength+1];
|
|
|
|
XkbDescPtr descr;
|
|
|
|
|
|
|
|
// Clear the LUT
|
|
|
|
for (i = 0; i < 256; ++i)
|
|
|
|
{
|
|
|
|
_glfwLibrary.X11.Xkb.keyCodeLUT[i] = -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// This functionality requires the Xkb extension
|
|
|
|
if (!_glfwLibrary.X11.Xkb.available)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get keyboard description
|
|
|
|
descr = XkbGetKeyboard(_glfwLibrary.X11.display,
|
|
|
|
XkbAllComponentsMask,
|
|
|
|
XkbUseCoreKbd);
|
|
|
|
|
|
|
|
// Find the X11 key code -> GLFW key code mapping
|
|
|
|
for (keyCode = descr->min_key_code; keyCode <= descr->max_key_code; ++keyCode)
|
|
|
|
{
|
|
|
|
// Get the key name
|
|
|
|
for (i = 0; i < XkbKeyNameLength; ++i)
|
|
|
|
{
|
|
|
|
name[i] = descr->names->keys[keyCode].name[i];
|
|
|
|
}
|
|
|
|
name[XkbKeyNameLength] = 0;
|
|
|
|
|
|
|
|
// Map the key name to a GLFW key code. Note: We only map printable
|
|
|
|
// keys here, and we use the US keyboard layout.
|
|
|
|
if (strcmp(name, "TLDE") == 0) keyCodeGLFW = GLFW_KEY_GRAVE_ACCENT;
|
|
|
|
else if (strcmp(name, "AE01") == 0) keyCodeGLFW = GLFW_KEY_1;
|
|
|
|
else if (strcmp(name, "AE02") == 0) keyCodeGLFW = GLFW_KEY_2;
|
|
|
|
else if (strcmp(name, "AE03") == 0) keyCodeGLFW = GLFW_KEY_3;
|
|
|
|
else if (strcmp(name, "AE04") == 0) keyCodeGLFW = GLFW_KEY_4;
|
|
|
|
else if (strcmp(name, "AE05") == 0) keyCodeGLFW = GLFW_KEY_5;
|
|
|
|
else if (strcmp(name, "AE06") == 0) keyCodeGLFW = GLFW_KEY_6;
|
|
|
|
else if (strcmp(name, "AE07") == 0) keyCodeGLFW = GLFW_KEY_7;
|
|
|
|
else if (strcmp(name, "AE08") == 0) keyCodeGLFW = GLFW_KEY_8;
|
|
|
|
else if (strcmp(name, "AE09") == 0) keyCodeGLFW = GLFW_KEY_9;
|
|
|
|
else if (strcmp(name, "AE10") == 0) keyCodeGLFW = GLFW_KEY_0;
|
|
|
|
else if (strcmp(name, "AE11") == 0) keyCodeGLFW = GLFW_KEY_MINUS;
|
|
|
|
else if (strcmp(name, "AE12") == 0) keyCodeGLFW = GLFW_KEY_EQUAL;
|
|
|
|
else if (strcmp(name, "AD01") == 0) keyCodeGLFW = GLFW_KEY_Q;
|
|
|
|
else if (strcmp(name, "AD02") == 0) keyCodeGLFW = GLFW_KEY_W;
|
|
|
|
else if (strcmp(name, "AD03") == 0) keyCodeGLFW = GLFW_KEY_E;
|
|
|
|
else if (strcmp(name, "AD04") == 0) keyCodeGLFW = GLFW_KEY_R;
|
|
|
|
else if (strcmp(name, "AD05") == 0) keyCodeGLFW = GLFW_KEY_T;
|
|
|
|
else if (strcmp(name, "AD06") == 0) keyCodeGLFW = GLFW_KEY_Y;
|
|
|
|
else if (strcmp(name, "AD07") == 0) keyCodeGLFW = GLFW_KEY_U;
|
|
|
|
else if (strcmp(name, "AD08") == 0) keyCodeGLFW = GLFW_KEY_I;
|
|
|
|
else if (strcmp(name, "AD09") == 0) keyCodeGLFW = GLFW_KEY_O;
|
|
|
|
else if (strcmp(name, "AD10") == 0) keyCodeGLFW = GLFW_KEY_P;
|
|
|
|
else if (strcmp(name, "AD11") == 0) keyCodeGLFW = GLFW_KEY_LEFT_SQUARE_BRACKET;
|
|
|
|
else if (strcmp(name, "AD12") == 0) keyCodeGLFW = GLFW_KEY_RIGHT_SQUARE_BRACKET;
|
|
|
|
else if (strcmp(name, "AC01") == 0) keyCodeGLFW = GLFW_KEY_A;
|
|
|
|
else if (strcmp(name, "AC02") == 0) keyCodeGLFW = GLFW_KEY_S;
|
|
|
|
else if (strcmp(name, "AC03") == 0) keyCodeGLFW = GLFW_KEY_D;
|
|
|
|
else if (strcmp(name, "AC04") == 0) keyCodeGLFW = GLFW_KEY_F;
|
|
|
|
else if (strcmp(name, "AC05") == 0) keyCodeGLFW = GLFW_KEY_G;
|
|
|
|
else if (strcmp(name, "AC06") == 0) keyCodeGLFW = GLFW_KEY_H;
|
|
|
|
else if (strcmp(name, "AC07") == 0) keyCodeGLFW = GLFW_KEY_J;
|
|
|
|
else if (strcmp(name, "AC08") == 0) keyCodeGLFW = GLFW_KEY_K;
|
|
|
|
else if (strcmp(name, "AC09") == 0) keyCodeGLFW = GLFW_KEY_L;
|
|
|
|
else if (strcmp(name, "AC10") == 0) keyCodeGLFW = GLFW_KEY_SEMICOLON;
|
|
|
|
else if (strcmp(name, "AC11") == 0) keyCodeGLFW = GLFW_KEY_APOSTROPHE;
|
|
|
|
else if (strcmp(name, "AB01") == 0) keyCodeGLFW = GLFW_KEY_Z;
|
|
|
|
else if (strcmp(name, "AB02") == 0) keyCodeGLFW = GLFW_KEY_X;
|
|
|
|
else if (strcmp(name, "AB03") == 0) keyCodeGLFW = GLFW_KEY_C;
|
|
|
|
else if (strcmp(name, "AB04") == 0) keyCodeGLFW = GLFW_KEY_V;
|
|
|
|
else if (strcmp(name, "AB05") == 0) keyCodeGLFW = GLFW_KEY_B;
|
|
|
|
else if (strcmp(name, "AB06") == 0) keyCodeGLFW = GLFW_KEY_N;
|
|
|
|
else if (strcmp(name, "AB07") == 0) keyCodeGLFW = GLFW_KEY_M;
|
|
|
|
else if (strcmp(name, "AB08") == 0) keyCodeGLFW = GLFW_KEY_COMMA;
|
|
|
|
else if (strcmp(name, "AB09") == 0) keyCodeGLFW = GLFW_KEY_PERIOD;
|
|
|
|
else if (strcmp(name, "AB10") == 0) keyCodeGLFW = GLFW_KEY_SLASH;
|
|
|
|
else if (strcmp(name, "BKSL") == 0) keyCodeGLFW = GLFW_KEY_BACKSLASH;
|
|
|
|
else if (strcmp(name, "LSGT") == 0) keyCodeGLFW = GLFW_KEY_WORLD_1;
|
|
|
|
else keyCodeGLFW = -1;
|
|
|
|
|
|
|
|
// Update the key code LUT
|
|
|
|
if ((keyCode >= 0) && (keyCode < 256))
|
|
|
|
{
|
|
|
|
_glfwLibrary.X11.Xkb.keyCodeLUT[keyCode] = keyCodeGLFW;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Free the keyboard description
|
|
|
|
XkbFreeKeyboard(descr, 0, True);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-07 15:34:51 +00:00
|
|
|
//========================================================================
|
2010-10-13 02:04:43 +00:00
|
|
|
// Initialize X11 display and look for supported X11 extensions
|
2010-09-07 15:34:51 +00:00
|
|
|
//========================================================================
|
|
|
|
|
2010-09-09 19:52:31 +00:00
|
|
|
static GLboolean initDisplay(void)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
2010-09-09 16:15:32 +00:00
|
|
|
_glfwLibrary.X11.display = XOpenDisplay(0);
|
|
|
|
if (!_glfwLibrary.X11.display)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr, "Failed to open X display\n");
|
2010-10-04 22:08:19 +00:00
|
|
|
_glfwSetError(GLFW_OPENGL_UNAVAILABLE);
|
2010-09-07 15:34:51 +00:00
|
|
|
return GL_FALSE;
|
|
|
|
}
|
|
|
|
|
2010-09-09 21:37:50 +00:00
|
|
|
// As the API currently doesn't understand multiple display devices, we hardcode
|
|
|
|
// this choice and hope for the best
|
|
|
|
_glfwLibrary.X11.screen = DefaultScreen(_glfwLibrary.X11.display);
|
|
|
|
_glfwLibrary.X11.root = RootWindow(_glfwLibrary.X11.display,
|
|
|
|
_glfwLibrary.X11.screen);
|
|
|
|
|
2010-09-07 15:34:51 +00:00
|
|
|
// Check for XF86VidMode extension
|
|
|
|
#ifdef _GLFW_HAS_XF86VIDMODE
|
2010-10-13 21:05:17 +00:00
|
|
|
_glfwLibrary.X11.VidMode.available =
|
2010-09-09 16:15:32 +00:00
|
|
|
XF86VidModeQueryExtension(_glfwLibrary.X11.display,
|
2010-10-13 21:05:17 +00:00
|
|
|
&_glfwLibrary.X11.VidMode.eventBase,
|
|
|
|
&_glfwLibrary.X11.VidMode.errorBase);
|
2010-09-07 15:34:51 +00:00
|
|
|
#else
|
2010-10-13 21:05:17 +00:00
|
|
|
_glfwLibrary.X11.VidMode.available = GL_FALSE;
|
2010-10-14 13:13:51 +00:00
|
|
|
#endif /*_GLFW_HAS_XF86VIDMODE*/
|
2010-09-07 15:34:51 +00:00
|
|
|
|
|
|
|
// Check for XRandR extension
|
|
|
|
#ifdef _GLFW_HAS_XRANDR
|
2010-10-13 21:05:17 +00:00
|
|
|
_glfwLibrary.X11.RandR.available =
|
2010-09-09 16:15:32 +00:00
|
|
|
XRRQueryExtension(_glfwLibrary.X11.display,
|
2010-10-13 21:05:17 +00:00
|
|
|
&_glfwLibrary.X11.RandR.eventBase,
|
|
|
|
&_glfwLibrary.X11.RandR.errorBase);
|
2010-10-13 02:04:43 +00:00
|
|
|
|
|
|
|
if (!XRRQueryVersion(_glfwLibrary.X11.display,
|
2010-10-13 21:05:17 +00:00
|
|
|
&_glfwLibrary.X11.RandR.majorVersion,
|
|
|
|
&_glfwLibrary.X11.RandR.minorVersion))
|
2010-10-13 02:04:43 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr, "Unable to query RandR version number\n");
|
|
|
|
}
|
2010-09-07 15:34:51 +00:00
|
|
|
#else
|
2010-10-13 21:05:17 +00:00
|
|
|
_glfwLibrary.X11.RandR.available = GL_FALSE;
|
2010-10-14 13:13:51 +00:00
|
|
|
#endif /*_GLFW_HAS_XRANDR*/
|
2010-09-07 15:34:51 +00:00
|
|
|
|
|
|
|
// Check if GLX is supported on this display
|
2010-09-09 16:15:32 +00:00
|
|
|
if (!glXQueryExtension(_glfwLibrary.X11.display, NULL, NULL))
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr, "GLX not supported\n");
|
2010-10-04 22:08:19 +00:00
|
|
|
_glfwSetError(GLFW_OPENGL_UNAVAILABLE);
|
2010-09-07 15:34:51 +00:00
|
|
|
return GL_FALSE;
|
|
|
|
}
|
|
|
|
|
2010-09-09 16:15:32 +00:00
|
|
|
if (!glXQueryVersion(_glfwLibrary.X11.display,
|
|
|
|
&_glfwLibrary.X11.glxMajor,
|
|
|
|
&_glfwLibrary.X11.glxMinor))
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
|
|
|
fprintf(stderr, "Unable to query GLX version\n");
|
2010-10-04 22:08:19 +00:00
|
|
|
_glfwSetError(GLFW_OPENGL_UNAVAILABLE);
|
2010-09-07 15:34:51 +00:00
|
|
|
return GL_FALSE;
|
|
|
|
}
|
|
|
|
|
2011-01-02 10:18:14 +00:00
|
|
|
// Check if Xkb is supported on this display
|
|
|
|
_glfwLibrary.X11.Xkb.majorVersion = 1;
|
|
|
|
_glfwLibrary.X11.Xkb.minorVersion = 0;
|
|
|
|
_glfwLibrary.X11.Xkb.available =
|
|
|
|
XkbQueryExtension(_glfwLibrary.X11.display,
|
|
|
|
&_glfwLibrary.X11.Xkb.majorOpcode,
|
|
|
|
&_glfwLibrary.X11.Xkb.eventBase,
|
|
|
|
&_glfwLibrary.X11.Xkb.errorBase,
|
|
|
|
&_glfwLibrary.X11.Xkb.majorVersion,
|
|
|
|
&_glfwLibrary.X11.Xkb.minorVersion);
|
|
|
|
|
|
|
|
// Update the key code LUT
|
|
|
|
// FIXME: We should listen to XkbMapNotify events to track changes to
|
|
|
|
// the keyboard mapping.
|
|
|
|
updateKeyCodeLUT();
|
|
|
|
|
2010-09-07 15:34:51 +00:00
|
|
|
return GL_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-10-13 02:04:43 +00:00
|
|
|
//========================================================================
|
|
|
|
// Detect gamma ramp support and save original gamma ramp, if available
|
|
|
|
//========================================================================
|
|
|
|
|
|
|
|
static void initGammaRamp(void)
|
|
|
|
{
|
|
|
|
#ifdef _GLFW_HAS_XRANDR
|
|
|
|
// RandR gamma support is only available with version 1.2 and above
|
2010-10-13 21:05:17 +00:00
|
|
|
if (_glfwLibrary.X11.RandR.available &&
|
|
|
|
(_glfwLibrary.X11.RandR.majorVersion > 1 ||
|
|
|
|
_glfwLibrary.X11.RandR.majorVersion == 1 &&
|
|
|
|
_glfwLibrary.X11.RandR.minorVersion >= 2))
|
2010-10-13 02:04:43 +00:00
|
|
|
{
|
|
|
|
// FIXME: Assumes that all monitors have the same size gamma tables
|
|
|
|
// This is reasonable as I suspect the that if they did differ, it
|
|
|
|
// would imply that setting the gamma size to an arbitary size is
|
|
|
|
// possible as well.
|
|
|
|
XRRScreenResources* rr = XRRGetScreenResources(_glfwLibrary.X11.display,
|
|
|
|
_glfwLibrary.X11.root);
|
|
|
|
|
|
|
|
_glfwLibrary.originalRampSize = XRRGetCrtcGammaSize(_glfwLibrary.X11.display,
|
|
|
|
rr->crtcs[0]);
|
|
|
|
if (!_glfwLibrary.originalRampSize)
|
|
|
|
{
|
|
|
|
// This is probably Nvidia RandR with broken gamma support
|
|
|
|
// Flag it as useless and try Xf86VidMode below, if available
|
2010-10-13 21:05:17 +00:00
|
|
|
_glfwLibrary.X11.RandR.gammaBroken = GL_TRUE;
|
2010-10-13 20:42:49 +00:00
|
|
|
fprintf(stderr, "Ignoring broken nVidia implementation of RandR 1.2+ gamma\n");
|
2010-10-13 02:04:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
XRRFreeScreenResources(rr);
|
|
|
|
}
|
2010-10-14 13:13:51 +00:00
|
|
|
#endif /*_GLFW_HAS_XRANDR*/
|
2010-10-13 02:04:43 +00:00
|
|
|
|
|
|
|
#if defined(_GLFW_HAS_XF86VIDMODE)
|
2010-10-13 21:05:17 +00:00
|
|
|
if (_glfwLibrary.X11.VidMode.available &&
|
2010-10-13 02:04:43 +00:00
|
|
|
!_glfwLibrary.originalRampSize)
|
|
|
|
{
|
|
|
|
// Get the gamma size using XF86VidMode
|
|
|
|
XF86VidModeGetGammaRampSize(_glfwLibrary.X11.display,
|
|
|
|
_glfwLibrary.X11.screen,
|
|
|
|
&_glfwLibrary.originalRampSize);
|
|
|
|
}
|
2010-10-14 13:13:51 +00:00
|
|
|
#endif /*_GLFW_HAS_XF86VIDMODE*/
|
2010-10-13 02:04:43 +00:00
|
|
|
|
|
|
|
if (!_glfwLibrary.originalRampSize)
|
2010-10-24 14:07:02 +00:00
|
|
|
fprintf(stderr, "No supported gamma ramp API found\n");
|
2010-10-13 02:04:43 +00:00
|
|
|
|
|
|
|
// Save the original gamma ramp
|
2010-10-13 20:42:31 +00:00
|
|
|
_glfwPlatformGetGammaRamp(&_glfwLibrary.originalRamp);
|
2010-10-13 02:04:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-11 13:39:21 +00:00
|
|
|
//========================================================================
|
|
|
|
// Create a blank cursor (for locked mouse mode)
|
|
|
|
//========================================================================
|
|
|
|
|
|
|
|
static Cursor createNULLCursor(void)
|
|
|
|
{
|
|
|
|
Pixmap cursormask;
|
|
|
|
XGCValues xgc;
|
|
|
|
GC gc;
|
|
|
|
XColor col;
|
|
|
|
Cursor cursor;
|
|
|
|
|
|
|
|
// TODO: Add error checks
|
|
|
|
|
|
|
|
cursormask = XCreatePixmap(_glfwLibrary.X11.display, _glfwLibrary.X11.root, 1, 1, 1);
|
|
|
|
xgc.function = GXclear;
|
|
|
|
gc = XCreateGC(_glfwLibrary.X11.display, cursormask, GCFunction, &xgc);
|
|
|
|
XFillRectangle(_glfwLibrary.X11.display, cursormask, gc, 0, 0, 1, 1);
|
|
|
|
col.pixel = 0;
|
|
|
|
col.red = 0;
|
|
|
|
col.flags = 4;
|
|
|
|
cursor = XCreatePixmapCursor(_glfwLibrary.X11.display, cursormask, cursormask,
|
|
|
|
&col, &col, 0, 0);
|
|
|
|
XFreePixmap(_glfwLibrary.X11.display, cursormask);
|
|
|
|
XFreeGC(_glfwLibrary.X11.display, gc);
|
|
|
|
|
|
|
|
return cursor;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-07 15:34:51 +00:00
|
|
|
//========================================================================
|
|
|
|
// Terminate X11 display
|
|
|
|
//========================================================================
|
|
|
|
|
2010-09-08 13:51:25 +00:00
|
|
|
static void terminateDisplay(void)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
2010-10-13 02:04:43 +00:00
|
|
|
if (_glfwLibrary.originalRampSize)
|
|
|
|
_glfwPlatformSetGammaRamp(&_glfwLibrary.originalRamp);
|
|
|
|
|
2010-09-09 16:15:32 +00:00
|
|
|
if (_glfwLibrary.X11.display)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
2010-09-09 16:15:32 +00:00
|
|
|
XCloseDisplay(_glfwLibrary.X11.display);
|
|
|
|
_glfwLibrary.X11.display = NULL;
|
2010-09-07 15:34:51 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-09-15 14:44:43 +00:00
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
////// GLFW platform API //////
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
2010-09-07 15:34:51 +00:00
|
|
|
|
|
|
|
//========================================================================
|
|
|
|
// Initialize various GLFW state
|
|
|
|
//========================================================================
|
|
|
|
|
2010-09-08 13:51:25 +00:00
|
|
|
int _glfwPlatformInit(void)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
2010-09-08 13:51:25 +00:00
|
|
|
if (!initDisplay())
|
2010-09-07 15:34:51 +00:00
|
|
|
return GL_FALSE;
|
|
|
|
|
2010-10-13 02:04:43 +00:00
|
|
|
initGammaRamp();
|
|
|
|
|
2010-09-11 13:39:21 +00:00
|
|
|
_glfwLibrary.X11.cursor = createNULLCursor();
|
|
|
|
|
2010-09-07 15:34:51 +00:00
|
|
|
// Try to load libGL.so if necessary
|
|
|
|
initLibraries();
|
|
|
|
|
|
|
|
_glfwInitJoysticks();
|
|
|
|
|
|
|
|
// Start the timer
|
|
|
|
_glfwInitTimer();
|
|
|
|
|
|
|
|
return GL_TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//========================================================================
|
|
|
|
// Close window and shut down library
|
|
|
|
//========================================================================
|
|
|
|
|
2010-09-08 13:51:25 +00:00
|
|
|
int _glfwPlatformTerminate(void)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
2010-09-11 13:42:32 +00:00
|
|
|
if (_glfwLibrary.X11.cursor)
|
|
|
|
{
|
|
|
|
XFreeCursor(_glfwLibrary.X11.display, _glfwLibrary.X11.cursor);
|
|
|
|
_glfwLibrary.X11.cursor = (Cursor) 0;
|
|
|
|
}
|
|
|
|
|
2010-09-07 15:34:51 +00:00
|
|
|
terminateDisplay();
|
|
|
|
|
|
|
|
_glfwTerminateJoysticks();
|
|
|
|
|
|
|
|
// Unload libGL.so if necessary
|
|
|
|
#ifdef _GLFW_DLOPEN_LIBGL
|
2010-09-09 18:25:33 +00:00
|
|
|
if (_glfwLibrary.X11.libGL != NULL)
|
2010-09-07 15:34:51 +00:00
|
|
|
{
|
2010-09-09 18:25:33 +00:00
|
|
|
dlclose(_glfwLibrary.X11.libGL);
|
|
|
|
_glfwLibrary.X11.libGL = NULL;
|
2010-09-07 15:34:51 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
return GL_TRUE;
|
|
|
|
}
|
|
|
|
|
2010-09-15 14:44:43 +00:00
|
|
|
|
2010-09-13 16:05:59 +00:00
|
|
|
//========================================================================
|
|
|
|
// Get the GLFW version string
|
|
|
|
//========================================================================
|
|
|
|
|
|
|
|
const char* _glfwPlatformGetVersionString(void)
|
|
|
|
{
|
|
|
|
const char* version = "GLFW " _GLFW_VERSION_FULL
|
|
|
|
#if defined(_GLFW_HAS_XRANDR)
|
|
|
|
" XRandR"
|
2010-10-13 02:04:43 +00:00
|
|
|
#endif
|
|
|
|
#if defined(_GLFW_HAS_XF86VIDMODE)
|
2010-09-13 16:05:59 +00:00
|
|
|
" Xf86VidMode"
|
2010-11-06 12:48:24 +00:00
|
|
|
#endif
|
|
|
|
#if !defined(_GLFW_HAS_XRANDR) && !defined(_GLFW_HAS_XF86VIDMODE)
|
|
|
|
" no-mode-switching-support"
|
2010-09-13 16:05:59 +00:00
|
|
|
#endif
|
|
|
|
#if defined(_GLFW_HAS_GLXGETPROCADDRESS)
|
|
|
|
" glXGetProcAddress"
|
|
|
|
#elif defined(_GLFW_HAS_GLXGETPROCADDRESSARB)
|
|
|
|
" glXGetProcAddressARB"
|
|
|
|
#elif defined(_GLFW_HAS_GLXGETPROCADDRESSEXT)
|
|
|
|
" glXGetProcAddressEXT"
|
|
|
|
#elif defined(_GLFW_DLOPEN_LIBGL)
|
2010-10-24 12:31:46 +00:00
|
|
|
" dlsym(libGL)"
|
2010-09-13 16:08:59 +00:00
|
|
|
#else
|
2010-11-06 12:48:24 +00:00
|
|
|
" no-extension-support"
|
2010-09-13 16:05:59 +00:00
|
|
|
#endif
|
|
|
|
#if defined(_GLFW_USE_LINUX_JOYSTICKS)
|
2010-10-24 12:35:37 +00:00
|
|
|
" Linux-joystick-API"
|
2010-09-13 16:08:59 +00:00
|
|
|
#else
|
2010-10-24 12:35:37 +00:00
|
|
|
" no-joystick-support"
|
2010-09-13 16:05:59 +00:00
|
|
|
#endif
|
|
|
|
;
|
|
|
|
|
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|