2020-09-30 16:53:55 +02:00

261 lines
5.9 KiB
C++

#include "precomp.h"
#include <oprahcom.h>
#include <regentry.h>
#include "mperror.h"
#include <sehcall.h>
#define LEGACY_DIVISOR 8
extern "C" WORD _cdecl is_cyrix(void);
extern "C" DWORD _cdecl get_nxcpu_type(void);
#ifndef _M_IX86
DWORD WINAPI CallWithSEH(EXCEPTPROC pfn, void *pv, INEXCEPTION InException)
{
// we don't have a native version of SEH for the alpha,
// use __try and __except
pfn(pv);
return 0;
}
#endif
#ifdef _M_IX86
DWORD NMINTERNAL FindTSC (LPVOID pvRefData)
{
_asm
{
mov eax,1
_emit 00Fh ;; CPUID
_emit 0A2h
// The ref data is 2 DWORDS, the first is the flags,
// the second the family
mov ecx,pvRefData
mov [ecx],edx
mov [ecx][4],eax
}
return 1;
}
DWORD NMINTERNAL NoCPUID (LPEXCEPTION_RECORD per,PCONTEXT pctx)
{
return 0;
}
//
// GetProcessorSpeed(dwFamily)
//
// get the processor speed in MHz, only works on Pentium or better
// machines.
//
// Will put 3, or 4 in dwFamily for 386/486, but no speed.
// returns speed and family for 586+
//
// - thanks to toddla, modified by mikeg
//
int NMINTERNAL GetProcessorSpeed(int *pdwFamily)
{
SYSTEM_INFO si;
__int64 start, end, freq;
int flags,family;
int time;
int clocks;
DWORD oldclass;
HANDLE hprocess;
int pRef[2];
ZeroMemory(&si, sizeof(si));
GetSystemInfo(&si);
//Set the family. If wProcessorLevel is not specified, dig it out of dwProcessorType
//Because wProcessor level is not implemented on Win95
if (si.wProcessorLevel) {
*pdwFamily=si.wProcessorLevel;
}else {
//Ok, we're on Win95
switch (si.dwProcessorType) {
case PROCESSOR_INTEL_386:
*pdwFamily=3;
break;
case PROCESSOR_INTEL_486:
*pdwFamily=4;
break;
default:
*pdwFamily=0;
break;
}
}
//
// make sure this is a INTEL Pentium (or clone) or higher.
//
if (si.wProcessorArchitecture != PROCESSOR_ARCHITECTURE_INTEL)
return 0;
if (si.dwProcessorType < PROCESSOR_INTEL_PENTIUM)
return 0;
//
// see if this chip supports rdtsc before using it.
//
if (!CallWithSEH (FindTSC, pRef, NoCPUID)) {
flags=0;
} else {
// The ref data is 2 DWORDS, the first is the flags,
// the second the family. Pull them out and use them
flags=pRef[0];
family=pRef[1];
}
if (!(flags & 0x10))
return 0;
//If we don't have a family, set it now
//Family is bits 11:8 of eax from CPU, with eax=1
if (!(*pdwFamily)) {
*pdwFamily=(family& 0x0F00) >> 8;
}
hprocess = GetCurrentProcess();
oldclass = GetPriorityClass(hprocess);
SetPriorityClass(hprocess, REALTIME_PRIORITY_CLASS);
Sleep(10);
QueryPerformanceFrequency((LARGE_INTEGER*)&freq);
QueryPerformanceCounter((LARGE_INTEGER*)&start);
_asm
{
_emit 0Fh ;; RDTSC
_emit 31h
mov ecx,100000
x: dec ecx
jnz x
mov ebx,eax
_emit 0Fh ;; RDTSC
_emit 31h
sub eax,ebx
mov dword ptr clocks[0],eax
}
QueryPerformanceCounter((LARGE_INTEGER*)&end);
SetPriorityClass(hprocess, oldclass);
time = MulDiv((int)(end-start),1000000,(int)freq);
return (clocks + time/2) / time;
}
HRESULT NMINTERNAL GetNormalizedCPUSpeed (int *pdwNormalizedSpeed, int *dwFamily)
{
int dwProcessorSpeed;
dwProcessorSpeed=GetProcessorSpeed (dwFamily);
*pdwNormalizedSpeed=dwProcessorSpeed;
if (*dwFamily > 5) {
//Ok, TWO things.
// ONE DO NOT DO FP!
// Two for the same Mhz assume a 686 is 1.3 times as fast as a 586 and a 786 is 1.6 times, etc.
*pdwNormalizedSpeed=(ULONG) (((10+3*(*dwFamily-5))*dwProcessorSpeed)/10);
}
if (*dwFamily < 5) {
//bugbug until we have 386/486 timing code, assume
//486=50,386=37
if (*dwFamily > 3) {
//Cyrix, (5x86)? check before making default assignment
if (is_cyrix()) {
if (*pdwNormalizedSpeed==0) {
*dwFamily=5;
*pdwNormalizedSpeed=100;
return hrSuccess;
}
}
}
*pdwNormalizedSpeed= (*dwFamily*100)/LEGACY_DIVISOR;
if (get_nxcpu_type ()) {
//Double the perceived value on a NexGen
*pdwNormalizedSpeed *=2;
}
}
return hrSuccess;
}
#endif //_M_IX86
BOOL WINAPI IsFloatingPointEmulated(void)
{
long lRegValue;
SYSTEM_INFO si;
OSVERSIONINFO osi;
BOOL fEmulation, bNT;
// are we a Pentium
ZeroMemory(&si, sizeof(si));
GetSystemInfo(&si);
if (si.dwProcessorType != PROCESSOR_INTEL_PENTIUM)
{
return FALSE;
}
// Which OS: NT or 95 ?
ZeroMemory(&osi, sizeof(osi));
osi.dwOSVersionInfoSize = sizeof(osi);
GetVersionEx(&osi);
bNT = (osi.dwPlatformId == VER_PLATFORM_WIN32_NT);
// Windows NT
if (bNT)
{
RegEntry re(TEXT("System\\CurrentControlSet\\Control\\Session Manager"),
HKEY_LOCAL_MACHINE, FALSE);
// try to get a definitive answer from the registry
lRegValue = re.GetNumber(TEXT("ForceNpxEmulation"), -1);
// registry: 0: no
// 1: conditional (not definitive!)
// 2: yes
if (lRegValue == 2)
{
return TRUE;
}
// we could load "IsProcessorFeaturePresent from kernel32.dll,
// but the version that shipped with NT 4 has a bug in it that
// returns the exact opposite of what it should be. It was
// fixed in NT 5. Since this API isn't the same across platforms,
// we won't use it.
return FALSE;
}
// Windows 95 - to be added later
return FALSE;
}