2006-11-08 11:47:44 +00:00
|
|
|
/*
|
|
|
|
* PROJECT: ReactOS Kernel
|
|
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
|
|
|
* FILE: ntoskrnl/ke/i386/cpu.c
|
|
|
|
* PURPOSE: Routines for CPU-level support
|
|
|
|
* PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
|
|
|
|
#include <ntoskrnl.h>
|
|
|
|
#define NDEBUG
|
|
|
|
#include <debug.h>
|
|
|
|
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
|
|
|
|
/* The TSS to use for Double Fault Traps (INT 0x9) */
|
|
|
|
UCHAR KiDoubleFaultTSS[KTSS_IO_MAPS];
|
|
|
|
|
|
|
|
/* The TSS to use for NMI Fault Traps (INT 0x2) */
|
|
|
|
UCHAR KiNMITSS[KTSS_IO_MAPS];
|
|
|
|
|
|
|
|
/* CPU Features and Flags */
|
|
|
|
ULONG KeI386CpuType;
|
|
|
|
ULONG KeI386CpuStep;
|
2011-09-25 09:12:26 +00:00
|
|
|
ULONG KiFastSystemCallDisable = 0;
|
2006-11-08 11:47:44 +00:00
|
|
|
ULONG KeI386NpxPresent = 0;
|
|
|
|
ULONG KiMXCsrMask = 0;
|
|
|
|
ULONG MxcsrFeatureMask = 0;
|
|
|
|
ULONG KeI386XMMIPresent = 0;
|
|
|
|
ULONG KeI386FxsrPresent = 0;
|
|
|
|
ULONG KeI386MachineType;
|
|
|
|
ULONG Ke386Pae = FALSE;
|
|
|
|
ULONG Ke386NoExecute = FALSE;
|
|
|
|
ULONG KeLargestCacheLine = 0x40;
|
|
|
|
ULONG KeDcacheFlushCount = 0;
|
|
|
|
ULONG KeIcacheFlushCount = 0;
|
|
|
|
ULONG KiDmaIoCoherency = 0;
|
2010-01-28 20:43:25 +00:00
|
|
|
ULONG KePrefetchNTAGranularity = 32;
|
2006-11-08 11:47:44 +00:00
|
|
|
BOOLEAN KiI386PentiumLockErrataPresent;
|
|
|
|
BOOLEAN KiSMTProcessorsPresent;
|
|
|
|
|
2010-01-27 03:05:10 +00:00
|
|
|
/* The distance between SYSEXIT and IRETD return modes */
|
|
|
|
UCHAR KiSystemCallExitAdjust;
|
|
|
|
|
|
|
|
/* The offset that was applied -- either 0 or the value above */
|
|
|
|
UCHAR KiSystemCallExitAdjusted;
|
|
|
|
|
|
|
|
/* Whether the adjustment was already done once */
|
|
|
|
BOOLEAN KiFastCallCopyDoneOnce;
|
|
|
|
|
2007-09-26 16:41:35 +00:00
|
|
|
/* Flush data */
|
|
|
|
volatile LONG KiTbFlushTimeStamp;
|
|
|
|
|
2006-11-08 11:47:44 +00:00
|
|
|
/* CPU Signatures */
|
|
|
|
static const CHAR CmpIntelID[] = "GenuineIntel";
|
|
|
|
static const CHAR CmpAmdID[] = "AuthenticAMD";
|
|
|
|
static const CHAR CmpCyrixID[] = "CyrixInstead";
|
|
|
|
static const CHAR CmpTransmetaID[] = "GenuineTMx86";
|
|
|
|
static const CHAR CmpCentaurID[] = "CentaurHauls";
|
|
|
|
static const CHAR CmpRiseID[] = "RiseRiseRise";
|
|
|
|
|
|
|
|
/* SUPPORT ROUTINES FOR MSVC COMPATIBILITY ***********************************/
|
|
|
|
|
2010-09-18 09:14:45 +00:00
|
|
|
/* NSC/Cyrix CPU configuration register index */
|
|
|
|
#define CX86_CCR1 0xc1
|
|
|
|
|
|
|
|
/* NSC/Cyrix CPU indexed register access macros */
|
2011-02-11 13:48:41 +00:00
|
|
|
static __inline
|
[HAL/NDK]
- Make Vector parameter in HalEnableSystemInterrupt, HalDisableSystemInterrupt and HalBeginSystemInterrupt an ULONG, not an UCHAR
[NDK]
- 64bit fixes for HANDLE_TABLE, KPROCESS, SECTION_IMAGE_INFORMATION, MMADDRESS_LIST, MMVAD_FLAGS, MMVAD, MMVAD_LONG, MMVAD_SHORT, MEMORY_DESCRIPTOR, MEMORY_ALLOCATION_DESCRIPTOR, LdrVerifyMappedImageMatchesChecksum
- KDPC_DATA::DpcQueueDepth is signed on amd64, unsigned on x86
[NTOSKRNL]
- Fix hundreds of MSVC and amd64 warnings
- add a pragma message to FstubFixupEfiPartition, since it looks broken
- Move portable Ke constants from <arch>/cpu.c to krnlinit.c
- Fixed a bug in amd64 KiGeneralProtectionFaultHandler
svn path=/trunk/; revision=53734
2011-09-18 13:11:45 +00:00
|
|
|
UCHAR
|
2011-02-11 13:48:41 +00:00
|
|
|
getCx86(UCHAR reg)
|
|
|
|
{
|
|
|
|
WRITE_PORT_UCHAR((PUCHAR)(ULONG_PTR)0x22, reg);
|
|
|
|
return READ_PORT_UCHAR((PUCHAR)(ULONG_PTR)0x23);
|
|
|
|
}
|
2010-09-18 09:14:45 +00:00
|
|
|
|
[HAL/NDK]
- Make Vector parameter in HalEnableSystemInterrupt, HalDisableSystemInterrupt and HalBeginSystemInterrupt an ULONG, not an UCHAR
[NDK]
- 64bit fixes for HANDLE_TABLE, KPROCESS, SECTION_IMAGE_INFORMATION, MMADDRESS_LIST, MMVAD_FLAGS, MMVAD, MMVAD_LONG, MMVAD_SHORT, MEMORY_DESCRIPTOR, MEMORY_ALLOCATION_DESCRIPTOR, LdrVerifyMappedImageMatchesChecksum
- KDPC_DATA::DpcQueueDepth is signed on amd64, unsigned on x86
[NTOSKRNL]
- Fix hundreds of MSVC and amd64 warnings
- add a pragma message to FstubFixupEfiPartition, since it looks broken
- Move portable Ke constants from <arch>/cpu.c to krnlinit.c
- Fixed a bug in amd64 KiGeneralProtectionFaultHandler
svn path=/trunk/; revision=53734
2011-09-18 13:11:45 +00:00
|
|
|
static __inline
|
|
|
|
void
|
|
|
|
setCx86(UCHAR reg, UCHAR data)
|
|
|
|
{
|
|
|
|
WRITE_PORT_UCHAR((PUCHAR)(ULONG_PTR)0x22, reg);
|
|
|
|
WRITE_PORT_UCHAR((PUCHAR)(ULONG_PTR)0x23, data);
|
|
|
|
}
|
|
|
|
|
2010-09-18 09:14:45 +00:00
|
|
|
|
2006-11-08 11:47:44 +00:00
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiSetProcessorType(VOID)
|
|
|
|
{
|
2009-09-21 15:20:18 +00:00
|
|
|
ULONG EFlags, NewEFlags;
|
2015-05-02 23:11:50 +00:00
|
|
|
CPU_INFO CpuInfo;
|
2006-11-08 11:47:44 +00:00
|
|
|
ULONG Stepping, Type;
|
|
|
|
|
|
|
|
/* Start by assuming no CPUID data */
|
|
|
|
KeGetCurrentPrcb()->CpuID = 0;
|
|
|
|
|
|
|
|
/* Save EFlags */
|
2009-09-21 15:20:18 +00:00
|
|
|
EFlags = __readeflags();
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* XOR out the ID bit and update EFlags */
|
2008-08-18 17:05:31 +00:00
|
|
|
NewEFlags = EFlags ^ EFLAGS_ID;
|
2009-09-21 15:20:18 +00:00
|
|
|
__writeeflags(NewEFlags);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Get them back and see if they were modified */
|
2009-09-21 15:20:18 +00:00
|
|
|
NewEFlags = __readeflags();
|
2006-11-08 11:47:44 +00:00
|
|
|
if (NewEFlags != EFlags)
|
|
|
|
{
|
|
|
|
/* The modification worked, so CPUID exists. Set the ID Bit again. */
|
2008-08-18 17:05:31 +00:00
|
|
|
EFlags |= EFLAGS_ID;
|
2009-09-21 15:20:18 +00:00
|
|
|
__writeeflags(EFlags);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Peform CPUID 0 to see if CPUID 1 is supported */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 0);
|
|
|
|
if (CpuInfo.Eax > 0)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
|
|
|
/* Do CPUID 1 now */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 1);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Get the Stepping and Type. The stepping contains both the
|
|
|
|
* Model and the Step, while the Type contains the returned Type.
|
|
|
|
* We ignore the family.
|
|
|
|
*
|
|
|
|
* For the stepping, we convert this: zzzzzzxy into this: x0y
|
|
|
|
*/
|
2015-05-02 23:11:50 +00:00
|
|
|
Stepping = CpuInfo.Eax & 0xF0;
|
2006-11-08 11:47:44 +00:00
|
|
|
Stepping <<= 4;
|
2015-05-02 23:11:50 +00:00
|
|
|
Stepping += (CpuInfo.Eax & 0xFF);
|
2006-11-08 11:47:44 +00:00
|
|
|
Stepping &= 0xF0F;
|
2015-05-02 23:11:50 +00:00
|
|
|
Type = CpuInfo.Eax & 0xF00;
|
2006-11-08 11:47:44 +00:00
|
|
|
Type >>= 8;
|
|
|
|
|
|
|
|
/* Save them in the PRCB */
|
|
|
|
KeGetCurrentPrcb()->CpuID = TRUE;
|
|
|
|
KeGetCurrentPrcb()->CpuType = (UCHAR)Type;
|
|
|
|
KeGetCurrentPrcb()->CpuStep = (USHORT)Stepping;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DPRINT1("CPUID Support lacking\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
DPRINT1("CPUID Support lacking\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Restore EFLAGS */
|
2009-09-21 15:20:18 +00:00
|
|
|
__writeeflags(EFlags);
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ULONG
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiGetCpuVendor(VOID)
|
|
|
|
{
|
|
|
|
PKPRCB Prcb = KeGetCurrentPrcb();
|
2015-05-02 23:11:50 +00:00
|
|
|
CPU_INFO CpuInfo;
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Assume no Vendor ID and fail if no CPUID Support. */
|
|
|
|
Prcb->VendorString[0] = 0;
|
|
|
|
if (!Prcb->CpuID) return 0;
|
|
|
|
|
2015-05-02 23:11:50 +00:00
|
|
|
/* Get the Vendor ID */
|
|
|
|
KiCpuId(&CpuInfo, 0);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2015-05-02 23:11:50 +00:00
|
|
|
/* Copy it to the PRCB and null-terminate it */
|
|
|
|
*(ULONG*)&Prcb->VendorString[0] = CpuInfo.Ebx;
|
|
|
|
*(ULONG*)&Prcb->VendorString[4] = CpuInfo.Edx;
|
|
|
|
*(ULONG*)&Prcb->VendorString[8] = CpuInfo.Ecx;
|
|
|
|
Prcb->VendorString[12] = 0;
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Now check the CPU Type */
|
|
|
|
if (!strcmp(Prcb->VendorString, CmpIntelID))
|
|
|
|
{
|
|
|
|
return CPU_INTEL;
|
|
|
|
}
|
|
|
|
else if (!strcmp(Prcb->VendorString, CmpAmdID))
|
|
|
|
{
|
|
|
|
return CPU_AMD;
|
|
|
|
}
|
|
|
|
else if (!strcmp(Prcb->VendorString, CmpCyrixID))
|
|
|
|
{
|
2008-10-02 21:57:36 +00:00
|
|
|
DPRINT1("Cyrix CPU support not fully tested!\n");
|
|
|
|
return CPU_CYRIX;
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
else if (!strcmp(Prcb->VendorString, CmpTransmetaID))
|
|
|
|
{
|
2008-10-02 21:57:36 +00:00
|
|
|
DPRINT1("Transmeta CPU support not fully tested!\n");
|
|
|
|
return CPU_TRANSMETA;
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
else if (!strcmp(Prcb->VendorString, CmpCentaurID))
|
|
|
|
{
|
2008-10-02 21:57:36 +00:00
|
|
|
DPRINT1("Centaur CPU support not fully tested!\n");
|
|
|
|
return CPU_CENTAUR;
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
else if (!strcmp(Prcb->VendorString, CmpRiseID))
|
|
|
|
{
|
2008-10-02 21:57:36 +00:00
|
|
|
DPRINT1("Rise CPU support not fully tested!\n");
|
|
|
|
return CPU_RISE;
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
2011-09-22 06:08:35 +00:00
|
|
|
/* Unknown CPU */
|
|
|
|
DPRINT1("%s CPU support not fully tested!\n", Prcb->VendorString);
|
|
|
|
return CPU_UNKNOWN;
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ULONG
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiGetFeatureBits(VOID)
|
|
|
|
{
|
|
|
|
PKPRCB Prcb = KeGetCurrentPrcb();
|
|
|
|
ULONG Vendor;
|
|
|
|
ULONG FeatureBits = KF_WORKING_PTE;
|
2015-05-02 23:11:50 +00:00
|
|
|
CPU_INFO CpuInfo, DummyCpuInfo;
|
[HAL/NDK]
- Make Vector parameter in HalEnableSystemInterrupt, HalDisableSystemInterrupt and HalBeginSystemInterrupt an ULONG, not an UCHAR
[NDK]
- 64bit fixes for HANDLE_TABLE, KPROCESS, SECTION_IMAGE_INFORMATION, MMADDRESS_LIST, MMVAD_FLAGS, MMVAD, MMVAD_LONG, MMVAD_SHORT, MEMORY_DESCRIPTOR, MEMORY_ALLOCATION_DESCRIPTOR, LdrVerifyMappedImageMatchesChecksum
- KDPC_DATA::DpcQueueDepth is signed on amd64, unsigned on x86
[NTOSKRNL]
- Fix hundreds of MSVC and amd64 warnings
- add a pragma message to FstubFixupEfiPartition, since it looks broken
- Move portable Ke constants from <arch>/cpu.c to krnlinit.c
- Fixed a bug in amd64 KiGeneralProtectionFaultHandler
svn path=/trunk/; revision=53734
2011-09-18 13:11:45 +00:00
|
|
|
UCHAR Ccr1;
|
2006-11-08 11:47:44 +00:00
|
|
|
BOOLEAN ExtendedCPUID = TRUE;
|
|
|
|
ULONG CpuFeatures = 0;
|
|
|
|
|
|
|
|
/* Get the Vendor ID */
|
|
|
|
Vendor = KiGetCpuVendor();
|
|
|
|
|
|
|
|
/* Make sure we got a valid vendor ID at least. */
|
|
|
|
if (!Vendor) return FeatureBits;
|
|
|
|
|
|
|
|
/* Get the CPUID Info. Features are in Reg[3]. */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 1);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Set the initial APIC ID */
|
2015-05-02 23:11:50 +00:00
|
|
|
Prcb->InitialApicId = (UCHAR)(CpuInfo.Ebx >> 24);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
switch (Vendor)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
2008-10-02 21:57:36 +00:00
|
|
|
/* Intel CPUs */
|
|
|
|
case CPU_INTEL:
|
2009-09-27 10:09:38 +00:00
|
|
|
|
2009-10-04 16:53:15 +00:00
|
|
|
/* Check if it's a P6 */
|
2008-10-02 21:57:36 +00:00
|
|
|
if (Prcb->CpuType == 6)
|
|
|
|
{
|
|
|
|
/* Perform the special sequence to get the MicroCode Signature */
|
2015-05-02 23:11:50 +00:00
|
|
|
__writemsr(0x8B, 0);
|
|
|
|
KiCpuId(&DummyCpuInfo, 1);
|
|
|
|
Prcb->UpdateSignature.QuadPart = __readmsr(0x8B);
|
2008-10-02 21:57:36 +00:00
|
|
|
}
|
|
|
|
else if (Prcb->CpuType == 5)
|
|
|
|
{
|
|
|
|
/* On P5, enable workaround for the LOCK errata. */
|
|
|
|
KiI386PentiumLockErrataPresent = TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check for broken P6 with bad SMP PTE implementation */
|
2015-05-02 23:11:50 +00:00
|
|
|
if (((CpuInfo.Eax & 0x0FF0) == 0x0610 && (CpuInfo.Eax & 0x000F) <= 0x9) ||
|
|
|
|
((CpuInfo.Eax & 0x0FF0) == 0x0630 && (CpuInfo.Eax & 0x000F) <= 0x4))
|
2008-10-02 21:57:36 +00:00
|
|
|
{
|
|
|
|
/* Remove support for correct PTE support. */
|
|
|
|
FeatureBits &= ~KF_WORKING_PTE;
|
|
|
|
}
|
|
|
|
|
2011-03-22 16:35:51 +00:00
|
|
|
/* Check if the CPU is too old to support SYSENTER */
|
|
|
|
if ((Prcb->CpuType < 6) ||
|
2015-03-06 11:49:28 +00:00
|
|
|
((Prcb->CpuType == 6) && (Prcb->CpuStep < 0x0303)))
|
2008-10-02 21:57:36 +00:00
|
|
|
{
|
|
|
|
/* Disable it */
|
2015-05-02 23:11:50 +00:00
|
|
|
CpuInfo.Edx &= ~0x800;
|
2008-10-02 21:57:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
/* AMD CPUs */
|
|
|
|
case CPU_AMD:
|
|
|
|
|
|
|
|
/* Check if this is a K5 or K6. (family 5) */
|
2015-05-02 23:11:50 +00:00
|
|
|
if ((CpuInfo.Eax & 0x0F00) == 0x0500)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
|
|
|
/* Get the Model Number */
|
2015-05-02 23:11:50 +00:00
|
|
|
switch (CpuInfo.Eax & 0x00F0)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
2008-10-02 21:57:36 +00:00
|
|
|
/* Model 1: K5 - 5k86 (initial models) */
|
2006-11-08 11:47:44 +00:00
|
|
|
case 0x0010:
|
|
|
|
|
|
|
|
/* Check if this is Step 0 or 1. They don't support PGE */
|
2015-05-02 23:11:50 +00:00
|
|
|
if ((CpuInfo.Eax & 0x000F) > 0x03) break;
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
/* Model 0: K5 - SSA5 */
|
2006-11-08 11:47:44 +00:00
|
|
|
case 0x0000:
|
|
|
|
|
|
|
|
/* Model 0 doesn't support PGE at all. */
|
2015-05-02 23:11:50 +00:00
|
|
|
CpuInfo.Edx &= ~0x2000;
|
2006-11-08 11:47:44 +00:00
|
|
|
break;
|
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
/* Model 8: K6-2 */
|
2006-11-08 11:47:44 +00:00
|
|
|
case 0x0080:
|
|
|
|
|
|
|
|
/* K6-2, Step 8 and over have support for MTRR. */
|
2015-05-02 23:11:50 +00:00
|
|
|
if ((CpuInfo.Eax & 0x000F) >= 0x8) FeatureBits |= KF_AMDK6MTRR;
|
2006-11-08 11:47:44 +00:00
|
|
|
break;
|
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
/* Model 9: K6-III
|
|
|
|
Model D: K6-2+, K6-III+ */
|
2006-11-08 11:47:44 +00:00
|
|
|
case 0x0090:
|
2008-10-02 21:57:36 +00:00
|
|
|
case 0x00D0:
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
FeatureBits |= KF_AMDK6MTRR;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
2015-05-02 23:11:50 +00:00
|
|
|
else if((CpuInfo.Eax & 0x0F00) < 0x0500)
|
2008-10-02 21:57:36 +00:00
|
|
|
{
|
|
|
|
/* Families below 5 don't support PGE, PSE or CMOV at all */
|
2015-05-02 23:11:50 +00:00
|
|
|
CpuInfo.Edx &= ~(0x08 | 0x2000 | 0x8000);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
/* They also don't support advanced CPUID functions. */
|
|
|
|
ExtendedCPUID = FALSE;
|
|
|
|
}
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
break;
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
/* Cyrix CPUs */
|
|
|
|
case CPU_CYRIX:
|
2009-09-27 10:09:38 +00:00
|
|
|
|
2010-09-18 09:14:45 +00:00
|
|
|
/* Workaround the "COMA" bug on 6x family of Cyrix CPUs */
|
|
|
|
if (Prcb->CpuType == 6 &&
|
|
|
|
Prcb->CpuStep <= 1)
|
|
|
|
{
|
|
|
|
/* Get CCR1 value */
|
|
|
|
Ccr1 = getCx86(CX86_CCR1);
|
|
|
|
|
|
|
|
/* Enable the NO_LOCK bit */
|
|
|
|
Ccr1 |= 0x10;
|
|
|
|
|
|
|
|
/* Set the new CCR1 value */
|
|
|
|
setCx86(CX86_CCR1, Ccr1);
|
|
|
|
}
|
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
break;
|
2008-10-01 11:54:29 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
/* Transmeta CPUs */
|
|
|
|
case CPU_TRANSMETA:
|
2009-09-27 10:09:38 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
/* Enable CMPXCHG8B if the family (>= 5), model and stepping (>= 4.2) support it */
|
2015-05-02 23:11:50 +00:00
|
|
|
if ((CpuInfo.Eax & 0x0FFF) >= 0x0542)
|
2008-10-01 18:19:42 +00:00
|
|
|
{
|
2015-05-02 23:11:50 +00:00
|
|
|
__writemsr(0x80860004, __readmsr(0x80860004) | 0x0100);
|
2008-10-01 18:19:42 +00:00
|
|
|
FeatureBits |= KF_CMPXCHG8B;
|
|
|
|
}
|
2008-10-01 11:54:29 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
/* Centaur, IDT, Rise and VIA CPUs */
|
|
|
|
case CPU_CENTAUR:
|
|
|
|
case CPU_RISE:
|
2009-09-27 10:09:38 +00:00
|
|
|
|
2008-10-02 21:57:36 +00:00
|
|
|
/* These CPUs don't report the presence of CMPXCHG8B through CPUID.
|
|
|
|
However, this feature exists and operates properly without any additional steps. */
|
|
|
|
FeatureBits |= KF_CMPXCHG8B;
|
|
|
|
|
|
|
|
break;
|
2008-10-01 18:19:42 +00:00
|
|
|
}
|
2008-10-01 11:54:29 +00:00
|
|
|
|
2011-09-22 06:08:35 +00:00
|
|
|
/* Set the current features */
|
2015-05-02 23:11:50 +00:00
|
|
|
CpuFeatures = CpuInfo.Edx;
|
2011-09-22 06:08:35 +00:00
|
|
|
|
2006-11-08 11:47:44 +00:00
|
|
|
/* Convert all CPUID Feature bits into our format */
|
|
|
|
if (CpuFeatures & 0x00000002) FeatureBits |= KF_V86_VIS | KF_CR4;
|
|
|
|
if (CpuFeatures & 0x00000008) FeatureBits |= KF_LARGE_PAGE | KF_CR4;
|
|
|
|
if (CpuFeatures & 0x00000010) FeatureBits |= KF_RDTSC;
|
|
|
|
if (CpuFeatures & 0x00000100) FeatureBits |= KF_CMPXCHG8B;
|
|
|
|
if (CpuFeatures & 0x00000800) FeatureBits |= KF_FAST_SYSCALL;
|
|
|
|
if (CpuFeatures & 0x00001000) FeatureBits |= KF_MTRR;
|
|
|
|
if (CpuFeatures & 0x00002000) FeatureBits |= KF_GLOBAL_PAGE | KF_CR4;
|
|
|
|
if (CpuFeatures & 0x00008000) FeatureBits |= KF_CMOV;
|
|
|
|
if (CpuFeatures & 0x00010000) FeatureBits |= KF_PAT;
|
2007-01-18 06:23:14 +00:00
|
|
|
if (CpuFeatures & 0x00200000) FeatureBits |= KF_DTS;
|
2006-11-08 11:47:44 +00:00
|
|
|
if (CpuFeatures & 0x00800000) FeatureBits |= KF_MMX;
|
|
|
|
if (CpuFeatures & 0x01000000) FeatureBits |= KF_FXSR;
|
|
|
|
if (CpuFeatures & 0x02000000) FeatureBits |= KF_XMMI;
|
|
|
|
if (CpuFeatures & 0x04000000) FeatureBits |= KF_XMMI64;
|
|
|
|
|
|
|
|
/* Check if the CPU has hyper-threading */
|
|
|
|
if (CpuFeatures & 0x10000000)
|
|
|
|
{
|
|
|
|
/* Set the number of logical CPUs */
|
2015-05-02 23:11:50 +00:00
|
|
|
Prcb->LogicalProcessorsPerPhysicalProcessor = (UCHAR)(CpuInfo.Ebx >> 16);
|
2006-11-08 11:47:44 +00:00
|
|
|
if (Prcb->LogicalProcessorsPerPhysicalProcessor > 1)
|
|
|
|
{
|
|
|
|
/* We're on dual-core */
|
|
|
|
KiSMTProcessorsPresent = TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* We only have a single CPU */
|
|
|
|
Prcb->LogicalProcessorsPerPhysicalProcessor = 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Check if CPUID 0x80000000 is supported */
|
|
|
|
if (ExtendedCPUID)
|
|
|
|
{
|
|
|
|
/* Do the call */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 0x80000000);
|
|
|
|
if ((CpuInfo.Eax & 0xffffff00) == 0x80000000)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
|
|
|
/* Check if CPUID 0x80000001 is supported */
|
2015-05-02 23:11:50 +00:00
|
|
|
if (CpuInfo.Eax >= 0x80000001)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
|
|
|
/* Check which extended features are available. */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 0x80000001);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2007-01-18 06:23:14 +00:00
|
|
|
/* Check if NX-bit is supported */
|
2015-05-02 23:11:50 +00:00
|
|
|
if (CpuInfo.Edx & 0x00100000) FeatureBits |= KF_NX_BIT;
|
2007-01-18 06:23:14 +00:00
|
|
|
|
2006-11-08 11:47:44 +00:00
|
|
|
/* Now handle each features for each CPU Vendor */
|
|
|
|
switch (Vendor)
|
|
|
|
{
|
|
|
|
case CPU_AMD:
|
2008-10-02 21:57:36 +00:00
|
|
|
case CPU_CENTAUR:
|
2015-05-02 23:11:50 +00:00
|
|
|
if (CpuInfo.Edx & 0x80000000) FeatureBits |= KF_3DNOW;
|
2006-11-08 11:47:44 +00:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-05-14 08:33:30 +00:00
|
|
|
|
|
|
|
#define print_supported(kf_value) ((FeatureBits & kf_value) ? #kf_value : "")
|
2011-03-07 20:39:33 +00:00
|
|
|
DPRINT1("Supported CPU features : %s %s %s %s %s %s %s %s %s %s %s %s %s %s %s %s %s %s %s %s %s\n",
|
|
|
|
print_supported(KF_V86_VIS),
|
|
|
|
print_supported(KF_RDTSC),
|
|
|
|
print_supported(KF_CR4),
|
|
|
|
print_supported(KF_CMOV),
|
|
|
|
print_supported(KF_GLOBAL_PAGE),
|
|
|
|
print_supported(KF_LARGE_PAGE),
|
|
|
|
print_supported(KF_MTRR),
|
|
|
|
print_supported(KF_CMPXCHG8B),
|
|
|
|
print_supported(KF_MMX),
|
|
|
|
print_supported(KF_WORKING_PTE),
|
|
|
|
print_supported(KF_PAT),
|
|
|
|
print_supported(KF_FXSR),
|
|
|
|
print_supported(KF_FAST_SYSCALL),
|
|
|
|
print_supported(KF_XMMI),
|
|
|
|
print_supported(KF_3DNOW),
|
|
|
|
print_supported(KF_AMDK6MTRR),
|
|
|
|
print_supported(KF_XMMI64),
|
|
|
|
print_supported(KF_DTS),
|
|
|
|
print_supported(KF_NX_BIT),
|
|
|
|
print_supported(KF_NX_DISABLED),
|
|
|
|
print_supported(KF_NX_ENABLED));
|
2011-03-07 13:33:10 +00:00
|
|
|
#undef print_supported
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Return the Feature Bits */
|
|
|
|
return FeatureBits;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiGetCacheInformation(VOID)
|
|
|
|
{
|
|
|
|
PKIPCR Pcr = (PKIPCR)KeGetPcr();
|
|
|
|
ULONG Vendor;
|
2015-05-02 23:11:50 +00:00
|
|
|
CPU_INFO CpuInfo;
|
2006-11-08 11:47:44 +00:00
|
|
|
ULONG CacheRequests = 0, i;
|
|
|
|
ULONG CurrentRegister;
|
[HAL/NDK]
- Make Vector parameter in HalEnableSystemInterrupt, HalDisableSystemInterrupt and HalBeginSystemInterrupt an ULONG, not an UCHAR
[NDK]
- 64bit fixes for HANDLE_TABLE, KPROCESS, SECTION_IMAGE_INFORMATION, MMADDRESS_LIST, MMVAD_FLAGS, MMVAD, MMVAD_LONG, MMVAD_SHORT, MEMORY_DESCRIPTOR, MEMORY_ALLOCATION_DESCRIPTOR, LdrVerifyMappedImageMatchesChecksum
- KDPC_DATA::DpcQueueDepth is signed on amd64, unsigned on x86
[NTOSKRNL]
- Fix hundreds of MSVC and amd64 warnings
- add a pragma message to FstubFixupEfiPartition, since it looks broken
- Move portable Ke constants from <arch>/cpu.c to krnlinit.c
- Fixed a bug in amd64 KiGeneralProtectionFaultHandler
svn path=/trunk/; revision=53734
2011-09-18 13:11:45 +00:00
|
|
|
UCHAR RegisterByte, Associativity = 0;
|
|
|
|
ULONG Size, CacheLine = 64, CurrentSize = 0;
|
2006-11-08 11:47:44 +00:00
|
|
|
BOOLEAN FirstPass = TRUE;
|
|
|
|
|
|
|
|
/* Set default L2 size */
|
|
|
|
Pcr->SecondLevelCacheSize = 0;
|
|
|
|
|
|
|
|
/* Get the Vendor ID and make sure we support CPUID */
|
|
|
|
Vendor = KiGetCpuVendor();
|
|
|
|
if (!Vendor) return;
|
|
|
|
|
|
|
|
/* Check the Vendor ID */
|
|
|
|
switch (Vendor)
|
|
|
|
{
|
|
|
|
/* Handle Intel case */
|
|
|
|
case CPU_INTEL:
|
|
|
|
|
|
|
|
/*Check if we support CPUID 2 */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 0);
|
|
|
|
if (CpuInfo.Eax >= 2)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
|
|
|
/* We need to loop for the number of times CPUID will tell us to */
|
|
|
|
do
|
|
|
|
{
|
|
|
|
/* Do the CPUID call */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 2);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Check if it was the first call */
|
|
|
|
if (FirstPass)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* The number of times to loop is the first byte. Read
|
|
|
|
* it and then destroy it so we don't get confused.
|
|
|
|
*/
|
2015-05-02 23:11:50 +00:00
|
|
|
CacheRequests = CpuInfo.Eax & 0xFF;
|
|
|
|
CpuInfo.Eax &= 0xFFFFFF00;
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Don't go over this again */
|
|
|
|
FirstPass = FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Loop all 4 registers */
|
|
|
|
for (i = 0; i < 4; i++)
|
|
|
|
{
|
|
|
|
/* Get the current register */
|
2015-05-02 23:11:50 +00:00
|
|
|
CurrentRegister = CpuInfo.AsUINT32[i];
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/*
|
|
|
|
* If the upper bit is set, then this register should
|
|
|
|
* be skipped.
|
|
|
|
*/
|
|
|
|
if (CurrentRegister & 0x80000000) continue;
|
|
|
|
|
|
|
|
/* Keep looping for every byte inside this register */
|
|
|
|
while (CurrentRegister)
|
|
|
|
{
|
|
|
|
/* Read a byte, skip a byte. */
|
|
|
|
RegisterByte = (UCHAR)(CurrentRegister & 0xFF);
|
|
|
|
CurrentRegister >>= 8;
|
|
|
|
if (!RegisterByte) continue;
|
|
|
|
|
2015-09-04 20:33:37 +00:00
|
|
|
Size = 0;
|
|
|
|
switch (RegisterByte)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
2015-09-04 20:33:37 +00:00
|
|
|
case 0x06:
|
|
|
|
case 0x08:
|
|
|
|
KePrefetchNTAGranularity = 32;
|
|
|
|
break;
|
|
|
|
case 0x09:
|
|
|
|
KePrefetchNTAGranularity = 64;
|
|
|
|
break;
|
|
|
|
case 0x0a:
|
|
|
|
case 0x0c:
|
|
|
|
KePrefetchNTAGranularity = 32;
|
|
|
|
break;
|
|
|
|
case 0x0d:
|
|
|
|
case 0x0e:
|
|
|
|
KePrefetchNTAGranularity = 64;
|
|
|
|
break;
|
|
|
|
case 0x1d:
|
|
|
|
Size = 128 * 1024;
|
|
|
|
Associativity = 2;
|
|
|
|
break;
|
|
|
|
case 0x21:
|
|
|
|
Size = 256 * 1024;
|
|
|
|
Associativity = 8;
|
|
|
|
break;
|
|
|
|
case 0x24:
|
|
|
|
Size = 1024 * 1024;
|
|
|
|
Associativity = 16;
|
|
|
|
break;
|
|
|
|
case 0x2c:
|
|
|
|
case 0x30:
|
|
|
|
KePrefetchNTAGranularity = 64;
|
|
|
|
break;
|
|
|
|
case 0x41:
|
|
|
|
case 0x42:
|
|
|
|
case 0x43:
|
|
|
|
case 0x44:
|
|
|
|
case 0x45:
|
|
|
|
Size = (1 << (RegisterByte - 0x41)) * 128 * 1024;
|
|
|
|
Associativity = 4;
|
|
|
|
break;
|
|
|
|
case 0x48:
|
|
|
|
Size = 3 * 1024 * 1024;
|
|
|
|
Associativity = 12;
|
|
|
|
break;
|
|
|
|
case 0x49:
|
|
|
|
Size = 4 * 1024 * 1024;
|
|
|
|
Associativity = 16;
|
|
|
|
break;
|
|
|
|
case 0x4e:
|
|
|
|
Size = 6 * 1024 * 1024;
|
|
|
|
Associativity = 24;
|
|
|
|
break;
|
|
|
|
case 0x60:
|
|
|
|
case 0x66:
|
|
|
|
case 0x67:
|
|
|
|
case 0x68:
|
|
|
|
KePrefetchNTAGranularity = 64;
|
|
|
|
break;
|
|
|
|
case 0x78:
|
|
|
|
Size = 1024 * 1024;
|
|
|
|
Associativity = 4;
|
|
|
|
break;
|
|
|
|
case 0x79:
|
|
|
|
case 0x7a:
|
|
|
|
case 0x7b:
|
|
|
|
case 0x7c:
|
|
|
|
case 0x7d:
|
|
|
|
Size = (1 << (RegisterByte - 0x79)) * 128 * 1024;
|
|
|
|
Associativity = 8;
|
|
|
|
break;
|
|
|
|
case 0x7f:
|
|
|
|
Size = 512 * 1024;
|
|
|
|
Associativity = 2;
|
|
|
|
break;
|
|
|
|
case 0x80:
|
|
|
|
Size = 512 * 1024;
|
|
|
|
Associativity = 8;
|
|
|
|
break;
|
|
|
|
case 0x82:
|
|
|
|
case 0x83:
|
|
|
|
case 0x84:
|
|
|
|
case 0x85:
|
|
|
|
Size = (1 << (RegisterByte - 0x82)) * 256 * 1024;
|
|
|
|
Associativity = 8;
|
|
|
|
break;
|
|
|
|
case 0x86:
|
|
|
|
Size = 512 * 1024;
|
|
|
|
Associativity = 4;
|
|
|
|
break;
|
|
|
|
case 0x87:
|
|
|
|
Size = 1024 * 1024;
|
|
|
|
Associativity = 8;
|
|
|
|
break;
|
|
|
|
case 0xf0:
|
|
|
|
KePrefetchNTAGranularity = 64;
|
|
|
|
break;
|
|
|
|
case 0xf1:
|
|
|
|
KePrefetchNTAGranularity = 128;
|
|
|
|
break;
|
2010-01-28 20:43:25 +00:00
|
|
|
}
|
2015-09-04 20:33:37 +00:00
|
|
|
if (Size && (Size / Associativity) > CurrentSize)
|
2010-01-28 20:43:25 +00:00
|
|
|
{
|
2015-09-04 20:33:37 +00:00
|
|
|
/* Set the L2 Cache Size and Associativity */
|
|
|
|
CurrentSize = Size / Associativity;
|
|
|
|
Pcr->SecondLevelCacheSize = Size;
|
|
|
|
Pcr->SecondLevelCacheAssociativity = Associativity;
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} while (--CacheRequests);
|
|
|
|
}
|
2007-05-26 16:37:11 +00:00
|
|
|
break;
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2007-05-26 16:37:11 +00:00
|
|
|
case CPU_AMD:
|
2006-11-08 11:47:44 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Check if we support CPUID 0x80000005 */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 0x80000000);
|
2015-06-27 10:58:32 +00:00
|
|
|
if (CpuInfo.Eax >= 0x80000005)
|
2007-05-26 16:37:11 +00:00
|
|
|
{
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Get L1 size first */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 0x80000005);
|
|
|
|
KePrefetchNTAGranularity = CpuInfo.Ecx & 0xFF;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Check if we support CPUID 0x80000006 */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 0x80000000);
|
|
|
|
if (CpuInfo.Eax >= 0x80000006)
|
2011-05-14 08:33:30 +00:00
|
|
|
{
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Get 2nd level cache and tlb size */
|
2015-05-02 23:11:50 +00:00
|
|
|
KiCpuId(&CpuInfo, 0x80000006);
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Cache line size */
|
2015-05-02 23:11:50 +00:00
|
|
|
CacheLine = CpuInfo.Ecx & 0xFF;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Hardcode associativity */
|
2015-05-02 23:11:50 +00:00
|
|
|
RegisterByte = (CpuInfo.Ecx >> 12) & 0xFF;
|
2010-01-28 20:43:25 +00:00
|
|
|
switch (RegisterByte)
|
|
|
|
{
|
|
|
|
case 2:
|
|
|
|
Associativity = 2;
|
|
|
|
break;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
case 4:
|
|
|
|
Associativity = 4;
|
|
|
|
break;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
case 6:
|
|
|
|
Associativity = 8;
|
|
|
|
break;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
case 8:
|
|
|
|
case 15:
|
|
|
|
Associativity = 16;
|
|
|
|
break;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
default:
|
|
|
|
Associativity = 1;
|
|
|
|
break;
|
|
|
|
}
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Compute size */
|
2015-05-02 23:11:50 +00:00
|
|
|
Size = (CpuInfo.Ecx >> 16) << 10;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Hack for Model 6, Steping 300 */
|
|
|
|
if ((KeGetCurrentPrcb()->CpuType == 6) &&
|
|
|
|
(KeGetCurrentPrcb()->CpuStep == 0x300))
|
|
|
|
{
|
|
|
|
/* Stick 64K in there */
|
|
|
|
Size = 64 * 1024;
|
|
|
|
}
|
2007-05-26 16:37:11 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Set the L2 Cache Size and associativity */
|
|
|
|
Pcr->SecondLevelCacheSize = Size;
|
|
|
|
Pcr->SecondLevelCacheAssociativity = Associativity;
|
|
|
|
}
|
2007-05-26 16:37:11 +00:00
|
|
|
}
|
|
|
|
break;
|
2009-09-27 10:09:38 +00:00
|
|
|
|
|
|
|
case CPU_CYRIX:
|
|
|
|
case CPU_TRANSMETA:
|
|
|
|
case CPU_CENTAUR:
|
|
|
|
case CPU_RISE:
|
|
|
|
|
|
|
|
/* FIXME */
|
|
|
|
break;
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-28 20:43:25 +00:00
|
|
|
/* Set the cache line */
|
|
|
|
if (CacheLine > KeLargestCacheLine) KeLargestCacheLine = CacheLine;
|
2013-08-31 16:02:13 +00:00
|
|
|
DPRINT1("Prefetch Cache: %lu bytes\tL2 Cache: %lu bytes\tL2 Cache Line: %lu bytes\tL2 Cache Associativity: %lu\n",
|
2010-01-28 20:43:25 +00:00
|
|
|
KePrefetchNTAGranularity,
|
|
|
|
Pcr->SecondLevelCacheSize,
|
|
|
|
KeLargestCacheLine,
|
|
|
|
Pcr->SecondLevelCacheAssociativity);
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiSetCR0Bits(VOID)
|
|
|
|
{
|
|
|
|
ULONG Cr0;
|
|
|
|
|
|
|
|
/* Save current CR0 */
|
|
|
|
Cr0 = __readcr0();
|
|
|
|
|
|
|
|
/* If this is a 486, enable Write-Protection */
|
|
|
|
if (KeGetCurrentPrcb()->CpuType > 3) Cr0 |= CR0_WP;
|
|
|
|
|
|
|
|
/* Set new Cr0 */
|
|
|
|
__writecr0(Cr0);
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiInitializeTSS2(IN PKTSS Tss,
|
|
|
|
IN PKGDTENTRY TssEntry OPTIONAL)
|
|
|
|
{
|
|
|
|
PUCHAR p;
|
|
|
|
|
|
|
|
/* Make sure the GDT Entry is valid */
|
|
|
|
if (TssEntry)
|
|
|
|
{
|
|
|
|
/* Set the Limit */
|
|
|
|
TssEntry->LimitLow = sizeof(KTSS) - 1;
|
2007-01-21 18:20:06 +00:00
|
|
|
TssEntry->HighWord.Bits.LimitHi = 0;
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* Now clear the I/O Map */
|
2009-12-31 23:51:26 +00:00
|
|
|
ASSERT(IOPM_COUNT == 1);
|
|
|
|
RtlFillMemory(Tss->IoMaps[0].IoMap, IOPM_FULL_SIZE, 0xFF);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Initialize Interrupt Direction Maps */
|
|
|
|
p = (PUCHAR)(Tss->IoMaps[0].DirectionMap);
|
2009-12-31 23:51:26 +00:00
|
|
|
RtlZeroMemory(p, IOPM_DIRECTION_MAP_SIZE);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Add DPMI support for interrupts */
|
|
|
|
p[0] = 4;
|
|
|
|
p[3] = 0x18;
|
|
|
|
p[4] = 0x18;
|
|
|
|
|
|
|
|
/* Initialize the default Interrupt Direction Map */
|
|
|
|
p = Tss->IntDirectionMap;
|
2009-12-31 23:51:26 +00:00
|
|
|
RtlZeroMemory(Tss->IntDirectionMap, IOPM_DIRECTION_MAP_SIZE);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Add DPMI support */
|
|
|
|
p[0] = 4;
|
|
|
|
p[3] = 0x18;
|
|
|
|
p[4] = 0x18;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KiInitializeTSS(IN PKTSS Tss)
|
|
|
|
{
|
|
|
|
/* Set an invalid map base */
|
|
|
|
Tss->IoMapBase = KiComputeIopmOffset(IO_ACCESS_MAP_NONE);
|
|
|
|
|
|
|
|
/* Disable traps during Task Switches */
|
|
|
|
Tss->Flags = 0;
|
|
|
|
|
|
|
|
/* Set LDT and Ring 0 SS */
|
|
|
|
Tss->LDT = 0;
|
|
|
|
Tss->Ss0 = KGDT_R0_DATA;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
FASTCALL
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
Ki386InitializeTss(IN PKTSS Tss,
|
|
|
|
IN PKIDTENTRY Idt,
|
|
|
|
IN PKGDTENTRY Gdt)
|
|
|
|
{
|
|
|
|
PKGDTENTRY TssEntry, TaskGateEntry;
|
|
|
|
|
|
|
|
/* Initialize the boot TSS. */
|
|
|
|
TssEntry = &Gdt[KGDT_TSS / sizeof(KGDTENTRY)];
|
|
|
|
TssEntry->HighWord.Bits.Type = I386_TSS;
|
|
|
|
TssEntry->HighWord.Bits.Pres = 1;
|
|
|
|
TssEntry->HighWord.Bits.Dpl = 0;
|
|
|
|
KiInitializeTSS2(Tss, TssEntry);
|
|
|
|
KiInitializeTSS(Tss);
|
|
|
|
|
|
|
|
/* Load the task register */
|
|
|
|
Ke386SetTr(KGDT_TSS);
|
|
|
|
|
|
|
|
/* Setup the Task Gate for Double Fault Traps */
|
|
|
|
TaskGateEntry = (PKGDTENTRY)&Idt[8];
|
|
|
|
TaskGateEntry->HighWord.Bits.Type = I386_TASK_GATE;
|
|
|
|
TaskGateEntry->HighWord.Bits.Pres = 1;
|
|
|
|
TaskGateEntry->HighWord.Bits.Dpl = 0;
|
|
|
|
((PKIDTENTRY)TaskGateEntry)->Selector = KGDT_DF_TSS;
|
|
|
|
|
|
|
|
/* Initialize the TSS used for handling double faults. */
|
|
|
|
Tss = (PKTSS)KiDoubleFaultTSS;
|
|
|
|
KiInitializeTSS(Tss);
|
|
|
|
Tss->CR3 = __readcr3();
|
2010-01-19 18:27:24 +00:00
|
|
|
Tss->Esp0 = KiDoubleFaultStack;
|
|
|
|
Tss->Esp = KiDoubleFaultStack;
|
Patch that fixes VMWare boot (and should fix QEMu/KVM boot on the testbot):
[NTOS]: A trap can get us into a state where DS/ES are invalid, making any pointer dereference (on DS/ES segmented memory, not SS, the stack) crash (and probably double-fault). Therefore, we have to be careful to switch to a good DS/ES before touching the TrapFrame pointer, which we don't have in ESP like the ASM code, but in a DS/ES-segmented register. For V8086 traps we can switch to the good DS/ES immediately, but for other kinds of traps, we actually need to save the current (bad) segments first. So we save them on the stack now, then switch to the good ones, then store the stack values into the trap frame. This is what happens on a non-optimized (-O0) build. On an optimized build, the segments will end up in registers instead, which is fine too (they'll be direct values). The order of instructions is guaranteed since the segment macros are volatile.
[NTOS]: The GPF and Invalid Opcode handlers are performance critical when talking about V8086 traps, because they control the main flow of execution during that mode (GPFs will be issued for any privileged instruction we need to emulate, and invalid opcode might be generated for BOPs). Because of this, we employ a fast entry/exit macro into V8086 mode since we can make certain assumptions. We detect, and use, such scenarios when the V8086 flag is enabled in EFLAGS. However, because we can land in a GPF handler with an invalid DS/ES, as some V8086 code could trample this during BIOS calls for example, we must make sure that we are on a valid DS/ES before dereferencing any pointer. We fixup DS/ES either in KiEnterTrap (for normal entry/exit) or, for V86, in KiEnterV86Trap. Notice the problem: we need to detect which of these to use early on but we can't touch the EFLAGS in the frame because DS/ES could be invalid. Thankfully SS is always guaranteed valid, so stack dereferences are game! We therefore read the EFLAGS here, in assembly, where we can touch ESP as we please. We save this in EDX, which will be used as the second argument for the FASTCALL C trap entry. When we make the fast V86 check, we use the parameter instead of the trap frame, leading us to using the correct trap entry function, which fixes up DS/ES and lets us go on our merry way...
[NTOS]: Make appropriate changes to GENERATE_TRAP_HANDLERS macro.
[NTOS]: Switch to using well-known NT trap handler names (hex-based, double-zeroed) instead of decimal-based trap handler names which are confusing.
[NTOS]: Clean up some debug spew.
svn path=/trunk/; revision=45052
2010-01-12 05:50:45 +00:00
|
|
|
Tss->Eip = PtrToUlong(KiTrap08);
|
2006-11-08 11:47:44 +00:00
|
|
|
Tss->Cs = KGDT_R0_CODE;
|
|
|
|
Tss->Fs = KGDT_R0_PCR;
|
|
|
|
Tss->Ss = Ke386GetSs();
|
|
|
|
Tss->Es = KGDT_R3_DATA | RPL_MASK;
|
|
|
|
Tss->Ds = KGDT_R3_DATA | RPL_MASK;
|
|
|
|
|
|
|
|
/* Setup the Double Trap TSS entry in the GDT */
|
|
|
|
TssEntry = &Gdt[KGDT_DF_TSS / sizeof(KGDTENTRY)];
|
|
|
|
TssEntry->HighWord.Bits.Type = I386_TSS;
|
|
|
|
TssEntry->HighWord.Bits.Pres = 1;
|
|
|
|
TssEntry->HighWord.Bits.Dpl = 0;
|
|
|
|
TssEntry->BaseLow = (USHORT)((ULONG_PTR)Tss & 0xFFFF);
|
|
|
|
TssEntry->HighWord.Bytes.BaseMid = (UCHAR)((ULONG_PTR)Tss >> 16);
|
|
|
|
TssEntry->HighWord.Bytes.BaseHi = (UCHAR)((ULONG_PTR)Tss >> 24);
|
|
|
|
TssEntry->LimitLow = KTSS_IO_MAPS;
|
|
|
|
|
|
|
|
/* Now setup the NMI Task Gate */
|
|
|
|
TaskGateEntry = (PKGDTENTRY)&Idt[2];
|
|
|
|
TaskGateEntry->HighWord.Bits.Type = I386_TASK_GATE;
|
|
|
|
TaskGateEntry->HighWord.Bits.Pres = 1;
|
|
|
|
TaskGateEntry->HighWord.Bits.Dpl = 0;
|
|
|
|
((PKIDTENTRY)TaskGateEntry)->Selector = KGDT_NMI_TSS;
|
|
|
|
|
|
|
|
/* Initialize the actual TSS */
|
|
|
|
Tss = (PKTSS)KiNMITSS;
|
|
|
|
KiInitializeTSS(Tss);
|
|
|
|
Tss->CR3 = __readcr3();
|
2010-01-19 18:27:24 +00:00
|
|
|
Tss->Esp0 = KiDoubleFaultStack;
|
|
|
|
Tss->Esp = KiDoubleFaultStack;
|
Patch that fixes VMWare boot (and should fix QEMu/KVM boot on the testbot):
[NTOS]: A trap can get us into a state where DS/ES are invalid, making any pointer dereference (on DS/ES segmented memory, not SS, the stack) crash (and probably double-fault). Therefore, we have to be careful to switch to a good DS/ES before touching the TrapFrame pointer, which we don't have in ESP like the ASM code, but in a DS/ES-segmented register. For V8086 traps we can switch to the good DS/ES immediately, but for other kinds of traps, we actually need to save the current (bad) segments first. So we save them on the stack now, then switch to the good ones, then store the stack values into the trap frame. This is what happens on a non-optimized (-O0) build. On an optimized build, the segments will end up in registers instead, which is fine too (they'll be direct values). The order of instructions is guaranteed since the segment macros are volatile.
[NTOS]: The GPF and Invalid Opcode handlers are performance critical when talking about V8086 traps, because they control the main flow of execution during that mode (GPFs will be issued for any privileged instruction we need to emulate, and invalid opcode might be generated for BOPs). Because of this, we employ a fast entry/exit macro into V8086 mode since we can make certain assumptions. We detect, and use, such scenarios when the V8086 flag is enabled in EFLAGS. However, because we can land in a GPF handler with an invalid DS/ES, as some V8086 code could trample this during BIOS calls for example, we must make sure that we are on a valid DS/ES before dereferencing any pointer. We fixup DS/ES either in KiEnterTrap (for normal entry/exit) or, for V86, in KiEnterV86Trap. Notice the problem: we need to detect which of these to use early on but we can't touch the EFLAGS in the frame because DS/ES could be invalid. Thankfully SS is always guaranteed valid, so stack dereferences are game! We therefore read the EFLAGS here, in assembly, where we can touch ESP as we please. We save this in EDX, which will be used as the second argument for the FASTCALL C trap entry. When we make the fast V86 check, we use the parameter instead of the trap frame, leading us to using the correct trap entry function, which fixes up DS/ES and lets us go on our merry way...
[NTOS]: Make appropriate changes to GENERATE_TRAP_HANDLERS macro.
[NTOS]: Switch to using well-known NT trap handler names (hex-based, double-zeroed) instead of decimal-based trap handler names which are confusing.
[NTOS]: Clean up some debug spew.
svn path=/trunk/; revision=45052
2010-01-12 05:50:45 +00:00
|
|
|
Tss->Eip = PtrToUlong(KiTrap02);
|
2006-11-08 11:47:44 +00:00
|
|
|
Tss->Cs = KGDT_R0_CODE;
|
|
|
|
Tss->Fs = KGDT_R0_PCR;
|
|
|
|
Tss->Ss = Ke386GetSs();
|
|
|
|
Tss->Es = KGDT_R3_DATA | RPL_MASK;
|
|
|
|
Tss->Ds = KGDT_R3_DATA | RPL_MASK;
|
|
|
|
|
|
|
|
/* And its associated TSS Entry */
|
|
|
|
TssEntry = &Gdt[KGDT_NMI_TSS / sizeof(KGDTENTRY)];
|
|
|
|
TssEntry->HighWord.Bits.Type = I386_TSS;
|
|
|
|
TssEntry->HighWord.Bits.Pres = 1;
|
|
|
|
TssEntry->HighWord.Bits.Dpl = 0;
|
|
|
|
TssEntry->BaseLow = (USHORT)((ULONG_PTR)Tss & 0xFFFF);
|
|
|
|
TssEntry->HighWord.Bytes.BaseMid = (UCHAR)((ULONG_PTR)Tss >> 16);
|
|
|
|
TssEntry->HighWord.Bytes.BaseHi = (UCHAR)((ULONG_PTR)Tss >> 24);
|
|
|
|
TssEntry->LimitLow = KTSS_IO_MAPS;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KeFlushCurrentTb(VOID)
|
|
|
|
{
|
[NTOS]
Add super-complicated handling of global pages to KeFlushCurrentTb (pretty much the same code which has been in HalpFlushTLB for the past ~6 years). This should be all that is required to make this feature work (everything else being in place already), and *seems* to work fine but is disabled under a switch until tested thoroughly.
Global pages, an important optimization that allows for not flushing the whole x86 TLB every time CR3 is changed (typically on context switch to a new process, or during process attach/detach), relies on us doing extra work whenever we do alter a global page. This is likely where any bugs will have to be flushed out!
Fixup Ki386EnableGlobalPage while we are at it -- disable/restore interrupts properly, and verify PGE-bit isn't set (nothing should have touched it before this routine, which is responsible for initializing it, so we shouldn't have to disable it). Fix, but disable, the CPU-sync spin as well as there should be no particular reason to do this for PGE-enabling during initialization (no other processor will be messing with PTEs at this stage, as compared to a call to KeFlushEntireTb).
Everyone, repeat after me: Global pages are awesome!
svn path=/trunk/; revision=69528
2015-10-14 19:33:35 +00:00
|
|
|
|
|
|
|
#if !defined(_GLOBAL_PAGES_ARE_AWESOME_)
|
|
|
|
|
2006-11-08 11:47:44 +00:00
|
|
|
/* Flush the TLB by resetting CR3 */
|
2007-03-02 16:26:45 +00:00
|
|
|
__writecr3(__readcr3());
|
[NTOS]
Add super-complicated handling of global pages to KeFlushCurrentTb (pretty much the same code which has been in HalpFlushTLB for the past ~6 years). This should be all that is required to make this feature work (everything else being in place already), and *seems* to work fine but is disabled under a switch until tested thoroughly.
Global pages, an important optimization that allows for not flushing the whole x86 TLB every time CR3 is changed (typically on context switch to a new process, or during process attach/detach), relies on us doing extra work whenever we do alter a global page. This is likely where any bugs will have to be flushed out!
Fixup Ki386EnableGlobalPage while we are at it -- disable/restore interrupts properly, and verify PGE-bit isn't set (nothing should have touched it before this routine, which is responsible for initializing it, so we shouldn't have to disable it). Fix, but disable, the CPU-sync spin as well as there should be no particular reason to do this for PGE-enabling during initialization (no other processor will be messing with PTEs at this stage, as compared to a call to KeFlushEntireTb).
Everyone, repeat after me: Global pages are awesome!
svn path=/trunk/; revision=69528
2015-10-14 19:33:35 +00:00
|
|
|
|
|
|
|
#else
|
|
|
|
|
|
|
|
/* Check if global pages are enabled */
|
|
|
|
if (KeFeatureBits & KF_GLOBAL_PAGE)
|
|
|
|
{
|
|
|
|
ULONG Cr4;
|
|
|
|
|
2015-10-14 20:42:18 +00:00
|
|
|
/* Disable PGE (Note: may not have been enabled yet) */
|
|
|
|
Cr4 = __readcr4();
|
|
|
|
__writecr4(Cr4 & ~CR4_PGE);
|
[NTOS]
Add super-complicated handling of global pages to KeFlushCurrentTb (pretty much the same code which has been in HalpFlushTLB for the past ~6 years). This should be all that is required to make this feature work (everything else being in place already), and *seems* to work fine but is disabled under a switch until tested thoroughly.
Global pages, an important optimization that allows for not flushing the whole x86 TLB every time CR3 is changed (typically on context switch to a new process, or during process attach/detach), relies on us doing extra work whenever we do alter a global page. This is likely where any bugs will have to be flushed out!
Fixup Ki386EnableGlobalPage while we are at it -- disable/restore interrupts properly, and verify PGE-bit isn't set (nothing should have touched it before this routine, which is responsible for initializing it, so we shouldn't have to disable it). Fix, but disable, the CPU-sync spin as well as there should be no particular reason to do this for PGE-enabling during initialization (no other processor will be messing with PTEs at this stage, as compared to a call to KeFlushEntireTb).
Everyone, repeat after me: Global pages are awesome!
svn path=/trunk/; revision=69528
2015-10-14 19:33:35 +00:00
|
|
|
|
|
|
|
/* Flush everything */
|
|
|
|
__writecr3(__readcr3());
|
|
|
|
|
|
|
|
/* Re-enable PGE */
|
2015-10-14 20:42:18 +00:00
|
|
|
__writecr4(Cr4);
|
[NTOS]
Add super-complicated handling of global pages to KeFlushCurrentTb (pretty much the same code which has been in HalpFlushTLB for the past ~6 years). This should be all that is required to make this feature work (everything else being in place already), and *seems* to work fine but is disabled under a switch until tested thoroughly.
Global pages, an important optimization that allows for not flushing the whole x86 TLB every time CR3 is changed (typically on context switch to a new process, or during process attach/detach), relies on us doing extra work whenever we do alter a global page. This is likely where any bugs will have to be flushed out!
Fixup Ki386EnableGlobalPage while we are at it -- disable/restore interrupts properly, and verify PGE-bit isn't set (nothing should have touched it before this routine, which is responsible for initializing it, so we shouldn't have to disable it). Fix, but disable, the CPU-sync spin as well as there should be no particular reason to do this for PGE-enabling during initialization (no other processor will be messing with PTEs at this stage, as compared to a call to KeFlushEntireTb).
Everyone, repeat after me: Global pages are awesome!
svn path=/trunk/; revision=69528
2015-10-14 19:33:35 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* No global pages, resetting CR3 is enough */
|
|
|
|
__writecr3(__readcr3());
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
- Add EXCEPTION_RECORD64 and LIST_ENTRY64, KeTryToAcquireSpinLockAtDpcLevel, BREAKPOINT_COMMAND_STRING, Ke386SetCr2, Ke386SetDr3, Ke386SetDr6.
- Remove non-kernel routines from kdfuncs.h and remove deprecated routines from ke.h.
- Implement KiRestoreProcessorControlState, KeFreezeExecution, KeThawExecution, ExAcquireTimeRefreshLock, ExReleaseTimeRefreshLock.
- Rename ModuleLoadList to PsLoadedModuleList. Add PsNtosImageBase and set value in it.
- Add skeleton wdbgexts.h with what's needed until now, this is a PSDK header.
- Add kddll.h for KDCOM/1394/USB2.DLL prototypes.
- Add windbgkd.h with KD protocol definitions. Used to be an NT5 DDK header, but was removed, so this goes into include\reactos.
svn path=/branches/alex-kd-branch/; revision=25833
2007-02-18 07:21:03 +00:00
|
|
|
KiRestoreProcessorControlState(PKPROCESSOR_STATE ProcessorState)
|
|
|
|
{
|
2009-10-04 16:53:15 +00:00
|
|
|
PKGDTENTRY TssEntry;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Restore the CR registers
|
|
|
|
//
|
- Add EXCEPTION_RECORD64 and LIST_ENTRY64, KeTryToAcquireSpinLockAtDpcLevel, BREAKPOINT_COMMAND_STRING, Ke386SetCr2, Ke386SetDr3, Ke386SetDr6.
- Remove non-kernel routines from kdfuncs.h and remove deprecated routines from ke.h.
- Implement KiRestoreProcessorControlState, KeFreezeExecution, KeThawExecution, ExAcquireTimeRefreshLock, ExReleaseTimeRefreshLock.
- Rename ModuleLoadList to PsLoadedModuleList. Add PsNtosImageBase and set value in it.
- Add skeleton wdbgexts.h with what's needed until now, this is a PSDK header.
- Add kddll.h for KDCOM/1394/USB2.DLL prototypes.
- Add windbgkd.h with KD protocol definitions. Used to be an NT5 DDK header, but was removed, so this goes into include\reactos.
svn path=/branches/alex-kd-branch/; revision=25833
2007-02-18 07:21:03 +00:00
|
|
|
__writecr0(ProcessorState->SpecialRegisters.Cr0);
|
|
|
|
Ke386SetCr2(ProcessorState->SpecialRegisters.Cr2);
|
|
|
|
__writecr3(ProcessorState->SpecialRegisters.Cr3);
|
2007-02-18 22:32:32 +00:00
|
|
|
if (KeFeatureBits & KF_CR4) __writecr4(ProcessorState->SpecialRegisters.Cr4);
|
- Add EXCEPTION_RECORD64 and LIST_ENTRY64, KeTryToAcquireSpinLockAtDpcLevel, BREAKPOINT_COMMAND_STRING, Ke386SetCr2, Ke386SetDr3, Ke386SetDr6.
- Remove non-kernel routines from kdfuncs.h and remove deprecated routines from ke.h.
- Implement KiRestoreProcessorControlState, KeFreezeExecution, KeThawExecution, ExAcquireTimeRefreshLock, ExReleaseTimeRefreshLock.
- Rename ModuleLoadList to PsLoadedModuleList. Add PsNtosImageBase and set value in it.
- Add skeleton wdbgexts.h with what's needed until now, this is a PSDK header.
- Add kddll.h for KDCOM/1394/USB2.DLL prototypes.
- Add windbgkd.h with KD protocol definitions. Used to be an NT5 DDK header, but was removed, so this goes into include\reactos.
svn path=/branches/alex-kd-branch/; revision=25833
2007-02-18 07:21:03 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// Restore the DR registers
|
|
|
|
//
|
2009-09-21 15:20:18 +00:00
|
|
|
__writedr(0, ProcessorState->SpecialRegisters.KernelDr0);
|
|
|
|
__writedr(1, ProcessorState->SpecialRegisters.KernelDr1);
|
|
|
|
__writedr(2, ProcessorState->SpecialRegisters.KernelDr2);
|
|
|
|
__writedr(3, ProcessorState->SpecialRegisters.KernelDr3);
|
|
|
|
__writedr(6, ProcessorState->SpecialRegisters.KernelDr6);
|
|
|
|
__writedr(7, ProcessorState->SpecialRegisters.KernelDr7);
|
- Add EXCEPTION_RECORD64 and LIST_ENTRY64, KeTryToAcquireSpinLockAtDpcLevel, BREAKPOINT_COMMAND_STRING, Ke386SetCr2, Ke386SetDr3, Ke386SetDr6.
- Remove non-kernel routines from kdfuncs.h and remove deprecated routines from ke.h.
- Implement KiRestoreProcessorControlState, KeFreezeExecution, KeThawExecution, ExAcquireTimeRefreshLock, ExReleaseTimeRefreshLock.
- Rename ModuleLoadList to PsLoadedModuleList. Add PsNtosImageBase and set value in it.
- Add skeleton wdbgexts.h with what's needed until now, this is a PSDK header.
- Add kddll.h for KDCOM/1394/USB2.DLL prototypes.
- Add windbgkd.h with KD protocol definitions. Used to be an NT5 DDK header, but was removed, so this goes into include\reactos.
svn path=/branches/alex-kd-branch/; revision=25833
2007-02-18 07:21:03 +00:00
|
|
|
|
|
|
|
//
|
2009-10-04 16:53:15 +00:00
|
|
|
// Restore GDT and IDT
|
- Add EXCEPTION_RECORD64 and LIST_ENTRY64, KeTryToAcquireSpinLockAtDpcLevel, BREAKPOINT_COMMAND_STRING, Ke386SetCr2, Ke386SetDr3, Ke386SetDr6.
- Remove non-kernel routines from kdfuncs.h and remove deprecated routines from ke.h.
- Implement KiRestoreProcessorControlState, KeFreezeExecution, KeThawExecution, ExAcquireTimeRefreshLock, ExReleaseTimeRefreshLock.
- Rename ModuleLoadList to PsLoadedModuleList. Add PsNtosImageBase and set value in it.
- Add skeleton wdbgexts.h with what's needed until now, this is a PSDK header.
- Add kddll.h for KDCOM/1394/USB2.DLL prototypes.
- Add windbgkd.h with KD protocol definitions. Used to be an NT5 DDK header, but was removed, so this goes into include\reactos.
svn path=/branches/alex-kd-branch/; revision=25833
2007-02-18 07:21:03 +00:00
|
|
|
//
|
2009-09-21 15:20:18 +00:00
|
|
|
Ke386SetGlobalDescriptorTable(&ProcessorState->SpecialRegisters.Gdtr.Limit);
|
|
|
|
__lidt(&ProcessorState->SpecialRegisters.Idtr.Limit);
|
2009-10-04 16:53:15 +00:00
|
|
|
|
|
|
|
//
|
|
|
|
// Clear the busy flag so we don't crash if we reload the same selector
|
|
|
|
//
|
|
|
|
TssEntry = (PKGDTENTRY)(ProcessorState->SpecialRegisters.Gdtr.Base +
|
|
|
|
ProcessorState->SpecialRegisters.Tr);
|
|
|
|
TssEntry->HighWord.Bytes.Flags1 &= ~0x2;
|
|
|
|
|
|
|
|
//
|
|
|
|
// Restore TSS and LDT
|
|
|
|
//
|
- Add EXCEPTION_RECORD64 and LIST_ENTRY64, KeTryToAcquireSpinLockAtDpcLevel, BREAKPOINT_COMMAND_STRING, Ke386SetCr2, Ke386SetDr3, Ke386SetDr6.
- Remove non-kernel routines from kdfuncs.h and remove deprecated routines from ke.h.
- Implement KiRestoreProcessorControlState, KeFreezeExecution, KeThawExecution, ExAcquireTimeRefreshLock, ExReleaseTimeRefreshLock.
- Rename ModuleLoadList to PsLoadedModuleList. Add PsNtosImageBase and set value in it.
- Add skeleton wdbgexts.h with what's needed until now, this is a PSDK header.
- Add kddll.h for KDCOM/1394/USB2.DLL prototypes.
- Add windbgkd.h with KD protocol definitions. Used to be an NT5 DDK header, but was removed, so this goes into include\reactos.
svn path=/branches/alex-kd-branch/; revision=25833
2007-02-18 07:21:03 +00:00
|
|
|
Ke386SetTr(ProcessorState->SpecialRegisters.Tr);
|
|
|
|
Ke386SetLocalDescriptorTable(ProcessorState->SpecialRegisters.Ldtr);
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KiSaveProcessorControlState(OUT PKPROCESSOR_STATE ProcessorState)
|
2006-11-08 11:47:44 +00:00
|
|
|
{
|
|
|
|
/* Save the CR registers */
|
|
|
|
ProcessorState->SpecialRegisters.Cr0 = __readcr0();
|
|
|
|
ProcessorState->SpecialRegisters.Cr2 = __readcr2();
|
|
|
|
ProcessorState->SpecialRegisters.Cr3 = __readcr3();
|
2007-02-18 22:32:32 +00:00
|
|
|
ProcessorState->SpecialRegisters.Cr4 = (KeFeatureBits & KF_CR4) ?
|
|
|
|
__readcr4() : 0;
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Save the DR registers */
|
2009-09-21 15:20:18 +00:00
|
|
|
ProcessorState->SpecialRegisters.KernelDr0 = __readdr(0);
|
|
|
|
ProcessorState->SpecialRegisters.KernelDr1 = __readdr(1);
|
|
|
|
ProcessorState->SpecialRegisters.KernelDr2 = __readdr(2);
|
|
|
|
ProcessorState->SpecialRegisters.KernelDr3 = __readdr(3);
|
|
|
|
ProcessorState->SpecialRegisters.KernelDr6 = __readdr(6);
|
|
|
|
ProcessorState->SpecialRegisters.KernelDr7 = __readdr(7);
|
|
|
|
__writedr(7, 0);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Save GDT, IDT, LDT and TSS */
|
2009-09-21 15:20:18 +00:00
|
|
|
Ke386GetGlobalDescriptorTable(&ProcessorState->SpecialRegisters.Gdtr.Limit);
|
|
|
|
__sidt(&ProcessorState->SpecialRegisters.Idtr.Limit);
|
2009-09-27 10:09:38 +00:00
|
|
|
ProcessorState->SpecialRegisters.Tr = Ke386GetTr();
|
|
|
|
ProcessorState->SpecialRegisters.Ldtr = Ke386GetLocalDescriptorTable();
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiInitializeMachineType(VOID)
|
|
|
|
{
|
|
|
|
/* Set the Machine Type we got from NTLDR */
|
|
|
|
KeI386MachineType = KeLoaderBlock->u.I386.MachineType & 0x000FF;
|
|
|
|
}
|
|
|
|
|
|
|
|
ULONG_PTR
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiLoadFastSyscallMachineSpecificRegisters(IN ULONG_PTR Context)
|
|
|
|
{
|
|
|
|
/* Set CS and ESP */
|
2015-05-02 23:11:50 +00:00
|
|
|
__writemsr(0x174, KGDT_R0_CODE);
|
|
|
|
__writemsr(0x175, (ULONG_PTR)KeGetCurrentPrcb()->DpcStack);
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Set LSTAR */
|
2015-05-02 23:11:50 +00:00
|
|
|
__writemsr(0x176, (ULONG_PTR)KiFastCallEntry);
|
2006-11-08 11:47:44 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2010-01-27 03:05:10 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2010-01-27 03:05:10 +00:00
|
|
|
KiRestoreFastSyscallReturnState(VOID)
|
|
|
|
{
|
2006-11-08 11:47:44 +00:00
|
|
|
/* Check if the CPU Supports fast system call */
|
|
|
|
if (KeFeatureBits & KF_FAST_SYSCALL)
|
2010-01-27 03:05:10 +00:00
|
|
|
{
|
|
|
|
/* Check if it has been disabled */
|
2011-09-25 09:12:26 +00:00
|
|
|
if (KiFastSystemCallDisable)
|
|
|
|
{
|
|
|
|
/* Disable fast system call */
|
|
|
|
KeFeatureBits &= ~KF_FAST_SYSCALL;
|
|
|
|
KiFastCallExitHandler = KiSystemCallTrapReturn;
|
|
|
|
DPRINT1("Support for SYSENTER disabled.\n");
|
|
|
|
}
|
|
|
|
else
|
2010-01-27 03:05:10 +00:00
|
|
|
{
|
2010-03-17 13:12:46 +00:00
|
|
|
/* Do an IPI to enable it */
|
|
|
|
KeIpiGenericCall(KiLoadFastSyscallMachineSpecificRegisters, 0);
|
|
|
|
|
|
|
|
/* It's enabled, so use the proper exit stub */
|
|
|
|
KiFastCallExitHandler = KiSystemCallSysExitReturn;
|
2011-09-25 09:12:26 +00:00
|
|
|
DPRINT("Support for SYSENTER detected.\n");
|
2010-01-27 03:05:10 +00:00
|
|
|
}
|
|
|
|
}
|
2010-03-17 16:17:16 +00:00
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Use the IRET handler */
|
|
|
|
KiFastCallExitHandler = KiSystemCallTrapReturn;
|
2010-03-17 21:26:04 +00:00
|
|
|
DPRINT1("No support for SYSENTER detected.\n");
|
2010-03-17 16:17:16 +00:00
|
|
|
}
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
ULONG_PTR
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
Ki386EnableDE(IN ULONG_PTR Context)
|
|
|
|
{
|
|
|
|
/* Enable DE */
|
|
|
|
__writecr4(__readcr4() | CR4_DE);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ULONG_PTR
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
Ki386EnableFxsr(IN ULONG_PTR Context)
|
|
|
|
{
|
|
|
|
/* Enable FXSR */
|
|
|
|
__writecr4(__readcr4() | CR4_FXSR);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
ULONG_PTR
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
Ki386EnableXMMIExceptions(IN ULONG_PTR Context)
|
|
|
|
{
|
2007-01-18 06:23:14 +00:00
|
|
|
PKIDTENTRY IdtEntry;
|
|
|
|
|
Patch that fixes VMWare boot (and should fix QEMu/KVM boot on the testbot):
[NTOS]: A trap can get us into a state where DS/ES are invalid, making any pointer dereference (on DS/ES segmented memory, not SS, the stack) crash (and probably double-fault). Therefore, we have to be careful to switch to a good DS/ES before touching the TrapFrame pointer, which we don't have in ESP like the ASM code, but in a DS/ES-segmented register. For V8086 traps we can switch to the good DS/ES immediately, but for other kinds of traps, we actually need to save the current (bad) segments first. So we save them on the stack now, then switch to the good ones, then store the stack values into the trap frame. This is what happens on a non-optimized (-O0) build. On an optimized build, the segments will end up in registers instead, which is fine too (they'll be direct values). The order of instructions is guaranteed since the segment macros are volatile.
[NTOS]: The GPF and Invalid Opcode handlers are performance critical when talking about V8086 traps, because they control the main flow of execution during that mode (GPFs will be issued for any privileged instruction we need to emulate, and invalid opcode might be generated for BOPs). Because of this, we employ a fast entry/exit macro into V8086 mode since we can make certain assumptions. We detect, and use, such scenarios when the V8086 flag is enabled in EFLAGS. However, because we can land in a GPF handler with an invalid DS/ES, as some V8086 code could trample this during BIOS calls for example, we must make sure that we are on a valid DS/ES before dereferencing any pointer. We fixup DS/ES either in KiEnterTrap (for normal entry/exit) or, for V86, in KiEnterV86Trap. Notice the problem: we need to detect which of these to use early on but we can't touch the EFLAGS in the frame because DS/ES could be invalid. Thankfully SS is always guaranteed valid, so stack dereferences are game! We therefore read the EFLAGS here, in assembly, where we can touch ESP as we please. We save this in EDX, which will be used as the second argument for the FASTCALL C trap entry. When we make the fast V86 check, we use the parameter instead of the trap frame, leading us to using the correct trap entry function, which fixes up DS/ES and lets us go on our merry way...
[NTOS]: Make appropriate changes to GENERATE_TRAP_HANDLERS macro.
[NTOS]: Switch to using well-known NT trap handler names (hex-based, double-zeroed) instead of decimal-based trap handler names which are confusing.
[NTOS]: Clean up some debug spew.
svn path=/trunk/; revision=45052
2010-01-12 05:50:45 +00:00
|
|
|
/* Get the IDT Entry for Interrupt 0x13 */
|
|
|
|
IdtEntry = &((PKIPCR)KeGetPcr())->IDT[0x13];
|
2007-01-18 06:23:14 +00:00
|
|
|
|
|
|
|
/* Set it up */
|
|
|
|
IdtEntry->Selector = KGDT_R0_CODE;
|
Patch that fixes VMWare boot (and should fix QEMu/KVM boot on the testbot):
[NTOS]: A trap can get us into a state where DS/ES are invalid, making any pointer dereference (on DS/ES segmented memory, not SS, the stack) crash (and probably double-fault). Therefore, we have to be careful to switch to a good DS/ES before touching the TrapFrame pointer, which we don't have in ESP like the ASM code, but in a DS/ES-segmented register. For V8086 traps we can switch to the good DS/ES immediately, but for other kinds of traps, we actually need to save the current (bad) segments first. So we save them on the stack now, then switch to the good ones, then store the stack values into the trap frame. This is what happens on a non-optimized (-O0) build. On an optimized build, the segments will end up in registers instead, which is fine too (they'll be direct values). The order of instructions is guaranteed since the segment macros are volatile.
[NTOS]: The GPF and Invalid Opcode handlers are performance critical when talking about V8086 traps, because they control the main flow of execution during that mode (GPFs will be issued for any privileged instruction we need to emulate, and invalid opcode might be generated for BOPs). Because of this, we employ a fast entry/exit macro into V8086 mode since we can make certain assumptions. We detect, and use, such scenarios when the V8086 flag is enabled in EFLAGS. However, because we can land in a GPF handler with an invalid DS/ES, as some V8086 code could trample this during BIOS calls for example, we must make sure that we are on a valid DS/ES before dereferencing any pointer. We fixup DS/ES either in KiEnterTrap (for normal entry/exit) or, for V86, in KiEnterV86Trap. Notice the problem: we need to detect which of these to use early on but we can't touch the EFLAGS in the frame because DS/ES could be invalid. Thankfully SS is always guaranteed valid, so stack dereferences are game! We therefore read the EFLAGS here, in assembly, where we can touch ESP as we please. We save this in EDX, which will be used as the second argument for the FASTCALL C trap entry. When we make the fast V86 check, we use the parameter instead of the trap frame, leading us to using the correct trap entry function, which fixes up DS/ES and lets us go on our merry way...
[NTOS]: Make appropriate changes to GENERATE_TRAP_HANDLERS macro.
[NTOS]: Switch to using well-known NT trap handler names (hex-based, double-zeroed) instead of decimal-based trap handler names which are confusing.
[NTOS]: Clean up some debug spew.
svn path=/trunk/; revision=45052
2010-01-12 05:50:45 +00:00
|
|
|
IdtEntry->Offset = ((ULONG_PTR)KiTrap13 & 0xFFFF);
|
|
|
|
IdtEntry->ExtendedOffset = ((ULONG_PTR)KiTrap13 >> 16) & 0xFFFF;
|
2007-01-18 06:23:14 +00:00
|
|
|
((PKIDT_ACCESS)&IdtEntry->Access)->Dpl = 0;
|
|
|
|
((PKIDT_ACCESS)&IdtEntry->Access)->Present = 1;
|
|
|
|
((PKIDT_ACCESS)&IdtEntry->Access)->SegmentType = I386_INTERRUPT_GATE;
|
|
|
|
|
|
|
|
/* Enable XMMI exceptions */
|
|
|
|
__writecr4(__readcr4() | CR4_XMMEXCPT);
|
2006-11-08 11:47:44 +00:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2006-11-08 11:47:44 +00:00
|
|
|
KiI386PentiumLockErrataFixup(VOID)
|
|
|
|
{
|
[NTOSKRNL]
Coverity code defects fixes :
- Cache: CID 701441
- Config: CIDs 716570, 716669, 716760
- Dbgk: Kdbg: CIDs 716571, 515128/9, 500432
- Ex: CIDs 500156/7, 515122, 716200/67, 701301, 514669
- Fsrtl: Fstub: CIDs 701341/2, 701288, 716770, 701302, and CIDs 716576/7/8 + 514636 + 716805 thanks to Thomas Faber
- Io: CIDs 514576, 514643, 514672/3, 716203, 716269, 716581, 716591, 716713
- Ke: CIDs 515125, 716592
- Ps: CIDs 716603/4, 701422
- Ob: Po: CIDs 514671/680, 701419/420/421, 716763, 716601/2
All the details are given in the different bug reports.
CORE-6677 CORE-6679 CORE-6680 CORE-6683 CORE-6686 CORE-6692 CORE-6693 CORE-6694 CORE-6695 CORE-6696 #comment Committed in rev.57400 #resolve #close
svn path=/trunk/; revision=57400
2012-09-27 17:16:31 +00:00
|
|
|
KDESCRIPTOR IdtDescriptor = {0, 0, 0};
|
2007-01-18 06:23:14 +00:00
|
|
|
PKIDTENTRY NewIdt, NewIdt2;
|
|
|
|
|
|
|
|
/* Allocate memory for a new IDT */
|
|
|
|
NewIdt = ExAllocatePool(NonPagedPool, 2 * PAGE_SIZE);
|
|
|
|
|
|
|
|
/* Put everything after the first 7 entries on a new page */
|
|
|
|
NewIdt2 = (PVOID)((ULONG_PTR)NewIdt + PAGE_SIZE - (7 * sizeof(KIDTENTRY)));
|
|
|
|
|
|
|
|
/* Disable interrupts */
|
|
|
|
_disable();
|
|
|
|
|
|
|
|
/* Get the current IDT and copy it */
|
2009-09-21 15:20:18 +00:00
|
|
|
__sidt(&IdtDescriptor.Limit);
|
2007-01-18 06:23:14 +00:00
|
|
|
RtlCopyMemory(NewIdt2,
|
|
|
|
(PVOID)IdtDescriptor.Base,
|
|
|
|
IdtDescriptor.Limit + 1);
|
|
|
|
IdtDescriptor.Base = (ULONG)NewIdt2;
|
|
|
|
|
|
|
|
/* Set the new IDT */
|
2009-09-21 15:20:18 +00:00
|
|
|
__lidt(&IdtDescriptor.Limit);
|
2007-01-18 06:23:14 +00:00
|
|
|
((PKIPCR)KeGetPcr())->IDT = NewIdt2;
|
|
|
|
|
|
|
|
/* Restore interrupts */
|
|
|
|
_enable();
|
|
|
|
|
|
|
|
/* Set the first 7 entries as read-only to produce a fault */
|
|
|
|
MmSetPageProtect(NULL, NewIdt, PAGE_READONLY);
|
2006-11-08 11:47:44 +00:00
|
|
|
}
|
|
|
|
|
2007-09-26 16:41:35 +00:00
|
|
|
BOOLEAN
|
|
|
|
NTAPI
|
|
|
|
KeInvalidateAllCaches(VOID)
|
|
|
|
{
|
|
|
|
/* Only supported on Pentium Pro and higher */
|
|
|
|
if (KeI386CpuType < 6) return FALSE;
|
|
|
|
|
|
|
|
/* Invalidate all caches */
|
2007-11-25 22:33:50 +00:00
|
|
|
__wbinvd();
|
2007-09-26 16:41:35 +00:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
|
|
|
FASTCALL
|
|
|
|
KeZeroPages(IN PVOID Address,
|
|
|
|
IN ULONG Size)
|
|
|
|
{
|
|
|
|
/* Not using XMMI in this routine */
|
|
|
|
RtlZeroMemory(Address, Size);
|
|
|
|
}
|
|
|
|
|
2010-01-02 01:34:27 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KiSaveProcessorState(IN PKTRAP_FRAME TrapFrame,
|
|
|
|
IN PKEXCEPTION_FRAME ExceptionFrame)
|
|
|
|
{
|
|
|
|
PKPRCB Prcb = KeGetCurrentPrcb();
|
|
|
|
|
|
|
|
//
|
|
|
|
// Save full context
|
|
|
|
//
|
|
|
|
Prcb->ProcessorState.ContextFrame.ContextFlags = CONTEXT_FULL |
|
|
|
|
CONTEXT_DEBUG_REGISTERS;
|
|
|
|
KeTrapFrameToContext(TrapFrame, NULL, &Prcb->ProcessorState.ContextFrame);
|
|
|
|
|
|
|
|
//
|
|
|
|
// Save control registers
|
|
|
|
//
|
|
|
|
KiSaveProcessorControlState(&Prcb->ProcessorState);
|
|
|
|
}
|
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
BOOLEAN
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2010-01-20 04:05:08 +00:00
|
|
|
KiIsNpxPresent(VOID)
|
|
|
|
{
|
|
|
|
ULONG Cr0;
|
|
|
|
USHORT Magic;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Set magic */
|
|
|
|
Magic = 0xFFFF;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Read CR0 and mask out FPU flags */
|
|
|
|
Cr0 = __readcr0() & ~(CR0_MP | CR0_TS | CR0_EM | CR0_ET);
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Store on FPU stack */
|
2010-07-26 21:32:38 +00:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
__asm fninit;
|
|
|
|
__asm fnstsw Magic;
|
|
|
|
#else
|
2010-01-20 04:05:08 +00:00
|
|
|
asm volatile ("fninit;" "fnstsw %0" : "+m"(Magic));
|
2010-07-26 21:32:38 +00:00
|
|
|
#endif
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Magic should now be cleared */
|
|
|
|
if (Magic & 0xFF)
|
|
|
|
{
|
|
|
|
/* You don't have an FPU -- enable emulation for now */
|
|
|
|
__writecr0(Cr0 | CR0_EM | CR0_TS);
|
|
|
|
return FALSE;
|
|
|
|
}
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* You have an FPU, enable it */
|
|
|
|
Cr0 |= CR0_ET;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Enable INT 16 on 486 and higher */
|
|
|
|
if (KeGetCurrentPrcb()->CpuType >= 3) Cr0 |= CR0_NE;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Set FPU state */
|
|
|
|
__writecr0(Cr0 | CR0_EM | CR0_TS);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
BOOLEAN
|
|
|
|
NTAPI
|
2010-11-02 16:29:06 +00:00
|
|
|
INIT_FUNCTION
|
2010-01-20 04:05:08 +00:00
|
|
|
KiIsNpxErrataPresent(VOID)
|
|
|
|
{
|
2011-09-30 21:03:29 +00:00
|
|
|
static double Value1 = 4195835.0, Value2 = 3145727.0;
|
|
|
|
INT ErrataPresent;
|
2010-01-20 04:05:08 +00:00
|
|
|
ULONG Cr0;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Disable interrupts */
|
|
|
|
_disable();
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Read CR0 and remove FPU flags */
|
|
|
|
Cr0 = __readcr0();
|
|
|
|
__writecr0(Cr0 & ~(CR0_MP | CR0_TS | CR0_EM));
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Initialize FPU state */
|
2010-07-26 21:32:38 +00:00
|
|
|
Ke386FnInit();
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Multiply the magic values and divide, we should get the result back */
|
2011-09-30 21:03:29 +00:00
|
|
|
#ifdef __GNUC__
|
|
|
|
__asm__ __volatile__
|
|
|
|
(
|
|
|
|
"fldl %1\n\t"
|
|
|
|
"fdivl %2\n\t"
|
|
|
|
"fmull %2\n\t"
|
|
|
|
"fldl %1\n\t"
|
|
|
|
"fsubp\n\t"
|
|
|
|
"fistpl %0\n\t"
|
|
|
|
: "=m" (ErrataPresent)
|
|
|
|
: "m" (Value1),
|
|
|
|
"m" (Value2)
|
|
|
|
);
|
|
|
|
#else
|
|
|
|
__asm
|
|
|
|
{
|
|
|
|
fld Value1
|
|
|
|
fdiv Value2
|
|
|
|
fmul Value2
|
|
|
|
fld Value1
|
2011-10-02 19:08:11 +00:00
|
|
|
fsubp st(1), st(0)
|
2011-09-30 21:03:29 +00:00
|
|
|
fistp ErrataPresent
|
|
|
|
};
|
|
|
|
#endif
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Restore CR0 */
|
|
|
|
__writecr0(Cr0);
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Enable interrupts */
|
|
|
|
_enable();
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Return if there's an errata */
|
2011-09-30 21:03:29 +00:00
|
|
|
return ErrataPresent != 0;
|
2010-01-20 04:05:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
VOID
|
2010-07-26 21:32:38 +00:00
|
|
|
NTAPI
|
2010-01-20 04:05:08 +00:00
|
|
|
KiFlushNPXState(IN PFLOATING_SAVE_AREA SaveArea)
|
|
|
|
{
|
|
|
|
ULONG EFlags, Cr0;
|
|
|
|
PKTHREAD Thread, NpxThread;
|
|
|
|
PFX_SAVE_AREA FxSaveArea;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Save volatiles and disable interrupts */
|
|
|
|
EFlags = __readeflags();
|
|
|
|
_disable();
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Save the PCR and get the current thread */
|
|
|
|
Thread = KeGetCurrentThread();
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Check if we're already loaded */
|
|
|
|
if (Thread->NpxState != NPX_STATE_LOADED)
|
|
|
|
{
|
|
|
|
/* If there's nothing to load, quit */
|
2012-04-06 20:13:33 +00:00
|
|
|
if (!SaveArea)
|
|
|
|
{
|
|
|
|
/* Restore interrupt state and return */
|
|
|
|
__writeeflags(EFlags);
|
|
|
|
return;
|
|
|
|
}
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Need FXSR support for this */
|
|
|
|
ASSERT(KeI386FxsrPresent == TRUE);
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Check for sane CR0 */
|
|
|
|
Cr0 = __readcr0();
|
|
|
|
if (Cr0 & (CR0_MP | CR0_TS | CR0_EM))
|
|
|
|
{
|
|
|
|
/* Mask out FPU flags */
|
|
|
|
__writecr0(Cr0 & ~(CR0_MP | CR0_TS | CR0_EM));
|
|
|
|
}
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Get the NPX thread and check its FPU state */
|
|
|
|
NpxThread = KeGetCurrentPrcb()->NpxThread;
|
|
|
|
if ((NpxThread) && (NpxThread->NpxState == NPX_STATE_LOADED))
|
|
|
|
{
|
|
|
|
/* Get the FX frame and store the state there */
|
|
|
|
FxSaveArea = KiGetThreadNpxArea(NpxThread);
|
|
|
|
Ke386FxSave(FxSaveArea);
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* NPX thread has lost its state */
|
|
|
|
NpxThread->NpxState = NPX_STATE_NOT_LOADED;
|
|
|
|
}
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Now load NPX state from the NPX area */
|
|
|
|
FxSaveArea = KiGetThreadNpxArea(Thread);
|
2011-05-14 08:33:30 +00:00
|
|
|
Ke386FxStore(FxSaveArea);
|
2010-01-20 04:05:08 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Check for sane CR0 */
|
|
|
|
Cr0 = __readcr0();
|
|
|
|
if (Cr0 & (CR0_MP | CR0_TS | CR0_EM))
|
|
|
|
{
|
|
|
|
/* Mask out FPU flags */
|
|
|
|
__writecr0(Cr0 & ~(CR0_MP | CR0_TS | CR0_EM));
|
|
|
|
}
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Get FX frame */
|
|
|
|
FxSaveArea = KiGetThreadNpxArea(Thread);
|
|
|
|
Thread->NpxState = NPX_STATE_NOT_LOADED;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Save state if supported by CPU */
|
|
|
|
if (KeI386FxsrPresent) Ke386FxSave(FxSaveArea);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Now save the FN state wherever it was requested */
|
|
|
|
if (SaveArea) Ke386FnSave(SaveArea);
|
|
|
|
|
|
|
|
/* Clear NPX thread */
|
|
|
|
KeGetCurrentPrcb()->NpxThread = NULL;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Add the CR0 from the NPX frame */
|
|
|
|
Cr0 |= NPX_STATE_NOT_LOADED;
|
|
|
|
Cr0 |= FxSaveArea->Cr0NpxState;
|
|
|
|
__writecr0(Cr0);
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-20 04:05:08 +00:00
|
|
|
/* Restore interrupt state */
|
|
|
|
__writeeflags(EFlags);
|
|
|
|
}
|
|
|
|
|
2010-01-11 06:08:11 +00:00
|
|
|
/* PUBLIC FUNCTIONS **********************************************************/
|
2010-01-06 00:40:07 +00:00
|
|
|
|
2010-01-11 06:08:11 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
VOID
|
2010-01-06 00:40:07 +00:00
|
|
|
NTAPI
|
2010-01-11 06:08:11 +00:00
|
|
|
KiCoprocessorError(VOID)
|
2010-01-06 00:40:07 +00:00
|
|
|
{
|
2010-01-11 06:08:11 +00:00
|
|
|
PFX_SAVE_AREA NpxArea;
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-11 06:08:11 +00:00
|
|
|
/* Get the FPU area */
|
|
|
|
NpxArea = KiGetThreadNpxArea(KeGetCurrentThread());
|
2011-09-07 13:43:54 +00:00
|
|
|
|
2010-01-11 06:08:11 +00:00
|
|
|
/* Set CR0_TS */
|
|
|
|
NpxArea->Cr0NpxState = CR0_TS;
|
|
|
|
__writecr0(__readcr0() | CR0_TS);
|
2010-01-06 00:40:07 +00:00
|
|
|
}
|
|
|
|
|
2006-11-08 11:47:44 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
KeSaveFloatingPointState(OUT PKFLOATING_SAVE Save)
|
|
|
|
{
|
|
|
|
PFNSAVE_FORMAT FpState;
|
2009-06-09 07:15:37 +00:00
|
|
|
ASSERT(KeGetCurrentIrql() <= DISPATCH_LEVEL);
|
2006-11-08 11:47:44 +00:00
|
|
|
DPRINT1("%s is not really implemented\n", __FUNCTION__);
|
|
|
|
|
|
|
|
/* check if we are doing software emulation */
|
|
|
|
if (!KeI386NpxPresent) return STATUS_ILLEGAL_FLOAT_CONTEXT;
|
|
|
|
|
|
|
|
FpState = ExAllocatePool(NonPagedPool, sizeof (FNSAVE_FORMAT));
|
|
|
|
if (!FpState) return STATUS_INSUFFICIENT_RESOURCES;
|
|
|
|
|
|
|
|
*((PVOID *) Save) = FpState;
|
|
|
|
#ifdef __GNUC__
|
|
|
|
asm volatile("fnsave %0\n\t" : "=m" (*FpState));
|
|
|
|
#else
|
|
|
|
__asm
|
|
|
|
{
|
2014-05-20 22:33:26 +00:00
|
|
|
mov eax, [FpState]
|
|
|
|
fnsave [eax]
|
2006-11-08 11:47:44 +00:00
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
2011-10-07 16:18:52 +00:00
|
|
|
KeGetCurrentThread()->Header.NpxIrql = KeGetCurrentIrql();
|
2006-11-08 11:47:44 +00:00
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
NTSTATUS
|
|
|
|
NTAPI
|
|
|
|
KeRestoreFloatingPointState(IN PKFLOATING_SAVE Save)
|
|
|
|
{
|
|
|
|
PFNSAVE_FORMAT FpState = *((PVOID *) Save);
|
2011-10-07 16:18:52 +00:00
|
|
|
ASSERT(KeGetCurrentThread()->Header.NpxIrql == KeGetCurrentIrql());
|
2006-11-08 11:47:44 +00:00
|
|
|
DPRINT1("%s is not really implemented\n", __FUNCTION__);
|
|
|
|
|
|
|
|
#ifdef __GNUC__
|
|
|
|
asm volatile("fnclex\n\t");
|
|
|
|
asm volatile("frstor %0\n\t" : "=m" (*FpState));
|
|
|
|
#else
|
|
|
|
__asm
|
|
|
|
{
|
|
|
|
fnclex
|
2014-05-20 22:51:38 +00:00
|
|
|
mov eax, [FpState]
|
|
|
|
frstor [eax]
|
2006-11-08 11:47:44 +00:00
|
|
|
};
|
|
|
|
#endif
|
|
|
|
|
|
|
|
ExFreePool(FpState);
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
ULONG
|
|
|
|
NTAPI
|
|
|
|
KeGetRecommendedSharedDataAlignment(VOID)
|
|
|
|
{
|
|
|
|
/* Return the global variable */
|
|
|
|
return KeLargestCacheLine;
|
|
|
|
}
|
|
|
|
|
2008-11-01 12:01:58 +00:00
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KiFlushTargetEntireTb(IN PKIPI_CONTEXT PacketContext,
|
|
|
|
IN PVOID Ignored1,
|
|
|
|
IN PVOID Ignored2,
|
|
|
|
IN PVOID Ignored3)
|
|
|
|
{
|
|
|
|
/* Signal this packet as done */
|
|
|
|
KiIpiSignalPacketDone(PacketContext);
|
|
|
|
|
|
|
|
/* Flush the TB for the Current CPU */
|
|
|
|
KeFlushCurrentTb();
|
|
|
|
}
|
|
|
|
|
2006-11-08 11:47:44 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KeFlushEntireTb(IN BOOLEAN Invalid,
|
|
|
|
IN BOOLEAN AllProcessors)
|
|
|
|
{
|
|
|
|
KIRQL OldIrql;
|
2008-11-01 12:01:58 +00:00
|
|
|
#ifdef CONFIG_SMP
|
|
|
|
KAFFINITY TargetAffinity;
|
|
|
|
PKPRCB Prcb = KeGetCurrentPrcb();
|
|
|
|
#endif
|
2006-11-08 11:47:44 +00:00
|
|
|
|
|
|
|
/* Raise the IRQL for the TB Flush */
|
|
|
|
OldIrql = KeRaiseIrqlToSynchLevel();
|
|
|
|
|
|
|
|
#ifdef CONFIG_SMP
|
2008-11-01 12:01:58 +00:00
|
|
|
/* FIXME: Use KiTbFlushTimeStamp to synchronize TB flush */
|
|
|
|
|
|
|
|
/* Get the current processor affinity, and exclude ourselves */
|
|
|
|
TargetAffinity = KeActiveProcessors;
|
2008-11-01 19:22:43 +00:00
|
|
|
TargetAffinity &= ~Prcb->SetMember;
|
2008-11-01 12:01:58 +00:00
|
|
|
|
|
|
|
/* Make sure this is MP */
|
|
|
|
if (TargetAffinity)
|
|
|
|
{
|
|
|
|
/* Send an IPI TB flush to the other processors */
|
|
|
|
KiIpiSendPacket(TargetAffinity,
|
|
|
|
KiFlushTargetEntireTb,
|
|
|
|
NULL,
|
|
|
|
0,
|
|
|
|
NULL);
|
|
|
|
}
|
2006-11-08 11:47:44 +00:00
|
|
|
#endif
|
|
|
|
|
2008-11-01 12:01:58 +00:00
|
|
|
/* Flush the TB for the Current CPU, and update the flush stamp */
|
2006-11-08 11:47:44 +00:00
|
|
|
KeFlushCurrentTb();
|
|
|
|
|
2008-11-01 12:01:58 +00:00
|
|
|
#ifdef CONFIG_SMP
|
|
|
|
/* If this is MP, wait for the other processors to finish */
|
|
|
|
if (TargetAffinity)
|
|
|
|
{
|
|
|
|
/* Sanity check */
|
2013-10-19 11:33:34 +00:00
|
|
|
ASSERT(Prcb == KeGetCurrentPrcb());
|
2008-11-01 12:01:58 +00:00
|
|
|
|
|
|
|
/* FIXME: TODO */
|
|
|
|
ASSERTMSG("Not yet implemented\n", FALSE);
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/* Update the flush stamp and return to original IRQL */
|
|
|
|
InterlockedExchangeAdd(&KiTbFlushTimeStamp, 1);
|
2006-11-08 11:47:44 +00:00
|
|
|
KeLowerIrql(OldIrql);
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
VOID
|
|
|
|
NTAPI
|
|
|
|
KeSetDmaIoCoherency(IN ULONG Coherency)
|
|
|
|
{
|
|
|
|
/* Save the coherency globally */
|
|
|
|
KiDmaIoCoherency = Coherency;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
KAFFINITY
|
|
|
|
NTAPI
|
|
|
|
KeQueryActiveProcessors(VOID)
|
|
|
|
{
|
|
|
|
PAGED_CODE();
|
|
|
|
|
|
|
|
/* Simply return the number of active processors */
|
|
|
|
return KeActiveProcessors;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
|
|
|
VOID
|
|
|
|
__cdecl
|
|
|
|
KeSaveStateForHibernate(IN PKPROCESSOR_STATE State)
|
|
|
|
{
|
|
|
|
/* Capture the context */
|
|
|
|
RtlCaptureContext(&State->ContextFrame);
|
|
|
|
|
|
|
|
/* Capture the control state */
|
|
|
|
KiSaveProcessorControlState(State);
|
|
|
|
}
|