2020-09-30 16:53:55 +02:00

621 lines
17 KiB
C
Raw Blame History

This file contains invisible Unicode characters

This file contains invisible Unicode characters that are indistinguishable to humans but may be processed differently by a computer. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*++
Copyright (c) 1989 Microsoft Corporation
Module Name:
semphore.c
Abstract:
This module implements the executive semaphore object. Functions are
provided to create, open, release, and query semaphore objects.
Author:
David N. Cutler (davec) 8-May-1989
Environment:
Kernel mode only.
Revision History:
--*/
#include "exp.h"
//
// Temporary so boost is patchable
//
ULONG ExpSemaphoreBoost = SEMAPHORE_INCREMENT;
//
// Address of semaphore object type descriptor.
//
POBJECT_TYPE ExSemaphoreObjectType;
//
// Structure that describes the mapping of generic access rights to object
// specific access rights for semaphore objects.
//
#ifdef ALLOC_DATA_PRAGMA
#pragma const_seg("INITCONST")
#endif
const GENERIC_MAPPING ExpSemaphoreMapping = {
STANDARD_RIGHTS_READ |
SEMAPHORE_QUERY_STATE,
STANDARD_RIGHTS_WRITE |
SEMAPHORE_MODIFY_STATE,
STANDARD_RIGHTS_EXECUTE |
SYNCHRONIZE,
SEMAPHORE_ALL_ACCESS
};
#ifdef ALLOC_DATA_PRAGMA
#pragma const_seg()
#endif
#ifdef ALLOC_PRAGMA
#pragma alloc_text(INIT, ExpSemaphoreInitialization)
#pragma alloc_text(PAGE, NtCreateSemaphore)
#pragma alloc_text(PAGE, NtOpenSemaphore)
#pragma alloc_text(PAGE, NtQuerySemaphore)
#pragma alloc_text(PAGE, NtReleaseSemaphore)
#endif
BOOLEAN
ExpSemaphoreInitialization (
)
/*++
Routine Description:
This function creates the semaphore object type descriptor at system
initialization and stores the address of the object type descriptor
in local static storage.
Arguments:
None.
Return Value:
A value of TRUE is returned if the semaphore object type descriptor is
successfully created. Otherwise a value of FALSE is returned.
--*/
{
OBJECT_TYPE_INITIALIZER ObjectTypeInitializer;
NTSTATUS Status;
UNICODE_STRING TypeName;
//
// Initialize string descriptor.
//
RtlInitUnicodeString(&TypeName, L"Semaphore");
//
// Create semaphore object type descriptor.
//
RtlZeroMemory(&ObjectTypeInitializer, sizeof(ObjectTypeInitializer));
ObjectTypeInitializer.Length = sizeof(ObjectTypeInitializer);
ObjectTypeInitializer.InvalidAttributes = OBJ_OPENLINK;
ObjectTypeInitializer.GenericMapping = ExpSemaphoreMapping;
ObjectTypeInitializer.PoolType = NonPagedPool;
ObjectTypeInitializer.DefaultNonPagedPoolCharge = sizeof(KSEMAPHORE);
ObjectTypeInitializer.ValidAccessMask = SEMAPHORE_ALL_ACCESS;
Status = ObCreateObjectType(&TypeName,
&ObjectTypeInitializer,
(PSECURITY_DESCRIPTOR)NULL,
&ExSemaphoreObjectType);
//
// If the semaphore object type descriptor was successfully created, then
// return a value of TRUE. Otherwise return a value of FALSE.
//
return (BOOLEAN)(NT_SUCCESS(Status));
}
NTSTATUS
NtCreateSemaphore (
IN PHANDLE SemaphoreHandle,
IN ACCESS_MASK DesiredAccess,
IN POBJECT_ATTRIBUTES ObjectAttributes OPTIONAL,
IN LONG InitialCount,
IN LONG MaximumCount
)
/*++
Routine Description:
This function creates a semaphore object, sets its initial count to the
specified value, sets its maximum count to the specified value, and opens
a handle to the object with the specified desired access.
Arguments:
SemaphoreHandle - Supplies a pointer to a variable that will receive the
semaphore object handle.
DesiredAccess - Supplies the desired types of access for the semaphore
object.
ObjectAttributes - Supplies a pointer to an object attributes structure.
InitialCount - Supplies the initial count of the semaphore object.
MaximumCount - Supplies the maximum count of the semaphore object.
Return Value:
NTSTATUS.
--*/
{
HANDLE Handle;
KPROCESSOR_MODE PreviousMode;
PVOID Semaphore;
NTSTATUS Status;
//
// Establish an exception handler, probe the output handle address, and
// attempt to create a semaphore object. If the probe fails, then return
// the exception code as the service status. Otherwise return the status
// value returned by the object insertion routine.
//
// Get previous processor mode and probe output handle address if
// necessary.
//
PreviousMode = KeGetPreviousMode();
if (PreviousMode != KernelMode) {
try {
ProbeForWriteHandle(SemaphoreHandle);
} except(ExSystemExceptionFilter()) {
return GetExceptionCode();
}
}
//
// Check argument validity.
//
if ((MaximumCount <= 0) || (InitialCount < 0) ||
(InitialCount > MaximumCount)) {
return STATUS_INVALID_PARAMETER;
}
//
// Allocate semaphore object.
//
Status = ObCreateObject(PreviousMode,
ExSemaphoreObjectType,
ObjectAttributes,
PreviousMode,
NULL,
sizeof(KSEMAPHORE),
0,
0,
(PVOID *)&Semaphore);
//
// If the semaphore object was successfully allocated, then initialize
// the semaphore object and attempt to insert the semaphore object in
// the current process' handle table.
//
if (NT_SUCCESS(Status)) {
KeInitializeSemaphore((PKSEMAPHORE)Semaphore,
InitialCount,
MaximumCount);
Status = ObInsertObject(Semaphore,
NULL,
DesiredAccess,
0,
(PVOID *)NULL,
&Handle);
//
// If the semaphore object was successfully inserted in the current
// process' handle table, then attempt to write the semaphore handle
// value. If the write attempt fails, then do not report an error.
// When the caller attempts to access the handle value, an access
// violation will occur.
//
if (NT_SUCCESS(Status)) {
if (PreviousMode != KernelMode) {
try {
*SemaphoreHandle = Handle;
} except(ExSystemExceptionFilter()) {
NOTHING;
}
}
else {
*SemaphoreHandle = Handle;
}
}
}
//
// Return service status.
//
return Status;
}
NTSTATUS
NtOpenSemaphore (
OUT PHANDLE SemaphoreHandle,
IN ACCESS_MASK DesiredAccess,
IN POBJECT_ATTRIBUTES ObjectAttributes
)
/*++
Routine Description:
This function opens a handle to a semaphore object with the specified
desired access.
Arguments:
SemaphoreHandle - Supplies a pointer to a variable that will receive the
semaphore object handle.
DesiredAccess - Supplies the desired types of access for the semaphore
object.
ObjectAttributes - Supplies a pointer to an object attributes structure.
Return Value:
NTSTATUS.
--*/
{
HANDLE Handle;
KPROCESSOR_MODE PreviousMode;
NTSTATUS Status;
//
// Establish an exception handler, probe the output handle address, and
// attempt to open a semaphore object. If the probe fails, then return
// the exception code as the service status. Otherwise return the status
// value returned by the object open routine.
//
// Get previous processor mode and probe output handle address if
// necessary.
//
PreviousMode = KeGetPreviousMode();
if (PreviousMode != KernelMode) {
try {
ProbeForWriteHandle(SemaphoreHandle);
} except(ExSystemExceptionFilter()) {
return GetExceptionCode();
}
}
//
// Open handle to the semaphore object with the specified desired access.
//
Status = ObOpenObjectByName(ObjectAttributes,
ExSemaphoreObjectType,
PreviousMode,
NULL,
DesiredAccess,
NULL,
&Handle);
//
// If the open was successful, then attempt to write the semaphore
// object handle value. If the write attempt fails, then do not report
// an error. When the caller attempts to access the handle value, an
// access violation will occur.
//
if (NT_SUCCESS(Status)) {
if (PreviousMode != KernelMode) {
try {
*SemaphoreHandle = Handle;
} except(ExSystemExceptionFilter()) {
NOTHING;
}
}
else {
*SemaphoreHandle = Handle;
}
}
//
// Return service status.
//
return Status;
}
NTSTATUS
NtQuerySemaphore (
IN HANDLE SemaphoreHandle,
IN SEMAPHORE_INFORMATION_CLASS SemaphoreInformationClass,
OUT PVOID SemaphoreInformation,
IN ULONG SemaphoreInformationLength,
OUT PULONG ReturnLength OPTIONAL
)
/*++
Routine Description:
This function queries the state of a semaphore object and returns the
requested information in the specified record structure.
Arguments:
SemaphoreHandle - Supplies a handle to a semaphore object.
SemaphoreInformationClass - Supplies the class of information being
requested.
SemaphoreInformation - Supplies a pointer to a record that is to receive
the requested information.
SemaphoreInformationLength - Supplies the length of the record that is
to receive the requested information.
ReturnLength - Supplies an optional pointer to a variable that will
receive the actual length of the information that is returned.
Return Value:
NTSTATUS.
--*/
{
PVOID Semaphore;
LONG Count;
LONG Maximum;
KPROCESSOR_MODE PreviousMode;
NTSTATUS Status;
//
// Establish an exception handler, probe the output arguments, reference
// the semaphore object, and return the specified information. If the probe
// fails, then return the exception code as the service status. Otherwise
// return the status value returned by the reference object by handle
// routine.
//
// Get previous processor mode and probe output arguments if necessary.
//
PreviousMode = KeGetPreviousMode();
if (PreviousMode != KernelMode) {
try {
ProbeForWriteSmallStructure (SemaphoreInformation,
sizeof(SEMAPHORE_BASIC_INFORMATION),
sizeof(ULONG));
if (ARGUMENT_PRESENT(ReturnLength)) {
ProbeForWriteUlong(ReturnLength);
}
} except(ExSystemExceptionFilter()) {
return GetExceptionCode();
}
}
//
// Check argument validity.
//
if (SemaphoreInformationClass != SemaphoreBasicInformation) {
return STATUS_INVALID_INFO_CLASS;
}
if (SemaphoreInformationLength != sizeof(SEMAPHORE_BASIC_INFORMATION)) {
return STATUS_INFO_LENGTH_MISMATCH;
}
//
// Reference semaphore object by handle.
//
Status = ObReferenceObjectByHandle(SemaphoreHandle,
SEMAPHORE_QUERY_STATE,
ExSemaphoreObjectType,
PreviousMode,
&Semaphore,
NULL);
//
// If the reference was successful, then read the current state and
// maximum count of the semaphore object, dereference semaphore object,
// fill in the information structure, and return the length of the
// information structure if specified. If the write of the semaphore
// information or the return length fails, then do not report an error.
// When the caller accesses the information structure or length an
// access violation will occur.
//
if (NT_SUCCESS(Status)) {
Count = KeReadStateSemaphore((PKSEMAPHORE)Semaphore);
Maximum = ((PKSEMAPHORE)Semaphore)->Limit;
ObDereferenceObject(Semaphore);
if (PreviousMode != KernelMode) {
try {
((PSEMAPHORE_BASIC_INFORMATION)SemaphoreInformation)->CurrentCount = Count;
((PSEMAPHORE_BASIC_INFORMATION)SemaphoreInformation)->MaximumCount = Maximum;
if (ARGUMENT_PRESENT(ReturnLength)) {
*ReturnLength = sizeof(SEMAPHORE_BASIC_INFORMATION);
}
} except(ExSystemExceptionFilter()) {
NOTHING;
}
}
else {
((PSEMAPHORE_BASIC_INFORMATION)SemaphoreInformation)->CurrentCount = Count;
((PSEMAPHORE_BASIC_INFORMATION)SemaphoreInformation)->MaximumCount = Maximum;
if (ARGUMENT_PRESENT(ReturnLength)) {
*ReturnLength = sizeof(SEMAPHORE_BASIC_INFORMATION);
}
}
}
//
// Return service status.
//
return Status;
}
NTSTATUS
NtReleaseSemaphore (
IN HANDLE SemaphoreHandle,
IN LONG ReleaseCount,
OUT PLONG PreviousCount OPTIONAL
)
/*++
Routine Description:
This function releases a semaphore object by adding the specified release
count to the current value.
Arguments:
Semaphore - Supplies a handle to a semaphore object.
ReleaseCount - Supplies the release count that is to be added to the
current semaphore count.
PreviousCount - Supplies an optional pointer to a variable that will
receive the previous semaphore count.
Return Value:
NTSTATUS.
--*/
{
LONG Count;
PVOID Semaphore;
KPROCESSOR_MODE PreviousMode;
NTSTATUS Status;
//
// Establish an exception handler, probe the previous count address if
// specified, reference the semaphore object, and release the semaphore
// object. If the probe fails, then return the exception code as the
// service status. Otherwise return the status value returned by the
// reference object by handle routine.
//
// Get previous processor mode and probe previous count address
// if necessary.
//
PreviousMode = KeGetPreviousMode();
if ((ARGUMENT_PRESENT(PreviousCount)) && (PreviousMode != KernelMode)) {
try {
ProbeForWriteLong(PreviousCount);
} except(ExSystemExceptionFilter()) {
return GetExceptionCode();
}
}
//
// Check argument validity.
//
if (ReleaseCount <= 0) {
return STATUS_INVALID_PARAMETER;
}
//
// Reference semaphore object by handle.
//
Status = ObReferenceObjectByHandle(SemaphoreHandle,
SEMAPHORE_MODIFY_STATE,
ExSemaphoreObjectType,
PreviousMode,
&Semaphore,
NULL);
//
// If the reference was successful, then release the semaphore object.
// If an exception occurs because the maximum count of the semaphore
// has been exceeded, then dereference the semaphore object and return
// the exception code as the service status. Otherwise write the previous
// count value if specified. If the write of the previous count fails,
// then do not report an error. When the caller attempts to access the
// previous count value, an access violation will occur.
//
if (NT_SUCCESS(Status)) {
//
// Initialize Count to keep W4 compiler happy.
//
Count = 0;
try {
PERFINFO_DECLARE_OBJECT(Semaphore);
Count = KeReleaseSemaphore((PKSEMAPHORE)Semaphore,
ExpSemaphoreBoost,
ReleaseCount,
FALSE);
} except(ExSystemExceptionFilter()) {
Status = GetExceptionCode();
}
ObDereferenceObject(Semaphore);
if (NT_SUCCESS(Status) && ARGUMENT_PRESENT(PreviousCount)) {
if (PreviousMode != KernelMode) {
try {
*PreviousCount = Count;
} except(ExSystemExceptionFilter()) {
NOTHING;
}
}
else {
*PreviousCount = Count;
}
}
}
//
// Return service status.
//
return Status;
}