1 /*
2 * PROJECT: ReactOS Kernel
3 * LICENSE: GPL - See COPYING in the top level directory
4 * FILE: ntoskrnl/ke/mutex.c
5 * PURPOSE: Implements the Mutant Dispatcher Object
6 * PROGRAMMERS: Alex Ionescu (alex.ionescu@reactos.org)
7 */
8
9 /* INCLUDES ******************************************************************/
10
11 #include <ntoskrnl.h>
12 #define NDEBUG
13 #include <debug.h>
14
15 /* FUNCTIONS *****************************************************************/
16
17 /*
18 * @implemented
19 */
20 VOID
21 NTAPI
KeInitializeMutant(IN PKMUTANT Mutant,IN BOOLEAN InitialOwner)22 KeInitializeMutant(IN PKMUTANT Mutant,
23 IN BOOLEAN InitialOwner)
24 {
25 PKTHREAD CurrentThread;
26 KIRQL OldIrql;
27
28 /* Check if we have an initial owner */
29 if (InitialOwner)
30 {
31 /* We also need to associate a thread */
32 CurrentThread = KeGetCurrentThread();
33 Mutant->OwnerThread = CurrentThread;
34
35 /* We're about to touch the Thread, so lock the Dispatcher */
36 OldIrql = KiAcquireDispatcherLock();
37
38 /* And insert it into its list */
39 InsertTailList(&CurrentThread->MutantListHead,
40 &Mutant->MutantListEntry);
41
42 /* Release Dispatcher Lock */
43 KiReleaseDispatcherLock(OldIrql);
44 }
45 else
46 {
47 /* In this case, we don't have an owner yet */
48 Mutant->OwnerThread = NULL;
49 }
50
51 /* Now we set up the Dispatcher Header */
52 Mutant->Header.Type = MutantObject;
53 Mutant->Header.Size = sizeof(KMUTANT) / sizeof(ULONG);
54 Mutant->Header.SignalState = InitialOwner ? 0 : 1;
55 InitializeListHead(&(Mutant->Header.WaitListHead));
56
57 /* Initialize the default data */
58 Mutant->Abandoned = FALSE;
59 Mutant->ApcDisable = 0;
60 }
61
62 /*
63 * @implemented
64 */
65 VOID
66 NTAPI
KeInitializeMutex(IN PKMUTEX Mutex,IN ULONG Level)67 KeInitializeMutex(IN PKMUTEX Mutex,
68 IN ULONG Level)
69 {
70 /* Set up the Dispatcher Header */
71 Mutex->Header.Type = MutantObject;
72 Mutex->Header.Size = sizeof(KMUTEX) / sizeof(ULONG);
73 Mutex->Header.SignalState = 1;
74 InitializeListHead(&(Mutex->Header.WaitListHead));
75
76 /* Initialize the default data */
77 Mutex->OwnerThread = NULL;
78 Mutex->Abandoned = FALSE;
79 Mutex->ApcDisable = 1;
80 }
81
82 /*
83 * @implemented
84 */
85 LONG
86 NTAPI
KeReadStateMutant(IN PKMUTANT Mutant)87 KeReadStateMutant(IN PKMUTANT Mutant)
88 {
89 /* Return the Signal State */
90 return Mutant->Header.SignalState;
91 }
92
93 /*
94 * @implemented
95 */
96 LONG
97 NTAPI
KeReleaseMutant(IN PKMUTANT Mutant,IN KPRIORITY Increment,IN BOOLEAN Abandon,IN BOOLEAN Wait)98 KeReleaseMutant(IN PKMUTANT Mutant,
99 IN KPRIORITY Increment,
100 IN BOOLEAN Abandon,
101 IN BOOLEAN Wait)
102 {
103 KIRQL OldIrql;
104 LONG PreviousState;
105 PKTHREAD CurrentThread = KeGetCurrentThread();
106 BOOLEAN EnableApc = FALSE;
107 ASSERT_MUTANT(Mutant);
108 ASSERT_IRQL_LESS_OR_EQUAL(DISPATCH_LEVEL);
109
110 /* Lock the Dispatcher Database */
111 OldIrql = KiAcquireDispatcherLock();
112
113 /* Save the Previous State */
114 PreviousState = Mutant->Header.SignalState;
115
116 /* Check if it is to be abandonned */
117 if (Abandon == FALSE)
118 {
119 /* Make sure that the Owner Thread is the current Thread */
120 if (Mutant->OwnerThread != CurrentThread)
121 {
122 /* Release the lock */
123 KiReleaseDispatcherLock(OldIrql);
124
125 /* Raise an exception */
126 ExRaiseStatus(Mutant->Abandoned ? STATUS_ABANDONED :
127 STATUS_MUTANT_NOT_OWNED);
128 }
129
130 /* If the thread owns it, then increase the signal state */
131 Mutant->Header.SignalState++;
132 }
133 else
134 {
135 /* It's going to be abandonned */
136 Mutant->Header.SignalState = 1;
137 Mutant->Abandoned = TRUE;
138 }
139
140 /* Check if the signal state is only single */
141 if (Mutant->Header.SignalState == 1)
142 {
143 /* Check if it's below 0 now */
144 if (PreviousState <= 0)
145 {
146 /* Remove the mutant from the list */
147 RemoveEntryList(&Mutant->MutantListEntry);
148
149 /* Save if we need to re-enable APCs */
150 EnableApc = Mutant->ApcDisable;
151 }
152
153 /* Remove the Owning Thread and wake it */
154 Mutant->OwnerThread = NULL;
155
156 /* Check if the Wait List isn't empty */
157 if (!IsListEmpty(&Mutant->Header.WaitListHead))
158 {
159 /* Wake the Mutant */
160 KiWaitTest(&Mutant->Header, Increment);
161 }
162 }
163
164 /* Check if the caller wants to wait after this release */
165 if (Wait == FALSE)
166 {
167 /* Release the Lock */
168 KiReleaseDispatcherLock(OldIrql);
169 }
170 else
171 {
172 /* Set a wait */
173 CurrentThread->WaitNext = TRUE;
174 CurrentThread->WaitIrql = OldIrql;
175 }
176
177 /* Check if we need to re-enable APCs */
178 if (EnableApc) KeLeaveCriticalRegion();
179
180 /* Return the previous state */
181 return PreviousState;
182 }
183
184 /*
185 * @implemented
186 */
187 LONG
188 NTAPI
KeReleaseMutex(IN PKMUTEX Mutex,IN BOOLEAN Wait)189 KeReleaseMutex(IN PKMUTEX Mutex,
190 IN BOOLEAN Wait)
191 {
192 ASSERT_MUTANT(Mutex);
193
194 /* There's no difference at this level between the two */
195 return KeReleaseMutant(Mutex, MUTANT_INCREMENT, FALSE, Wait);
196 }
197
198 /* EOF */
199