xref: /reactos/sdk/lib/rtl/condvar.c (revision 3e1f4074)
1 /*
2  * COPYRIGHT:         See COPYING in the top level directory
3  * PROJECT:           ReactOS system libraries
4  * PURPOSE:           Condition Variable Routines
5  * PROGRAMMERS:       Thomas Weidenmueller <w3seek@reactos.com>
6  *                    Stephan A. R�ger
7  */
8 
9 /* NOTE: This functionality can be optimized for releasing single
10    threads or for releasing all waiting threads at once. This
11    implementation is optimized for releasing a single thread at a time.
12    It wakes up sleeping threads in FIFO order. */
13 
14 /* INCLUDES ******************************************************************/
15 
16 #include <rtl_vista.h>
17 
18 #define NDEBUG
19 #include <debug.h>
20 
21 /* INTERNAL TYPES ************************************************************/
22 
23 #define COND_VAR_UNUSED_FLAG         ((ULONG_PTR)1)
24 #define COND_VAR_LOCKED_FLAG         ((ULONG_PTR)2)
25 #define COND_VAR_FLAGS_MASK          ((ULONG_PTR)3)
26 #define COND_VAR_ADDRESS_MASK        (~COND_VAR_FLAGS_MASK)
27 
28 typedef struct _COND_VAR_WAIT_ENTRY
29 {
30     /* ListEntry must have an alignment of at least 32-bits, since we
31        want COND_VAR_ADDRESS_MASK to cover all of the address. */
32     LIST_ENTRY ListEntry;
33     PVOID WaitKey;
34     BOOLEAN ListRemovalHandled;
35 } COND_VAR_WAIT_ENTRY, * PCOND_VAR_WAIT_ENTRY;
36 
37 #define CONTAINING_COND_VAR_WAIT_ENTRY(address, field) \
38     CONTAINING_RECORD(address, COND_VAR_WAIT_ENTRY, field)
39 
40 /* GLOBALS *******************************************************************/
41 
42 static HANDLE CondVarKeyedEventHandle = NULL;
43 
44 /* INTERNAL FUNCTIONS ********************************************************/
45 
46 FORCEINLINE
47 ULONG_PTR
48 InternalCmpXChgCondVarAcq(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable,
49                           IN ULONG_PTR Exchange,
50                           IN ULONG_PTR Comperand)
51 {
52     return (ULONG_PTR)InterlockedCompareExchangePointerAcquire(&ConditionVariable->Ptr,
53                                                                (PVOID)Exchange,
54                                                                (PVOID)Comperand);
55 }
56 
57 FORCEINLINE
58 ULONG_PTR
59 InternalCmpXChgCondVarRel(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable,
60                           IN ULONG_PTR Exchange,
61                           IN ULONG_PTR Comperand)
62 {
63     return (ULONG_PTR)InterlockedCompareExchangePointerRelease(&ConditionVariable->Ptr,
64                                                                (PVOID)Exchange,
65                                                                (PVOID)Comperand);
66 }
67 
68 FORCEINLINE
69 BOOLEAN *
70 InternalGetListRemovalHandledFlag(IN PCOND_VAR_WAIT_ENTRY Entry)
71 {
72     return (BOOLEAN *)&Entry->ListRemovalHandled;
73 }
74 
75 static
76 PCOND_VAR_WAIT_ENTRY
77 InternalLockCondVar(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable,
78                     IN PCOND_VAR_WAIT_ENTRY InsertEntry OPTIONAL,
79                     IN BOOLEAN * AbortIfLocked OPTIONAL)
80 {
81     /* InsertEntry and AbortIfLocked may be NULL on entry. This routine
82        will return NULL if the lock was not acquired. Otherwise it has
83        successfully acquired the lock and the return value is a valid
84        reference to the list head associated with ConditionVariable.
85        The caller must in this case call InternalUnlockCondVar later
86        in order to unlock the condition variable.
87 
88        If InsertEntry is NULL and there are no entries on the list, this
89        routine will not acquire the lock and return NULL. If InsertEntry
90        is not NULL this routine ensures that InsertEntry will be on the
91        list when it returns successfully.
92 
93        If the lock is owned by another thread and AbortIfLocked is NULL,
94        this routine will block until it acquires the lock. If AbortIfLocked
95        is not NULL and the lock is owned by another thread, this routine
96        will periodically check if *AbortIfLocked is nonzero and if so, will
97        return NULL instead of continuing the wait. */
98 
99     ULONG_PTR OldVal = (ULONG_PTR)ConditionVariable->Ptr;
100 
101     for (;;)
102     {
103         ULONG_PTR NewVal, LockRes;
104         PLIST_ENTRY OldListHead;
105 
106         if (OldVal & COND_VAR_LOCKED_FLAG)
107         {
108             /* The locked flag is set, indicating someone else currently
109                holds the lock. We'll spin until this flag becomes
110                clear or we're asked to abort. */
111             YieldProcessor();
112 
113             if ((AbortIfLocked != NULL) && *AbortIfLocked)
114             {
115                 /* The caller wants us to abort in this case. */
116                 return NULL;
117             }
118 
119             /* Refresh OldVal and try again. */
120             OldVal = *(ULONG_PTR *)&ConditionVariable->Ptr;
121             continue;
122         }
123 
124         /* Retrieve the list head currently associated with the
125            condition variable. */
126         OldListHead = (PLIST_ENTRY)(OldVal & COND_VAR_ADDRESS_MASK);
127         if (InsertEntry == NULL)
128         {
129             /* The caller doesn't want to put any entry on the list. */
130             if (OldListHead == NULL)
131             {
132                 /* The list is empty, so there is nothing to lock. */
133                 return NULL;
134             }
135 
136             /* The list isn't empty. In this case we need to preserve
137                all of OldVal. */
138             NewVal = OldVal;
139         }
140         else
141         {
142             /* Let InsertEntry be the new list head. Preserve only the
143                bits inside the COND_VAR_FLAGS_MASK range. */
144             NewVal = ((OldVal & COND_VAR_FLAGS_MASK) |
145                       (ULONG_PTR)&InsertEntry->ListEntry);
146         }
147 
148         /* Set the flag that indicates someone is holding the lock and
149            try to update the condition variable thread-safe. */
150         NewVal |= COND_VAR_LOCKED_FLAG;
151         LockRes = InternalCmpXChgCondVarAcq(ConditionVariable, NewVal, OldVal);
152         if (LockRes == OldVal)
153         {
154             /* We successfully updated ConditionVariable the way we
155                wanted and now hold the lock. */
156             if (InsertEntry == NULL)
157             {
158                 /* We know that OldVal contains a valid address in
159                    this case. */
160                 ASSERT(OldListHead != NULL);
161                 return CONTAINING_COND_VAR_WAIT_ENTRY(OldListHead, ListEntry);
162             }
163 
164             /* InsertEntry is not on the list yet, so add it. In any
165                case InsertEntry will be the new list head. */
166             if (OldListHead == NULL)
167             {
168                 /* List was empty before. */
169                 InitializeListHead(&InsertEntry->ListEntry);
170             }
171             else
172             {
173                 /* Make InsertEntry the last entry of the old list.
174                    As InsertEntry will take the role as new list head,
175                    OldListHead will become the second entry (InsertEntry->Flink)
176                    on the new list. */
177                 InsertTailList(OldListHead, &InsertEntry->ListEntry);
178             }
179 
180             return InsertEntry;
181         }
182 
183         /* We didn't manage to update ConditionVariable, so try again. */
184         OldVal = LockRes;
185     }
186 }
187 
188 static
189 VOID
190 InternalUnlockCondVar(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable,
191                       IN PCOND_VAR_WAIT_ENTRY RemoveEntry OPTIONAL)
192 {
193     /* This routine assumes that the lock is being held on entry.
194        RemoveEntry may be NULL. If it is not NULL, this routine
195        assumes that RemoveEntry is on the list and will remove it
196        before releasing the lock. */
197     ULONG_PTR OldVal = (ULONG_PTR)ConditionVariable->Ptr;
198     PLIST_ENTRY NewHeadEntry;
199 
200     ASSERT((OldVal & COND_VAR_LOCKED_FLAG) &&
201            (OldVal & COND_VAR_ADDRESS_MASK));
202 
203     NewHeadEntry = (PLIST_ENTRY)(OldVal & COND_VAR_ADDRESS_MASK);
204     if (RemoveEntry != NULL)
205     {
206         /* We have to drop RemoveEntry from the list. */
207         if (&RemoveEntry->ListEntry == NewHeadEntry)
208         {
209             /* RemoveEntry is the list head. */
210             if (!IsListEmpty(NewHeadEntry))
211             {
212                 /* The second entry in the list will become the new
213                    list head. It's from the thread that arrived
214                    right before the owner of RemoveEntry. */
215                 NewHeadEntry = NewHeadEntry->Flink;
216                 RemoveEntryList(&RemoveEntry->ListEntry);
217             }
218             else
219             {
220                 /* The list will be empty, so discard the list. */
221                 NewHeadEntry = NULL;
222             }
223         }
224         else
225         {
226             /* RemoveEntry is not the list head. The current list head
227                will remain. */
228             RemoveEntryList(&RemoveEntry->ListEntry);
229         }
230 
231         /* Indicate to the owner of RemoveEntry that the entry
232            was removed from the list. RemoveEntry may not be touched
233            from here on. We don't use volatile semantics here since
234            the cache will anyway be flushed soon when we update
235            ConditionVariable. */
236         RemoveEntry->ListRemovalHandled = TRUE;
237     }
238 
239     /* Now unlock thread-safe, while preserving any flags within the
240        COND_VAR_FLAGS_MASK range except for COND_VAR_LOCKED_FLAG. */
241     for (;;)
242     {
243         ULONG_PTR NewVal = ((OldVal & (COND_VAR_FLAGS_MASK ^ COND_VAR_LOCKED_FLAG)) |
244                             (ULONG_PTR)NewHeadEntry);
245         ULONG_PTR LockRes = InternalCmpXChgCondVarRel(ConditionVariable, NewVal, OldVal);
246         if (LockRes == OldVal)
247         {
248             /* We unlocked. */
249             break;
250         }
251 
252         /* Try again. */
253         OldVal = LockRes;
254     }
255 }
256 
257 static
258 VOID
259 InternalWake(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable,
260              IN BOOLEAN ReleaseAll)
261 {
262     /* If ReleaseAll is zero on entry, one thread at most will be woken.
263        Otherwise all waiting threads are woken. Wakeups happen in FIFO
264        order. */
265     PCOND_VAR_WAIT_ENTRY CONST HeadEntry = InternalLockCondVar(ConditionVariable, NULL, NULL);
266     PCOND_VAR_WAIT_ENTRY Entry;
267     PCOND_VAR_WAIT_ENTRY NextEntry;
268     LARGE_INTEGER Timeout;
269     PCOND_VAR_WAIT_ENTRY RemoveOnUnlockEntry;
270 
271     ASSERT(CondVarKeyedEventHandle != NULL);
272 
273     if (HeadEntry == NULL)
274     {
275         /* There is noone there to wake up. In this case do nothing
276            and return immediately. We don't stockpile releases. */
277         return;
278     }
279 
280     Timeout.QuadPart = 0;
281     RemoveOnUnlockEntry = NULL;
282 
283     /* Release sleeping threads. We will iterate from the last entry on
284        the list to the first. Note that the loop condition is always
285        true for the initial test. */
286     for (Entry = CONTAINING_COND_VAR_WAIT_ENTRY(HeadEntry->ListEntry.Blink, ListEntry);
287          Entry != NULL;
288          Entry = NextEntry)
289     {
290         NTSTATUS Status;
291 
292         if (HeadEntry == Entry)
293         {
294             /* After the current entry we've iterated through the
295                entire list in backward direction. Then exit.*/
296             NextEntry = NULL;
297         }
298         else
299         {
300             /* Store away the next reference right now, since we may
301                not touch Entry anymore at the end of the block. */
302             NextEntry = CONTAINING_COND_VAR_WAIT_ENTRY(Entry->ListEntry.Blink, ListEntry);
303         }
304 
305         /* Wake the thread associated with this event. We will
306            immediately return if we failed (zero timeout). */
307         Status = NtReleaseKeyedEvent(CondVarKeyedEventHandle,
308                                      &Entry->WaitKey,
309                                      FALSE,
310                                      &Timeout);
311 
312         if (!NT_SUCCESS(Status))
313         {
314             /* We failed to wake a thread. We'll keep trying. */
315             ASSERT(STATUS_INVALID_HANDLE != Status);
316             continue;
317         }
318 
319         /* We've woken a thread and will make sure this thread
320            is removed from the list. */
321         if (HeadEntry == Entry)
322         {
323             /* This is the list head. We can't remove it as easily as
324                other entries and will pass it to the unlock routine
325                later (we will exit the loop after this round anyway). */
326             RemoveOnUnlockEntry = HeadEntry;
327         }
328         else
329         {
330             /* We can remove the entry right away. */
331             RemoveEntryList(&Entry->ListEntry);
332 
333             /* Now tell the woken thread that removal from the list was
334                already taken care of here so that this thread can resume
335                its normal operation more quickly. We may not touch
336                Entry after signaling this, since it may lie in invalid
337                memory from there on. */
338             *InternalGetListRemovalHandledFlag(Entry) = TRUE;
339         }
340 
341         if (!ReleaseAll)
342         {
343             /* We've successfully woken one thread as the caller
344                demanded. */
345             break;
346         }
347     }
348 
349     InternalUnlockCondVar(ConditionVariable, RemoveOnUnlockEntry);
350 }
351 
352 VOID
353 NTAPI
354 RtlAcquireSRWLockExclusive(IN OUT PRTL_SRWLOCK SRWLock);
355 VOID
356 NTAPI
357 RtlAcquireSRWLockShared(IN OUT PRTL_SRWLOCK SRWLock);
358 VOID
359 NTAPI
360 RtlReleaseSRWLockExclusive(IN OUT PRTL_SRWLOCK SRWLock);
361 VOID
362 NTAPI
363 RtlReleaseSRWLockShared(IN OUT PRTL_SRWLOCK SRWLock);
364 
365 static
366 NTSTATUS
367 InternalSleep(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable,
368               IN OUT PRTL_CRITICAL_SECTION CriticalSection OPTIONAL,
369               IN OUT PRTL_SRWLOCK SRWLock OPTIONAL,
370               IN ULONG SRWFlags,
371               IN const LARGE_INTEGER * TimeOut OPTIONAL)
372 {
373     /* Either CriticalSection or SRWLock must be NULL, but not both.
374        These caller provided lock must be held on entry and will be
375        held again on return. */
376 
377     COND_VAR_WAIT_ENTRY OwnEntry;
378     NTSTATUS Status;
379 
380     ASSERT(CondVarKeyedEventHandle != NULL);
381     ASSERT((CriticalSection == NULL) != (SRWLock == NULL));
382 
383     RtlZeroMemory(&OwnEntry, sizeof(OwnEntry));
384 
385     /* Put OwnEntry on the list. */
386     InternalLockCondVar(ConditionVariable, &OwnEntry, NULL);
387     InternalUnlockCondVar(ConditionVariable, NULL);
388 
389     /* We can now drop the caller provided lock as a preparation for
390        going to sleep. */
391     if (CriticalSection == NULL)
392     {
393         if (0 == (RTL_CONDITION_VARIABLE_LOCKMODE_SHARED & SRWFlags))
394         {
395             RtlReleaseSRWLockExclusive(SRWLock);
396         }
397         else
398         {
399             RtlReleaseSRWLockShared(SRWLock);
400         }
401     }
402     else
403     {
404         RtlLeaveCriticalSection(CriticalSection);
405     }
406 
407     /* Now sleep using the caller provided timeout. */
408     Status = NtWaitForKeyedEvent(CondVarKeyedEventHandle,
409                                  &OwnEntry.WaitKey,
410                                  FALSE,
411                                  (PLARGE_INTEGER)TimeOut);
412 
413     ASSERT(STATUS_INVALID_HANDLE != Status);
414 
415     if (!*InternalGetListRemovalHandledFlag(&OwnEntry))
416     {
417         /* Remove OwnEntry from the list again, since it still seems to
418            be on the list. We will know for sure once we've acquired
419            the lock. */
420         if (InternalLockCondVar(ConditionVariable,
421                                 NULL,
422                                 InternalGetListRemovalHandledFlag(&OwnEntry)))
423         {
424             /* Unlock and potentially remove OwnEntry. Self-removal is
425                usually only necessary when a timeout occurred. */
426             InternalUnlockCondVar(ConditionVariable,
427                                   !OwnEntry.ListRemovalHandled ?
428                                   &OwnEntry : NULL);
429         }
430     }
431 
432 #ifdef _DEBUG
433     /* Clear OwnEntry to aid in detecting bugs. */
434     RtlZeroMemory(&OwnEntry, sizeof(OwnEntry));
435 #endif
436 
437     /* Reacquire the caller provided lock, as we are about to return. */
438     if (CriticalSection == NULL)
439     {
440         if (0 == (RTL_CONDITION_VARIABLE_LOCKMODE_SHARED & SRWFlags))
441         {
442             RtlAcquireSRWLockExclusive(SRWLock);
443         }
444         else
445         {
446             RtlAcquireSRWLockShared(SRWLock);
447         }
448     }
449     else
450     {
451         RtlEnterCriticalSection(CriticalSection);
452     }
453 
454     /* Return whatever NtWaitForKeyedEvent returned. */
455     return Status;
456 }
457 
458 VOID
459 RtlpInitializeKeyedEvent(VOID)
460 {
461     ASSERT(CondVarKeyedEventHandle == NULL);
462     NtCreateKeyedEvent(&CondVarKeyedEventHandle, EVENT_ALL_ACCESS, NULL, 0);
463 }
464 
465 VOID
466 RtlpCloseKeyedEvent(VOID)
467 {
468     ASSERT(CondVarKeyedEventHandle != NULL);
469     NtClose(CondVarKeyedEventHandle);
470     CondVarKeyedEventHandle = NULL;
471 }
472 
473 /* EXPORTED FUNCTIONS ********************************************************/
474 
475 VOID
476 NTAPI
477 RtlInitializeConditionVariable(OUT PRTL_CONDITION_VARIABLE ConditionVariable)
478 {
479     ConditionVariable->Ptr = NULL;
480 }
481 
482 VOID
483 NTAPI
484 RtlWakeConditionVariable(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable)
485 {
486     InternalWake(ConditionVariable, FALSE);
487 }
488 
489 VOID
490 NTAPI
491 RtlWakeAllConditionVariable(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable)
492 {
493     InternalWake(ConditionVariable, TRUE);
494 }
495 
496 NTSTATUS
497 NTAPI
498 RtlSleepConditionVariableCS(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable,
499                             IN OUT PRTL_CRITICAL_SECTION CriticalSection,
500                             IN const LARGE_INTEGER * TimeOut OPTIONAL)
501 {
502     return InternalSleep(ConditionVariable,
503                          CriticalSection,
504                          (PRTL_SRWLOCK)NULL,
505                          0,
506                          TimeOut);
507 }
508 
509 NTSTATUS
510 NTAPI
511 RtlSleepConditionVariableSRW(IN OUT PRTL_CONDITION_VARIABLE ConditionVariable,
512                              IN OUT PRTL_SRWLOCK SRWLock,
513                              IN const LARGE_INTEGER * TimeOut OPTIONAL,
514                              IN ULONG Flags)
515 {
516     return InternalSleep(ConditionVariable,
517                          (PRTL_CRITICAL_SECTION)NULL,
518                          SRWLock,
519                          Flags,
520                          TimeOut);
521 }
522 
523 /* EOF */
524