mirror of
https://github.com/reactos/reactos.git
synced 2025-04-04 20:50:41 +00:00
[NTOS:KE/x64] Handle NMI vs swapgs race condition
This commit is contained in:
parent
72fd54a7f4
commit
6ac260dcec
2 changed files with 30 additions and 1 deletions
|
@ -91,6 +91,10 @@ static __inline__ __attribute__((always_inline)) void __str(unsigned short *Dest
|
||||||
__asm__ __volatile__("str %0" : : "m"(*Destination) : "memory");
|
__asm__ __volatile__("str %0" : : "m"(*Destination) : "memory");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static __inline__ __attribute__((always_inline)) void __swapgs(void)
|
||||||
|
{
|
||||||
|
__asm__ __volatile__("swapgs" : : : "memory");
|
||||||
|
}
|
||||||
|
|
||||||
#elif defined(_MSC_VER)
|
#elif defined(_MSC_VER)
|
||||||
|
|
||||||
|
@ -106,6 +110,7 @@ void __ltr(unsigned short Source);
|
||||||
|
|
||||||
void __str(unsigned short *Destination);
|
void __str(unsigned short *Destination);
|
||||||
|
|
||||||
|
void __swapgs(void);
|
||||||
|
|
||||||
#else
|
#else
|
||||||
#error Unknown compiler for inline assembler
|
#error Unknown compiler for inline assembler
|
||||||
|
|
|
@ -89,15 +89,39 @@ KiNmiInterruptHandler(
|
||||||
_In_ PKTRAP_FRAME TrapFrame,
|
_In_ PKTRAP_FRAME TrapFrame,
|
||||||
_In_ PKEXCEPTION_FRAME ExceptionFrame)
|
_In_ PKEXCEPTION_FRAME ExceptionFrame)
|
||||||
{
|
{
|
||||||
|
BOOLEAN ManualSwapGs = FALSE;
|
||||||
|
|
||||||
|
/* Check if the NMI came from kernel mode */
|
||||||
|
if ((TrapFrame->SegCs & MODE_MASK) == 0)
|
||||||
|
{
|
||||||
|
/* Check if GS base is already kernel mode. This is needed, because
|
||||||
|
we might be interrupted during an interrupt/exception from user-mode
|
||||||
|
before the swapgs instruction. */
|
||||||
|
if ((LONG64)__readmsr(MSR_GS_BASE) >= 0)
|
||||||
|
{
|
||||||
|
/* Swap GS to kernel */
|
||||||
|
__swapgs();
|
||||||
|
ManualSwapGs = TRUE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
/* Check if this is a freeze */
|
/* Check if this is a freeze */
|
||||||
if (KiProcessorFreezeHandler(TrapFrame, ExceptionFrame))
|
if (KiProcessorFreezeHandler(TrapFrame, ExceptionFrame))
|
||||||
{
|
{
|
||||||
/* NMI was handled */
|
/* NMI was handled */
|
||||||
return;
|
goto Exit;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Handle the NMI */
|
/* Handle the NMI */
|
||||||
KiHandleNmi();
|
KiHandleNmi();
|
||||||
|
|
||||||
|
Exit:
|
||||||
|
/* Check if we need to swap GS back */
|
||||||
|
if (ManualSwapGs)
|
||||||
|
{
|
||||||
|
/* Swap GS back to user */
|
||||||
|
__swapgs();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#define MAX_SYSCALL_PARAMS 16
|
#define MAX_SYSCALL_PARAMS 16
|
||||||
|
|
Loading…
Reference in a new issue