halflife/utils/procinfo/procinfo.cpp
2013-08-30 13:34:05 -07:00

189 lines
5.1 KiB
C++

/***************************************************************************
; Copyright (C) 1998 Intel Corp.
;
; Subject to the terms and conditions set forth below, Intel hereby
; grants you a nonexclusive, nontransferable license, to use,
; reproduce and distribute the example code sequences contained
; herein, in object code format, solely as part of your computer
; program(s) and solely in order to allow your computer program(s) to
; implement the multimedia instruction extensions contained in such
; sequences solely with respect to the Intel instruction set
; architecture. No other license, express, implied, statutory, by
; estoppel or otherwise, to any other intellectual property rights is
; granted herein.
;
; ALL INFORMATION, SAMPLES AND OTHER MATERIALS PROVIDED HEREIN
; INCLUDING, WITHOUT LIMITATION, THE EXAMPLE CODE SEQUENCES ARE
; PROVIDED "AS IS" WITH NO WARRANTIES, EXPRESS, IMPLIED, STATUTORY OR
; OTHERWISE, AND INTEL SPECIFICALLY DISCLAIMS ANY IMPLIED WARRANTY OF
; MERCHANTABILITY, NONINFRINGEMENT OR FITNESS FOR ANY PARTICULAR
; PURPOSE.
;
; THE MATERIALS PROVIDED HEREIN ARE PROVIDED WITHOUT CHARGE.
; THEREFORE, IN NO EVENT WILL INTEL BE LIABLE FOR ANY DAMAGES OF ANY
; KIND, INCLUDING DIRECT OR INDIRECT DAMAGES, LOSS OF DATA, LOST
; PROFITS, COST OF COVER OR SPECIAL, INCIDENTAL, CONSEQUENTIAL,
; DAMAGES ARISING FROM THE USE OF THE MATERIALS PROVIDED HEREIN,
; INCLUDING WITHOUT LIMITATION THE EXAMPLE CODE SEQUENCES, HOWEVER
; CAUSED AND ON ANY THEORY OF LIABILITY. THIS LIMITATION WILL APPLY
; EVEN IF INTEL OR ANY AUTHORIZED AGENT OF INTEL HAS BEEN ADVISED OF
; THE POSSIBILITY OF SUCH DAMAGE.
;
;***************************************************************************/
#include "windows.h"
typedef unsigned __int64 QWORD;
#define RDTSC _asm _emit 0fh _asm _emit 031h
#define CPUID _asm _emit 0fh _asm _emit 0a2h
#define EMMS _asm _emit 0fh _asm _emit 077h
// If CPUID supported return 1, else return 0
unsigned int HasCpuid()
{
unsigned int supported = 0;
_asm {
push ebx ; save ebx
pushfd ; save flags
pushfd
pop eax
mov ebx,eax ; save old flags
xor eax,(1 SHL 21) ; flip id bit
push eax
popfd
pushfd
pop eax
popfd ; restore origional flags
cmp ebx,eax ; did ID bit change?
pop ebx
je Done ; if no change fail
mov supported,1 ; CPUID is supported
Done:
}
return supported;
}
// If RDTSC supported return 1, else return 0
unsigned int HasTsc()
{
unsigned int supported = 0;
if(! HasCpuid())
return 0;
_asm {
; check that max cpuid levels must be at least 1
mov eax,0
CPUID
cmp eax,1
jl Done
; check TSC bit in feature flags
mov eax,1
CPUID
bt edx,4 ; check TSD bit
jnc Done
; RDTSC is supported
mov supported,1
Done:
}
return supported;
}
// --------------------------------------------------------------------------
QWORD GetRDTSC(void)
{
QWORD clock;
_asm
{
// cpuid serializes for out of order processors
push ebx
push ecx
xor eax, eax
CPUID
RDTSC
mov dword ptr clock, eax
mov dword ptr clock+4, edx
xor eax, eax
CPUID
pop ecx
pop ebx
}
return clock;
}
// -----------------------------------------------------------------------
int PROC_GetSpeed(void)
{
QWORD StartClock, ElapClock;
DWORD StartTime, times = 0;
int RetVal;
if (!HasTsc())
return 166;
// try to get rid of the variability
StartClock = GetRDTSC();
StartTime = timeGetTime();
// this loop should take 1 sec +- 1 ms
while (timeGetTime() < StartTime + 250)
;
ElapClock = GetRDTSC() - StartClock + 500000;
// try to get rid of the variability
StartClock = GetRDTSC();
StartTime = timeGetTime();
// this loop should take 1 sec +- 1 ms
while (timeGetTime() < StartTime + 1000)
;
ElapClock = GetRDTSC() - StartClock + 500000;
RetVal = (DWORD)(ElapClock/1000000);
return RetVal;
}
#pragma optimize( "", off )
// --------------------------------------------------------------------------
int PROC_IsMMX(void)
{
int retval = 1;
DWORD RegEDX;
__try
{
_asm
{
mov eax, 1 // set up CPUID to return processor version and features
// 0 = vendor string, 1 = version info, 2 = cache info
CPUID // code bytes = 0fh, 0a2h
mov RegEDX, edx // features returned in edx
}
} __except(EXCEPTION_EXECUTE_HANDLER) { retval = FALSE; }
if (retval == FALSE)
return 0; // processor does not support CPUID
if (RegEDX & 0x800000) // bit 23 is set for MMX technology
{
__try {
EMMS // _asm emms
} // try executing the MMX instruction "emms"
__except(EXCEPTION_EXECUTE_HANDLER) { retval = FALSE; }
}
else
return 0; // processor supports CPUID but does not support MMX technology
// if retval == 0 here, it means the processor has MMX technology but
// floating-point emulation is on; so MMX technology is unavailable
return retval;
}
// --------------------------------------------------------------------------
#pragma optimize( "", on )