mirror of
https://github.com/reactos/reactos.git
synced 2025-05-28 13:38:19 +00:00
319 lines
6 KiB
C
319 lines
6 KiB
C
/*
|
|
* PROJECT: ReactOS Hardware Abstraction Layer (HAL)
|
|
* LICENSE: BSD - See COPYING.ARM in the top level directory
|
|
* FILE: halx86/generic/misc.c
|
|
* PURPOSE: NMI, I/O Mapping and x86 Subs
|
|
* PROGRAMMERS: ReactOS Portable Systems Group
|
|
*/
|
|
|
|
/* INCLUDES *******************************************************************/
|
|
|
|
#include <hal.h>
|
|
#define NDEBUG
|
|
#include <debug.h>
|
|
|
|
/* GLOBALS *******************************************************************/
|
|
|
|
BOOLEAN HalpNMIInProgress;
|
|
|
|
/* PRIVATE FUNCTIONS **********************************************************/
|
|
|
|
NTSTATUS
|
|
NTAPI
|
|
HalpOpenRegistryKey(IN PHANDLE KeyHandle,
|
|
IN HANDLE RootKey,
|
|
IN PUNICODE_STRING KeyName,
|
|
IN ACCESS_MASK DesiredAccess,
|
|
IN BOOLEAN Create)
|
|
{
|
|
NTSTATUS Status;
|
|
ULONG Disposition;
|
|
OBJECT_ATTRIBUTES ObjectAttributes;
|
|
|
|
/* Setup the attributes we received */
|
|
InitializeObjectAttributes(&ObjectAttributes,
|
|
KeyName,
|
|
OBJ_CASE_INSENSITIVE,
|
|
RootKey,
|
|
NULL);
|
|
|
|
/* What to do? */
|
|
if ( Create )
|
|
{
|
|
/* Create the key */
|
|
Status = ZwCreateKey(KeyHandle,
|
|
DesiredAccess,
|
|
&ObjectAttributes,
|
|
0,
|
|
NULL,
|
|
REG_OPTION_VOLATILE,
|
|
&Disposition);
|
|
}
|
|
else
|
|
{
|
|
/* Open the key */
|
|
Status = ZwOpenKey(KeyHandle, DesiredAccess, &ObjectAttributes);
|
|
}
|
|
|
|
/* We're done */
|
|
return Status;
|
|
}
|
|
|
|
VOID
|
|
NTAPI
|
|
HalpCheckPowerButton(VOID)
|
|
{
|
|
//
|
|
// Nothing to do on non-ACPI
|
|
//
|
|
return;
|
|
}
|
|
|
|
VOID
|
|
NTAPI
|
|
HalpFlushTLB(VOID)
|
|
{
|
|
ULONG Flags, Cr4;
|
|
INT CpuInfo[4];
|
|
ULONG_PTR PageDirectory;
|
|
|
|
//
|
|
// Disable interrupts
|
|
//
|
|
Flags = __readeflags();
|
|
_disable();
|
|
|
|
//
|
|
// Get page table directory base
|
|
//
|
|
PageDirectory = __readcr3();
|
|
|
|
//
|
|
// Check for CPUID support
|
|
//
|
|
if (KeGetCurrentPrcb()->CpuID)
|
|
{
|
|
//
|
|
// Check for global bit in CPU features
|
|
//
|
|
__cpuid(CpuInfo, 1);
|
|
if (CpuInfo[3] & 0x2000)
|
|
{
|
|
//
|
|
// Get current CR4 value
|
|
//
|
|
Cr4 = __readcr4();
|
|
|
|
//
|
|
// Disable global bit
|
|
//
|
|
__writecr4(Cr4 & ~CR4_PGE);
|
|
|
|
//
|
|
// Flush TLB and re-enable global bit
|
|
//
|
|
__writecr3(PageDirectory);
|
|
__writecr4(Cr4);
|
|
|
|
//
|
|
// Restore interrupts
|
|
//
|
|
__writeeflags(Flags);
|
|
return;
|
|
}
|
|
}
|
|
|
|
//
|
|
// Legacy: just flush TLB
|
|
//
|
|
__writecr3(PageDirectory);
|
|
__writeeflags(Flags);
|
|
}
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
HalHandleNMI(IN PVOID NmiInfo)
|
|
{
|
|
#ifndef _MINIHAL_
|
|
SYSTEM_CONTROL_PORT_B_REGISTER SystemControl;
|
|
|
|
//
|
|
// Don't recurse
|
|
//
|
|
if (HalpNMIInProgress++) while (TRUE);
|
|
|
|
//
|
|
// Read the system control register B
|
|
//
|
|
SystemControl.Bits = __inbyte(SYSTEM_CONTROL_PORT_B);
|
|
|
|
//
|
|
// Switch to boot vieo
|
|
//
|
|
if (InbvIsBootDriverInstalled())
|
|
{
|
|
//
|
|
// Acquire ownership
|
|
//
|
|
InbvAcquireDisplayOwnership();
|
|
InbvResetDisplay();
|
|
|
|
//
|
|
// Fill the screen
|
|
//
|
|
InbvSolidColorFill(0, 0, 639, 479, 1);
|
|
InbvSetScrollRegion(0, 0, 639, 479);
|
|
|
|
//
|
|
// Enable text
|
|
//
|
|
InbvSetTextColor(15);
|
|
InbvInstallDisplayStringFilter(NULL);
|
|
InbvEnableDisplayString(TRUE);
|
|
}
|
|
|
|
//
|
|
// Display NMI failure string
|
|
//
|
|
InbvDisplayString("\n*** Hardware Malfunction\n\n");
|
|
InbvDisplayString("Call your hardware vendor for support\n\n");
|
|
|
|
//
|
|
// Check for parity error
|
|
//
|
|
if (SystemControl.ParityCheck)
|
|
{
|
|
//
|
|
// Display message
|
|
//
|
|
InbvDisplayString("NMI: Parity Check / Memory Parity Error\n");
|
|
}
|
|
|
|
//
|
|
// Check for I/O failure
|
|
//
|
|
if (SystemControl.ChannelCheck)
|
|
{
|
|
//
|
|
// Display message
|
|
//
|
|
InbvDisplayString("NMI: Channel Check / IOCHK\n");
|
|
}
|
|
|
|
//
|
|
// Check for EISA systems
|
|
//
|
|
if (HalpBusType == MACHINE_TYPE_EISA)
|
|
{
|
|
//
|
|
// FIXME: Not supported
|
|
//
|
|
UNIMPLEMENTED;
|
|
}
|
|
|
|
//
|
|
// Halt the system
|
|
//
|
|
InbvDisplayString("\n*** The system has halted ***\n");
|
|
#endif
|
|
|
|
//
|
|
// Enter the debugger if possible
|
|
//
|
|
//if (!(KdDebuggerNotPresent) && (KdDebuggerEnabled)) KeEnterKernelDebugger();
|
|
|
|
//
|
|
// Freeze the system
|
|
//
|
|
while (TRUE);
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
UCHAR
|
|
FASTCALL
|
|
HalSystemVectorDispatchEntry(IN ULONG Vector,
|
|
OUT PKINTERRUPT_ROUTINE **FlatDispatch,
|
|
OUT PKINTERRUPT_ROUTINE *NoConnection)
|
|
{
|
|
//
|
|
// Not implemented on x86
|
|
//
|
|
return 0;
|
|
}
|
|
|
|
/*
|
|
* @implemented
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
KeFlushWriteBuffer(VOID)
|
|
{
|
|
//
|
|
// Not implemented on x86
|
|
//
|
|
return;
|
|
}
|
|
|
|
#ifdef _M_IX86
|
|
/* x86 fastcall wrappers */
|
|
|
|
#undef KeRaiseIrql
|
|
/*
|
|
* @implemented
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
KeRaiseIrql(KIRQL NewIrql,
|
|
PKIRQL OldIrql)
|
|
{
|
|
/* Call the fastcall function */
|
|
*OldIrql = KfRaiseIrql(NewIrql);
|
|
}
|
|
|
|
#undef KeLowerIrql
|
|
/*
|
|
* @implemented
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
KeLowerIrql(KIRQL NewIrql)
|
|
{
|
|
/* Call the fastcall function */
|
|
KfLowerIrql(NewIrql);
|
|
}
|
|
|
|
#undef KeAcquireSpinLock
|
|
/*
|
|
* @implemented
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
KeAcquireSpinLock(PKSPIN_LOCK SpinLock,
|
|
PKIRQL OldIrql)
|
|
{
|
|
/* Call the fastcall function */
|
|
*OldIrql = KfAcquireSpinLock(SpinLock);
|
|
}
|
|
|
|
#undef KeReleaseSpinLock
|
|
/*
|
|
* @implemented
|
|
*/
|
|
VOID
|
|
NTAPI
|
|
KeReleaseSpinLock(PKSPIN_LOCK SpinLock,
|
|
KIRQL NewIrql)
|
|
{
|
|
/* Call the fastcall function */
|
|
KfReleaseSpinLock(SpinLock, NewIrql);
|
|
}
|
|
|
|
#endif
|
|
|