2005-05-09 01:38:29 +00:00
|
|
|
/*
|
2005-01-07 06:54:27 +00:00
|
|
|
* COPYRIGHT: See COPYING in the top level directory
|
2005-12-29 18:47:14 +00:00
|
|
|
* PROJECT: ReactOS Kernel
|
2005-01-07 06:54:27 +00:00
|
|
|
* FILE: ntoskrnl/ex/mutant.c
|
2005-03-14 02:08:17 +00:00
|
|
|
* PURPOSE: Executive Management of Mutants
|
2005-12-29 18:47:14 +00:00
|
|
|
* PROGRAMMERS: Alex Ionescu (alex@relsoft.net)
|
|
|
|
* Thomas Weidenmueller
|
2005-01-07 06:54:27 +00:00
|
|
|
*/
|
|
|
|
|
|
|
|
/* INCLUDES *****************************************************************/
|
|
|
|
|
|
|
|
#include <ntoskrnl.h>
|
|
|
|
#define NDEBUG
|
2008-08-30 16:31:06 +00:00
|
|
|
#include <debug.h>
|
2005-01-07 06:54:27 +00:00
|
|
|
|
2005-11-28 23:25:31 +00:00
|
|
|
#if defined (ALLOC_PRAGMA)
|
|
|
|
#pragma alloc_text(INIT, ExpInitializeMutantImplementation)
|
|
|
|
#endif
|
|
|
|
|
2005-12-29 18:47:14 +00:00
|
|
|
/* DATA **********************************************************************/
|
2005-06-18 18:32:29 +00:00
|
|
|
|
2005-01-07 06:54:27 +00:00
|
|
|
POBJECT_TYPE ExMutantObjectType = NULL;
|
|
|
|
|
2005-12-29 18:47:14 +00:00
|
|
|
GENERIC_MAPPING ExpMutantMapping =
|
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
STANDARD_RIGHTS_READ | SYNCHRONIZE | MUTANT_QUERY_STATE,
|
|
|
|
STANDARD_RIGHTS_WRITE | SYNCHRONIZE,
|
|
|
|
STANDARD_RIGHTS_EXECUTE | SYNCHRONIZE | MUTANT_QUERY_STATE,
|
2005-12-29 18:47:14 +00:00
|
|
|
MUTANT_ALL_ACCESS
|
|
|
|
};
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-12-29 18:47:14 +00:00
|
|
|
static const INFORMATION_CLASS_INFO ExMutantInfoClass[] =
|
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
/* MutantBasicInformation */
|
2005-12-29 18:47:14 +00:00
|
|
|
ICI_SQ_SAME( sizeof(MUTANT_BASIC_INFORMATION), sizeof(ULONG), ICIF_QUERY),
|
2005-01-22 03:54:23 +00:00
|
|
|
};
|
|
|
|
|
2005-01-07 06:54:27 +00:00
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
2005-05-09 01:38:29 +00:00
|
|
|
VOID
|
2005-12-29 18:47:14 +00:00
|
|
|
NTAPI
|
2005-02-28 17:40:15 +00:00
|
|
|
ExpDeleteMutant(PVOID ObjectBody)
|
2005-01-07 06:54:27 +00:00
|
|
|
{
|
2005-07-04 22:11:00 +00:00
|
|
|
DPRINT("ExpDeleteMutant(ObjectBody 0x%p)\n", ObjectBody);
|
2005-01-07 06:54:27 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Make sure to release the Mutant */
|
|
|
|
KeReleaseMutant((PKMUTANT)ObjectBody,
|
|
|
|
MUTANT_INCREMENT,
|
|
|
|
TRUE,
|
|
|
|
FALSE);
|
|
|
|
}
|
2005-02-28 17:40:15 +00:00
|
|
|
|
2005-05-09 01:38:29 +00:00
|
|
|
VOID
|
2005-03-14 02:08:17 +00:00
|
|
|
INIT_FUNCTION
|
2005-12-29 18:47:14 +00:00
|
|
|
NTAPI
|
2005-02-28 17:40:15 +00:00
|
|
|
ExpInitializeMutantImplementation(VOID)
|
|
|
|
{
|
2005-05-15 17:59:33 +00:00
|
|
|
OBJECT_TYPE_INITIALIZER ObjectTypeInitializer;
|
|
|
|
UNICODE_STRING Name;
|
2005-05-21 16:33:23 +00:00
|
|
|
DPRINT("Creating Mutant Object Type\n");
|
2005-12-29 18:47:14 +00:00
|
|
|
|
2005-05-15 17:59:33 +00:00
|
|
|
/* Create the Event Pair Object Type */
|
|
|
|
RtlZeroMemory(&ObjectTypeInitializer, sizeof(ObjectTypeInitializer));
|
|
|
|
RtlInitUnicodeString(&Name, L"Mutant");
|
|
|
|
ObjectTypeInitializer.Length = sizeof(ObjectTypeInitializer);
|
|
|
|
ObjectTypeInitializer.DefaultNonPagedPoolCharge = sizeof(KMUTANT);
|
|
|
|
ObjectTypeInitializer.GenericMapping = ExpMutantMapping;
|
|
|
|
ObjectTypeInitializer.PoolType = NonPagedPool;
|
|
|
|
ObjectTypeInitializer.DeleteProcedure = ExpDeleteMutant;
|
|
|
|
ObjectTypeInitializer.ValidAccessMask = MUTANT_ALL_ACCESS;
|
2006-06-05 06:31:42 +00:00
|
|
|
ObCreateObjectType(&Name, &ObjectTypeInitializer, NULL, &ExMutantObjectType);
|
2005-03-14 02:08:17 +00:00
|
|
|
}
|
2005-02-28 17:40:15 +00:00
|
|
|
|
2005-01-22 03:54:23 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2005-05-09 01:38:29 +00:00
|
|
|
NTSTATUS
|
2005-12-29 18:47:14 +00:00
|
|
|
NTAPI
|
2005-01-07 06:54:27 +00:00
|
|
|
NtCreateMutant(OUT PHANDLE MutantHandle,
|
2005-03-14 02:08:17 +00:00
|
|
|
IN ACCESS_MASK DesiredAccess,
|
|
|
|
IN POBJECT_ATTRIBUTES ObjectAttributes OPTIONAL,
|
|
|
|
IN BOOLEAN InitialOwner)
|
2005-01-07 06:54:27 +00:00
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
KPROCESSOR_MODE PreviousMode = ExGetPreviousMode();
|
|
|
|
HANDLE hMutant;
|
|
|
|
PKMUTANT Mutant;
|
2009-08-26 17:31:02 +00:00
|
|
|
NTSTATUS Status;
|
2005-03-14 02:08:17 +00:00
|
|
|
PAGED_CODE();
|
2005-12-29 18:47:14 +00:00
|
|
|
DPRINT("NtCreateMutant(0x%p, 0x%x, 0x%p)\n",
|
|
|
|
MutantHandle, DesiredAccess, ObjectAttributes);
|
|
|
|
|
|
|
|
/* Check if we were called from user-mode */
|
2009-08-26 17:31:02 +00:00
|
|
|
if (PreviousMode != KernelMode)
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Enter SEH Block */
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Check handle pointer */
|
2005-08-21 17:38:07 +00:00
|
|
|
ProbeForWriteHandle(MutantHandle);
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2009-08-26 17:31:02 +00:00
|
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
2009-08-26 17:31:02 +00:00
|
|
|
/* Return the exception code */
|
|
|
|
_SEH2_YIELD(return _SEH2_GetExceptionCode());
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2005-02-22 22:19:14 +00:00
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Create the Mutant Object*/
|
|
|
|
Status = ObCreateObject(PreviousMode,
|
|
|
|
ExMutantObjectType,
|
|
|
|
ObjectAttributes,
|
|
|
|
PreviousMode,
|
|
|
|
NULL,
|
|
|
|
sizeof(KMUTANT),
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
(PVOID*)&Mutant);
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Check for success */
|
2005-12-29 18:47:14 +00:00
|
|
|
if(NT_SUCCESS(Status))
|
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Initalize the Kernel Mutant */
|
|
|
|
DPRINT("Initializing the Mutant\n");
|
|
|
|
KeInitializeMutant(Mutant, InitialOwner);
|
|
|
|
|
|
|
|
/* Insert the Object */
|
|
|
|
Status = ObInsertObject((PVOID)Mutant,
|
|
|
|
NULL,
|
|
|
|
DesiredAccess,
|
|
|
|
0,
|
|
|
|
NULL,
|
|
|
|
&hMutant);
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-12-29 18:47:14 +00:00
|
|
|
/* Check for success */
|
2009-08-26 17:31:02 +00:00
|
|
|
if (NT_SUCCESS(Status))
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Enter SEH for return */
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Return the handle to the caller */
|
2005-03-14 02:08:17 +00:00
|
|
|
*MutantHandle = hMutant;
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_EXCEPT(ExSystemExceptionFilter())
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
2009-08-26 17:31:02 +00:00
|
|
|
/* Get the exception code */
|
2008-11-24 13:40:26 +00:00
|
|
|
Status = _SEH2_GetExceptionCode();
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2005-03-14 02:08:17 +00:00
|
|
|
}
|
2005-01-07 06:54:27 +00:00
|
|
|
}
|
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Return Status */
|
|
|
|
return Status;
|
2005-01-07 06:54:27 +00:00
|
|
|
}
|
|
|
|
|
2005-01-22 03:54:23 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2005-05-09 01:38:29 +00:00
|
|
|
NTSTATUS
|
2005-12-29 18:47:14 +00:00
|
|
|
NTAPI
|
2005-01-07 06:54:27 +00:00
|
|
|
NtOpenMutant(OUT PHANDLE MutantHandle,
|
2005-03-14 02:08:17 +00:00
|
|
|
IN ACCESS_MASK DesiredAccess,
|
|
|
|
IN POBJECT_ATTRIBUTES ObjectAttributes)
|
2005-01-07 06:54:27 +00:00
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
HANDLE hMutant;
|
|
|
|
KPROCESSOR_MODE PreviousMode = ExGetPreviousMode();
|
2009-08-26 17:31:02 +00:00
|
|
|
NTSTATUS Status;
|
2005-03-14 02:08:17 +00:00
|
|
|
PAGED_CODE();
|
2005-12-29 18:47:14 +00:00
|
|
|
DPRINT("NtOpenMutant(0x%p, 0x%x, 0x%p)\n",
|
|
|
|
MutantHandle, DesiredAccess, ObjectAttributes);
|
|
|
|
|
|
|
|
/* Check if we were called from user-mode */
|
2009-08-26 17:31:02 +00:00
|
|
|
if (PreviousMode != KernelMode)
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Enter SEH Block */
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Check handle pointer */
|
2005-08-21 17:38:07 +00:00
|
|
|
ProbeForWriteHandle(MutantHandle);
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2009-08-26 17:31:02 +00:00
|
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
2009-08-26 17:31:02 +00:00
|
|
|
/* Return the exception code */
|
|
|
|
_SEH2_YIELD(return _SEH2_GetExceptionCode());
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2005-02-28 17:40:15 +00:00
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Open the Object */
|
|
|
|
Status = ObOpenObjectByName(ObjectAttributes,
|
|
|
|
ExMutantObjectType,
|
|
|
|
PreviousMode,
|
2006-05-25 20:50:58 +00:00
|
|
|
NULL,
|
2005-03-14 02:08:17 +00:00
|
|
|
DesiredAccess,
|
|
|
|
NULL,
|
|
|
|
&hMutant);
|
|
|
|
|
2005-12-29 18:47:14 +00:00
|
|
|
/* Check for success */
|
|
|
|
if(NT_SUCCESS(Status))
|
|
|
|
{
|
|
|
|
/* Enter SEH for return */
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Return the handle to the caller */
|
2005-03-14 02:08:17 +00:00
|
|
|
*MutantHandle = hMutant;
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_EXCEPT(ExSystemExceptionFilter())
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
2008-11-24 13:40:26 +00:00
|
|
|
Status = _SEH2_GetExceptionCode();
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2005-01-07 06:54:27 +00:00
|
|
|
}
|
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Return Status */
|
|
|
|
return Status;
|
2005-01-07 06:54:27 +00:00
|
|
|
}
|
|
|
|
|
2005-01-22 03:54:23 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2005-05-09 01:38:29 +00:00
|
|
|
NTSTATUS
|
2005-12-29 18:47:14 +00:00
|
|
|
NTAPI
|
2005-01-22 03:54:23 +00:00
|
|
|
NtQueryMutant(IN HANDLE MutantHandle,
|
2005-03-14 02:08:17 +00:00
|
|
|
IN MUTANT_INFORMATION_CLASS MutantInformationClass,
|
|
|
|
OUT PVOID MutantInformation,
|
|
|
|
IN ULONG MutantInformationLength,
|
|
|
|
OUT PULONG ResultLength OPTIONAL)
|
2005-01-07 06:54:27 +00:00
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
PKMUTANT Mutant;
|
|
|
|
KPROCESSOR_MODE PreviousMode = ExGetPreviousMode();
|
2009-08-26 17:31:02 +00:00
|
|
|
NTSTATUS Status;
|
2005-12-29 18:47:14 +00:00
|
|
|
PMUTANT_BASIC_INFORMATION BasicInfo =
|
|
|
|
(PMUTANT_BASIC_INFORMATION)MutantInformation;
|
2005-03-14 02:08:17 +00:00
|
|
|
PAGED_CODE();
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Check buffers and parameters */
|
2005-10-10 13:03:55 +00:00
|
|
|
Status = DefaultQueryInfoBufferCheck(MutantInformationClass,
|
|
|
|
ExMutantInfoClass,
|
2005-12-29 18:47:14 +00:00
|
|
|
sizeof(ExMutantInfoClass) /
|
|
|
|
sizeof(ExMutantInfoClass[0]),
|
2005-10-10 13:03:55 +00:00
|
|
|
MutantInformation,
|
|
|
|
MutantInformationLength,
|
|
|
|
ResultLength,
|
2010-01-13 22:35:43 +00:00
|
|
|
NULL,
|
2005-10-10 13:03:55 +00:00
|
|
|
PreviousMode);
|
2005-12-29 18:47:14 +00:00
|
|
|
if(!NT_SUCCESS(Status))
|
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
DPRINT("NtQueryMutant() failed, Status: 0x%x\n", Status);
|
|
|
|
return Status;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Open the Object */
|
|
|
|
Status = ObReferenceObjectByHandle(MutantHandle,
|
|
|
|
MUTANT_QUERY_STATE,
|
|
|
|
ExMutantObjectType,
|
|
|
|
PreviousMode,
|
|
|
|
(PVOID*)&Mutant,
|
|
|
|
NULL);
|
|
|
|
/* Check for Status */
|
2009-08-26 17:31:02 +00:00
|
|
|
if (NT_SUCCESS(Status))
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Enter SEH Block for return */
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Fill out the Basic Information Requested */
|
|
|
|
DPRINT("Returning Mutant Information\n");
|
|
|
|
BasicInfo->CurrentCount = KeReadStateMutant(Mutant);
|
2005-12-29 18:47:14 +00:00
|
|
|
BasicInfo->OwnedByCaller = (Mutant->OwnerThread ==
|
|
|
|
KeGetCurrentThread());
|
2005-03-14 02:08:17 +00:00
|
|
|
BasicInfo->AbandonedState = Mutant->Abandoned;
|
|
|
|
|
|
|
|
/* Return the Result Length if requested */
|
2009-08-26 17:31:02 +00:00
|
|
|
if (ResultLength) *ResultLength = sizeof(MUTANT_BASIC_INFORMATION);
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_EXCEPT(ExSystemExceptionFilter())
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
2008-11-24 13:40:26 +00:00
|
|
|
Status = _SEH2_GetExceptionCode();
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2005-03-14 02:08:17 +00:00
|
|
|
|
|
|
|
/* Release the Object */
|
|
|
|
ObDereferenceObject(Mutant);
|
|
|
|
}
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Return Status */
|
|
|
|
return Status;
|
2005-01-22 03:54:23 +00:00
|
|
|
}
|
2005-01-07 06:54:27 +00:00
|
|
|
|
2005-01-22 03:54:23 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2005-05-09 01:38:29 +00:00
|
|
|
NTSTATUS
|
2005-12-29 18:47:14 +00:00
|
|
|
NTAPI
|
2005-01-22 03:54:23 +00:00
|
|
|
NtReleaseMutant(IN HANDLE MutantHandle,
|
2005-12-29 18:47:14 +00:00
|
|
|
IN PLONG PreviousCount OPTIONAL)
|
2005-01-22 03:54:23 +00:00
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
PKMUTANT Mutant;
|
2005-12-29 18:47:14 +00:00
|
|
|
KPROCESSOR_MODE PreviousMode = ExGetPreviousMode();
|
2009-08-26 17:31:02 +00:00
|
|
|
NTSTATUS Status;
|
2005-03-14 02:08:17 +00:00
|
|
|
PAGED_CODE();
|
2005-07-04 22:11:00 +00:00
|
|
|
DPRINT("NtReleaseMutant(MutantHandle 0x%p PreviousCount 0x%p)\n",
|
2005-05-09 01:38:29 +00:00
|
|
|
MutantHandle,
|
2005-03-14 02:08:17 +00:00
|
|
|
PreviousCount);
|
|
|
|
|
2005-12-29 18:47:14 +00:00
|
|
|
/* Check if we were called from user-mode */
|
2009-08-26 17:31:02 +00:00
|
|
|
if ((PreviousCount) && (PreviousMode != KernelMode))
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Entry SEH Block */
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Make sure the state pointer is valid */
|
2005-08-21 17:38:07 +00:00
|
|
|
ProbeForWriteLong(PreviousCount);
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2009-08-26 17:31:02 +00:00
|
|
|
_SEH2_EXCEPT(EXCEPTION_EXECUTE_HANDLER)
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
2009-08-26 17:31:02 +00:00
|
|
|
/* Return the exception code */
|
|
|
|
_SEH2_YIELD(return _SEH2_GetExceptionCode());
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2005-05-09 01:38:29 +00:00
|
|
|
}
|
2005-03-14 02:08:17 +00:00
|
|
|
|
|
|
|
/* Open the Object */
|
|
|
|
Status = ObReferenceObjectByHandle(MutantHandle,
|
|
|
|
MUTANT_QUERY_STATE,
|
|
|
|
ExMutantObjectType,
|
|
|
|
PreviousMode,
|
|
|
|
(PVOID*)&Mutant,
|
|
|
|
NULL);
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Check for Success and release if such */
|
2009-08-26 17:31:02 +00:00
|
|
|
if (NT_SUCCESS(Status))
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Release the mutant. doing so might raise an exception which we're
|
|
|
|
* required to catch!
|
|
|
|
*/
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_TRY
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
|
|
|
/* Release the mutant */
|
|
|
|
LONG Prev = KeReleaseMutant(Mutant,
|
|
|
|
MUTANT_INCREMENT,
|
|
|
|
FALSE,
|
|
|
|
FALSE);
|
|
|
|
|
|
|
|
/* Return the previous count if requested */
|
2009-08-26 17:31:02 +00:00
|
|
|
if (PreviousCount) *PreviousCount = Prev;
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_EXCEPT(ExSystemExceptionFilter())
|
2005-12-29 18:47:14 +00:00
|
|
|
{
|
2009-08-26 17:31:02 +00:00
|
|
|
/* Get the exception code */
|
2008-11-24 13:40:26 +00:00
|
|
|
Status = _SEH2_GetExceptionCode();
|
2005-12-29 18:47:14 +00:00
|
|
|
}
|
2008-11-24 13:40:26 +00:00
|
|
|
_SEH2_END;
|
2005-03-28 15:17:54 +00:00
|
|
|
|
2005-12-29 18:47:14 +00:00
|
|
|
/* Dereference it */
|
2005-03-14 02:08:17 +00:00
|
|
|
ObDereferenceObject(Mutant);
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Return Status */
|
|
|
|
return Status;
|
2005-01-07 06:54:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/* EOF */
|