xref: /reactos/ntoskrnl/ke/mutex.c (revision cdf90707)
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
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
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
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
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
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