mirror of
https://github.com/blendogames/thirtyflightsofloving.git
synced 2025-01-18 14:31:55 +00:00
2118 lines
51 KiB
C
2118 lines
51 KiB
C
/*
|
|
===========================================================================
|
|
Copyright (C) 1997-2001 Id Software, Inc.
|
|
|
|
This file is part of Quake 2 source code.
|
|
|
|
Quake 2 source code is free software; you can redistribute it
|
|
and/or modify it under the terms of the GNU General Public License as
|
|
published by the Free Software Foundation; either version 2 of the License,
|
|
or (at your option) any later version.
|
|
|
|
Quake 2 source code 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 Quake 2 source code; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
===========================================================================
|
|
*/
|
|
|
|
// sys_win.h
|
|
|
|
#include "../qcommon/qcommon.h"
|
|
#include "winquake.h"
|
|
#include "resource.h"
|
|
#include <errno.h>
|
|
#include <float.h>
|
|
#include <fcntl.h>
|
|
#include <stdio.h>
|
|
#include <direct.h>
|
|
#include <io.h>
|
|
#include <conio.h>
|
|
#include <shlobj.h>
|
|
#include "../win32/conproc.h"
|
|
|
|
// [Slipyx] mingw support for _controlfp. from float.h. these would be defined
|
|
// if __STRICT_ANSI__ was undefined before including float.h, but i just copied
|
|
// out these specific definitions instead since they're the only ones used.
|
|
#if defined(__MINGW32__)
|
|
#define _MCW_PC 0x00030000
|
|
#define _PC_24 0x00020000
|
|
_CRTIMP unsigned int __cdecl __MINGW_NOTHROW _controlfp (unsigned int unNew, unsigned int unMask);
|
|
#endif // __MINGW32__
|
|
|
|
#define MINIMUM_WIN_MEMORY 0x0a00000
|
|
#define MAXIMUM_WIN_MEMORY 0x1000000
|
|
|
|
qboolean s_win95;
|
|
|
|
qboolean s_win9X;
|
|
qboolean s_winNT;
|
|
qboolean s_winNT6;
|
|
|
|
int starttime;
|
|
int ActiveApp;
|
|
qboolean Minimized;
|
|
|
|
static HANDLE hinput, houtput;
|
|
|
|
unsigned sys_msg_time;
|
|
unsigned sys_frame_time;
|
|
|
|
|
|
static HANDLE qwclsemaphore;
|
|
|
|
#define MAX_NUM_ARGVS 128
|
|
int argc;
|
|
char *argv[MAX_NUM_ARGVS];
|
|
|
|
#define NT5_SAVEDIR "My Games/KMQuake2"
|
|
#define NT6_SAVEDIR "KMQuake2"
|
|
#define NT5_DLDIR "My Downloads/KMQuake2"
|
|
#define NT6_DLDIR "KMQuake2"
|
|
static char exe_dir[MAX_OSPATH];
|
|
static char pref_dir[MAX_OSPATH];
|
|
static char download_dir[MAX_OSPATH];
|
|
|
|
qboolean Detect_WinNT5orLater (void);
|
|
qboolean Detect_WinNT6orLater (void);
|
|
|
|
typedef HRESULT (WINAPI *SHGETFOLDERPATHW) (HWND hwnd, int CSIDL, HANDLE hToken, DWORD dwFlags, LPWSTR pszPath);
|
|
SHGETFOLDERPATHW fnSHGetFolderPathW = NULL;
|
|
|
|
// from DarkPlaces
|
|
const GUID qFOLDERID_SavedGames = {0x4C5C32FF, 0xBB9D, 0x43b0, {0xB5, 0xB4, 0x2D, 0x72, 0xE5, 0x4E, 0xAA, 0xA4}};
|
|
const GUID qFOLDERID_Downloads = {0x374DE290, 0x123F, 0x4565, {0x91, 0x64, 0x39, 0xC4, 0x92, 0x5E, 0x46, 0x7B}};
|
|
#define qREFKNOWNFOLDERID const GUID *
|
|
#define qKF_FLAG_CREATE 0x8000
|
|
#define qKF_FLAG_NO_ALIAS 0x1000
|
|
static HRESULT (WINAPI *fnSHGetKnownFolderPath) (qREFKNOWNFOLDERID rfid, DWORD dwFlags, HANDLE hToken, PWSTR *ppszPath);
|
|
static HRESULT (WINAPI *fnCoInitializeEx)(LPVOID pvReserved, DWORD dwCoInit);
|
|
static void (WINAPI *fnCoUninitialize)(void);
|
|
static void (WINAPI *fnCoTaskMemFree)(LPVOID pv);
|
|
// end DarkPlaces code
|
|
|
|
|
|
#ifndef NEW_DED_CONSOLE
|
|
/*
|
|
===============================================================================
|
|
|
|
DEDICATED CONSOLE
|
|
|
|
===============================================================================
|
|
*/
|
|
static char console_text[256];
|
|
static int console_textlen;
|
|
|
|
/*
|
|
================
|
|
Sys_InitConsole
|
|
================
|
|
*/
|
|
void Sys_InitConsole (void)
|
|
{
|
|
if (!dedicated->value)
|
|
return;
|
|
|
|
if (!AllocConsole ())
|
|
Sys_Error ("Couldn't create dedicated server console");
|
|
hinput = GetStdHandle (STD_INPUT_HANDLE);
|
|
houtput = GetStdHandle (STD_OUTPUT_HANDLE);
|
|
|
|
// let QHOST hook in
|
|
InitConProc (argc, argv);
|
|
}
|
|
|
|
|
|
/*
|
|
================
|
|
Sys_ConsoleInput
|
|
================
|
|
*/
|
|
char *Sys_ConsoleInput (void)
|
|
{
|
|
INPUT_RECORD recs[1024];
|
|
int dummy;
|
|
int ch, numread, numevents;
|
|
|
|
if (!dedicated || !dedicated->value)
|
|
return NULL;
|
|
|
|
for ( ;; )
|
|
{
|
|
if (!GetNumberOfConsoleInputEvents (hinput, &numevents))
|
|
Sys_Error ("Error getting # of console events");
|
|
|
|
if (numevents <= 0)
|
|
break;
|
|
|
|
if (!ReadConsoleInput(hinput, recs, 1, &numread))
|
|
Sys_Error ("Error reading console input");
|
|
|
|
if (numread != 1)
|
|
Sys_Error ("Couldn't read console input");
|
|
|
|
if (recs[0].EventType == KEY_EVENT)
|
|
{
|
|
if (!recs[0].Event.KeyEvent.bKeyDown)
|
|
{
|
|
ch = recs[0].Event.KeyEvent.uChar.AsciiChar;
|
|
|
|
switch (ch)
|
|
{
|
|
case '\r':
|
|
WriteFile(houtput, "\r\n", 2, &dummy, NULL);
|
|
|
|
if (console_textlen)
|
|
{
|
|
console_text[console_textlen] = 0;
|
|
console_textlen = 0;
|
|
return console_text;
|
|
}
|
|
break;
|
|
|
|
case '\b':
|
|
if (console_textlen)
|
|
{
|
|
console_textlen--;
|
|
WriteFile(houtput, "\b \b", 3, &dummy, NULL);
|
|
}
|
|
break;
|
|
|
|
default:
|
|
if (ch >= ' ')
|
|
{
|
|
if (console_textlen < sizeof(console_text)-2)
|
|
{
|
|
WriteFile(houtput, &ch, 1, &dummy, NULL);
|
|
console_text[console_textlen] = ch;
|
|
console_textlen++;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
/*
|
|
================
|
|
Sys_ConsoleOutput
|
|
|
|
Print text to the dedicated console
|
|
================
|
|
*/
|
|
void Sys_ConsoleOutput (char *string)
|
|
{
|
|
int dummy;
|
|
char text[256];
|
|
|
|
if (!dedicated || !dedicated->value)
|
|
return;
|
|
|
|
if (console_textlen)
|
|
{
|
|
text[0] = '\r';
|
|
memset(&text[1], ' ', console_textlen);
|
|
text[console_textlen+1] = '\r';
|
|
text[console_textlen+2] = 0;
|
|
WriteFile(houtput, text, console_textlen+2, &dummy, NULL);
|
|
}
|
|
|
|
WriteFile(houtput, string, strlen(string), &dummy, NULL);
|
|
|
|
if (console_textlen)
|
|
WriteFile(houtput, console_text, console_textlen, &dummy, NULL);
|
|
}
|
|
|
|
//================================================================
|
|
#endif // NEW_DED_CONSOLE
|
|
|
|
/*
|
|
================
|
|
Sys_Sleep
|
|
================
|
|
*/
|
|
void Sys_Sleep (int msec)
|
|
{
|
|
Sleep (msec);
|
|
}
|
|
|
|
/*
|
|
================
|
|
Sys_TickCount
|
|
================
|
|
*/
|
|
unsigned Sys_TickCount (void)
|
|
{
|
|
return GetTickCount();
|
|
}
|
|
|
|
/*
|
|
================
|
|
Sys_SendKeyEvents
|
|
|
|
Send Key_Event calls
|
|
================
|
|
*/
|
|
void Sys_SendKeyEvents (void)
|
|
{
|
|
MSG msg;
|
|
|
|
while (PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE))
|
|
{
|
|
if (!GetMessage (&msg, NULL, 0, 0))
|
|
Sys_Quit ();
|
|
sys_msg_time = msg.time;
|
|
TranslateMessage (&msg);
|
|
DispatchMessage (&msg);
|
|
}
|
|
|
|
// grab frame time
|
|
sys_frame_time = timeGetTime(); // FIXME: should this be at start?
|
|
}
|
|
|
|
|
|
/*
|
|
================
|
|
Sys_GetClipboardData
|
|
|
|
================
|
|
*/
|
|
char *Sys_GetClipboardData( void )
|
|
{
|
|
char *data = NULL;
|
|
char *cliptext;
|
|
|
|
if ( OpenClipboard( NULL ) != 0 )
|
|
{
|
|
HANDLE hClipboardData;
|
|
|
|
if ( ( hClipboardData = GetClipboardData( CF_TEXT ) ) != 0 )
|
|
{
|
|
if ( ( cliptext = GlobalLock( hClipboardData ) ) != 0 )
|
|
{
|
|
data = malloc( GlobalSize( hClipboardData ) + 1 );
|
|
// strncpy( data, cliptext );
|
|
Q_strncpyz( data, cliptext, GlobalSize( hClipboardData ) + 1 );
|
|
GlobalUnlock( hClipboardData );
|
|
}
|
|
}
|
|
CloseClipboard();
|
|
}
|
|
return data;
|
|
}
|
|
|
|
|
|
/*
|
|
===============================================================================
|
|
|
|
SYSTEM IO
|
|
|
|
===============================================================================
|
|
*/
|
|
|
|
#ifndef NEW_DED_CONSOLE
|
|
void Sys_Error (char *error, ...)
|
|
{
|
|
va_list argptr;
|
|
char text[1024];
|
|
|
|
CL_Shutdown ();
|
|
Qcommon_Shutdown ();
|
|
|
|
va_start (argptr, error);
|
|
// vsprintf (text, error, argptr);
|
|
Q_vsnprintf (text, sizeof(text), error, argptr);
|
|
va_end (argptr);
|
|
|
|
MessageBox(NULL, text, "Error", 0 /* MB_OK */ );
|
|
|
|
if (qwclsemaphore)
|
|
CloseHandle (qwclsemaphore);
|
|
|
|
// shut down QHOST hooks if necessary
|
|
DeinitConProc ();
|
|
|
|
exit (1);
|
|
}
|
|
#endif // NEW_DED_CONSOLE
|
|
|
|
|
|
void Sys_Quit (void)
|
|
{
|
|
timeEndPeriod( 1 );
|
|
|
|
CL_Shutdown();
|
|
Qcommon_Shutdown ();
|
|
CloseHandle (qwclsemaphore);
|
|
if (dedicated && dedicated->value)
|
|
FreeConsole ();
|
|
|
|
#ifdef NEW_DED_CONSOLE
|
|
Sys_ShutdownConsole();
|
|
#else
|
|
// shut down QHOST hooks if necessary
|
|
DeinitConProc ();
|
|
#endif
|
|
|
|
exit (0);
|
|
}
|
|
|
|
|
|
void WinError (void)
|
|
{
|
|
LPVOID lpMsgBuf;
|
|
|
|
FormatMessage(
|
|
FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM,
|
|
NULL,
|
|
GetLastError(),
|
|
MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), // Default language
|
|
(LPTSTR) &lpMsgBuf,
|
|
0,
|
|
NULL
|
|
);
|
|
|
|
// Display the string.
|
|
MessageBox( NULL, lpMsgBuf, "GetLastError", MB_OK|MB_ICONINFORMATION );
|
|
|
|
// Free the buffer.
|
|
LocalFree( lpMsgBuf );
|
|
}
|
|
|
|
//================================================================
|
|
|
|
/*
|
|
================
|
|
Sys_ScanForCD
|
|
|
|
================
|
|
*/
|
|
char *Sys_ScanForCD (void)
|
|
{
|
|
static char cddir[MAX_OSPATH];
|
|
static qboolean done;
|
|
char drive[4];
|
|
FILE *f;
|
|
char test[MAX_QPATH];
|
|
qboolean missionpack = false; // Knightmare added
|
|
int i; // Knightmare added
|
|
|
|
if (done) // don't re-check
|
|
return cddir;
|
|
|
|
// no abort/retry/fail errors
|
|
SetErrorMode (SEM_FAILCRITICALERRORS);
|
|
|
|
drive[0] = 'c';
|
|
drive[1] = ':';
|
|
drive[2] = '\\';
|
|
drive[3] = 0;
|
|
|
|
Com_Printf("\nScanning for game CD data path...");
|
|
|
|
done = true;
|
|
|
|
// Knightmare- check if mission pack gamedir is set
|
|
for (i=0; i<argc; i++)
|
|
if (!strcmp(argv[i], "game") && (i+1<argc))
|
|
{
|
|
if (!strcmp(argv[i+1], "rogue") || !strcmp(argv[i+1], "xatrix"))
|
|
missionpack = true;
|
|
break; // game parameter only appears once in command line
|
|
}
|
|
|
|
// scan the drives
|
|
for (drive[0] = 'c' ; drive[0] <= 'z' ; drive[0]++)
|
|
{
|
|
// where activision put the stuff...
|
|
if (missionpack) // Knightmare- mission packs have cinematics in different path
|
|
{
|
|
sprintf (cddir, "%sdata\\max", drive);
|
|
sprintf (test, "%sdata\\patch\\quake2.exe", drive);
|
|
}
|
|
else
|
|
{
|
|
sprintf (cddir, "%sinstall\\data", drive);
|
|
sprintf (test, "%sinstall\\data\\quake2.exe", drive);
|
|
}
|
|
f = fopen(test, "r");
|
|
if (f)
|
|
{
|
|
fclose (f);
|
|
if (GetDriveType (drive) == DRIVE_CDROM) {
|
|
Com_Printf(" found %s\n", cddir);
|
|
return cddir;
|
|
}
|
|
}
|
|
}
|
|
|
|
Com_Printf(" could not find %s on any CDROM drive!\n", test);
|
|
|
|
cddir[0] = 0;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
//================================================================
|
|
|
|
/*
|
|
=================
|
|
Sys_DetectCPU
|
|
l33t CPU detection
|
|
Borrowed from Q2E
|
|
=================
|
|
*/
|
|
static qboolean Sys_DetectCPU (char *cpuString, int maxSize)
|
|
{
|
|
#if ( defined (_M_IX86) || defined (_M_X64) || defined (_M_AMD64) || defined (__x86_64__) ) && !defined(__GNUC__) // [Slipyx] mingw support
|
|
|
|
char vendor[16];
|
|
// int numLogicalCores=1, numCores=1;
|
|
int maxExtFunc, stdBits, features, moreFeatures, extFeatures;
|
|
int family, extFamily, model, extModel;
|
|
unsigned __int64 start, end, counter, stop, frequency;
|
|
unsigned speed;
|
|
qboolean hasMMX, hasMMXExt, has3DNow, has3DNowExt, hasSSE, hasSSE2, hasSSE3, hasSSE41, hasSSE42, hasSSE4a, hasAVX;
|
|
SYSTEM_INFO sysInfo;
|
|
|
|
#if defined (_M_X64) || defined (_M_AMD64) || defined (__x86_64__)
|
|
#define rdtsc __asm __emit 0fh __asm __emit 031h
|
|
int registers[4];
|
|
|
|
// Get vendor identifier
|
|
__cpuid(registers, 0);
|
|
memset(vendor, 0, sizeof(vendor));
|
|
memcpy(vendor+0, ®isters[1], sizeof(char)*4); // ebx
|
|
memcpy(vendor+4, ®isters[3], sizeof(char)*4); // edx
|
|
memcpy(vendor+8, ®isters[2], sizeof(char)*4); // ecx
|
|
|
|
// Get standard bits and features
|
|
__cpuid(registers, 1);
|
|
stdBits = registers[0];
|
|
moreFeatures = registers[2];
|
|
features = registers[3];
|
|
// numLogicalCores = (registers[1] >> 16) & 255;
|
|
|
|
// Check if extended functions are present
|
|
__cpuid(registers, 0x80000000);
|
|
maxExtFunc = registers[0];
|
|
|
|
// Get extended features
|
|
if (maxExtFunc >= 0x80000001)
|
|
{
|
|
__cpuid(registers, 0x80000001);
|
|
extFeatures = registers[3];
|
|
}
|
|
/*
|
|
// Get non-threaded core count
|
|
if (!Q_stricmp(vendor, "AuthenticAMD"))
|
|
{
|
|
if (maxExtFunc >= 0x80000008) {
|
|
__cpuid(registers, 0x80000008);
|
|
numCores = 1 + (registers[2] & 255);
|
|
}
|
|
}
|
|
else if (!Q_stricmp(vendor, "GenuineIntel"))
|
|
{
|
|
__cpuid(registers, 4);
|
|
numCores = 1 + ( (registers[0] >> 26) & 63 );
|
|
}
|
|
*/
|
|
#elif defined (_M_IX86)
|
|
// int tempThreads, tempCores;
|
|
// Check if CPUID instruction is supported
|
|
__try {
|
|
__asm {
|
|
mov eax, 0
|
|
cpuid
|
|
}
|
|
}
|
|
__except (EXCEPTION_EXECUTE_HANDLER) {
|
|
return false;
|
|
}
|
|
|
|
// Get CPU info
|
|
__asm {
|
|
; // Get vendor identifier
|
|
mov eax, 0
|
|
cpuid
|
|
mov dword ptr[vendor+0], ebx
|
|
mov dword ptr[vendor+4], edx
|
|
mov dword ptr[vendor+8], ecx
|
|
mov dword ptr[vendor+12], 0
|
|
|
|
; // Get standard bits and features
|
|
mov eax, 1
|
|
cpuid
|
|
mov stdBits, eax
|
|
mov moreFeatures, ecx ; // Knightmare added
|
|
mov features, edx
|
|
// mov tempThreads, ebx ; // Knightmare added
|
|
|
|
; // Check if extended functions are present
|
|
mov extFeatures, 0
|
|
mov eax, 80000000h
|
|
cpuid
|
|
mov maxExtFunc, eax ; // Knightmare added
|
|
cmp eax, 80000000h
|
|
jbe NoExtFunction
|
|
|
|
; // Get extended features
|
|
mov eax, 80000001h
|
|
cpuid
|
|
mov extFeatures, edx
|
|
|
|
NoExtFunction:
|
|
}
|
|
/*
|
|
numLogicalCores = (tempThreads >> 16) & 255;
|
|
|
|
// Get non-threaded core count
|
|
if (!Q_stricmp(vendor, "AuthenticAMD"))
|
|
{
|
|
if (maxExtFunc >= 0x80000008)
|
|
{
|
|
__asm {
|
|
mov eax, 80000008h
|
|
cpuid
|
|
mov tempCores, ecx
|
|
}
|
|
numCores = 1 + (tempCores & 255);
|
|
}
|
|
}
|
|
else if (!Q_stricmp(vendor, "GenuineIntel"))
|
|
{
|
|
__asm {
|
|
mov eax, 4
|
|
cpuid
|
|
mov tempCores, eax
|
|
}
|
|
numCores = 1 + ( (tempCores >> 26) & 63 );
|
|
}
|
|
*/
|
|
#endif
|
|
|
|
// Get CPU name
|
|
family = (stdBits >> 8) & 15;
|
|
model = (stdBits >> 4) & 15;
|
|
if ( (family == 15) || (family == 6) ) {
|
|
// extFamily = ( ((stdBits >> 20) & 15) << 4 ) + family;
|
|
extFamily = ((stdBits >> 20) & 255) + family;
|
|
extModel = ( ((stdBits >> 16) & 15) << 4 ) + model;
|
|
}
|
|
else {
|
|
extFamily = (stdBits >> 20) & 255;
|
|
extModel = (stdBits >> 16) & 15;
|
|
}
|
|
|
|
if (!Q_stricmp(vendor, "AuthenticAMD"))
|
|
{
|
|
strncpy(cpuString, "AMD", maxSize);
|
|
// Com_sprintf(cpuString, maxSize, "AMD Family %i ExtFamily %i Model %i ExtModel %i", family, extFamily, model, extModel);
|
|
|
|
switch (family)
|
|
{
|
|
case 4:
|
|
Q_strncatz(cpuString, " 5x86", maxSize);
|
|
break;
|
|
case 5:
|
|
switch (model)
|
|
{
|
|
case 0:
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
Q_strncatz(cpuString, " K5", maxSize);
|
|
break;
|
|
case 6:
|
|
case 7:
|
|
Q_strncatz(cpuString, " K6", maxSize);
|
|
break;
|
|
case 8:
|
|
Q_strncatz(cpuString, " K6-2", maxSize);
|
|
break;
|
|
case 9:
|
|
case 10:
|
|
case 11:
|
|
case 12:
|
|
case 13:
|
|
case 14:
|
|
case 15:
|
|
Q_strncatz(cpuString, " K6-III", maxSize);
|
|
break;
|
|
}
|
|
break;
|
|
case 6: // K7 family
|
|
switch (model)
|
|
{
|
|
case 1: // 250nm core
|
|
case 2: // 180nm core
|
|
case 4: // Thunderbird core
|
|
case 6: // Palomino core
|
|
case 8: // Thoroughbred core
|
|
case 10: // Barton core
|
|
Q_strncatz(cpuString, " Athlon", maxSize);
|
|
break;
|
|
case 3: // Spitfire core
|
|
case 7: // Morgan core
|
|
Q_strncatz(cpuString, " Duron", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " K7", maxSize);
|
|
break;
|
|
}
|
|
break;
|
|
case 15: // refer to extended family
|
|
if (extFamily == 0x0F) // K8 family
|
|
{
|
|
switch (model)
|
|
{
|
|
// case 0:
|
|
// case 2:
|
|
// case 6:
|
|
case 4: // Clawhammer/Newark
|
|
case 7: // San Diego/Newcastle
|
|
case 12: // Newcastle/Albany
|
|
Q_strncatz(cpuString, " Athlon 64", maxSize);
|
|
break;
|
|
case 3: // Toledo
|
|
case 11: // Manchester/Brisbane
|
|
case 15: // Winchester/Venice
|
|
Q_strncatz(cpuString, " Athlon 64 X2", maxSize);
|
|
break;
|
|
case 1:
|
|
case 5:
|
|
Q_strncatz(cpuString, " Athlon 64 FX / Opteron", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " K8", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
else if (extFamily == 0x10) // K10 family
|
|
{
|
|
switch (model)
|
|
{
|
|
case 0: // Barcelona A0-A2
|
|
case 2: // Barcelona B0-B3
|
|
Q_strncatz(cpuString, " Phenom / Opteron", maxSize);
|
|
break;
|
|
case 4: // Deneb / Shanghai
|
|
case 10: // Thuban
|
|
Q_strncatz(cpuString, " Phenom II / Opteron", maxSize);
|
|
break;
|
|
case 5: // Propus
|
|
case 6: // Regor
|
|
Q_strncatz(cpuString, " Athlon II", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " K10", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
else if (extFamily == 0x12) // Stars Fusion family
|
|
{
|
|
switch (model)
|
|
{
|
|
case 1: // Llano
|
|
Q_strncatz(cpuString, " A8 APU", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " A series APU", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
else if (extFamily == 0x14) // Bobcat family
|
|
{
|
|
switch (model)
|
|
{
|
|
case 1: // Zacate
|
|
Q_strncatz(cpuString, " E-350 APU", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " E series APU", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
else if (extFamily == 0x15) // Bulldozer family
|
|
{
|
|
switch (model)
|
|
{
|
|
case 0:
|
|
if (extModel == 0x10) // Trinity
|
|
Q_strncatz(cpuString, " A series APU", maxSize);
|
|
else if (extModel == 0x30) // Kaveri
|
|
Q_strncatz(cpuString, " A series APU", maxSize);
|
|
else // Zambezi
|
|
Q_strncatz(cpuString, " FX series", maxSize);
|
|
break;
|
|
case 1:
|
|
if (extModel == 1) // Zambezi
|
|
Q_strncatz(cpuString, " FX series", maxSize);
|
|
else if (extModel == 0x60) // Carrizo
|
|
Q_strncatz(cpuString, " A series APU", maxSize);
|
|
break;
|
|
case 2:
|
|
if (extModel == 2) // Zambezi / Vishera
|
|
Q_strncatz(cpuString, " FX series", maxSize);
|
|
break;
|
|
case 3:
|
|
if (extModel == 0x13) // Richland
|
|
Q_strncatz(cpuString, " A series APU", maxSize);
|
|
break;
|
|
case 5:
|
|
if (extModel == 0x65) // Bristol Ridge
|
|
Q_strncatz(cpuString, " A series APU", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " FX series", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
else if (extFamily == 0x17) // Ryzen Family
|
|
{
|
|
switch (model)
|
|
{
|
|
case 1:
|
|
if (extModel == 1) // Summit Ridge
|
|
Q_strncatz(cpuString, " Ryzen 7/5/3 1x00", maxSize);
|
|
else if (extModel == 0x11) // Raven Ridge
|
|
Q_strncatz(cpuString, " Ryzen 5/3 2x00G", maxSize);
|
|
else if (extModel == 0x71) // Matisse (Zen 2)
|
|
Q_strncatz(cpuString, " Ryzen 9/7/5/3 3x00", maxSize);
|
|
break;
|
|
case 8:
|
|
if (extModel == 8) // Pinnacle Ridge
|
|
Q_strncatz(cpuString, " Ryzen 7/5/3 2x00", maxSize);
|
|
else if (extModel == 0x18) // Picasso
|
|
Q_strncatz(cpuString, " Ryzen 5/3 3x00G", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " Zen/Zen+/Zen2", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
else if (extFamily == 0x19) // Ryzen Zen3 Family
|
|
{
|
|
switch (model)
|
|
{
|
|
case 1:
|
|
if (extModel == 0x21) // Vermeer (Zen3)
|
|
Q_strncatz(cpuString, " Ryzen 9/7/5/3 5x00", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " Zen3", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
default: // unknown family
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_stricmp(vendor, "CyrixInstead"))
|
|
{
|
|
strncpy(cpuString, "Cyrix", maxSize);
|
|
|
|
switch (family)
|
|
{
|
|
case 4:
|
|
Q_strncatz(cpuString, " 5x86", maxSize);
|
|
break;
|
|
case 5:
|
|
switch (model)
|
|
{
|
|
case 2:
|
|
Q_strncatz(cpuString, " 6x86", maxSize);
|
|
break;
|
|
case 4:
|
|
Q_strncatz(cpuString, " MediaGX", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " 6x86 / MediaGX", maxSize);
|
|
break;
|
|
}
|
|
break;
|
|
case 6:
|
|
Q_strncatz(cpuString, " 6x86MX", maxSize);
|
|
break;
|
|
default: // unknown family
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_stricmp(vendor, "CentaurHauls"))
|
|
{
|
|
strncpy(cpuString, "Centaur", maxSize);
|
|
|
|
switch (family)
|
|
{
|
|
case 5:
|
|
switch (model)
|
|
{
|
|
case 4:
|
|
Q_strncatz(cpuString, " C6", maxSize);
|
|
break;
|
|
case 8:
|
|
Q_strncatz(cpuString, " C2", maxSize);
|
|
break;
|
|
case 9:
|
|
Q_strncatz(cpuString, " C3", maxSize);
|
|
break;
|
|
default: // unknown model
|
|
break;
|
|
}
|
|
break;
|
|
default: // unknown family
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_stricmp(vendor, "NexGenDriven"))
|
|
{
|
|
strncpy(cpuString, "NexGen", maxSize);
|
|
|
|
switch (family)
|
|
{
|
|
case 5:
|
|
switch (model)
|
|
{
|
|
case 0:
|
|
Q_strncatz(cpuString, " Nx586 / Nx586FPU", maxSize);
|
|
break;
|
|
default: // unknown model
|
|
break;
|
|
}
|
|
break;
|
|
default: // unknown family
|
|
break;
|
|
}
|
|
}
|
|
else if (!Q_stricmp(vendor, "GenuineIntel"))
|
|
{
|
|
strncpy(cpuString, "Intel", maxSize);
|
|
// Com_sprintf(cpuString, maxSize, "Intel Family %i ExtFamily %i Model %i ExtModel %i", family, extFamily, model, extModel);
|
|
|
|
switch (family)
|
|
{
|
|
case 5: // Pentium family
|
|
switch (model)
|
|
{
|
|
case 0:
|
|
case 1:
|
|
case 2:
|
|
case 3:
|
|
case 4:
|
|
case 7:
|
|
case 8:
|
|
default:
|
|
Q_strncatz(cpuString, " Pentium", maxSize);
|
|
break;
|
|
}
|
|
break;
|
|
case 6:
|
|
if (model == extModel) // P6 family, before extModel was used
|
|
{
|
|
switch (model)
|
|
{
|
|
case 0:
|
|
case 1:
|
|
Q_strncatz(cpuString, " Pentium Pro", maxSize);
|
|
break;
|
|
// Actual differentiation depends on cache settings
|
|
case 3: // Klamath
|
|
case 5: // Deschutes
|
|
Q_strncatz(cpuString, " Pentium II", maxSize);
|
|
break;
|
|
case 6:
|
|
Q_strncatz(cpuString, " Celeron", maxSize);
|
|
break;
|
|
// Actual differentiation depends on cache settings
|
|
case 7: // Katmai
|
|
case 8: // Coppermine
|
|
case 10: // Coppermine
|
|
case 11: // Tualatin
|
|
Q_strncatz(cpuString, " Pentium III", maxSize);
|
|
break;
|
|
case 12: // Silverthorne
|
|
Q_strncatz(cpuString, " Atom", maxSize);
|
|
break;
|
|
case 9: // Banias
|
|
case 13: // Dothan
|
|
Q_strncatz(cpuString, " Pentium M", maxSize);
|
|
break;
|
|
case 14: // Yonah
|
|
Q_strncatz(cpuString, " Core", maxSize);
|
|
break;
|
|
case 15: // Conroe / Kentsfield
|
|
Q_strncatz(cpuString, " Core 2", maxSize);
|
|
break;
|
|
default:
|
|
Q_strncatz(cpuString, " P6", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
else // Newer CPUs
|
|
{
|
|
switch (model)
|
|
{
|
|
case 5:
|
|
if (extModel == 0x25) // Clarkdale / Arrandale
|
|
Q_strncatz(cpuString, " Core i5/i3 6xx / Core i3 5xx", maxSize);
|
|
else if (extModel == 0x45) // Haswell ULT
|
|
Q_strncatz(cpuString, " Core i7/i5/i3 4xxxU", maxSize);
|
|
else if (extModel == 0x55) // Skylake-X, Cascade Lake-X
|
|
Q_strncatz(cpuString, " Core i9/i7 79xx / 78xx", maxSize);
|
|
else if (extModel == 0xA5) // Comet Lake
|
|
Q_strncatz(cpuString, " Core i9/i7/i5/i3 10xxx", maxSize);
|
|
break;
|
|
case 7:
|
|
if (extModel == 0x17) // Wolfdale / Yorkfield (Penryn)
|
|
Q_strncatz(cpuString, " Core 2", maxSize);
|
|
else if (extModel == 0x47) // Broadwell
|
|
Q_strncatz(cpuString, " Core i7/i5 5xxx", maxSize);
|
|
break;
|
|
case 10:
|
|
if (extModel == 0x2A) // Sandy Bridge
|
|
Q_strncatz(cpuString, " Core i7/i5/i3 2xxx", maxSize);
|
|
else if (extModel == 0x3A) // Ivy Bridge
|
|
Q_strncatz(cpuString, " Core i7/i5/i3 3xxx", maxSize);
|
|
else if (extModel == 0x1A) // Bloomfield
|
|
Q_strncatz(cpuString, " Core i7 9xx", maxSize);
|
|
break;
|
|
case 12:
|
|
if (extModel == 0x2C) // Gulftown
|
|
Q_strncatz(cpuString, " Core i7 9xx", maxSize);
|
|
else if (extModel == 0x3C) // Haswell
|
|
Q_strncatz(cpuString, " Core i7/i5/i3 4xxx", maxSize);
|
|
else // Silverthorne
|
|
Q_strncatz(cpuString, " Atom", maxSize);
|
|
break;
|
|
case 13:
|
|
if (extModel == 0x2D) // Sandy Bridge-E
|
|
Q_strncatz(cpuString, " Core i7 39xx / 38xx", maxSize);
|
|
break;
|
|
case 14:
|
|
if (extModel == 0x1E) // Lynnfield
|
|
Q_strncatz(cpuString, " Core i7 8xx / Core i5 7xx", maxSize);
|
|
else if (extModel == 0x3E) // Ivy Bridge-E
|
|
Q_strncatz(cpuString, " Core i7 49xx / 48xx", maxSize);
|
|
else if (extModel == 0x5E) // Skylake
|
|
Q_strncatz(cpuString, " Core i7/i5/i3 6xxx", maxSize);
|
|
else if (extModel == 0x7E) // Ice lake
|
|
Q_strncatz(cpuString, " Core i7/i5/i3 10xxG7", maxSize);
|
|
else if (extModel == 0x9E) // Coffee Lake
|
|
Q_strncatz(cpuString, " Core i7/i5/i3 8xxx", maxSize);
|
|
break;
|
|
case 15:
|
|
if (extModel == 0x0F) // Conroe / Kentsfield (Merom)
|
|
Q_strncatz(cpuString, " Core 2", maxSize);
|
|
else if (extModel == 0x3F) // Haswell-E
|
|
Q_strncatz(cpuString, " Core i7 59xx / 58xx", maxSize);
|
|
else if (extModel == 0x4F) // Broadwell-E
|
|
Q_strncatz(cpuString, " Core i7 69xx / 68xx", maxSize);
|
|
else if (extModel == 0x5E) // Kaby Lake
|
|
Q_strncatz(cpuString, " Core i7/i5/i3 7xxx", maxSize);
|
|
else if (extModel == 0x9E) // Kaby Lake-X
|
|
Q_strncatz(cpuString, " Core i7/i5 7xxx", maxSize);
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
case 7: // Itanium family
|
|
switch (model)
|
|
{
|
|
default:
|
|
Q_strncatz(cpuString, " Itanium", maxSize);
|
|
break;
|
|
}
|
|
break;
|
|
case 15: // refer to extended family
|
|
if (extFamily == 0x0F) // NetBurst family
|
|
{
|
|
switch (model)
|
|
{
|
|
case 0: // Williamette
|
|
case 1: // Williamette
|
|
case 2: // Northwood
|
|
case 3: // Prescott
|
|
case 4: // Smithfield
|
|
case 6: // Cedar Mill / Presler
|
|
Q_strncatz(cpuString, " Pentium 4", maxSize);
|
|
break;
|
|
// case 4: // Smithfield
|
|
// case 6: // Cedar Mill / Presler
|
|
// Q_strncatz(cpuString, " Pentium D", maxSize);
|
|
// break;
|
|
default:
|
|
Q_strncatz(cpuString, " NetBurst", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
else if (extFamily == 0x1F || extFamily == 0x2F) // Itanium 2 family
|
|
{
|
|
switch (model)
|
|
{
|
|
default:
|
|
Q_strncatz(cpuString, " Itanium 2", maxSize);
|
|
break;
|
|
}
|
|
}
|
|
break;
|
|
default: // unknown family
|
|
break;
|
|
}
|
|
}
|
|
else
|
|
return false;
|
|
|
|
// Check if RDTSC instruction is supported
|
|
if ((features >> 4) & 1)
|
|
{
|
|
// Measure CPU speed
|
|
QueryPerformanceFrequency((LARGE_INTEGER *)&frequency);
|
|
|
|
#if defined (_M_X64) || defined (_M_AMD64) || defined (__x86_64__)
|
|
start = __rdtsc();
|
|
#elif defined (_M_IX86)
|
|
__asm {
|
|
rdtsc
|
|
mov dword ptr[start+0], eax
|
|
mov dword ptr[start+4], edx
|
|
}
|
|
#endif
|
|
|
|
QueryPerformanceCounter((LARGE_INTEGER *)&stop);
|
|
stop += frequency;
|
|
|
|
do {
|
|
QueryPerformanceCounter((LARGE_INTEGER *)&counter);
|
|
} while (counter < stop);
|
|
|
|
#if defined (_M_X64) || defined (_M_AMD64) || defined (__x86_64__)
|
|
end = __rdtsc();
|
|
#elif defined (_M_IX86)
|
|
__asm {
|
|
rdtsc
|
|
mov dword ptr[end+0], eax
|
|
mov dword ptr[end+4], edx
|
|
}
|
|
#endif
|
|
|
|
speed = (unsigned)((end - start) / 1000000);
|
|
|
|
if (speed > 1000)
|
|
Q_strncatz(cpuString, va(" %4.2f GHz", ((float)speed/1000.0f)), maxSize);
|
|
else
|
|
Q_strncatz(cpuString, va(" %u MHz", speed), maxSize);
|
|
}
|
|
|
|
// get number of logical processors
|
|
GetSystemInfo(&sysInfo);
|
|
if (sysInfo.dwNumberOfProcessors > 1)
|
|
Q_strncatz(cpuString, va(" (%u logical CPUs)", sysInfo.dwNumberOfProcessors), maxSize);
|
|
/*
|
|
if (numLogicalCores >= 2 || numCores >= 2)
|
|
{
|
|
if (numLogicalCores > numCores) // Hyperthreading or SMT
|
|
Q_strncatz(cpuString, va(" (%u cores, %u threads)", numCores, numLogicalCores), maxSize);
|
|
else
|
|
Q_strncatz(cpuString, va(" (%u cores)", numCores), maxSize);
|
|
}
|
|
*/
|
|
|
|
// Get extended instruction sets supported
|
|
hasMMX = (features >> 23) & 1;
|
|
hasMMXExt = (extFeatures >> 22) & 1;
|
|
has3DNow = (extFeatures >> 31) & 1;
|
|
has3DNowExt = (extFeatures >> 30) & 1;
|
|
hasSSE = (features >> 25) & 1;
|
|
hasSSE2 = (features >> 26) & 1;
|
|
hasSSE3 = (moreFeatures >> 0) & 1;
|
|
hasSSE41 = (moreFeatures >> 19) & 1;
|
|
hasSSE42 = (moreFeatures >> 20) & 1;
|
|
hasSSE4a = (moreFeatures >> 6) & 1;
|
|
hasAVX = (moreFeatures >> 28) & 1;
|
|
|
|
if (hasMMX || has3DNow || hasSSE)
|
|
{
|
|
Q_strncatz(cpuString, " w/", maxSize);
|
|
|
|
if (hasMMX){
|
|
Q_strncatz(cpuString, " MMX", maxSize);
|
|
if (hasMMXExt)
|
|
Q_strncatz(cpuString, "+", maxSize);
|
|
}
|
|
if (has3DNow){
|
|
Q_strncatz(cpuString, " 3DNow!", maxSize);
|
|
if (has3DNowExt)
|
|
Q_strncatz(cpuString, "+", maxSize);
|
|
}
|
|
if (hasSSE){
|
|
Q_strncatz(cpuString, " SSE", maxSize);
|
|
if (hasSSE42)
|
|
Q_strncatz(cpuString, "4.2", maxSize);
|
|
else if (hasSSE41)
|
|
Q_strncatz(cpuString, "4.1", maxSize);
|
|
else if (hasSSE3)
|
|
Q_strncatz(cpuString, "3", maxSize);
|
|
else if (hasSSE2)
|
|
Q_strncatz(cpuString, "2", maxSize);
|
|
}
|
|
if (hasSSE4a){
|
|
Q_strncatz(cpuString, " SSE4a", maxSize);
|
|
}
|
|
if (hasAVX){
|
|
Q_strncatz(cpuString, " AVX", maxSize);
|
|
}
|
|
}
|
|
|
|
return true;
|
|
|
|
#else
|
|
|
|
// Q_strncpyz(cpuString, "Alpha AXP", maxSize);
|
|
Q_strncpyz(cpuString, CPUSTRING, maxSize); // [Slipyx] mingw support
|
|
return true;
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
================
|
|
Sys_Init
|
|
================
|
|
*/
|
|
void Sys_Init (void)
|
|
{
|
|
OSVERSIONINFOEX osInfo;
|
|
SYSTEM_INFO sysInfo;
|
|
MEMORYSTATUS memStatus; // Knightmare added
|
|
char string[128]; // Knightmare added
|
|
|
|
#if 0
|
|
// allocate a named semaphore on the client so the
|
|
// front end can tell if it is alive
|
|
|
|
// mutex will fail if semephore already exists
|
|
qwclsemaphore = CreateMutex(
|
|
NULL, /* Security attributes */
|
|
0, /* owner */
|
|
"qwcl"); /* Semaphore name */
|
|
if (!qwclsemaphore)
|
|
Sys_Error ("QWCL is already running on this system");
|
|
CloseHandle (qwclsemaphore);
|
|
|
|
qwclsemaphore = CreateSemaphore(
|
|
NULL, /* Security attributes */
|
|
0, /* Initial count */
|
|
1, /* Maximum count */
|
|
"qwcl"); /* Semaphore name */
|
|
#endif
|
|
|
|
timeBeginPeriod( 1 );
|
|
|
|
osInfo.dwOSVersionInfoSize = sizeof(osInfo);
|
|
|
|
if (!GetVersionEx ((OSVERSIONINFO*) &osInfo))
|
|
Sys_Error ("Couldn't get OS info");
|
|
|
|
GetSystemInfo(&sysInfo);
|
|
|
|
if (osInfo.dwMajorVersion < 4)
|
|
Sys_Error ("KMQuake2 requires windows version 4 or greater");
|
|
if (osInfo.dwPlatformId == VER_PLATFORM_WIN32s)
|
|
Sys_Error ("KMQuake2 doesn't run on Win32s");
|
|
else if ( osInfo.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS )
|
|
s_win95 = true;
|
|
|
|
// from Q2E - OS & CPU detection
|
|
if (osInfo.dwPlatformId == VER_PLATFORM_WIN32_NT)
|
|
{
|
|
|
|
// wProductType field not supported in MSVC6
|
|
#if (_MSC_VER < 1300)
|
|
if (osInfo.dwMajorVersion == 4) {
|
|
Q_strncpyz (string, "Windows NT", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 5 && osInfo.dwMinorVersion == 0) {
|
|
Q_strncpyz (string, "Windows 2000", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 5 && osInfo.dwMinorVersion == 1) {
|
|
Q_strncpyz (string, "Windows XP", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 5 && osInfo.dwMinorVersion == 2) {
|
|
Q_strncpyz (string, "Windows XP", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 6 && osInfo.dwMinorVersion == 0) {
|
|
Q_strncpyz (string, "Windows Vista", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 6 && osInfo.dwMinorVersion == 1) {
|
|
Q_strncpyz (string, "Windows 7", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 6 && osInfo.dwMinorVersion == 2) {
|
|
Q_strncpyz (string, "Windows 8", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 6 && osInfo.dwMinorVersion == 3) {
|
|
Q_strncpyz (string, "Windows 8.1", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 10 && osInfo.dwMinorVersion == 0) {
|
|
Q_strncpyz (string, "Windows 10", sizeof(string));
|
|
}
|
|
#else // (_MSC_VER < 1300)
|
|
if (osInfo.dwMajorVersion == 4) {
|
|
Q_strncpyz (string, "Windows NT", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 5 && osInfo.dwMinorVersion == 0) {
|
|
if (osInfo.wProductType == VER_NT_WORKSTATION)
|
|
Q_strncpyz (string, "Windows 2000", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows 2000 Server", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 5 && osInfo.dwMinorVersion == 1) {
|
|
Q_strncpyz (string, "Windows XP", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 5 && osInfo.dwMinorVersion == 2) {
|
|
// if ( (osInfo.wProductType == VER_NT_WORKSTATION) && (sysInfo.wProcessorArchitecture == PROCESSOR_ARCHITECTURE_AMD64) )
|
|
// Q_strncpyz (string, "Windows XP x64 Edition", sizeof(string));
|
|
if (osInfo.wProductType == VER_NT_WORKSTATION)
|
|
Q_strncpyz (string, "Windows XP", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows Server 2003", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 6 && osInfo.dwMinorVersion == 0) {
|
|
if (osInfo.wProductType == VER_NT_WORKSTATION)
|
|
Q_strncpyz (string, "Windows Vista", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows Server 2008", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 6 && osInfo.dwMinorVersion == 1) {
|
|
if (osInfo.wProductType == VER_NT_WORKSTATION)
|
|
Q_strncpyz (string, "Windows 7", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows Server 2008 R2", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 6 && osInfo.dwMinorVersion == 2) {
|
|
if (osInfo.wProductType == VER_NT_WORKSTATION)
|
|
Q_strncpyz (string, "Windows 8", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows Server 2012", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 6 && osInfo.dwMinorVersion == 3) {
|
|
if (osInfo.wProductType == VER_NT_WORKSTATION)
|
|
Q_strncpyz (string, "Windows 8.1", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows Server 2012 R2", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 10 && osInfo.dwMinorVersion == 0) {
|
|
if (osInfo.wProductType == VER_NT_WORKSTATION)
|
|
Q_strncpyz (string, "Windows 10", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows Server 2016", sizeof(string));
|
|
}
|
|
#endif // (_MSC_VER < 1300)
|
|
else {
|
|
Q_strncpyz (string, va("Windows %i.%i", osInfo.dwMajorVersion, osInfo.dwMinorVersion), sizeof(string));
|
|
}
|
|
|
|
if (sysInfo.wProcessorArchitecture == PROCESSOR_ARCHITECTURE_AMD64)
|
|
Q_strncatz (string, " x64", sizeof(string));
|
|
if (strlen(osInfo.szCSDVersion) > 0)
|
|
Q_strncatz (string, va(" %s", osInfo.szCSDVersion), sizeof(string));
|
|
}
|
|
else if (osInfo.dwPlatformId == VER_PLATFORM_WIN32_WINDOWS)
|
|
{
|
|
if (osInfo.dwMajorVersion == 4 && osInfo.dwMinorVersion == 0) {
|
|
if (osInfo.szCSDVersion[1] == 'C' || osInfo.szCSDVersion[1] == 'B')
|
|
Q_strncpyz (string, "Windows 95 OSR2", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows 95", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 4 && osInfo.dwMinorVersion == 10) {
|
|
if (osInfo.szCSDVersion[1] == 'A')
|
|
Q_strncpyz (string, "Windows 98 SE", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, "Windows 98", sizeof(string));
|
|
}
|
|
else if (osInfo.dwMajorVersion == 4 && osInfo.dwMinorVersion == 90)
|
|
Q_strncpyz (string, "Windows ME", sizeof(string));
|
|
else
|
|
Q_strncpyz (string, va("Windows %i.%i", osInfo.dwMajorVersion, osInfo.dwMinorVersion), sizeof(string));
|
|
}
|
|
else
|
|
Q_strncpyz (string, va("Windows %i.%i", osInfo.dwMajorVersion, osInfo.dwMinorVersion), sizeof(string));
|
|
|
|
if (osInfo.dwBuildNumber > 0)
|
|
Q_strncatz (string, va(", build %i", osInfo.dwBuildNumber), sizeof(string));
|
|
Com_Printf ("OS: %s\n", string);
|
|
Cvar_Get ("sys_osVersion", string, CVAR_NOSET|CVAR_LATCH|CVAR_SAVE_IGNORE);
|
|
|
|
// Detect CPU
|
|
Com_Printf("Detecting CPU... ");
|
|
if (Sys_DetectCPU(string, sizeof(string))) {
|
|
Com_Printf("Found %s\n", string);
|
|
Cvar_Get("sys_cpuString", string, CVAR_NOSET|CVAR_LATCH|CVAR_SAVE_IGNORE);
|
|
}
|
|
else {
|
|
Com_Printf("Unknown CPU found\n");
|
|
Cvar_Get("sys_cpuString", "Unknown", CVAR_NOSET|CVAR_LATCH|CVAR_SAVE_IGNORE);
|
|
}
|
|
|
|
// Get physical memory
|
|
GlobalMemoryStatus(&memStatus);
|
|
strncpy(string, va("%u",memStatus.dwTotalPhys >> 20), sizeof(string));
|
|
Com_Printf("Memory: %s MB\n", string);
|
|
Cvar_Get("sys_ramMegs", string, CVAR_NOSET|CVAR_LATCH|CVAR_SAVE_IGNORE);
|
|
// end Q2E detection
|
|
|
|
#ifndef NEW_DED_CONSOLE
|
|
Sys_InitConsole (); // show dedicated console, moved to function
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
==============================================================================
|
|
|
|
WINDOWS CRAP
|
|
|
|
==============================================================================
|
|
*/
|
|
|
|
/*
|
|
=================
|
|
Sys_AppActivate
|
|
=================
|
|
*/
|
|
void Sys_AppActivate (void)
|
|
{
|
|
ShowWindow ( cl_hwnd, SW_RESTORE);
|
|
SetForegroundWindow ( cl_hwnd );
|
|
}
|
|
|
|
/*
|
|
========================================================================
|
|
|
|
GAME DLL
|
|
|
|
========================================================================
|
|
*/
|
|
|
|
static HINSTANCE game_library;
|
|
|
|
/*
|
|
=================
|
|
Sys_UnloadGame
|
|
=================
|
|
*/
|
|
void Sys_UnloadGame (void)
|
|
{
|
|
if (!FreeLibrary (game_library))
|
|
Com_Error (ERR_FATAL, "FreeLibrary failed for game library");
|
|
game_library = NULL;
|
|
}
|
|
|
|
/*
|
|
=================
|
|
Sys_GetGameAPI
|
|
|
|
Loads the game dll
|
|
=================
|
|
*/
|
|
void *Sys_GetGameAPI (void *parms)
|
|
{
|
|
void *(*GetGameAPI) (void *);
|
|
char name[MAX_OSPATH];
|
|
char *path;
|
|
char cwd[MAX_OSPATH];
|
|
|
|
#if defined (_M_X64) || defined (_M_AMD64) || defined (__x86_64__)
|
|
const char *gamename = "kmq2gamex64.dll";
|
|
|
|
#ifdef NDEBUG
|
|
const char *debugdir = "release";
|
|
#else
|
|
const char *debugdir = "debug";
|
|
#endif
|
|
|
|
#elif defined (_M_IX86)
|
|
// Knightmare- changed DLL name for better cohabitation
|
|
const char *gamename = "kmq2gamex86.dll";
|
|
|
|
#ifdef NDEBUG
|
|
const char *debugdir = "release";
|
|
#else
|
|
const char *debugdir = "debug";
|
|
#endif
|
|
|
|
#elif defined _M_ALPHA
|
|
const char *gamename = "kmq2gameaxp.dll";
|
|
|
|
#ifdef NDEBUG
|
|
const char *debugdir = "releaseaxp";
|
|
#else
|
|
const char *debugdir = "debugaxp";
|
|
#endif
|
|
|
|
#endif
|
|
|
|
if (game_library)
|
|
Com_Error (ERR_FATAL, "Sys_GetGameAPI without Sys_UnloadingGame");
|
|
|
|
// check the current debug directory first for development purposes
|
|
_getcwd (cwd, sizeof(cwd));
|
|
Com_sprintf (name, sizeof(name), "%s/%s/%s", cwd, debugdir, gamename);
|
|
game_library = LoadLibrary ( name );
|
|
if (game_library)
|
|
{
|
|
Com_DPrintf ("LoadLibrary (%s)\n", name);
|
|
}
|
|
else
|
|
{
|
|
#ifdef DEBUG
|
|
// check the current directory for other development purposes
|
|
Com_sprintf (name, sizeof(name), "%s/%s", cwd, gamename);
|
|
game_library = LoadLibrary ( name );
|
|
if (game_library)
|
|
{
|
|
Com_DPrintf ("LoadLibrary (%s)\n", name);
|
|
}
|
|
else
|
|
#endif
|
|
{
|
|
// now run through the search paths
|
|
path = NULL;
|
|
while (1)
|
|
{
|
|
// path = FS_NextPath (path);
|
|
path = FS_NextGamePath (path);
|
|
if (!path)
|
|
return NULL; // couldn't find one anywhere
|
|
Com_sprintf (name, sizeof(name), "%s/%s", path, gamename);
|
|
game_library = LoadLibrary (name);
|
|
if (game_library)
|
|
{
|
|
Com_DPrintf ("LoadLibrary (%s)\n",name);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
GetGameAPI = (void *)GetProcAddress (game_library, "GetGameAPI");
|
|
if (!GetGameAPI)
|
|
{
|
|
Sys_UnloadGame ();
|
|
return NULL;
|
|
}
|
|
|
|
return GetGameAPI (parms);
|
|
}
|
|
|
|
//=======================================================================
|
|
|
|
|
|
/*
|
|
==================
|
|
ParseCommandLine
|
|
|
|
==================
|
|
*/
|
|
void ParseCommandLine (LPSTR lpCmdLine)
|
|
{
|
|
argc = 1;
|
|
argv[0] = "exe";
|
|
|
|
while (*lpCmdLine && (argc < MAX_NUM_ARGVS))
|
|
{
|
|
while (*lpCmdLine && ((*lpCmdLine <= 32) || (*lpCmdLine > 126)))
|
|
lpCmdLine++;
|
|
|
|
if (*lpCmdLine)
|
|
{
|
|
argv[argc] = lpCmdLine;
|
|
argc++;
|
|
|
|
while (*lpCmdLine && ((*lpCmdLine > 32) && (*lpCmdLine <= 126)))
|
|
lpCmdLine++;
|
|
|
|
if (*lpCmdLine)
|
|
{
|
|
*lpCmdLine = 0;
|
|
lpCmdLine++;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
ReplaceBackSlashes
|
|
|
|
Replaces backslashes in a path with slashes.
|
|
==================
|
|
*/
|
|
static void ReplaceBackSlashes (char *path)
|
|
{
|
|
char *cur, *old;
|
|
|
|
cur = old = path;
|
|
if (strstr(cur, "\\") != NULL)
|
|
{
|
|
while (cur != NULL)
|
|
{
|
|
if ((cur - old) > 1) {
|
|
*cur = '/';
|
|
}
|
|
old = cur;
|
|
cur = strchr(old + 1, '\\');
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Sys_ExeDir
|
|
==================
|
|
*/
|
|
const char *Sys_ExeDir (void)
|
|
{
|
|
return exe_dir;
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Sys_PrefDir
|
|
==================
|
|
*/
|
|
const char *Sys_PrefDir (void)
|
|
{
|
|
return pref_dir;
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Sys_DownloadDir
|
|
==================
|
|
*/
|
|
const char *Sys_DownloadDir (void)
|
|
{
|
|
return download_dir;
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Sys_InitPrefDir
|
|
|
|
Adapted from DK 1.3 source
|
|
==================
|
|
*/
|
|
void Sys_InitPrefDir (void)
|
|
{
|
|
if ( win_use_profile_dir && win_use_profile_dir->integer )
|
|
{
|
|
char profile[MAX_PATH], dlPath[MAX_PATH];
|
|
const char *reason = "No error!";
|
|
const char *reason_dl = "No error!";
|
|
int len, len_dl;
|
|
qboolean bGotNT6SavedGames=false, bGotNT6Downloads=false;
|
|
|
|
// Use Saved Games/KMQuake2 on Win Vista and later, unless "mygames" parameter is set
|
|
if ( Detect_WinNT6orLater() && !COM_CheckParm ("-mygames") && !COM_CheckParm ("+mygames") )
|
|
{
|
|
WCHAR *wprofile, *wDLPath;
|
|
HMODULE hShell32 = LoadLibrary("shell32");
|
|
HMODULE hOle32 = LoadLibrary("ole32");
|
|
|
|
if ( !hShell32 || !hOle32 ) {
|
|
reason = reason_dl = "shell32.dll or ole32.dll couldn't be loaded";
|
|
}
|
|
else
|
|
{
|
|
fnSHGetKnownFolderPath = (void *)GetProcAddress (hShell32, "SHGetKnownFolderPath");
|
|
fnCoInitializeEx = (void *)GetProcAddress (hOle32, "CoInitializeEx");
|
|
fnCoUninitialize = (void *)GetProcAddress (hOle32, "CoUninitialize");
|
|
fnCoTaskMemFree = (void *)GetProcAddress (hOle32, "CoTaskMemFree");
|
|
if ( !fnSHGetKnownFolderPath || !fnCoInitializeEx || !fnCoUninitialize || !fnCoTaskMemFree ) {
|
|
reason = reason_dl = "functions SHGetKnownFolderPath / CoInitializeEx / CoUninitialize / CoTaskMemFree couldn't be mapped";
|
|
}
|
|
else
|
|
{
|
|
// from DarkPlaces
|
|
memset (profile, 0, sizeof(profile));
|
|
fnCoInitializeEx(NULL, COINIT_APARTMENTTHREADED);
|
|
if (fnSHGetKnownFolderPath(&qFOLDERID_SavedGames, qKF_FLAG_CREATE | qKF_FLAG_NO_ALIAS, NULL, &wprofile) == S_OK)
|
|
{
|
|
memset(profile, 0, sizeof(profile));
|
|
#if _MSC_VER >= 1400
|
|
wcstombs_s(NULL, profile, sizeof(profile), wprofile, sizeof(profile)-1);
|
|
#else
|
|
wcstombs(profile, wprofile, sizeof(profile)-1);
|
|
#endif
|
|
fnCoTaskMemFree(wprofile);
|
|
}
|
|
// Also get Downloads Folder from qFOLDERID_Downloads
|
|
if (fnSHGetKnownFolderPath(&qFOLDERID_Downloads, qKF_FLAG_CREATE | qKF_FLAG_NO_ALIAS, NULL, &wDLPath) == S_OK)
|
|
{
|
|
memset(dlPath, 0, sizeof(dlPath));
|
|
#if _MSC_VER >= 1400
|
|
wcstombs_s(NULL, dlPath, sizeof(dlPath), wDLPath, sizeof(dlPath)-1);
|
|
#else
|
|
wcstombs(dlPath, wDLPath, sizeof(dlPath)-1);
|
|
#endif
|
|
fnCoTaskMemFree(wDLPath);
|
|
}
|
|
fnCoUninitialize();
|
|
// end DarkPlaces code
|
|
|
|
len = (int)strlen(profile);
|
|
if (len == 0) {
|
|
reason = "SHGetKnownFolderPath()/wcstombs() returned 0 length string";
|
|
}
|
|
else
|
|
{
|
|
// Check if path is too long
|
|
if ( (len + strlen(NT6_SAVEDIR) + 3) >= 256 ) {
|
|
reason = "the resulting path would be too long (>= 256 chars)";
|
|
}
|
|
else
|
|
{
|
|
// Replace backslashes with slashes
|
|
ReplaceBackSlashes (profile);
|
|
Com_sprintf (pref_dir, sizeof(pref_dir), "%s/%s", profile, NT6_SAVEDIR);
|
|
bGotNT6SavedGames = true;
|
|
// return;
|
|
}
|
|
}
|
|
|
|
len_dl = (int)strlen(dlPath);
|
|
if (len_dl == 0) {
|
|
reason_dl = "SHGetKnownFolderPath()/wcstombs() returned 0 length string";
|
|
}
|
|
else
|
|
{
|
|
// Check if path is too long
|
|
if ( (len_dl + strlen(NT6_DLDIR) + 3) >= 256 ) {
|
|
reason_dl = "the resulting path would be too long (>= 256 chars)";
|
|
}
|
|
else
|
|
{
|
|
// Replace backslashes with slashes
|
|
ReplaceBackSlashes (dlPath);
|
|
Com_sprintf (download_dir, sizeof(download_dir), "%s/%s", dlPath, NT6_DLDIR);
|
|
bGotNT6Downloads = true;
|
|
// return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if (bGotNT6SavedGames && bGotNT6Downloads) // successfully got both dirs, return now
|
|
return;
|
|
|
|
// Output reason for why one or both dirs couldn't be found, then fall back on NT5 path
|
|
if (!bGotNT6SavedGames)
|
|
Com_Printf("Couldn't get PrefDir (Saved Games/KMQuake2), because %s.\n", reason);
|
|
if (!bGotNT6Downloads)
|
|
Com_Printf("Couldn't get DownloadDir (Downloads/KMQuake2), because %s.\n", reason_dl);
|
|
}
|
|
|
|
// Use My Documents/My Games/KMQuake2 on Win2K / XP
|
|
if ( Detect_WinNT5orLater() )
|
|
{
|
|
WCHAR sprofile[MAX_PATH];
|
|
WCHAR uprofile[MAX_PATH];
|
|
HMODULE hShell32 = LoadLibrary("shell32");
|
|
|
|
if (!hShell32) {
|
|
reason = "shell32.dll couldn't be loaded";
|
|
}
|
|
else
|
|
{
|
|
fnSHGetFolderPathW = (SHGETFOLDERPATHW)GetProcAddress (hShell32, "SHGetFolderPathW");
|
|
if (!fnSHGetFolderPathW) {
|
|
reason = "function SHGetFolderPathW couldn't be mapped";
|
|
}
|
|
else
|
|
{
|
|
memset (pref_dir, 0, sizeof(pref_dir));
|
|
|
|
/* The following lines implement a horrible
|
|
hack to connect the UTF-16 WinAPI to the
|
|
ASCII Quake II. While this should work in
|
|
most cases, it'll fail if the "Windows to
|
|
DOS filename translation" is switched off.
|
|
In that case the function will return NULL
|
|
and no homedir is used. */
|
|
|
|
// Get path to "My Documents" folder
|
|
fnSHGetFolderPathW (NULL, CSIDL_PERSONAL, NULL, 8, uprofile);
|
|
|
|
// Create a UTF-16 DOS path
|
|
len = GetShortPathNameW (uprofile, sprofile, sizeof(sprofile));
|
|
|
|
if (len == 0) {
|
|
reason = "GetShortPathNameW() returned 0";
|
|
}
|
|
else
|
|
{
|
|
// Since the DOS path contains no UTF-16 characters, just convert it to ASCII
|
|
len = WideCharToMultiByte (CP_ACP, 0, sprofile, -1, profile, sizeof(profile), NULL, NULL);
|
|
|
|
if (len == 0) {
|
|
reason = "WideCharToMultiByte() returned 0";
|
|
}
|
|
else
|
|
{
|
|
// Check if path is too long
|
|
if ( ((len + strlen(NT5_SAVEDIR) + 3) >= 256) || ((len + strlen(NT5_DLDIR) + 3) >= 256) ) {
|
|
reason = "The resulting path would be too long (>= 256 chars)";
|
|
}
|
|
else
|
|
{
|
|
// Replace backslashes with slashes
|
|
ReplaceBackSlashes (profile);
|
|
// Allow splitting of dirs from above NT6 section if only one failed
|
|
if (!bGotNT6SavedGames)
|
|
Com_sprintf (pref_dir, sizeof(pref_dir), "%s/%s", profile, NT5_SAVEDIR);
|
|
if (!bGotNT6Downloads)
|
|
Com_sprintf (download_dir, sizeof(download_dir), "%s/%s", profile, NT5_DLDIR);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
Com_Printf("Couldn't get PrefDir (My Documents/My Games/KMQuake2), because %s.\n", reason);
|
|
}
|
|
}
|
|
|
|
Com_sprintf (pref_dir, sizeof(pref_dir), exe_dir);
|
|
Com_sprintf (download_dir, sizeof(download_dir), exe_dir);
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Init_ExeDir
|
|
==================
|
|
*/
|
|
static void Init_ExeDir (void)
|
|
{
|
|
#if 1
|
|
memset(exe_dir, 0, sizeof(exe_dir));
|
|
Q_snprintfz (exe_dir, sizeof(exe_dir), ".");
|
|
#else
|
|
char buf[MAX_PATH];
|
|
const char *lastSlash;
|
|
int dirLen;
|
|
|
|
memset(buf, 0, sizeof(buf));
|
|
memset(exe_dir, 0, sizeof(exe_dir));
|
|
|
|
GetModuleFileName (NULL, buf, sizeof(buf)-1);
|
|
|
|
// get path up to last backslash
|
|
lastSlash = strrchr(buf, '\\');
|
|
if (lastSlash == NULL)
|
|
lastSlash = strrchr(buf, '/');
|
|
|
|
dirLen = lastSlash ? (lastSlash - buf) : 0;
|
|
if ( lastSlash == NULL || dirLen == 0 || dirLen >= sizeof(exe_dir) ) {
|
|
Q_snprintfz (exe_dir, sizeof(exe_dir), ".");
|
|
}
|
|
else {
|
|
memcpy(exe_dir, buf, dirLen);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
FixWorkingDirectory
|
|
==================
|
|
*/
|
|
void FixWorkingDirectory (void)
|
|
{
|
|
int i;
|
|
char *p;
|
|
char curDir[MAX_PATH];
|
|
|
|
GetModuleFileName (NULL, curDir, sizeof(curDir)-1);
|
|
|
|
p = strrchr (curDir, '\\');
|
|
p[0] = 0;
|
|
|
|
for (i = 1; i < argc; i++)
|
|
{
|
|
if (!strcmp (argv[i], "-nopathcheck"))
|
|
goto skipPathCheck;
|
|
|
|
if (!strcmp (argv[i], "-nocwdcheck"))
|
|
return;
|
|
}
|
|
|
|
if (strlen(curDir) > (MAX_OSPATH - MAX_QPATH))
|
|
Sys_Error ("Current path is too long. Please move your Quake II installation to a shorter path.");
|
|
|
|
skipPathCheck:
|
|
|
|
SetCurrentDirectory (curDir);
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Detect_WinNT5orLater
|
|
==================
|
|
*/
|
|
qboolean Detect_WinNT5orLater (void)
|
|
{
|
|
DWORD WinVersion;
|
|
DWORD WinLowByte, WinHighByte;
|
|
|
|
WinVersion = GetVersion();
|
|
WinLowByte = (DWORD)(LOBYTE(LOWORD(WinVersion)));
|
|
WinHighByte = (DWORD)(HIBYTE(HIWORD(WinVersion)));
|
|
|
|
if (WinLowByte <= 4) {
|
|
Com_DPrintf("Windows 9x or NT 4 detected.\n");
|
|
return false;
|
|
}
|
|
|
|
if (WinLowByte >= 5) {
|
|
Com_DPrintf("Windows 5.x or later detected.\n");
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Detect_WinNT6orLater
|
|
==================
|
|
*/
|
|
qboolean Detect_WinNT6orLater (void)
|
|
{
|
|
DWORD WinVersion;
|
|
DWORD WinLowByte, WinHighByte;
|
|
|
|
WinVersion = GetVersion();
|
|
WinLowByte = (DWORD)(LOBYTE(LOWORD(WinVersion)));
|
|
WinHighByte = (DWORD)(HIBYTE(HIWORD(WinVersion)));
|
|
|
|
if (WinLowByte <= 4) {
|
|
Com_DPrintf("Windows 9x or NT 4 detected.\n");
|
|
return false;
|
|
}
|
|
|
|
if (WinLowByte == 5) {
|
|
Com_DPrintf("Windows 5.x (Win2K / XP / Server 2003) detected.\n");
|
|
return false;
|
|
}
|
|
|
|
if (WinLowByte >= 6) {
|
|
Com_DPrintf("Windows 6.x (WinVista / 7 / 8) detected.\n");
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
Sys_SetHighDPIMode
|
|
|
|
From Yamagi Quake2
|
|
==================
|
|
*/
|
|
typedef enum KMQ2_PROCESS_DPI_AWARENESS {
|
|
KMQ2_PROCESS_DPI_UNAWARE = 0,
|
|
KMQ2_PROCESS_SYSTEM_DPI_AWARE = 1,
|
|
KMQ2_PROCESS_PER_MONITOR_DPI_AWARE = 2
|
|
} KMQ2_PROCESS_DPI_AWARENESS;
|
|
|
|
void Sys_SetHighDPIMode (void)
|
|
{
|
|
HINSTANCE userDLL, shcoreDLL;
|
|
|
|
/* For Vista, Win7 and Win8 */
|
|
BOOL(WINAPI *SetProcessDPIAware)(void) = NULL;
|
|
|
|
/* Win8.1 and later */
|
|
HRESULT(WINAPI *SetProcessDpiAwareness)(KMQ2_PROCESS_DPI_AWARENESS dpiAwareness) = NULL;
|
|
|
|
userDLL = LoadLibrary("USER32.DLL");
|
|
if (userDLL)
|
|
{
|
|
SetProcessDPIAware = (BOOL(WINAPI *)(void)) GetProcAddress(userDLL,
|
|
"SetProcessDPIAware");
|
|
}
|
|
|
|
shcoreDLL = LoadLibrary("SHCORE.DLL");
|
|
if (shcoreDLL)
|
|
{
|
|
SetProcessDpiAwareness = (HRESULT(WINAPI *)(KMQ2_PROCESS_DPI_AWARENESS))
|
|
GetProcAddress(shcoreDLL, "SetProcessDpiAwareness");
|
|
}
|
|
|
|
if (SetProcessDpiAwareness) {
|
|
SetProcessDpiAwareness(KMQ2_PROCESS_PER_MONITOR_DPI_AWARE);
|
|
}
|
|
else if (SetProcessDPIAware) {
|
|
SetProcessDPIAware();
|
|
}
|
|
}
|
|
|
|
|
|
/*
|
|
==================
|
|
WinMain
|
|
==================
|
|
*/
|
|
HINSTANCE global_hInstance;
|
|
HWND hwnd_dialog; // Knightmare added
|
|
|
|
int WINAPI WinMain (HINSTANCE hInstance, HINSTANCE hPrevInstance, LPSTR lpCmdLine, int nCmdShow)
|
|
{
|
|
MSG msg;
|
|
int time, oldtime, newtime;
|
|
char *cddir;
|
|
int i; // Knightmare added
|
|
qboolean cdscan = false; // Knightmare added
|
|
|
|
/* previous instances do not exist in Win32 */
|
|
if (hPrevInstance)
|
|
return 0;
|
|
|
|
global_hInstance = hInstance;
|
|
|
|
ParseCommandLine (lpCmdLine);
|
|
|
|
Sys_SetHighDPIMode (); // setup DPI awareness
|
|
|
|
// r1ch: always change to our directory (ugh)
|
|
FixWorkingDirectory ();
|
|
|
|
Init_ExeDir (); // Knightmare added
|
|
|
|
#ifndef NEW_DED_CONSOLE
|
|
// Knightmare- startup logo, code from TomazQuake
|
|
//if (!(dedicated && dedicated->value))
|
|
{
|
|
hwnd_dialog = CreateDialog(hInstance, MAKEINTRESOURCE(IDD_DIALOG1), NULL, NULL);
|
|
RECT rect; // Knightmare added
|
|
|
|
if (hwnd_dialog)
|
|
{
|
|
if (GetWindowRect (hwnd_dialog, &rect))
|
|
{
|
|
if (rect.left > (rect.top * 2))
|
|
{
|
|
SetWindowPos (hwnd_dialog, 0, (rect.left/2) - ((rect.right - rect.left)/2), rect.top, 0, 0, SWP_NOZORDER | SWP_NOSIZE);
|
|
}
|
|
}
|
|
|
|
ShowWindow (hwnd_dialog, SW_SHOWDEFAULT);
|
|
UpdateWindow (hwnd_dialog);
|
|
SetForegroundWindow (hwnd_dialog);
|
|
}
|
|
}
|
|
// end Knightmare
|
|
#endif
|
|
|
|
#ifdef NEW_DED_CONSOLE // init debug console
|
|
Sys_InitDedConsole ();
|
|
Com_Printf("KMQ2 %4.2f %s %s %s %s\n", VERSION, CPUSTRING, OS_STRING, COMPILETYPE_STRING, __DATE__);
|
|
#endif
|
|
|
|
// Knightmare- scan for cd command line option
|
|
for (i=0; i<argc; i++)
|
|
if (!strcmp(argv[i], "scanforcd")) {
|
|
cdscan = true;
|
|
break;
|
|
}
|
|
|
|
// if we find the CD, add a +set cddir xxx command line
|
|
if (cdscan)
|
|
{
|
|
cddir = Sys_ScanForCD ();
|
|
if (cddir && argc < MAX_NUM_ARGVS - 3)
|
|
{
|
|
int i;
|
|
|
|
// don't override a cddir on the command line
|
|
for (i=0 ; i<argc ; i++)
|
|
if (!strcmp(argv[i], "cddir"))
|
|
break;
|
|
if (i == argc)
|
|
{
|
|
argv[argc++] = "+set";
|
|
argv[argc++] = "cddir";
|
|
argv[argc++] = cddir;
|
|
}
|
|
}
|
|
}
|
|
|
|
Qcommon_Init (argc, argv);
|
|
oldtime = Sys_Milliseconds ();
|
|
|
|
/* main window message loop */
|
|
while (1)
|
|
{
|
|
// if at a full screen console, don't update unless needed
|
|
if (Minimized || (dedicated && dedicated->value) )
|
|
{
|
|
Sleep (1);
|
|
}
|
|
|
|
while (PeekMessage (&msg, NULL, 0, 0, PM_NOREMOVE))
|
|
{
|
|
if (!GetMessage (&msg, NULL, 0, 0))
|
|
Com_Quit ();
|
|
sys_msg_time = msg.time;
|
|
TranslateMessage (&msg);
|
|
DispatchMessage (&msg);
|
|
}
|
|
|
|
// DarkOne's CPU usage fix
|
|
while (1)
|
|
{
|
|
newtime = Sys_Milliseconds();
|
|
time = newtime - oldtime;
|
|
if (time > 0) break;
|
|
Sleep(0); // may also use Sleep(1); to free more CPU, but it can lower your fps
|
|
}
|
|
/*do
|
|
{
|
|
newtime = Sys_Milliseconds ();
|
|
time = newtime - oldtime;
|
|
} while (time < 1);*/
|
|
// Con_Printf ("time:%5.2f - %5.2f = %5.2f\n", newtime, oldtime, time);
|
|
|
|
// _controlfp( ~( _EM_ZERODIVIDE /*| _EM_INVALID*/ ), _MCW_EM );
|
|
_controlfp( _PC_24, _MCW_PC );
|
|
Qcommon_Frame (time);
|
|
|
|
oldtime = newtime;
|
|
}
|
|
|
|
// never gets here
|
|
return TRUE;
|
|
}
|