Windows2000/private/ntos/ps/psspnd.c

380 lines
6.8 KiB
C
Raw Normal View History

2001-01-01 00:00:00 +01:00
/*++
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;
}
}