2005-10-02 08:31:45 +00:00
|
|
|
/*
|
2006-09-10 18:47:53 +00:00
|
|
|
* PROJECT: ReactOS Kernel
|
|
|
|
* LICENSE: GPL - See COPYING in the top level directory
|
1998-09-05 17:34:23 +00:00
|
|
|
* FILE: ntoskrnl/ke/sem.c
|
2006-09-10 18:47:53 +00:00
|
|
|
* PURPOSE: Implements the Semaphore Dispatcher Object
|
|
|
|
* PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
|
1998-08-25 04:27:26 +00:00
|
|
|
*/
|
|
|
|
|
2006-09-10 19:14:03 +00:00
|
|
|
/* INCLUDES ******************************************************************/
|
1998-08-25 04:27:26 +00:00
|
|
|
|
2004-08-15 16:39:12 +00:00
|
|
|
#include <ntoskrnl.h>
|
1999-11-12 12:01:17 +00:00
|
|
|
#define NDEBUG
|
2008-08-30 16:31:06 +00:00
|
|
|
#include <debug.h>
|
1998-08-25 04:27:26 +00:00
|
|
|
|
|
|
|
/* FUNCTIONS *****************************************************************/
|
|
|
|
|
2003-07-10 17:44:06 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2005-03-14 02:08:17 +00:00
|
|
|
VOID
|
2006-09-10 18:47:53 +00:00
|
|
|
NTAPI
|
|
|
|
KeInitializeSemaphore(IN PKSEMAPHORE Semaphore,
|
|
|
|
IN LONG Count,
|
|
|
|
IN LONG Limit)
|
1998-08-25 04:27:26 +00:00
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Simply Initialize the Header */
|
|
|
|
KeInitializeDispatcherHeader(&Semaphore->Header,
|
|
|
|
SemaphoreObject,
|
2005-10-02 08:31:45 +00:00
|
|
|
sizeof(KSEMAPHORE) / sizeof(ULONG),
|
2005-03-14 02:08:17 +00:00
|
|
|
Count);
|
|
|
|
|
|
|
|
/* Set the Limit */
|
2005-05-09 01:38:29 +00:00
|
|
|
Semaphore->Limit = Limit;
|
1998-08-25 04:27:26 +00:00
|
|
|
}
|
|
|
|
|
2003-07-10 17:44:06 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
|
|
|
*/
|
2005-05-09 01:38:29 +00:00
|
|
|
LONG
|
2006-09-10 18:47:53 +00:00
|
|
|
NTAPI
|
|
|
|
KeReadStateSemaphore(IN PKSEMAPHORE Semaphore)
|
1998-08-25 04:27:26 +00:00
|
|
|
{
|
2006-09-10 18:47:53 +00:00
|
|
|
ASSERT_SEMAPHORE(Semaphore);
|
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Just return the Signal State */
|
2005-10-02 08:31:45 +00:00
|
|
|
return Semaphore->Header.SignalState;
|
1998-08-25 04:27:26 +00:00
|
|
|
}
|
|
|
|
|
2003-07-10 17:44:06 +00:00
|
|
|
/*
|
|
|
|
* @implemented
|
1999-11-12 12:01:17 +00:00
|
|
|
*/
|
2005-05-09 01:38:29 +00:00
|
|
|
LONG
|
2006-09-10 18:47:53 +00:00
|
|
|
NTAPI
|
|
|
|
KeReleaseSemaphore(IN PKSEMAPHORE Semaphore,
|
|
|
|
IN KPRIORITY Increment,
|
|
|
|
IN LONG Adjustment,
|
|
|
|
IN BOOLEAN Wait)
|
1998-08-25 04:27:26 +00:00
|
|
|
{
|
2005-10-02 12:27:54 +00:00
|
|
|
LONG InitialState, State;
|
2005-03-14 02:08:17 +00:00
|
|
|
KIRQL OldIrql;
|
|
|
|
PKTHREAD CurrentThread;
|
2006-09-10 18:47:53 +00:00
|
|
|
ASSERT_SEMAPHORE(Semaphore);
|
|
|
|
ASSERT_IRQL_LESS_OR_EQUAL(DISPATCH_LEVEL);
|
2003-11-02 01:16:21 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Lock the Dispatcher Database */
|
2006-09-10 18:26:50 +00:00
|
|
|
OldIrql = KiAcquireDispatcherLock();
|
2003-11-02 01:16:21 +00:00
|
|
|
|
2005-10-02 08:31:45 +00:00
|
|
|
/* Save the Old State and get new one */
|
2005-03-14 02:08:17 +00:00
|
|
|
InitialState = Semaphore->Header.SignalState;
|
2005-10-02 12:27:54 +00:00
|
|
|
State = InitialState + Adjustment;
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Check if the Limit was exceeded */
|
2005-10-02 08:31:45 +00:00
|
|
|
if ((Semaphore->Limit < State) || (InitialState > State))
|
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Raise an error if it was exceeded */
|
2006-09-10 18:26:50 +00:00
|
|
|
KiReleaseDispatcherLock(OldIrql);
|
2005-03-14 02:08:17 +00:00
|
|
|
ExRaiseStatus(STATUS_SEMAPHORE_LIMIT_EXCEEDED);
|
2003-11-02 01:16:21 +00:00
|
|
|
}
|
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Now set the new state */
|
2005-10-02 08:31:45 +00:00
|
|
|
Semaphore->Header.SignalState = State;
|
2005-05-09 01:38:29 +00:00
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Check if we should wake it */
|
2005-10-02 12:27:54 +00:00
|
|
|
if (!(InitialState) && !(IsListEmpty(&Semaphore->Header.WaitListHead)))
|
2005-10-02 08:31:45 +00:00
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Wake the Semaphore */
|
2005-03-14 06:44:31 +00:00
|
|
|
KiWaitTest(&Semaphore->Header, Increment);
|
2003-11-02 01:16:21 +00:00
|
|
|
}
|
|
|
|
|
2006-09-10 18:47:53 +00:00
|
|
|
/* Check if the caller wants to wait after this release */
|
2005-10-02 08:31:45 +00:00
|
|
|
if (Wait == FALSE)
|
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Release the Lock */
|
2006-09-10 18:26:50 +00:00
|
|
|
KiReleaseDispatcherLock(OldIrql);
|
2005-10-02 08:31:45 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Set a wait */
|
|
|
|
CurrentThread = KeGetCurrentThread();
|
|
|
|
CurrentThread->WaitNext = TRUE;
|
|
|
|
CurrentThread->WaitIrql = OldIrql;
|
2003-11-02 01:16:21 +00:00
|
|
|
}
|
|
|
|
|
2005-03-14 02:08:17 +00:00
|
|
|
/* Return the previous state */
|
|
|
|
return InitialState;
|
1998-08-25 04:27:26 +00:00
|
|
|
}
|
|
|
|
|
2000-06-04 19:51:05 +00:00
|
|
|
/* EOF */
|