mirror of
https://github.com/ptitSeb/Serious-Engine
synced 2024-12-01 14:15:53 +01:00
385 lines
12 KiB
C++
385 lines
12 KiB
C++
/* Copyright (c) 2002-2012 Croteam Ltd.
|
|
This program is free software; you can redistribute it and/or modify
|
|
it under the terms of version 2 of the GNU General Public License as published by
|
|
the Free Software Foundation
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
GNU General Public License for more details.
|
|
|
|
You should have received a copy of the GNU General Public License along
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */
|
|
|
|
#include "Engine/StdH.h"
|
|
|
|
#include <Engine/Graphics/Adapter.h>
|
|
#include <Engine/Graphics/GfxLibrary.h>
|
|
#include <Engine/Base/Translation.h>
|
|
#include <Engine/Base/Console.h>
|
|
|
|
|
|
// !!! FIXME : rcg11052001 move this somewhere.
|
|
#ifdef PLATFORM_UNIX
|
|
#include "SDL.h"
|
|
#endif
|
|
|
|
extern BOOL _bDedicatedServer;
|
|
#ifdef SE1_D3D
|
|
extern const D3DDEVTYPE d3dDevType;
|
|
#endif // SE1_D3D
|
|
|
|
// list of all modes avaliable through CDS
|
|
static CListHead _lhCDSModes;
|
|
|
|
class CResolution {
|
|
public:
|
|
PIX re_pixSizeI;
|
|
PIX re_pixSizeJ;
|
|
};
|
|
|
|
static CResolution _areResolutions[] =
|
|
{
|
|
{ 320, 240 },
|
|
{ 400, 300 },
|
|
{ 480, 360 },
|
|
{ 512, 384 },
|
|
{ 640, 480 },
|
|
{ 720, 540 },
|
|
{ 720, 576 },
|
|
{ 800, 600 },
|
|
{ 960, 720 },
|
|
{ 1024, 768 },
|
|
{ 1152, 864 },
|
|
{ 1280, 960 },
|
|
{ 1280, 1024 },
|
|
{ 1600, 1200 },
|
|
{ 1792, 1344 },
|
|
{ 1856, 1392 },
|
|
{ 1920, 1440 },
|
|
{ 2048, 1536 },
|
|
|
|
// matrox dualhead modes
|
|
{ 1280, 480 },
|
|
{ 1600, 600 },
|
|
{ 2048, 768 },
|
|
|
|
// NTSC HDTV widescreen
|
|
{ 848, 480 },
|
|
{ 856, 480 },
|
|
};
|
|
// THIS NUMBER MUST NOT BE OVER 25! (otherwise change it in adapter.h)
|
|
static const INDEX MAX_RESOLUTIONS = sizeof(_areResolutions)/sizeof(_areResolutions[0]);
|
|
|
|
|
|
#ifdef PLATFORM_WIN32
|
|
|
|
// initialize CDS support (enumerate modes at startup)
|
|
void CGfxLibrary::InitAPIs(void)
|
|
{
|
|
// no need for gfx when dedicated server is on
|
|
if( _bDedicatedServer) return;
|
|
|
|
CDisplayAdapter *pda;
|
|
INDEX iResolution;
|
|
|
|
// detect current mode and print to console
|
|
DEVMODE devmode;
|
|
memset( &devmode, 0, sizeof(devmode));
|
|
devmode.dmSize = sizeof(devmode);
|
|
LONG lRes = EnumDisplaySettings(NULL, ENUM_CURRENT_SETTINGS, &devmode);
|
|
CPrintF( TRANS("Current display: '%s' version %d - %dx%dx%d\n\n"),
|
|
devmode.dmDeviceName, devmode.dmDriverVersion,
|
|
devmode.dmPelsWidth, devmode.dmPelsHeight, devmode.dmBitsPerPel);
|
|
|
|
// fill OpenGL adapter info
|
|
gl_gaAPI[GAT_OGL].ga_ctAdapters = 1;
|
|
gl_gaAPI[GAT_OGL].ga_iCurrentAdapter = 0;
|
|
pda = &gl_gaAPI[GAT_OGL].ga_adaAdapter[0];
|
|
pda->da_ulFlags = DAF_USEGDIFUNCTIONS;
|
|
pda->da_strVendor = TRANS( "unknown");
|
|
pda->da_strRenderer = TRANS( "Default ICD");
|
|
pda->da_strVersion = "1.1+";
|
|
|
|
// detect modes for OpenGL ICD
|
|
pda->da_ctDisplayModes = 0;
|
|
pda->da_iCurrentDisplayMode = -1;
|
|
|
|
// enumerate modes thru resolution list
|
|
for( iResolution=0; iResolution<MAX_RESOLUTIONS; iResolution++)
|
|
{
|
|
DEVMODE devmode;
|
|
memset( &devmode, 0, sizeof(devmode));
|
|
CResolution &re = _areResolutions[iResolution];
|
|
|
|
// ask windows if they could set the mode
|
|
devmode.dmSize = sizeof(devmode);
|
|
devmode.dmPelsWidth = re.re_pixSizeI;
|
|
devmode.dmPelsHeight = re.re_pixSizeJ;
|
|
devmode.dmDisplayFlags = CDS_FULLSCREEN;
|
|
devmode.dmFields = DM_PELSWIDTH|DM_PELSHEIGHT|DM_DISPLAYFLAGS;
|
|
LONG lRes = ChangeDisplaySettings( &devmode, CDS_TEST|CDS_FULLSCREEN);
|
|
// skip if not successfull
|
|
if( lRes!=DISP_CHANGE_SUCCESSFUL) continue;
|
|
|
|
// make a new display mode
|
|
CDisplayMode &dm = pda->da_admDisplayModes[pda->da_ctDisplayModes];
|
|
dm.dm_pixSizeI = re.re_pixSizeI;
|
|
dm.dm_pixSizeJ = re.re_pixSizeJ;
|
|
dm.dm_ddDepth = DD_DEFAULT;
|
|
pda->da_ctDisplayModes++;
|
|
}
|
|
|
|
// detect presence of 3Dfx standalone OpenGL driver (for Voodoo1/2)
|
|
char *strDummy;
|
|
char strBuffer[_MAX_PATH+1];
|
|
int iRes = SearchPathA( NULL, "3DFXVGL.DLL", NULL, _MAX_PATH, strBuffer, &strDummy);
|
|
// if present
|
|
if(iRes) {
|
|
// set adapter and force some enumeration of voodoo1/2 display modes
|
|
gl_gaAPI[GAT_OGL].ga_ctAdapters++;
|
|
pda = &gl_gaAPI[GAT_OGL].ga_adaAdapter[1];
|
|
pda->da_ctDisplayModes = 4; // voodoos have only 4 display modes
|
|
pda->da_ulFlags = DAF_ONEWINDOW | DAF_FULLSCREENONLY | DAF_16BITONLY;
|
|
pda->da_strVendor = "3Dfx";
|
|
pda->da_strRenderer = "3Dfx Voodoo2";
|
|
pda->da_strVersion = "1.1+";
|
|
CDisplayMode *adm = &pda->da_admDisplayModes[0];
|
|
adm[0].dm_pixSizeI = 512; adm[0].dm_pixSizeJ = 384; adm[0].dm_ddDepth = DD_16BIT;
|
|
adm[1].dm_pixSizeI = 640; adm[1].dm_pixSizeJ = 480; adm[1].dm_ddDepth = DD_16BIT;
|
|
adm[2].dm_pixSizeI = 800; adm[2].dm_pixSizeJ = 600; adm[2].dm_ddDepth = DD_16BIT;
|
|
adm[3].dm_pixSizeI = 1024; adm[3].dm_pixSizeJ = 768; adm[3].dm_ddDepth = DD_16BIT;
|
|
}
|
|
|
|
// try to init Direct3D 8
|
|
#ifdef SE1_D3D
|
|
BOOL bRes = InitDriver_D3D();
|
|
if( !bRes) return; // didn't made it?
|
|
|
|
// determine DX8 adapters and display modes
|
|
const INDEX ctMaxAdapters = gl_pD3D->GetAdapterCount();
|
|
INDEX &ctAdapters = gl_gaAPI[GAT_D3D].ga_ctAdapters;
|
|
ctAdapters = 0;
|
|
|
|
for( INDEX iAdapter=0; iAdapter<ctMaxAdapters; iAdapter++)
|
|
{
|
|
pda = &gl_gaAPI[1].ga_adaAdapter[ctAdapters];
|
|
pda->da_ulFlags = NONE;
|
|
pda->da_ctDisplayModes = 0;
|
|
INDEX ctModes = gl_pD3D->GetAdapterModeCount(iAdapter);
|
|
INDEX iMode;
|
|
HRESULT hr;
|
|
|
|
// check whether 32-bits rendering modes are supported
|
|
hr = gl_pD3D->CheckDeviceType( iAdapter, d3dDevType, D3DFMT_A8R8G8B8, D3DFMT_A8R8G8B8, FALSE);
|
|
if( hr!=D3D_OK) {
|
|
hr = gl_pD3D->CheckDeviceType( iAdapter, d3dDevType, D3DFMT_X8R8G8B8, D3DFMT_X8R8G8B8, FALSE);
|
|
if( hr!=D3D_OK) pda->da_ulFlags |= DAF_16BITONLY;
|
|
}
|
|
|
|
// check whether windowed rendering modes are supported
|
|
D3DCAPS8 d3dCaps;
|
|
gl_pD3D->GetDeviceCaps( iAdapter, d3dDevType, &d3dCaps);
|
|
if( !(d3dCaps.Caps2 & D3DCAPS2_CANRENDERWINDOWED)) pda->da_ulFlags |= DAF_FULLSCREENONLY;
|
|
|
|
// enumerate modes thru resolution list
|
|
for( iResolution=0; iResolution<MAX_RESOLUTIONS; iResolution++)
|
|
{
|
|
CResolution &re = _areResolutions[iResolution];
|
|
for( iMode=0; iMode<ctModes; iMode++) {
|
|
// if resolution matches and display depth is 16 or 32 bit
|
|
D3DDISPLAYMODE d3dDisplayMode;
|
|
gl_pD3D->EnumAdapterModes( iAdapter, iMode, &d3dDisplayMode);
|
|
if( d3dDisplayMode.Width==re.re_pixSizeI && d3dDisplayMode.Height==re.re_pixSizeJ
|
|
&& (d3dDisplayMode.Format==D3DFMT_A8R8G8B8 || d3dDisplayMode.Format==D3DFMT_X8R8G8B8
|
|
|| d3dDisplayMode.Format==D3DFMT_A1R5G5B5 || d3dDisplayMode.Format==D3DFMT_X1R5G5B5
|
|
|| d3dDisplayMode.Format==D3DFMT_R5G6B5)) {
|
|
hr = gl_pD3D->CheckDeviceType( iAdapter, d3dDevType, d3dDisplayMode.Format, d3dDisplayMode.Format, FALSE);
|
|
if( hr!=D3D_OK) continue;
|
|
|
|
// make a new display mode
|
|
CDisplayMode &dm = pda->da_admDisplayModes[pda->da_ctDisplayModes];
|
|
dm.dm_pixSizeI = re.re_pixSizeI;
|
|
dm.dm_pixSizeJ = re.re_pixSizeJ;
|
|
dm.dm_ddDepth = DD_DEFAULT;
|
|
pda->da_ctDisplayModes++;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
// get adapter identifier
|
|
ctAdapters++;
|
|
D3DADAPTER_IDENTIFIER8 d3dAdapterIdentifier;
|
|
gl_pD3D->GetAdapterIdentifier( iAdapter, D3DENUM_NO_WHQL_LEVEL, &d3dAdapterIdentifier);
|
|
pda->da_strVendor = "MS DirectX 8";
|
|
pda->da_strRenderer = d3dAdapterIdentifier.Description;
|
|
pda->da_strVersion.PrintF("%d.%d.%d.%d", d3dAdapterIdentifier.DriverVersion.HighPart >>16,
|
|
d3dAdapterIdentifier.DriverVersion.HighPart & 0xFFFF,
|
|
d3dAdapterIdentifier.DriverVersion.LowPart >>16,
|
|
d3dAdapterIdentifier.DriverVersion.LowPart & 0xFFFF);
|
|
}
|
|
// shutdown DX8 (we'll start it again if needed)
|
|
D3DRELEASE( gl_pD3D, TRUE);
|
|
#endif
|
|
|
|
if( gl_hiDriver!=NONE) FreeLibrary(gl_hiDriver);
|
|
gl_hiDriver = NONE;
|
|
}
|
|
|
|
#else
|
|
|
|
|
|
// initialize CDS support (enumerate modes at startup)
|
|
void CGfxLibrary::InitAPIs(void)
|
|
{
|
|
// no need for gfx when dedicated server is on
|
|
if( _bDedicatedServer) return;
|
|
|
|
// fill OpenGL adapter info
|
|
CDisplayAdapter *pda;
|
|
INDEX iResolution;
|
|
|
|
gl_gaAPI[GAT_OGL].ga_ctAdapters = 1;
|
|
gl_gaAPI[GAT_OGL].ga_iCurrentAdapter = 0;
|
|
pda = &gl_gaAPI[GAT_OGL].ga_adaAdapter[0];
|
|
pda->da_ulFlags = 0;
|
|
pda->da_strVendor = TRANS( "unknown");
|
|
pda->da_strRenderer = TRANS( "Default ICD");
|
|
pda->da_strVersion = "1.1+";
|
|
|
|
// detect modes for OpenGL ICD
|
|
pda->da_ctDisplayModes = 0;
|
|
pda->da_iCurrentDisplayMode = -1;
|
|
|
|
const int dpy = 0; // !!! FIXME: hook up a cvar?
|
|
const int total = SDL_GetNumDisplayModes(dpy);
|
|
for (int i = 0; i < total; i++)
|
|
{
|
|
if (pda->da_ctDisplayModes >= ARRAYCOUNT(pda->da_admDisplayModes))
|
|
break;
|
|
|
|
SDL_DisplayMode mode;
|
|
if (SDL_GetDisplayMode(dpy, i, &mode) == 0)
|
|
{
|
|
const int bpp = (int) SDL_BITSPERPIXEL(mode.format);
|
|
if (bpp < 16) continue;
|
|
DisplayDepth bits = DD_DEFAULT;
|
|
switch (bpp)
|
|
{
|
|
case 16: bits = DD_16BIT; break;
|
|
case 32: bits = DD_32BIT; break;
|
|
case 24: bits = DD_24BIT; break;
|
|
default: break;
|
|
}
|
|
|
|
CDisplayMode &dm = pda->da_admDisplayModes[pda->da_ctDisplayModes];
|
|
dm.dm_pixSizeI = mode.w;
|
|
dm.dm_pixSizeJ = mode.h;
|
|
dm.dm_ddDepth = bits;
|
|
pda->da_ctDisplayModes++;
|
|
}
|
|
}
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// get list of all modes avaliable through CDS -- do not modify/free the returned list
|
|
CListHead &CDS_GetModes(void)
|
|
{
|
|
return _lhCDSModes;
|
|
}
|
|
|
|
|
|
// set given display mode
|
|
BOOL CDS_SetMode( PIX pixSizeI, PIX pixSizeJ, enum DisplayDepth dd)
|
|
{
|
|
// no need for gfx when dedicated server is on
|
|
if( _bDedicatedServer) return FALSE;
|
|
|
|
// !!! FIXME : rcg11052001 better abstraction!
|
|
#ifdef PLATFORM_WIN32
|
|
// prepare general mode parameters
|
|
DEVMODE devmode;
|
|
memset(&devmode, 0, sizeof(devmode));
|
|
devmode.dmSize = sizeof(devmode);
|
|
devmode.dmPelsWidth = pixSizeI;
|
|
devmode.dmPelsHeight = pixSizeJ;
|
|
devmode.dmDisplayFlags = CDS_FULLSCREEN;
|
|
devmode.dmFields = DM_PELSWIDTH|DM_PELSHEIGHT|DM_DISPLAYFLAGS;
|
|
extern INDEX gap_iRefreshRate;
|
|
if( gap_iRefreshRate>0) {
|
|
devmode.dmFields |= DM_DISPLAYFREQUENCY;
|
|
devmode.dmDisplayFrequency = gap_iRefreshRate;
|
|
}
|
|
// determine bits per pixel to try to set
|
|
SLONG slBPP2 = 0;
|
|
switch(dd) {
|
|
case DD_16BIT:
|
|
devmode.dmBitsPerPel = 16;
|
|
slBPP2 = 15;
|
|
devmode.dmFields |= DM_BITSPERPEL;
|
|
break;
|
|
case DD_32BIT:
|
|
devmode.dmBitsPerPel = 32;
|
|
slBPP2 = 24;
|
|
devmode.dmFields |= DM_BITSPERPEL;
|
|
break;
|
|
case DD_DEFAULT:
|
|
NOTHING;
|
|
break;
|
|
default:
|
|
ASSERT(FALSE);
|
|
NOTHING;
|
|
}
|
|
|
|
// try to set primary depth
|
|
LONG lRes = ChangeDisplaySettings(&devmode, CDS_FULLSCREEN);
|
|
|
|
// if failed
|
|
if( lRes!=DISP_CHANGE_SUCCESSFUL) {
|
|
// try to set secondary depth
|
|
devmode.dmBitsPerPel = slBPP2;
|
|
LONG lRes2 = ChangeDisplaySettings(&devmode, CDS_FULLSCREEN);
|
|
// if failed
|
|
if( lRes2!=DISP_CHANGE_SUCCESSFUL) {
|
|
CTString strError;
|
|
switch(lRes) {
|
|
case DISP_CHANGE_SUCCESSFUL: strError = "DISP_CHANGE_SUCCESSFUL"; break;
|
|
case DISP_CHANGE_RESTART: strError = "DISP_CHANGE_RESTART"; break;
|
|
case DISP_CHANGE_BADFLAGS: strError = "DISP_CHANGE_BADFLAGS"; break;
|
|
case DISP_CHANGE_BADPARAM: strError = "DISP_CHANGE_BADPARAM"; break;
|
|
case DISP_CHANGE_FAILED: strError = "DISP_CHANGE_FAILED"; break;
|
|
case DISP_CHANGE_BADMODE: strError = "DISP_CHANGE_BADMODE"; break;
|
|
case DISP_CHANGE_NOTUPDATED: strError = "DISP_CHANGE_NOTUPDATED"; break;
|
|
default: strError.PrintF("%d", lRes); break;
|
|
}
|
|
CPrintF(TRANSV("CDS error: %s\n"), strError);
|
|
return FALSE;
|
|
}
|
|
}
|
|
// report
|
|
CPrintF(TRANSV(" CDS: mode set to %dx%dx%d\n"), pixSizeI, pixSizeJ, devmode.dmBitsPerPel);
|
|
#endif
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
// reset windows to mode chosen by user within windows diplay properties
|
|
void CDS_ResetMode(void)
|
|
{
|
|
// no need for gfx when dedicated server is on
|
|
if( _bDedicatedServer) return;
|
|
|
|
#ifdef PLATFORM_WIN32
|
|
LONG lRes = ChangeDisplaySettings( NULL, 0);
|
|
ASSERT(lRes==DISP_CHANGE_SUCCESSFUL);
|
|
CPrintF(TRANSV(" CDS: mode reset to original desktop settings\n"));
|
|
#endif
|
|
}
|
|
|