380 lines
6.8 KiB
C
380 lines
6.8 KiB
C
/*++
|
|
|
|
Copyright (c) 1989 Microsoft Corporation
|
|
|
|
Module Name:
|
|
|
|
psspnd.c
|
|
|
|
Abstract:
|
|
|
|
This module implements NtSuspendThread and NtResumeThread
|
|
|
|
Author:
|
|
|
|
Mark Lucovsky (markl) 25-May-1989
|
|
|
|
Revision History:
|
|
|
|
--*/
|
|
|
|
#include "psp.h"
|
|
|
|
#ifdef ALLOC_PRAGMA
|
|
#pragma alloc_text(PAGE, NtSuspendThread)
|
|
#pragma alloc_text(PAGE, NtResumeThread)
|
|
#pragma alloc_text(PAGE, NtAlertThread)
|
|
#pragma alloc_text(PAGE, NtAlertResumeThread)
|
|
#pragma alloc_text(PAGE, NtTestAlert)
|
|
#endif
|
|
|
|
|
|
NTSTATUS
|
|
NtSuspendThread(
|
|
IN HANDLE ThreadHandle,
|
|
OUT PULONG PreviousSuspendCount OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function suspends the target thread, and optionally
|
|
returns the previous suspend count.
|
|
|
|
Arguments:
|
|
|
|
ThreadHandle - Supplies a handle to the thread object to suspend.
|
|
|
|
PreviousSuspendCount - An optional parameter, that if specified
|
|
points to a variable that receives the thread's previous suspend
|
|
count.
|
|
|
|
Return Value:
|
|
|
|
return-value - Description of conditions needed to return value. - or -
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
PETHREAD Thread;
|
|
NTSTATUS st;
|
|
ULONG LocalPreviousSuspendCount;
|
|
KPROCESSOR_MODE Mode;
|
|
|
|
PAGED_CODE();
|
|
|
|
try {
|
|
|
|
Mode = KeGetPreviousMode();
|
|
|
|
if ( Mode != KernelMode ) {
|
|
if (ARGUMENT_PRESENT(PreviousSuspendCount)) {
|
|
ProbeForWriteUlong(PreviousSuspendCount);
|
|
}
|
|
}
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
st = ObReferenceObjectByHandle(
|
|
ThreadHandle,
|
|
THREAD_SUSPEND_RESUME,
|
|
PsThreadType,
|
|
Mode,
|
|
(PVOID *)&Thread,
|
|
NULL
|
|
);
|
|
|
|
if ( !NT_SUCCESS(st) ) {
|
|
return st;
|
|
}
|
|
|
|
try {
|
|
|
|
if ( Thread != PsGetCurrentThread() ) {
|
|
if ( Thread->HasTerminated ) {
|
|
ObDereferenceObject(Thread);
|
|
return STATUS_THREAD_IS_TERMINATING;
|
|
}
|
|
|
|
LocalPreviousSuspendCount = (ULONG) KeSuspendThread(&Thread->Tcb);
|
|
|
|
} else {
|
|
LocalPreviousSuspendCount = (ULONG) KeSuspendThread(&Thread->Tcb);
|
|
}
|
|
|
|
ObDereferenceObject(Thread);
|
|
|
|
if (ARGUMENT_PRESENT(PreviousSuspendCount))
|
|
*PreviousSuspendCount = LocalPreviousSuspendCount;
|
|
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
|
|
st = GetExceptionCode();
|
|
|
|
|
|
// Either the suspend, or the store could cause an
|
|
// exception. The store is a partial success, while the
|
|
// suspend exception is an error
|
|
|
|
|
|
if ( st == STATUS_SUSPEND_COUNT_EXCEEDED ) {
|
|
ObDereferenceObject(Thread);
|
|
} else {
|
|
st = STATUS_SUCCESS;
|
|
}
|
|
|
|
return st;
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
NTSTATUS
|
|
NtResumeThread(
|
|
IN HANDLE ThreadHandle,
|
|
OUT PULONG PreviousSuspendCount OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
description-of-function.
|
|
|
|
Arguments:
|
|
|
|
argument-name - Supplies | Returns description of argument.
|
|
.
|
|
.
|
|
|
|
Return Value:
|
|
|
|
return-value - Description of conditions needed to return value. - or -
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
PETHREAD Thread;
|
|
NTSTATUS st;
|
|
ULONG LocalPreviousSuspendCount;
|
|
KPROCESSOR_MODE Mode;
|
|
|
|
PAGED_CODE();
|
|
|
|
try {
|
|
|
|
Mode = KeGetPreviousMode();
|
|
|
|
if ( Mode != KernelMode ) {
|
|
if (ARGUMENT_PRESENT(PreviousSuspendCount))
|
|
ProbeForWriteUlong(PreviousSuspendCount);
|
|
}
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
st = ObReferenceObjectByHandle(
|
|
ThreadHandle,
|
|
THREAD_SUSPEND_RESUME,
|
|
PsThreadType,
|
|
Mode,
|
|
(PVOID *)&Thread,
|
|
NULL
|
|
);
|
|
|
|
if ( !NT_SUCCESS(st) ) {
|
|
return st;
|
|
}
|
|
|
|
LocalPreviousSuspendCount = (ULONG) KeResumeThread(&Thread->Tcb);
|
|
|
|
ObDereferenceObject(Thread);
|
|
|
|
try {
|
|
if (ARGUMENT_PRESENT(PreviousSuspendCount))
|
|
*PreviousSuspendCount = LocalPreviousSuspendCount;
|
|
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
NTSTATUS
|
|
NtAlertThread(
|
|
IN HANDLE ThreadHandle
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function alerts the target thread using the previous mode
|
|
as the mode of the alert.
|
|
|
|
Arguments:
|
|
|
|
ThreadHandle - Supplies an open handle to the thread to be alerted
|
|
|
|
Return Value:
|
|
|
|
TBD
|
|
|
|
--*/
|
|
|
|
{
|
|
PETHREAD Thread;
|
|
NTSTATUS st;
|
|
KPROCESSOR_MODE Mode;
|
|
|
|
PAGED_CODE();
|
|
|
|
Mode = KeGetPreviousMode();
|
|
|
|
st = ObReferenceObjectByHandle(
|
|
ThreadHandle,
|
|
THREAD_ALERT,
|
|
PsThreadType,
|
|
Mode,
|
|
(PVOID *)&Thread,
|
|
NULL
|
|
);
|
|
|
|
if ( !NT_SUCCESS(st) ) {
|
|
return st;
|
|
}
|
|
|
|
(VOID) KeAlertThread(&Thread->Tcb,Mode);
|
|
|
|
ObDereferenceObject(Thread);
|
|
|
|
return STATUS_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
NtAlertResumeThread(
|
|
IN HANDLE ThreadHandle,
|
|
OUT PULONG PreviousSuspendCount OPTIONAL
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
description-of-function.
|
|
|
|
Arguments:
|
|
|
|
argument-name - Supplies | Returns description of argument.
|
|
.
|
|
.
|
|
|
|
Return Value:
|
|
|
|
return-value - Description of conditions needed to return value. - or -
|
|
None.
|
|
|
|
--*/
|
|
|
|
{
|
|
PETHREAD Thread;
|
|
NTSTATUS st;
|
|
ULONG LocalPreviousSuspendCount;
|
|
KPROCESSOR_MODE Mode;
|
|
|
|
PAGED_CODE();
|
|
|
|
try {
|
|
|
|
Mode = KeGetPreviousMode();
|
|
|
|
if ( Mode != KernelMode ) {
|
|
if (ARGUMENT_PRESENT(PreviousSuspendCount)) {
|
|
ProbeForWriteUlong(PreviousSuspendCount);
|
|
}
|
|
}
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
|
|
return GetExceptionCode();
|
|
}
|
|
|
|
st = ObReferenceObjectByHandle(
|
|
ThreadHandle,
|
|
THREAD_SUSPEND_RESUME,
|
|
PsThreadType,
|
|
Mode,
|
|
(PVOID *)&Thread,
|
|
NULL
|
|
);
|
|
|
|
if ( !NT_SUCCESS(st) ) {
|
|
return st;
|
|
}
|
|
|
|
LocalPreviousSuspendCount = (ULONG) KeAlertResumeThread(&Thread->Tcb);
|
|
|
|
ObDereferenceObject(Thread);
|
|
|
|
try {
|
|
|
|
if (ARGUMENT_PRESENT(PreviousSuspendCount))
|
|
*PreviousSuspendCount = LocalPreviousSuspendCount;
|
|
|
|
} except (EXCEPTION_EXECUTE_HANDLER) {
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
return STATUS_SUCCESS;
|
|
}
|
|
|
|
|
|
NTSTATUS
|
|
NtTestAlert(
|
|
VOID
|
|
)
|
|
|
|
/*++
|
|
|
|
Routine Description:
|
|
|
|
This function tests the alert flag inside the current thread. If
|
|
an alert is pending for the previous mode, then the alerted status
|
|
is returned, pending APC's may also be delivered at this time.
|
|
|
|
Arguments:
|
|
|
|
None
|
|
|
|
Return Value:
|
|
|
|
STATUS_ALERTED - An alert was pending for the current thread at the
|
|
time this function was called.
|
|
|
|
STATUS_SUCCESS - No alert was pending for this thread.
|
|
|
|
--*/
|
|
|
|
{
|
|
|
|
PAGED_CODE();
|
|
|
|
if ( KeTestAlertThread(KeGetPreviousMode()) ) {
|
|
return STATUS_ALERTED;
|
|
} else {
|
|
return STATUS_SUCCESS;
|
|
}
|
|
}
|