From 2630d4968c2de1079c775ea0856ea68dddf583ae Mon Sep 17 00:00:00 2001
From: Camilla Berglund
Date: Wed, 13 Oct 2010 04:04:43 +0200
Subject: [PATCH] Initial implementation of experimental gamma ramp API.
---
CMakeLists.txt | 15 ++--
include/GL/glfw3.h | 15 ++++
readme.html | 3 +
src/gamma.c | 115 +++++++++++++++++++++++++
src/internal.h | 18 ++--
src/win32/CMakeLists.txt | 1 +
src/win32/platform.h | 9 ++
src/win32/win32_gamma.c | 67 +++++++++++++++
src/win32/win32_init.c | 30 ++++++-
src/x11/CMakeLists.txt | 1 +
src/x11/platform.h | 16 ++--
src/x11/x11_fullscreen.c | 2 +-
src/x11/x11_gamma.c | 124 +++++++++++++++++++++++++++
src/x11/x11_init.c | 79 +++++++++++++++--
tests/CMakeLists.txt | 5 +-
tests/gamma.c | 178 +++++++++++++++++++++++++++++++++++++++
16 files changed, 647 insertions(+), 31 deletions(-)
create mode 100644 src/gamma.c
create mode 100644 src/win32/win32_gamma.c
create mode 100644 src/x11/x11_gamma.c
create mode 100644 tests/gamma.c
diff --git a/CMakeLists.txt b/CMakeLists.txt
index b2313305..ba17da7e 100644
--- a/CMakeLists.txt
+++ b/CMakeLists.txt
@@ -20,6 +20,7 @@ set(common_SOURCES
${GLFW_SOURCE_DIR}/src/enable.c
${GLFW_SOURCE_DIR}/src/error.c
${GLFW_SOURCE_DIR}/src/fullscreen.c
+ ${GLFW_SOURCE_DIR}/src/gamma.c
${GLFW_SOURCE_DIR}/src/glext.c
${GLFW_SOURCE_DIR}/src/init.c
${GLFW_SOURCE_DIR}/src/input.c
@@ -67,14 +68,12 @@ if (UNIX AND NOT APPLE AND NOT CYGWIN)
endif(X11_XRANDR_FOUND)
# Check for Xf86VidMode (fallback legacy resolution switching extension)
- if (NOT X11_XRANDR_FOUND)
- CHECK_X11_XF86VIDMODE()
- if (X11_XF86VIDMODE_FOUND)
- set(_GLFW_HAS_XF86VIDMODE 1)
- list(APPEND GLFW_INCLUDE_DIR ${X11_XF86VIDMODE_INCLUDE_DIR})
- list(APPEND GLFW_LIBRARIES ${X11_XF86VIDMODE_LIBRARIES})
- endif(X11_XF86VIDMODE_FOUND)
- endif (NOT X11_XRANDR_FOUND)
+ CHECK_X11_XF86VIDMODE()
+ if (X11_XF86VIDMODE_FOUND)
+ set(_GLFW_HAS_XF86VIDMODE 1)
+ list(APPEND GLFW_INCLUDE_DIR ${X11_XF86VIDMODE_INCLUDE_DIR})
+ list(APPEND GLFW_LIBRARIES ${X11_XF86VIDMODE_LIBRARIES})
+ endif(X11_XF86VIDMODE_FOUND)
CHECK_FUNCTION_EXISTS(glXGetProcAddress _GLFW_HAS_GLXGETPROCADDRESS)
diff --git a/include/GL/glfw3.h b/include/GL/glfw3.h
index 138c7c50..08e0d006 100644
--- a/include/GL/glfw3.h
+++ b/include/GL/glfw3.h
@@ -361,6 +361,8 @@ extern "C" {
#define GLFW_VERSION_UNAVAILABLE 0x00070007
#define GLFW_PLATFORM_ERROR 0x00070008
+/* Gamma ramps */
+#define GLFW_GAMMA_RAMP_SIZE 256
/*************************************************************************
* Typedefs
@@ -379,6 +381,14 @@ typedef struct
int greenBits;
} GLFWvidmode;
+/* Gamma ramp */
+typedef struct
+{
+ unsigned short red[GLFW_GAMMA_RAMP_SIZE];
+ unsigned short green[GLFW_GAMMA_RAMP_SIZE];
+ unsigned short blue[GLFW_GAMMA_RAMP_SIZE];
+} GLFWgammaramp;
+
/* Function pointer types */
typedef void (* GLFWwindowsizefun)(GLFWwindow,int,int);
typedef int (* GLFWwindowclosefun)(GLFWwindow);
@@ -410,6 +420,11 @@ GLFWAPI const char* glfwErrorString(int error);
GLFWAPI int glfwGetVideoModes(GLFWvidmode* list, int maxcount);
GLFWAPI void glfwGetDesktopMode(GLFWvidmode* mode);
+/* Gamma ramp functions */
+GLFWAPI void glfwSetGammaFormula(float gamma, float blacklevel, float gain);
+GLFWAPI void glfwGetGammaRamp(GLFWgammaramp* ramp);
+GLFWAPI void glfwSetGammaRamp(const GLFWgammaramp* ramp);
+
/* Window handling */
GLFWAPI GLFWwindow glfwOpenWindow(int width, int height, int mode, const char* title, GLFWwindow share);
GLFWAPI void glfwOpenWindowHint(int target, int hint);
diff --git a/readme.html b/readme.html
index a11b614d..1482a13d 100644
--- a/readme.html
+++ b/readme.html
@@ -802,6 +802,9 @@ their skills. Special thanks go out to:
Jonathan Dummer, for submitting a patch fixing an input bug on Win32 and
adding logic for the GLFW_ICON
resource
+ Ralph Eastwood, for the initial design of the gamma correction API and
+ the Win32 and X11 implementations of it
+
Gerald Franz, who made GLFW compile under IRIX, and supplied patches
for the X11 keyboard translation routine
diff --git a/src/gamma.c b/src/gamma.c
new file mode 100644
index 00000000..ff97d600
--- /dev/null
+++ b/src/gamma.c
@@ -0,0 +1,115 @@
+//========================================================================
+// GLFW - An OpenGL framework
+// Platform: Any
+// API version: 3.0
+// WWW: http://www.glfw.org/
+//------------------------------------------------------------------------
+// Copyright (c) 2010 Camilla Berglund
+//
+// 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"
+
+#include
+#include
+
+
+//////////////////////////////////////////////////////////////////////////
+////// GLFW public API //////
+//////////////////////////////////////////////////////////////////////////
+
+//========================================================================
+// Calculate the gamma ramp table from specified values
+//========================================================================
+
+GLFWAPI void glfwSetGammaFormula(float gamma, float blacklevel, float gain)
+{
+ int i, size = 256;
+ GLFWgammaramp ramp;
+
+ if (!_glfwInitialized)
+ {
+ _glfwSetError(GLFW_NOT_INITIALIZED);
+ return;
+ }
+
+ for (i = 0; i < size; i++)
+ {
+ float value = (float) i / ((float) (size - 1));
+
+ // Apply gamma
+ value = pow(value, gamma) * 65535.f + 0.5f;
+
+ // Apply gain
+ value = gain * (value - 32767.5f) + 32767.5f;
+
+ // Apply black-level
+ value += blacklevel * 65535.f;
+
+ // Clamp values
+ if (value < 0.f)
+ value = 0.f;
+ else if (value > 65535.f)
+ value = 65535.f;
+
+ // Set the gamma ramp values
+ ramp.red[i] = (unsigned short) value;
+ ramp.green[i] = (unsigned short) value;
+ ramp.blue[i] = (unsigned short) value;
+ }
+
+ glfwSetGammaRamp(&ramp);
+}
+
+
+//========================================================================
+// Return the currently set gamma ramp
+//========================================================================
+
+GLFWAPI void glfwGetGammaRamp(GLFWgammaramp* ramp)
+{
+ if (!_glfwInitialized)
+ {
+ _glfwSetError(GLFW_NOT_INITIALIZED);
+ return;
+ }
+
+ *ramp = _glfwLibrary.currentRamp;
+}
+
+
+//========================================================================
+// Make the specified gamma ramp current
+//========================================================================
+
+GLFWAPI void glfwSetGammaRamp(const GLFWgammaramp* ramp)
+{
+ if (!_glfwInitialized)
+ {
+ _glfwSetError(GLFW_NOT_INITIALIZED);
+ return;
+ }
+
+ _glfwPlatformSetGammaRamp(ramp);
+ _glfwLibrary.currentRamp = *ramp;
+}
+
diff --git a/src/internal.h b/src/internal.h
index d073d44b..fb7fef90 100644
--- a/src/internal.h
+++ b/src/internal.h
@@ -215,12 +215,16 @@ struct _GLFWwindow
//------------------------------------------------------------------------
struct _GLFWlibrary
{
- _GLFWhints hints;
+ _GLFWhints hints;
- _GLFWwindow* windowListHead;
- _GLFWwindow* currentWindow;
- _GLFWwindow* activeWindow;
- _GLFWwindow* cursorLockWindow;
+ _GLFWwindow* windowListHead;
+ _GLFWwindow* currentWindow;
+ _GLFWwindow* activeWindow;
+ _GLFWwindow* cursorLockWindow;
+
+ GLFWgammaramp currentRamp;
+ GLFWgammaramp originalRamp;
+ int originalRampSize;
_GLFW_PLATFORM_LIBRARY_STATE;
};
@@ -257,6 +261,10 @@ void _glfwPlatformDisableSystemKeys(_GLFWwindow* window);
int _glfwPlatformGetVideoModes(GLFWvidmode* list, int maxcount);
void _glfwPlatformGetDesktopMode(GLFWvidmode* mode);
+// Gamma ramp
+void _glfwPlatformSaveGammaRamp(void);
+void _glfwPlatformSetGammaRamp(const GLFWgammaramp* ramp);
+
// Joystick
int _glfwPlatformGetJoystickParam(int joy, int param);
int _glfwPlatformGetJoystickPos(int joy, float* pos, int numaxes);
diff --git a/src/win32/CMakeLists.txt b/src/win32/CMakeLists.txt
index 50b160d4..f7df1ff7 100644
--- a/src/win32/CMakeLists.txt
+++ b/src/win32/CMakeLists.txt
@@ -23,6 +23,7 @@ set(libglfw_SOURCES
${common_SOURCES}
win32_enable.c
win32_fullscreen.c
+ win32_gamma.c
win32_glext.c
win32_init.c
win32_joystick.c
diff --git a/src/win32/platform.h b/src/win32/platform.h
index 8a2a2ffa..fb9107b8 100644
--- a/src/win32/platform.h
+++ b/src/win32/platform.h
@@ -156,6 +156,8 @@ typedef int (WINAPI * DESCRIBEPIXELFORMAT_T) (HDC,int,UINT,LPPIXELFORMATDESCRIP
typedef int (WINAPI * GETPIXELFORMAT_T) (HDC);
typedef BOOL (WINAPI * SETPIXELFORMAT_T) (HDC,int,const PIXELFORMATDESCRIPTOR*);
typedef BOOL (WINAPI * SWAPBUFFERS_T) (HDC);
+typedef BOOL (WINAPI * GETDEVICEGAMMARAMP_T) (HDC,PVOID);
+typedef BOOL (WINAPI * SETDEVICEGAMMARAMP_T) (HDC,PVOID);
#endif // _GLFW_NO_DLOAD_GDI32
// winmm.dll function pointer typedefs
@@ -174,12 +176,16 @@ typedef DWORD (WINAPI * TIMEGETTIME_T) (void);
#define _glfw_GetPixelFormat _glfwLibrary.Win32.gdi.GetPixelFormat
#define _glfw_SetPixelFormat _glfwLibrary.Win32.gdi.SetPixelFormat
#define _glfw_SwapBuffers _glfwLibrary.Win32.gdi.SwapBuffers
+#define _glfw_GetDeviceGammaRamp _glfwLibrary.Win32.gdi.GetDeviceGammaRamp
+#define _glfw_SetDeviceGammaRamp _glfwLibrary.Win32.gdi.SetDeviceGammaRamp
#else
#define _glfw_ChoosePixelFormat ChoosePixelFormat
#define _glfw_DescribePixelFormat DescribePixelFormat
#define _glfw_GetPixelFormat GetPixelFormat
#define _glfw_SetPixelFormat SetPixelFormat
#define _glfw_SwapBuffers SwapBuffers
+#define _glfw_GetDeviceGammaRamp GetDeviceGammaRamp
+#define _glfw_SetDeviceGammaRamp SetDeviceGammaRamp
#endif // _GLFW_NO_DLOAD_GDI32
// winmm.dll shortcuts
@@ -264,6 +270,7 @@ typedef struct _GLFWlibraryWin32
ATOM classAtom; // Window class atom
HHOOK keyboardHook; // Keyboard hook handle
DWORD foregroundLockTimeout;
+ HDC desktopDC;
// Default monitor
struct {
@@ -291,6 +298,8 @@ typedef struct _GLFWlibraryWin32
GETPIXELFORMAT_T GetPixelFormat;
SETPIXELFORMAT_T SetPixelFormat;
SWAPBUFFERS_T SwapBuffers;
+ GETDEVICEGAMMARAMP_T GetDeviceGammaRamp;
+ SETDEVICEGAMMARAMP_T SetDeviceGammaRamp;
} gdi;
#endif // _GLFW_NO_DLOAD_GDI32
diff --git a/src/win32/win32_gamma.c b/src/win32/win32_gamma.c
new file mode 100644
index 00000000..71d2645f
--- /dev/null
+++ b/src/win32/win32_gamma.c
@@ -0,0 +1,67 @@
+//========================================================================
+// GLFW - An OpenGL framework
+// Platform: Win32/WGL
+// API version: 2.7
+// WWW: http://www.glfw.org/
+//------------------------------------------------------------------------
+// Copyright (c) 2002-2006 Marcus Geelnard
+// Copyright (c) 2006-2010 Camilla Berglund
+//
+// 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"
+
+#include
+
+
+//************************************************************************
+//**** GLFW internal functions ****
+//************************************************************************
+
+//========================================================================
+// Save the gamma ramp to our internal copy
+//========================================================================
+
+void _glfwPlatformSaveGammaRamp(int ramp)
+{
+ if (!_glfwLibrary.gammaSize)
+ {
+ return;
+ }
+ _glfw_GetDeviceGammaRamp(_glfwLibrary.Win32.desktopDC,
+ _glfwLibrary.gammaRamp[ramp]);
+}
+
+
+//========================================================================
+// Restore the gamma ramp to our internal copy of the gamma ramp
+//========================================================================
+
+void _glfwPlatformRestoreGammaRamp(int ramp)
+{
+ if (!_glfwLibrary.gammaSize)
+ {
+ return;
+ }
+ _glfw_SetDeviceGammaRamp(_glfwLibrary.Win32.desktopDC,
+ _glfwLibrary.gammaRamp[ramp]);
+}
diff --git a/src/win32/win32_init.c b/src/win32/win32_init.c
index 3e71ad6e..54939cc3 100644
--- a/src/win32/win32_init.c
+++ b/src/win32/win32_init.c
@@ -30,6 +30,8 @@
#include "internal.h"
+#include
+
#ifdef __BORLANDC__
// With the Borland C++ compiler, we want to disable FPU exceptions
#include
@@ -59,12 +61,18 @@ static GLboolean initLibraries(void)
GetProcAddress(_glfwLibrary.Win32.gdi.instance, "SetPixelFormat");
_glfwLibrary.Win32.gdi.SwapBuffers = (SWAPBUFFERS_T)
GetProcAddress(_glfwLibrary.Win32.gdi.instance, "SwapBuffers");
+ _glfwLibrary.Win32.gdi.GetDeviceGammaRamp = (GETDEVICEGAMMARAMP_T)
+ GetProcAddress(_glfwLibrary.Win32.gdi.instance, "GetDeviceGammaRamp");
+ _glfwLibrary.Win32.gdi.SetDeviceGammaRamp = (SETDEVICEGAMMARAMP_T)
+ GetProcAddress(_glfwLibrary.Win32.gdi.instance, "SetDeviceGammaRamp");
if (!_glfwLibrary.Win32.gdi.ChoosePixelFormat ||
!_glfwLibrary.Win32.gdi.DescribePixelFormat ||
!_glfwLibrary.Win32.gdi.GetPixelFormat ||
!_glfwLibrary.Win32.gdi.SetPixelFormat ||
- !_glfwLibrary.Win32.gdi.SwapBuffers)
+ !_glfwLibrary.Win32.gdi.SwapBuffers ||
+ !_glfwLibrary.Win32.gdi.GetDeviceGammaRamp ||
+ !_glfwLibrary.Win32.gdi.SetDeviceGammaRamp)
{
return GL_FALSE;
}
@@ -152,6 +160,19 @@ int _glfwPlatformInit(void)
_glfwLibrary.Win32.instance = GetModuleHandle(NULL);
+ // Initialise the internal gamma ramp
+ _glfwLibrary.gammaSize = 256;
+ _glfwLibrary.gammaRamp[GLFW_GAMMA_ORIG] =
+ malloc(256 * sizeof(unsigned short) * 3);
+ _glfwLibrary.gammaRamp[GLFW_GAMMA_CURR] =
+ malloc(256 * sizeof(unsigned short) * 3);
+
+ // Get the desktop DC
+ _glfwLibrary.Win32.desktopDC = GetDC(GetDesktopWindow());
+
+ // Save the original gamma ramp
+ _glfwPlatformSaveGammaRamp(GLFW_GAMMA_ORIG);
+
_glfwInitTimer();
return GL_TRUE;
@@ -164,6 +185,13 @@ int _glfwPlatformInit(void)
int _glfwPlatformTerminate(void)
{
+ // Restore the original gamma ramp
+ _glfwPlatformRestoreGammaRamp(GLFW_GAMMA_ORIG);
+
+ // Free the gamma ramps
+ free(_glfwLibrary.gammaRamp[GLFW_GAMMA_ORIG]);
+ free(_glfwLibrary.gammaRamp[GLFW_GAMMA_CURR]);
+
if (_glfwLibrary.Win32.classAtom)
{
UnregisterClass(_GLFW_WNDCLASSNAME, _glfwLibrary.Win32.instance);
diff --git a/src/x11/CMakeLists.txt b/src/x11/CMakeLists.txt
index 2e20dadc..a75adf7b 100644
--- a/src/x11/CMakeLists.txt
+++ b/src/x11/CMakeLists.txt
@@ -12,6 +12,7 @@ set(libglfw_SOURCES
${common_SOURCES}
x11_enable.c
x11_fullscreen.c
+ x11_gamma.c
x11_glext.c
x11_init.c
x11_joystick.c
diff --git a/src/x11/platform.h b/src/x11/platform.h
index 1e3668be..5c497777 100644
--- a/src/x11/platform.h
+++ b/src/x11/platform.h
@@ -52,10 +52,6 @@
#error "GLX header version 1.3 or above is required"
#endif
-#if defined(_GLFW_HAS_XF86VIDMODE) && defined(_GLFW_HAS_XRANDR)
- #error "Xf86VidMode and RandR extensions cannot both be enabled"
-#endif
-
// With XFree86, we can use the XF86VidMode extension
#if defined(_GLFW_HAS_XF86VIDMODE)
#include
@@ -156,15 +152,18 @@ typedef struct _GLFWlibraryX11
int glxMajor, glxMinor;
struct {
- int available;
+ GLboolean available;
int eventBase;
int errorBase;
} XF86VidMode;
struct {
- int available;
+ GLboolean available;
int eventBase;
int errorBase;
+ int majorVersion;
+ int minorVersion;
+ GLboolean gammaBroken;
} XRandR;
// Screensaver data
@@ -179,14 +178,13 @@ typedef struct _GLFWlibraryX11
// Fullscreen data
struct {
int modeChanged;
-#if defined(_GLFW_HAS_XF86VIDMODE)
- XF86VidModeModeInfo oldMode;
-#endif
#if defined(_GLFW_HAS_XRANDR)
SizeID oldSizeID;
int oldWidth;
int oldHeight;
Rotation oldRotation;
+#elif defined(_GLFW_HAS_XF86VIDMODE)
+ XF86VidModeModeInfo oldMode;
#endif
} FS;
diff --git a/src/x11/x11_fullscreen.c b/src/x11/x11_fullscreen.c
index a80ed918..17ce6584 100644
--- a/src/x11/x11_fullscreen.c
+++ b/src/x11/x11_fullscreen.c
@@ -480,7 +480,7 @@ void _glfwPlatformGetDesktopMode(GLFWvidmode* mode)
{
Display* dpy;
int bpp, screen;
-#if defined(_GLFW_HAS_XF86VIDMODE)
+#if !defined(_GLFW_HAS_XRANDR) && defined(_GLFW_HAS_XF86VIDMODE)
XF86VidModeModeInfo** modelist;
int modecount;
#endif
diff --git a/src/x11/x11_gamma.c b/src/x11/x11_gamma.c
new file mode 100644
index 00000000..779859f3
--- /dev/null
+++ b/src/x11/x11_gamma.c
@@ -0,0 +1,124 @@
+//========================================================================
+// GLFW - An OpenGL framework
+// Platform: X11/GLX
+// API version: 3.0
+// WWW: http://www.glfw.org/
+//------------------------------------------------------------------------
+// Copyright (c) 2010 Camilla Berglund
+//
+// 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"
+
+#include
+#include
+
+
+//************************************************************************
+//**** GLFW internal functions ****
+//************************************************************************
+
+//========================================================================
+// Save the original gamma ramp so that we can restore it later
+//========================================================================
+
+void _glfwPlatformSaveGammaRamp(void)
+{
+ if (_glfwLibrary.X11.XRandR.available &&
+ !_glfwLibrary.X11.XRandR.gammaBroken)
+ {
+#if defined (_GLFW_HAS_XRANDR)
+ size_t size = GLFW_GAMMA_RAMP_SIZE * sizeof(unsigned short);
+
+ XRRScreenResources* rr = XRRGetScreenResources(_glfwLibrary.X11.display,
+ _glfwLibrary.X11.root);
+
+ XRRCrtcGamma* gamma = XRRGetCrtcGamma(_glfwLibrary.X11.display,
+ rr->crtcs[0]);
+
+ memcpy(_glfwLibrary.originalRamp.red, gamma->red, size);
+ memcpy(_glfwLibrary.originalRamp.green, gamma->green, size);
+ memcpy(_glfwLibrary.originalRamp.blue, gamma->blue, size);
+
+ XRRFreeGamma(gamma);
+ XRRFreeScreenResources(rr);
+ }
+#endif
+ else if (_glfwLibrary.X11.XF86VidMode.available)
+ {
+#if defined (_GLFW_HAS_XF86VIDMODE)
+ XF86VidModeGetGammaRamp(_glfwLibrary.X11.display,
+ _glfwLibrary.X11.screen,
+ GLFW_GAMMA_RAMP_SIZE,
+ _glfwLibrary.originalRamp.red,
+ _glfwLibrary.originalRamp.green,
+ _glfwLibrary.originalRamp.blue);
+#endif
+ }
+}
+
+
+//========================================================================
+// Make the specified gamma ramp current
+//========================================================================
+
+void _glfwPlatformSetGammaRamp(const GLFWgammaramp* ramp)
+{
+ if (_glfwLibrary.X11.XRandR.available &&
+ !_glfwLibrary.X11.XRandR.gammaBroken)
+ {
+#if defined (_GLFW_HAS_XRANDR)
+ int i;
+ size_t size = GLFW_GAMMA_RAMP_SIZE * sizeof(unsigned short);
+
+ XRRScreenResources* rr = XRRGetScreenResources(_glfwLibrary.X11.display,
+ _glfwLibrary.X11.root);
+
+ // Update gamma per monitor
+ for (i = 0; i < rr->ncrtc; i++)
+ {
+ XRRCrtcGamma* gamma = XRRAllocGamma(GLFW_GAMMA_RAMP_SIZE);
+
+ memcpy(gamma->red, ramp->red, size);
+ memcpy(gamma->green, ramp->green, size);
+ memcpy(gamma->blue, ramp->blue, size);
+
+ XRRSetCrtcGamma(_glfwLibrary.X11.display, rr->crtcs[i], gamma);
+ XRRFreeGamma(gamma);
+ }
+
+ XRRFreeScreenResources(rr);
+ }
+#endif
+ else if (_glfwLibrary.X11.XF86VidMode.available)
+ {
+#if defined (_GLFW_HAS_XF86VIDMODE)
+ XF86VidModeSetGammaRamp(_glfwLibrary.X11.display,
+ _glfwLibrary.X11.screen,
+ GLFW_GAMMA_RAMP_SIZE,
+ (unsigned short*) ramp->red,
+ (unsigned short*) ramp->green,
+ (unsigned short*) ramp->blue);
+#endif
+ }
+}
+
diff --git a/src/x11/x11_init.c b/src/x11/x11_init.c
index 6eab3c5f..b6529168 100644
--- a/src/x11/x11_init.c
+++ b/src/x11/x11_init.c
@@ -31,6 +31,7 @@
#include "internal.h"
#include
+#include
//========================================================================
@@ -62,7 +63,7 @@ static void initLibraries(void)
//========================================================================
-// Initialize X11 display
+// Initialize X11 display and look for supported X11 extensions
//========================================================================
static GLboolean initDisplay(void)
@@ -88,7 +89,7 @@ static GLboolean initDisplay(void)
&_glfwLibrary.X11.XF86VidMode.eventBase,
&_glfwLibrary.X11.XF86VidMode.errorBase);
#else
- _glfwLibrary.X11.XF86VidMode.available = 0;
+ _glfwLibrary.X11.XF86VidMode.available = GL_FALSE;
#endif
// Check for XRandR extension
@@ -97,11 +98,17 @@ static GLboolean initDisplay(void)
XRRQueryExtension(_glfwLibrary.X11.display,
&_glfwLibrary.X11.XRandR.eventBase,
&_glfwLibrary.X11.XRandR.errorBase);
+
+ if (!XRRQueryVersion(_glfwLibrary.X11.display,
+ &_glfwLibrary.X11.XRandR.majorVersion,
+ &_glfwLibrary.X11.XRandR.minorVersion))
+ {
+ fprintf(stderr, "Unable to query RandR version number\n");
+ }
#else
- _glfwLibrary.X11.XRandR.available = 0;
+ _glfwLibrary.X11.XRandR.available = GL_FALSE;
#endif
- // Fullscreen & screen saver settings
// Check if GLX is supported on this display
if (!glXQueryExtension(_glfwLibrary.X11.display, NULL, NULL))
{
@@ -123,6 +130,58 @@ static GLboolean initDisplay(void)
}
+//========================================================================
+// 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
+ if (_glfwLibrary.X11.XRandR.available &&
+ (_glfwLibrary.X11.XRandR.majorVersion > 1 ||
+ _glfwLibrary.X11.XRandR.majorVersion == 1 &&
+ _glfwLibrary.X11.XRandR.minorVersion >= 2))
+ {
+ // 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
+ _glfwLibrary.X11.XRandR.gammaBroken = GL_TRUE;
+ }
+
+ XRRFreeScreenResources(rr);
+ }
+#endif
+
+#if defined(_GLFW_HAS_XF86VIDMODE)
+ if (_glfwLibrary.X11.XF86VidMode.available &&
+ !_glfwLibrary.originalRampSize)
+ {
+ // Get the gamma size using XF86VidMode
+ XF86VidModeGetGammaRampSize(_glfwLibrary.X11.display,
+ _glfwLibrary.X11.screen,
+ &_glfwLibrary.originalRampSize);
+ }
+#endif
+
+ if (!_glfwLibrary.originalRampSize)
+ fprintf(stderr, "Gamma ramp setting unsupported\n");
+
+ // Save the original gamma ramp
+ _glfwPlatformSaveGammaRamp();
+}
+
+
//========================================================================
// Create a blank cursor (for locked mouse mode)
//========================================================================
@@ -159,8 +218,15 @@ static Cursor createNULLCursor(void)
static void terminateDisplay(void)
{
+ if (_glfwLibrary.originalRampSize)
+ {
+ _glfwPlatformSetGammaRamp(&_glfwLibrary.originalRamp);
+ }
+
if (_glfwLibrary.X11.display)
{
+ _glfwPlatformSetGammaRamp(&_glfwLibrary.originalRamp);
+
XCloseDisplay(_glfwLibrary.X11.display);
_glfwLibrary.X11.display = NULL;
}
@@ -180,6 +246,8 @@ int _glfwPlatformInit(void)
if (!initDisplay())
return GL_FALSE;
+ initGammaRamp();
+
_glfwLibrary.X11.cursor = createNULLCursor();
// Try to load libGL.so if necessary
@@ -232,7 +300,8 @@ const char* _glfwPlatformGetVersionString(void)
const char* version = "GLFW " _GLFW_VERSION_FULL
#if defined(_GLFW_HAS_XRANDR)
" XRandR"
-#elif defined(_GLFW_HAS_XF86VIDMODE)
+#endif
+#if defined(_GLFW_HAS_XF86VIDMODE)
" Xf86VidMode"
#else
" (no mode switching support)"
diff --git a/tests/CMakeLists.txt b/tests/CMakeLists.txt
index f55a1900..9d53b9f2 100644
--- a/tests/CMakeLists.txt
+++ b/tests/CMakeLists.txt
@@ -6,6 +6,7 @@ include_directories(${CMAKE_CURRENT_SOURCE_DIR}/../include ${OPENGL_INCLUDE_DIR}
add_executable(defaults defaults.c)
add_executable(events events.c)
add_executable(fsfocus fsfocus.c)
+add_executable(gamma gamma.c getopt.c)
add_executable(iconify iconify.c getopt.c)
add_executable(joysticks joysticks.c)
add_executable(listmodes listmodes.c)
@@ -30,8 +31,8 @@ else()
endif(APPLE)
set(WINDOWS_BINARIES accuracy sharing tearing windows)
-set(CONSOLE_BINARIES defaults events fsaa fsfocus iconify joysticks listmodes
- peter reopen version)
+set(CONSOLE_BINARIES defaults events fsaa fsfocus gamma iconify joysticks
+ listmodes peter reopen version)
if(MSVC)
# Tell MSVC to use main instead of WinMain for Windows subsystem executables
diff --git a/tests/gamma.c b/tests/gamma.c
new file mode 100644
index 00000000..e918a156
--- /dev/null
+++ b/tests/gamma.c
@@ -0,0 +1,178 @@
+//========================================================================
+// Gamma correction test program
+// Copyright (c) Camilla Berglund
+//
+// 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.
+//
+//========================================================================
+//
+// This program is used to test the gamma correction functionality for
+// both fullscreen and windowed mode windows
+//
+//========================================================================
+
+#include
+
+#include
+#include
+
+#include "getopt.h"
+
+static GLfloat ggamma = 1.0f;
+static GLfloat ggain = 1.0f;
+static GLfloat gblacklevel = 0.0f;
+
+static void usage(void)
+{
+ printf("Usage: gammatest [-h] [-f]\n");
+}
+
+static void key_callback(GLFWwindow window, int key, int action)
+{
+ if (action != GLFW_PRESS)
+ return;
+
+ switch (key)
+ {
+ case GLFW_KEY_ESC:
+ glfwCloseWindow(window);
+ break;
+ case 'Q':
+ ggamma += 0.1f;
+ printf("Gamma: %f\n", ggamma);
+ glfwSetGammaFormula( ggamma, gblacklevel, ggain );
+ break;
+ case 'W':
+ ggamma -= 0.1f;
+ printf("Gamma: %f\n", ggamma);
+ glfwSetGammaFormula( ggamma, gblacklevel, ggain );
+ break;
+ case 'A':
+ ggain += 0.1f;
+ printf("Gain: %f\n", ggain);
+ glfwSetGammaFormula( ggamma, gblacklevel, ggain );
+ break;
+ case 'S':
+ ggain -= 0.1f;
+ printf("Gain: %f\n", ggain);
+ glfwSetGammaFormula( ggamma, gblacklevel, ggain );
+ break;
+ case 'Z':
+ gblacklevel += 0.1f;
+ printf("Black Level: %f\n", gblacklevel);
+ glfwSetGammaFormula( ggamma, gblacklevel, ggain );
+ break;
+ case 'X':
+ gblacklevel -= 0.1f;
+ printf("Black Level: %f\n", gblacklevel);
+ glfwSetGammaFormula( ggamma, gblacklevel, ggain );
+ break;
+
+ }
+}
+
+static void size_callback(GLFWwindow window, int width, int height)
+{
+ glViewport(0, 0, width, height);
+}
+
+int main(int argc, char** argv)
+{
+ int width, height, ch;
+ int mode = GLFW_WINDOWED;
+ GLFWwindow window;
+
+ while ((ch = getopt(argc, argv, "fh")) != -1)
+ {
+ switch (ch)
+ {
+ case 'h':
+ usage();
+ exit(EXIT_SUCCESS);
+
+ case 'f':
+ mode = GLFW_FULLSCREEN;
+ break;
+
+ default:
+ usage();
+ exit(EXIT_FAILURE);
+ }
+ }
+
+ if (!glfwInit())
+ {
+ fprintf(stderr, "Failed to initialize GLFW: %s\n", glfwErrorString(glfwGetError()));
+ exit(EXIT_FAILURE);
+ }
+
+ if (mode == GLFW_FULLSCREEN)
+ {
+ GLFWvidmode mode;
+ glfwGetDesktopMode(&mode);
+ width = mode.width;
+ height = mode.height;
+ }
+ else
+ {
+ width = 0;
+ height = 0;
+ }
+
+ window = glfwOpenWindow(width, height, mode, "Gamma Test", NULL);
+ if (!window)
+ {
+ glfwTerminate();
+
+ fprintf(stderr, "Failed to open GLFW window: %s\n", glfwErrorString(glfwGetError()));
+ exit(EXIT_FAILURE);
+ }
+
+ printf("Gamma: %f\nGain: %f\nBlack Level: %f\n",
+ ggamma, ggain, gblacklevel);
+
+ glfwSwapInterval(1);
+ glfwSetKeyCallback(window, key_callback);
+ glfwSetWindowSizeCallback(window, size_callback);
+
+ glEnable(GL_SCISSOR_TEST);
+
+ while (glfwIsWindow(window))
+ {
+ int width, height;
+
+ glfwGetWindowSize(window, &width, &height);
+
+ glScissor(0, 0, width, height);
+ glClearColor(0.5f, 0.5f, 0.5f, 0);
+ glClear(GL_COLOR_BUFFER_BIT);
+
+ glScissor(0, 0, 640, 480);
+ glClearColor(0.8f, 0.2f, 0.4f, 0);
+ glClear(GL_COLOR_BUFFER_BIT);
+
+ glfwSwapBuffers();
+ glfwPollEvents();
+ }
+
+ glfwTerminate();
+ exit(EXIT_SUCCESS);
+}
+