lwjgl/src/native/win32/display.c

318 lines
9.9 KiB
C
Raw Normal View History

2002-08-15 11:46:18 -04:00
/*
2004-06-12 16:28:34 -04:00
* Copyright (c) 2002-2004 LWJGL Project
2002-08-15 11:46:18 -04:00
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
2002-08-11 07:49:32 -04:00
*
2002-08-15 11:46:18 -04:00
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
2002-08-11 07:49:32 -04:00
*
2004-06-12 16:28:34 -04:00
* * Neither the name of 'LWJGL' nor the names of
2002-08-15 11:46:18 -04:00
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED
* TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
* EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
* PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR
* PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF
* LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
* SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
2002-08-15 11:46:18 -04:00
/**
* $Id$
2002-08-11 07:49:32 -04:00
*
2004-07-03 06:16:28 -04:00
* Win32 specific library for display handling.
2002-08-11 07:49:32 -04:00
*
2002-08-15 11:46:18 -04:00
* @author cix_foo <cix_foo@users.sourceforge.net>
* @version $Revision$
2002-08-11 07:49:32 -04:00
*/
2004-07-03 06:16:28 -04:00
#include <windows.h>
2005-02-21 09:46:47 -05:00
// Multimon.h enables multi monitor emulation on win95 and winnt4
// So we only need the extended, multi-monitor aware path
#define COMPILE_MULTIMON_STUBS
#include <Multimon.h>
2004-07-03 06:16:28 -04:00
#include <jni.h>
#include "display.h"
2003-10-11 12:29:40 -04:00
#include "common_tools.h"
2002-08-15 10:53:18 -04:00
2002-08-17 10:13:12 -04:00
#define WINDOWCLASSNAME "LWJGLWINDOW"
2002-08-11 07:49:32 -04:00
2004-11-09 06:00:49 -05:00
#define GAMMA_SIZE (3*256)
static jobjectArray GetAvailableDisplayModesEx(JNIEnv * env);
2004-05-23 09:54:55 -04:00
static char * getDriver();
2004-11-09 06:00:49 -05:00
static bool modeSet = false; // Whether we've done a display mode change
2004-11-13 17:27:19 -05:00
static WORD originalGamma[GAMMA_SIZE]; // Original gamma settings
static WORD currentGamma[GAMMA_SIZE]; // Current gamma settings
2004-11-09 06:00:49 -05:00
static DEVMODE devmode; // Now we'll remember this value for the future
extern HWND display_hwnd; // Handle to the window
2003-02-07 16:54:31 -05:00
2004-07-03 06:16:28 -04:00
jobjectArray getAvailableDisplayModes(JNIEnv *env)
2002-08-11 07:49:32 -04:00
{
jobjectArray result = GetAvailableDisplayModesEx(env);
return result;
}
/**
2003-09-10 18:35:06 -04:00
* Choose displaymodes using extended codepath (multiple displaydevices)
*/
static jobjectArray GetAvailableDisplayModesEx(JNIEnv * env) {
int i = 0, j = 0, n = 0;
DISPLAY_DEVICE DisplayDevice;
DEVMODE DevMode;
jobject *display_mode_objects = NULL;
int list_size = 0;
jclass displayModeClass;
jobjectArray ret;
jmethodID displayModeConstructor;
2003-09-02 07:30:36 -04:00
ZeroMemory(&DevMode, sizeof(DEVMODE));
ZeroMemory(&DisplayDevice, sizeof(DISPLAY_DEVICE));
2003-03-19 07:41:28 -05:00
DevMode.dmSize = sizeof(DEVMODE);
DisplayDevice.cb = sizeof(DISPLAY_DEVICE);
displayModeClass = (*env)->FindClass(env, "org/lwjgl/opengl/DisplayMode");
displayModeConstructor = (*env)->GetMethodID(env, displayModeClass, "<init>", "(IIII)V");
/* Multi-monitor stuff commented out since we're only really interested in the primary monitor */
/* while(EnumDisplayDevices(NULL, i++, &DisplayDevice, 0) != 0) {
// continue if mirroring device
2004-03-11 14:50:42 -05:00
if((DisplayDevice.StateFlags & DISPLAY_DEVICE_MIRRORING_DRIVER) != 0) {
continue;
}
2003-09-02 07:30:36 -04:00
j = 0;
while(EnumDisplaySettings((const char *) DisplayDevice.DeviceName, j++, &DevMode) != 0) {*/
while(EnumDisplaySettings(NULL, j++, &DevMode) != 0) {
2003-03-19 07:41:28 -05:00
// Filter out indexed modes
2004-03-10 11:47:04 -05:00
if (DevMode.dmBitsPerPel > 8 && ChangeDisplaySettings(&DevMode, CDS_FULLSCREEN | CDS_TEST) == DISP_CHANGE_SUCCESSFUL) {
2003-03-19 07:41:28 -05:00
jobject displayMode;
if (list_size <= n) {
list_size += 1;
display_mode_objects = (jobject *)realloc(display_mode_objects, sizeof(jobject)*list_size);
if (display_mode_objects == NULL)
return NULL;
}
displayMode = (*env)->NewObject(env, displayModeClass, displayModeConstructor,
DevMode.dmPelsWidth, DevMode.dmPelsHeight,
DevMode.dmBitsPerPel, DevMode.dmDisplayFrequency);
display_mode_objects[n++] = displayMode;
}
2003-03-19 07:41:28 -05:00
}
// }
printfDebugJava(env, "Found %d displaymodes", n);
ret = (*env)->NewObjectArray(env, n, displayModeClass, NULL);
for (i = 0; i < n; i++) {
(*env)->SetObjectArrayElement(env, ret, i, display_mode_objects[i]);
}
free(display_mode_objects);
2003-03-19 07:41:28 -05:00
return ret;
2002-08-11 07:49:32 -04:00
}
2004-07-03 06:16:28 -04:00
void switchDisplayMode(JNIEnv * env, jobject mode)
{
jclass cls_displayMode = (*env)->GetObjectClass(env, mode);
jfieldID fid_width = (*env)->GetFieldID(env, cls_displayMode, "width", "I");
jfieldID fid_height = (*env)->GetFieldID(env, cls_displayMode, "height", "I");
jfieldID fid_bpp = (*env)->GetFieldID(env, cls_displayMode, "bpp", "I");
jfieldID fid_freq = (*env)->GetFieldID(env, cls_displayMode, "freq", "I");
int width = (*env)->GetIntField(env, mode, fid_width);
int height = (*env)->GetIntField(env, mode, fid_height);
int bpp = (*env)->GetIntField(env, mode, fid_bpp);
int freq = (*env)->GetIntField(env, mode, fid_freq);
LONG cdsret;
devmode.dmSize = sizeof(DEVMODE);
devmode.dmBitsPerPel = bpp;
devmode.dmPelsWidth = width;
devmode.dmPelsHeight = height;
devmode.dmDisplayFrequency = freq;
devmode.dmFields = DM_BITSPERPEL | DM_PELSWIDTH | DM_PELSHEIGHT;
if (freq != 0)
devmode.dmFields |= DM_DISPLAYFREQUENCY;
cdsret = ChangeDisplaySettings(&devmode, CDS_FULLSCREEN);
if (cdsret != DISP_CHANGE_SUCCESSFUL) {
/* What's the proper way to do this multiply with 2 thing, if at all necessary? */
/* // Failed: so let's check to see if it's a wierd dual screen display
printfDebugJava(env, "Failed to set display mode (%ld) ... assuming dual monitors", cdsret);
devmode.dmPelsWidth = width * 2;
cdsret = ChangeDisplaySettings(&devmode, CDS_FULLSCREEN);
if (cdsret != DISP_CHANGE_SUCCESSFUL) {
printfDebugJava(env, "Failed to set display mode using dual monitors (%ld)", cdsret);*/
throwFormattedException(env, "Failed to set display mode (%ld).", cdsret);
2003-03-28 14:02:24 -05:00
return;
// }
2002-08-11 07:49:32 -04:00
}
modeSet = true;
2002-08-11 07:49:32 -04:00
}
2004-07-03 06:16:28 -04:00
int getGammaRampLength(void)
2003-02-07 16:54:31 -05:00
{
return 256;
}
2003-02-07 16:54:31 -05:00
2004-07-03 06:16:28 -04:00
void setGammaRamp(JNIEnv * env, jobject gammaRampBuffer)
{
int i;
float scaledRampEntry;
WORD rampEntry;
HDC screenDC;
const float *gammaRamp = (const float *)(*env)->GetDirectBufferAddress(env, gammaRampBuffer);
// Turn array of floats into array of RGB WORDs
for (i = 0; i < 256; i ++) {
scaledRampEntry = gammaRamp[i]*0xffff;
rampEntry = (WORD)scaledRampEntry;
currentGamma[i] = rampEntry;
currentGamma[i + 256] = rampEntry;
currentGamma[i + 512] = rampEntry;
2003-03-29 16:52:14 -05:00
}
screenDC = GetDC(NULL);
2003-10-23 03:18:07 -04:00
if (SetDeviceGammaRamp(screenDC, currentGamma) == FALSE) {
throwException(env, "Failed to set device gamma.");
2003-03-29 16:52:14 -05:00
}
ReleaseDC(NULL, screenDC);
2003-02-07 16:54:31 -05:00
}
2003-03-21 12:08:26 -05:00
2004-07-03 06:16:28 -04:00
jobject initDisplay(JNIEnv * env)
2003-03-21 12:08:26 -05:00
{
int width;
int height;
int bpp;
int freq;
jclass jclass_DisplayMode;
jmethodID ctor;
jobject newMode;
// Determine the current screen resolution
// Get the screen
2003-03-29 16:52:14 -05:00
HDC screenDC = GetDC(NULL);
2003-03-28 14:02:24 -05:00
if (!screenDC) {
2004-07-03 06:16:28 -04:00
throwException(env, "Couldn't get screen DC!");
return NULL;
2003-03-28 14:02:24 -05:00
}
// Get the device caps
width = GetDeviceCaps(screenDC, HORZRES);
height = GetDeviceCaps(screenDC, VERTRES);
bpp = GetDeviceCaps(screenDC, BITSPIXEL);
freq = GetDeviceCaps(screenDC, VREFRESH);
if (freq <= 1)
freq = 0; // Unknown
jclass_DisplayMode = (*env)->FindClass(env, "org/lwjgl/opengl/DisplayMode");
ctor = (*env)->GetMethodID(env, jclass_DisplayMode, "<init>", "(IIII)V");
newMode = (*env)->NewObject(env, jclass_DisplayMode, ctor, width, height, bpp, freq);
2003-03-29 16:52:14 -05:00
// Get the default gamma ramp
2004-05-10 04:55:09 -04:00
if (GetDeviceGammaRamp(screenDC, originalGamma) == FALSE) {
printfDebugJava(env, "Failed to get initial device gamma");
2003-03-29 16:52:14 -05:00
}
2004-11-09 06:00:49 -05:00
memcpy(currentGamma, originalGamma, sizeof(WORD)*GAMMA_SIZE);
2003-03-29 16:52:14 -05:00
ReleaseDC(NULL, screenDC);
2004-07-03 06:16:28 -04:00
return newMode;
}
void resetDisplayMode(JNIEnv * env) {
2004-07-03 06:16:28 -04:00
// Return device gamma to normal
HDC screenDC = GetDC(NULL);
if (!SetDeviceGammaRamp(screenDC, originalGamma)) {
printfDebugJava(env, "Could not reset device gamma");
2004-07-03 06:16:28 -04:00
}
ReleaseDC(NULL, screenDC);
if (modeSet) {
modeSet = false;
// Under Win32, all we have to do is:
ChangeDisplaySettings(NULL, 0);
// And we'll call init() again to put the correct mode back in Display
if (env != NULL)
initDisplay(env);
}
}
/*
* Put display settings back to what they were when the window is maximized.
*/
void restoreDisplayMode(void) {
// Restore gamma
HDC screenDC = GetDC(NULL);
LONG cdsret;
if (!SetDeviceGammaRamp(screenDC, currentGamma)) {
printfDebug("Could not restore device gamma\n");
}
ReleaseDC(NULL, screenDC);
if (!modeSet) {
printfDebug("Attempting to restore the display mode\n");
modeSet = true;
cdsret = ChangeDisplaySettings(&devmode, CDS_FULLSCREEN);
if (cdsret != DISP_CHANGE_SUCCESSFUL) {
printfDebug("Failed to restore display mode\n");
}
2004-07-03 06:16:28 -04:00
}
2003-09-28 02:55:01 -04:00
}
jstring getVersion(JNIEnv * env, char *driver)
2003-09-28 02:55:01 -04:00
{
jstring ret = NULL;
TCHAR driverDLL[256] = "\0";
DWORD var = 0;
DWORD dwInfoSize;
LPVOID lpInfoBuff;
BOOL bRetval;
2003-09-28 02:55:01 -04:00
if (driver == NULL) {
return NULL;
}
strcat(driverDLL, driver);
strcat(driverDLL, ".dll");
dwInfoSize = GetFileVersionInfoSize(driverDLL, &var);
lpInfoBuff = malloc(dwInfoSize);
bRetval = GetFileVersionInfo(driverDLL, 0, dwInfoSize, lpInfoBuff);
2003-09-28 02:55:01 -04:00
if (bRetval == 0) {
} else {
VS_FIXEDFILEINFO * fxdFileInfo;
UINT uiLen = 0;
bRetval = VerQueryValue(lpInfoBuff, TEXT("\\"), (void **) &fxdFileInfo, &uiLen);
if (bRetval != 0) {
2003-09-28 02:55:01 -04:00
TCHAR version[256];
TCHAR ms[10], ls[10];
sprintf(ms, "%d.%d\0", fxdFileInfo->dwProductVersionMS >> 16, fxdFileInfo->dwProductVersionMS & 0xFFFF);
sprintf(ls, "%d.%d\0", fxdFileInfo->dwProductVersionLS >> 16, fxdFileInfo->dwProductVersionLS & 0xFFFF);
sprintf(version, "%s.%s\0", ms, ls);
ret = NewStringNative(env, version);
2003-09-28 02:55:01 -04:00
}
}
free(lpInfoBuff);
2003-09-28 02:55:01 -04:00
return ret;
2003-03-21 12:08:26 -05:00
}