source: libcfa/src/concurrency/locks.hfa @ df932552

ADTast-experimentalpthread-emulationqualifiedEnum
Last change on this file since df932552 was df932552, checked in by caparsons <caparson@…>, 2 years ago

added atomic store/load for spinqueue

  • Property mode set to 100644
File size: 21.9 KB
Line 
1//
2// Cforall Version 1.0.0 Copyright (C) 2021 University of Waterloo
3//
4// The contents of this file are covered under the licence agreement in the
5// file "LICENCE" distributed with Cforall.
6//
7// locks.hfa -- PUBLIC
8// Runtime locks that used with the runtime thread system.
9//
10// Author           : Colby Alexander Parsons
11// Created On       : Thu Jan 21 19:46:50 2021
12// Last Modified By :
13// Last Modified On :
14// Update Count     :
15//
16
17#pragma once
18
19#include <stdbool.h>
20#include <stdio.h>
21
22#include "bits/weakso_locks.hfa"
23#include "containers/queueLockFree.hfa"
24#include "containers/list.hfa"
25
26#include "limits.hfa"
27#include "thread.hfa"
28
29#include "time_t.hfa"
30#include "time.hfa"
31
32//-----------------------------------------------------------------------------
33// Semaphore
34struct semaphore {
35        __spinlock_t lock;
36        int count;
37        __queue_t(thread$) waiting;
38};
39
40void  ?{}(semaphore & this, int count = 1);
41void ^?{}(semaphore & this);
42bool   P (semaphore & this);
43bool   V (semaphore & this);
44bool   V (semaphore & this, unsigned count);
45thread$ * V (semaphore & this, bool );
46
47//----------
48struct single_acquisition_lock {
49        inline blocking_lock;
50};
51
52static inline void  ?{}( single_acquisition_lock & this ) {((blocking_lock &)this){ false, false };}
53static inline void ^?{}( single_acquisition_lock & this ) {}
54static inline void   lock     ( single_acquisition_lock & this ) { lock    ( (blocking_lock &)this ); }
55static inline bool   try_lock ( single_acquisition_lock & this ) { return try_lock( (blocking_lock &)this ); }
56static inline void   unlock   ( single_acquisition_lock & this ) { unlock  ( (blocking_lock &)this ); }
57static inline size_t on_wait  ( single_acquisition_lock & this ) { return on_wait ( (blocking_lock &)this ); }
58static inline void   on_wakeup( single_acquisition_lock & this, size_t v ) { on_wakeup ( (blocking_lock &)this, v ); }
59static inline void   on_notify( single_acquisition_lock & this, struct thread$ * t ) { on_notify( (blocking_lock &)this, t ); }
60
61//----------
62struct owner_lock {
63        inline blocking_lock;
64};
65
66static inline void  ?{}( owner_lock & this ) {((blocking_lock &)this){ true, true };}
67static inline void ^?{}( owner_lock & this ) {}
68static inline void   lock     ( owner_lock & this ) { lock    ( (blocking_lock &)this ); }
69static inline bool   try_lock ( owner_lock & this ) { return try_lock( (blocking_lock &)this ); }
70static inline void   unlock   ( owner_lock & this ) { unlock  ( (blocking_lock &)this ); }
71static inline size_t on_wait  ( owner_lock & this ) { return on_wait ( (blocking_lock &)this ); }
72static inline void   on_wakeup( owner_lock & this, size_t v ) { on_wakeup ( (blocking_lock &)this, v ); }
73static inline void   on_notify( owner_lock & this, struct thread$ * t ) { on_notify( (blocking_lock &)this, t ); }
74
75//-----------------------------------------------------------------------------
76// MCS Lock
77struct mcs_node {
78        mcs_node * volatile next;
79        single_sem sem;
80};
81
82static inline void ?{}(mcs_node & this) { this.next = 0p; }
83
84static inline mcs_node * volatile & ?`next ( mcs_node * node ) {
85        return node->next;
86}
87
88struct mcs_lock {
89        mcs_queue(mcs_node) queue;
90};
91
92static inline void lock(mcs_lock & l, mcs_node & n) {
93        if(push(l.queue, &n))
94                wait(n.sem);
95}
96
97static inline void unlock(mcs_lock & l, mcs_node & n) {
98        mcs_node * next = advance(l.queue, &n);
99        if(next) post(next->sem);
100}
101
102//-----------------------------------------------------------------------------
103// MCS Spin Lock
104// - No recursive acquisition
105// - Needs to be released by owner
106
107struct mcs_spin_node {
108        mcs_spin_node * volatile next;
109        volatile bool locked;
110};
111
112struct mcs_spin_queue {
113        mcs_spin_node * volatile tail;
114};
115
116static inline void ?{}(mcs_spin_node & this) { this.next = 0p; this.locked = true; }
117
118static inline mcs_spin_node * volatile & ?`next ( mcs_spin_node * node ) {
119        return node->next;
120}
121
122struct mcs_spin_lock {
123        mcs_spin_queue queue;
124};
125
126static inline void lock(mcs_spin_lock & l, mcs_spin_node & n) {
127        mcs_spin_node * prev = __atomic_exchange_n(&l.queue.tail, &n, __ATOMIC_SEQ_CST);
128        n.locked = true;
129        if(prev == 0p) return;
130        prev->next = &n;
131        while(__atomic_load_n(&n.locked, __ATOMIC_RELAXED)) Pause();
132}
133
134static inline void unlock(mcs_spin_lock & l, mcs_spin_node & n) {
135        mcs_spin_node * n_ptr = &n;
136        if (__atomic_compare_exchange_n(&l.queue.tail, &n_ptr, 0p, false, __ATOMIC_SEQ_CST, __ATOMIC_SEQ_CST)) return;
137        while (__atomic_load_n(&n.next, __ATOMIC_RELAXED) == 0p) {}
138        n.next->locked = false;
139}
140
141//-----------------------------------------------------------------------------
142// CLH Spinlock
143// - No recursive acquisition
144// - Needs to be released by owner
145
146struct clh_lock {
147        volatile bool * volatile tail;
148};
149
150static inline void  ?{}( clh_lock & this ) { this.tail = malloc(); *this.tail = true; }
151static inline void ^?{}( clh_lock & this ) { free(this.tail); }
152
153static inline void lock(clh_lock & l) {
154        thread$ * curr_thd = active_thread();
155        *(curr_thd->clh_node) = false;
156        volatile bool * prev = __atomic_exchange_n((bool **)(&l.tail), (bool *)(curr_thd->clh_node), __ATOMIC_SEQ_CST);
157        while(!__atomic_load_n(prev, __ATOMIC_ACQUIRE)) Pause();
158        curr_thd->clh_prev = prev;
159}
160
161static inline void unlock(clh_lock & l) {
162        thread$ * curr_thd = active_thread();
163        __atomic_store_n(curr_thd->clh_node, true, __ATOMIC_RELEASE);
164        curr_thd->clh_node = curr_thd->clh_prev;
165}
166
167//-----------------------------------------------------------------------------
168// Linear backoff Spinlock
169struct linear_backoff_then_block_lock {
170        // Spin lock used for mutual exclusion
171        __spinlock_t spinlock;
172
173        // Current thread owning the lock
174        struct thread$ * owner;
175
176        // List of blocked threads
177        dlist( thread$ ) blocked_threads;
178
179        // Used for comparing and exchanging
180        volatile size_t lock_value;
181
182        // used for linear backoff spinning
183        int spin_start;
184        int spin_end;
185        int spin_count;
186
187        // after unsuccessful linear backoff yield this many times
188        int yield_count;
189};
190
191static inline void  ?{}( linear_backoff_then_block_lock & this, int spin_start, int spin_end, int spin_count, int yield_count ) {
192        this.spinlock{};
193        this.blocked_threads{};
194        this.lock_value = 0;
195        this.spin_start = spin_start;
196        this.spin_end = spin_end;
197        this.spin_count = spin_count;
198        this.yield_count = yield_count;
199}
200static inline void  ?{}( linear_backoff_then_block_lock & this ) { this{4, 1024, 16, 0}; }
201static inline void ^?{}( linear_backoff_then_block_lock & this ) {}
202static inline void ?{}( linear_backoff_then_block_lock & this, linear_backoff_then_block_lock this2 ) = void;
203static inline void ?=?( linear_backoff_then_block_lock & this, linear_backoff_then_block_lock this2 ) = void;
204
205static inline bool internal_try_lock(linear_backoff_then_block_lock & this, size_t & compare_val) with(this) {
206        if (__atomic_compare_exchange_n(&lock_value, &compare_val, 1, false, __ATOMIC_ACQUIRE, __ATOMIC_RELAXED)) {
207                owner = active_thread();
208                return true;
209        }
210        return false;
211}
212
213static inline bool try_lock(linear_backoff_then_block_lock & this) { size_t compare_val = 0; return internal_try_lock(this, compare_val); }
214
215static inline bool try_lock_contention(linear_backoff_then_block_lock & this) with(this) {
216        if (__atomic_exchange_n(&lock_value, 2, __ATOMIC_ACQUIRE) == 0) {
217                owner = active_thread();
218                return true;
219        }
220        return false;
221}
222
223static inline bool block(linear_backoff_then_block_lock & this) with(this) {
224        lock( spinlock __cfaabi_dbg_ctx2 );
225        if (lock_value != 2) {
226                unlock( spinlock );
227                return true;
228        }
229        insert_last( blocked_threads, *active_thread() );
230        unlock( spinlock );
231        park( );
232        return true;
233}
234
235static inline void lock(linear_backoff_then_block_lock & this) with(this) {
236        // if owner just return
237        if (active_thread() == owner) return;
238        size_t compare_val = 0;
239        int spin = spin_start;
240        // linear backoff
241        for( ;; ) {
242                compare_val = 0;
243                if (internal_try_lock(this, compare_val)) return;
244                if (2 == compare_val) break;
245                for (int i = 0; i < spin; i++) Pause();
246                if (spin >= spin_end) break;
247                spin += spin;
248        }
249
250        if(2 != compare_val && try_lock_contention(this)) return;
251        // block until signalled
252        while (block(this)) if(try_lock_contention(this)) return;
253}
254
255static inline void unlock(linear_backoff_then_block_lock & this) with(this) {
256        verify(lock_value > 0);
257    owner = 0p;
258    if (__atomic_exchange_n(&lock_value, 0, __ATOMIC_RELEASE) == 1) return;
259        lock( spinlock __cfaabi_dbg_ctx2 );
260        thread$ * t = &try_pop_front( blocked_threads );
261        unlock( spinlock );
262        unpark( t );
263}
264
265static inline void on_notify(linear_backoff_then_block_lock & this, struct thread$ * t ) { unpark(t); }
266static inline size_t on_wait(linear_backoff_then_block_lock & this) { unlock(this); return 0; }
267static inline void on_wakeup(linear_backoff_then_block_lock & this, size_t recursion ) { lock(this); }
268
269//-----------------------------------------------------------------------------
270// Fast Block Lock
271
272// minimal blocking lock
273// - No reacquire for cond var
274// - No recursive acquisition
275// - No ownership
276struct fast_block_lock {
277        // List of blocked threads
278        dlist( thread$ ) blocked_threads;
279
280        // Spin lock used for mutual exclusion
281        __spinlock_t lock;
282
283        // flag showing if lock is held
284        bool held:1;
285
286        #ifdef __CFA_DEBUG__
287        // for deadlock detection
288        struct thread$ * owner;
289        #endif
290};
291
292static inline void  ?{}( fast_block_lock & this ) with(this) {
293        lock{};
294        blocked_threads{};
295        held = false;
296}
297static inline void ^?{}( fast_block_lock & this ) {}
298static inline void ?{}( fast_block_lock & this, fast_block_lock this2 ) = void;
299static inline void ?=?( fast_block_lock & this, fast_block_lock this2 ) = void;
300
301// if this is called recursively IT WILL DEADLOCK!!!!!
302static inline void lock(fast_block_lock & this) with(this) {
303        lock( lock __cfaabi_dbg_ctx2 );
304
305        #ifdef __CFA_DEBUG__
306        assert(!(held && owner == active_thread()));
307        #endif
308        if (held) {
309                insert_last( blocked_threads, *active_thread() );
310                unlock( lock );
311                park( );
312                return;
313        }
314        held = true;
315        #ifdef __CFA_DEBUG__
316        owner = active_thread();
317        #endif
318        unlock( lock );
319}
320
321static inline void unlock(fast_block_lock & this) with(this) {
322        lock( lock __cfaabi_dbg_ctx2 );
323        /* paranoid */ verifyf( held != false, "Attempt to release lock %p that isn't held", &this );
324        thread$ * t = &try_pop_front( blocked_threads );
325        held = ( t ? true : false );
326        #ifdef __CFA_DEBUG__
327        owner = ( t ? t : 0p );
328        #endif
329        unpark( t );
330        unlock( lock );
331}
332
333static inline void on_notify(fast_block_lock & this, struct thread$ * t ) { unpark(t); }
334static inline size_t on_wait(fast_block_lock & this) { unlock(this); return 0; }
335static inline void on_wakeup(fast_block_lock & this, size_t recursion ) { }
336
337//-----------------------------------------------------------------------------
338// simple_owner_lock
339
340// pthread owner lock
341// - reacquire for cond var
342// - recursive acquisition
343// - ownership
344struct simple_owner_lock {
345        // List of blocked threads
346        dlist( thread$ ) blocked_threads;
347
348        // Spin lock used for mutual exclusion
349        __spinlock_t lock;
350
351        // owner showing if lock is held
352        struct thread$ * owner;
353
354        size_t recursion_count;
355};
356
357static inline void  ?{}( simple_owner_lock & this ) with(this) {
358        lock{};
359        blocked_threads{};
360        owner = 0p;
361        recursion_count = 0;
362}
363static inline void ^?{}( simple_owner_lock & this ) {}
364static inline void ?{}( simple_owner_lock & this, simple_owner_lock this2 ) = void;
365static inline void ?=?( simple_owner_lock & this, simple_owner_lock this2 ) = void;
366
367static inline void lock(simple_owner_lock & this) with(this) {
368        if (owner == active_thread()) {
369                recursion_count++;
370                return;
371        }
372        lock( lock __cfaabi_dbg_ctx2 );
373
374        if (owner != 0p) {
375                insert_last( blocked_threads, *active_thread() );
376                unlock( lock );
377                park( );
378                return;
379        }
380        owner = active_thread();
381        recursion_count = 1;
382        unlock( lock );
383}
384
385// TODO: fix duplicate def issue and bring this back
386// void pop_and_set_new_owner( simple_owner_lock & this ) with( this ) {
387        // thread$ * t = &try_pop_front( blocked_threads );
388        // owner = t;
389        // recursion_count = ( t ? 1 : 0 );
390        // unpark( t );
391// }
392
393static inline void unlock(simple_owner_lock & this) with(this) {
394        lock( lock __cfaabi_dbg_ctx2 );
395        /* paranoid */ verifyf( owner != 0p, "Attempt to release lock %p that isn't held", &this );
396        /* paranoid */ verifyf( owner == active_thread(), "Thread %p other than the owner %p attempted to release owner lock %p", owner, active_thread(), &this );
397        // if recursion count is zero release lock and set new owner if one is waiting
398        recursion_count--;
399        if ( recursion_count == 0 ) {
400                // pop_and_set_new_owner( this );
401                thread$ * t = &try_pop_front( blocked_threads );
402                owner = t;
403                recursion_count = ( t ? 1 : 0 );
404                unpark( t );
405        }
406        unlock( lock );
407}
408
409static inline void on_notify(simple_owner_lock & this, struct thread$ * t ) with(this) {
410        lock( lock __cfaabi_dbg_ctx2 );
411        // lock held
412        if ( owner != 0p ) {
413                insert_last( blocked_threads, *t );
414                unlock( lock );
415        }
416        // lock not held
417        else {
418                owner = t;
419                recursion_count = 1;
420                unpark( t );
421                unlock( lock );
422        }
423}
424
425static inline size_t on_wait(simple_owner_lock & this) with(this) {
426        lock( lock __cfaabi_dbg_ctx2 );
427        /* paranoid */ verifyf( owner != 0p, "Attempt to release lock %p that isn't held", &this );
428        /* paranoid */ verifyf( owner == active_thread(), "Thread %p other than the owner %p attempted to release owner lock %p", owner, active_thread(), &this );
429
430        size_t ret = recursion_count;
431
432        // pop_and_set_new_owner( this );
433
434        thread$ * t = &try_pop_front( blocked_threads );
435        owner = t;
436        recursion_count = ( t ? 1 : 0 );
437        unpark( t );
438
439        unlock( lock );
440        return ret;
441}
442
443static inline void on_wakeup(simple_owner_lock & this, size_t recursion ) with(this) { recursion_count = recursion; }
444
445//-----------------------------------------------------------------------------
446// Spin Queue Lock
447
448// - No reacquire for cond var
449// - No recursive acquisition
450// - No ownership
451// - spin lock with no locking/atomics in unlock
452struct spin_queue_lock {
453        // Spin lock used for mutual exclusion
454        mcs_spin_lock lock;
455
456        // flag showing if lock is held
457        volatile bool held;
458
459        #ifdef __CFA_DEBUG__
460        // for deadlock detection
461        struct thread$ * owner;
462        #endif
463};
464
465static inline void  ?{}( spin_queue_lock & this ) with(this) {
466        lock{};
467        held = false;
468}
469static inline void ^?{}( spin_queue_lock & this ) {}
470static inline void ?{}( spin_queue_lock & this, spin_queue_lock this2 ) = void;
471static inline void ?=?( spin_queue_lock & this, spin_queue_lock this2 ) = void;
472
473// if this is called recursively IT WILL DEADLOCK!!!!!
474static inline void lock(spin_queue_lock & this) with(this) {
475        mcs_spin_node node;
476        #ifdef __CFA_DEBUG__
477        assert(!(held && owner == active_thread()));
478        #endif
479        lock( lock, node );
480        while(__atomic_load_n(&held, __ATOMIC_SEQ_CST)) Pause();
481        __atomic_store_n(&held, true, __ATOMIC_SEQ_CST);
482        // printf("locked\n");
483        unlock( lock, node );
484        #ifdef __CFA_DEBUG__
485        owner = active_thread();
486        #endif
487}
488
489static inline void unlock(spin_queue_lock & this) with(this) {
490        // printf("unlocked\n");
491        #ifdef __CFA_DEBUG__
492        owner = 0p;
493        #endif
494        __atomic_store_n(&held, false, __ATOMIC_SEQ_CST);
495}
496
497static inline void on_notify(spin_queue_lock & this, struct thread$ * t ) { unpark(t); }
498static inline size_t on_wait(spin_queue_lock & this) { unlock(this); return 0; }
499static inline void on_wakeup(spin_queue_lock & this, size_t recursion ) { }
500
501
502//-----------------------------------------------------------------------------
503// MCS Block Spin Lock
504
505// - No reacquire for cond var
506// - No recursive acquisition
507// - No ownership
508// - Blocks but first node spins (like spin queue but blocking for not first thd)
509struct mcs_block_spin_lock {
510        // Spin lock used for mutual exclusion
511        mcs_lock lock;
512
513        // flag showing if lock is held
514        volatile bool held;
515
516        #ifdef __CFA_DEBUG__
517        // for deadlock detection
518        struct thread$ * owner;
519        #endif
520};
521
522static inline void  ?{}( mcs_block_spin_lock & this ) with(this) {
523        lock{};
524        held = false;
525}
526static inline void ^?{}( mcs_block_spin_lock & this ) {}
527static inline void ?{}( mcs_block_spin_lock & this, mcs_block_spin_lock this2 ) = void;
528static inline void ?=?( mcs_block_spin_lock & this, mcs_block_spin_lock this2 ) = void;
529
530// if this is called recursively IT WILL DEADLOCK!!!!!
531static inline void lock(mcs_block_spin_lock & this) with(this) {
532        mcs_node node;
533        #ifdef __CFA_DEBUG__
534        assert(!(held && owner == active_thread()));
535        #endif
536        lock( lock, node );
537        while(held) Pause();
538        held = true;
539        unlock( lock, node );
540        #ifdef __CFA_DEBUG__
541        owner = active_thread();
542        #endif
543}
544
545static inline void unlock(mcs_block_spin_lock & this) with(this) {
546        #ifdef __CFA_DEBUG__
547        owner = 0p;
548        #endif
549        held = false;
550}
551
552static inline void on_notify(mcs_block_spin_lock & this, struct thread$ * t ) { unpark(t); }
553static inline size_t on_wait(mcs_block_spin_lock & this) { unlock(this); return 0; }
554static inline void on_wakeup(mcs_block_spin_lock & this, size_t recursion ) { }
555
556//-----------------------------------------------------------------------------
557// Block Spin Lock
558
559// - No reacquire for cond var
560// - No recursive acquisition
561// - No ownership
562// - Blocks but first node spins (like spin queue but blocking for not first thd)
563struct block_spin_lock {
564        // Spin lock used for mutual exclusion
565        fast_block_lock lock;
566
567        // flag showing if lock is held
568        volatile bool held;
569
570        #ifdef __CFA_DEBUG__
571        // for deadlock detection
572        struct thread$ * owner;
573        #endif
574};
575
576static inline void  ?{}( block_spin_lock & this ) with(this) {
577        lock{};
578        held = false;
579}
580static inline void ^?{}( block_spin_lock & this ) {}
581static inline void ?{}( block_spin_lock & this, block_spin_lock this2 ) = void;
582static inline void ?=?( block_spin_lock & this, block_spin_lock this2 ) = void;
583
584// if this is called recursively IT WILL DEADLOCK!!!!!
585static inline void lock(block_spin_lock & this) with(this) {
586        #ifdef __CFA_DEBUG__
587        assert(!(held && owner == active_thread()));
588        #endif
589        lock( lock );
590        while(held) Pause();
591        held = true;
592        unlock( lock );
593        #ifdef __CFA_DEBUG__
594        owner = active_thread();
595        #endif
596}
597
598static inline void unlock(block_spin_lock & this) with(this) {
599        #ifdef __CFA_DEBUG__
600        owner = 0p;
601        #endif
602        held = false;
603}
604
605static inline void on_notify(block_spin_lock & this, struct thread$ * t ) { unpark(t); }
606static inline size_t on_wait(block_spin_lock & this) { unlock(this); return 0; }
607static inline void on_wakeup(block_spin_lock & this, size_t recursion ) { }
608
609//-----------------------------------------------------------------------------
610// is_blocking_lock
611trait is_blocking_lock(L & | sized(L)) {
612        // For synchronization locks to use when acquiring
613        void on_notify( L &, struct thread$ * );
614
615        // For synchronization locks to use when releasing
616        size_t on_wait( L & );
617
618        // to set recursion count after getting signalled;
619        void on_wakeup( L &, size_t recursion );
620};
621
622//-----------------------------------------------------------------------------
623// // info_thread
624// // the info thread is a wrapper around a thread used
625// // to store extra data for use in the condition variable
626forall(L & | is_blocking_lock(L)) {
627        struct info_thread;
628
629        // // for use by sequence
630        // info_thread(L) *& Back( info_thread(L) * this );
631        // info_thread(L) *& Next( info_thread(L) * this );
632}
633
634//-----------------------------------------------------------------------------
635// Synchronization Locks
636forall(L & | is_blocking_lock(L)) {
637
638        //-----------------------------------------------------------------------------
639        // condition_variable
640
641        // The multi-tool condition variable
642        // - can pass timeouts to wait for either a signal or timeout
643        // - can wait without passing a lock
644        // - can have waiters reacquire different locks while waiting on the same cond var
645        // - has shadow queue
646        // - can be signalled outside of critical sections with no locks held
647        struct condition_variable {
648                // Spin lock used for mutual exclusion
649                __spinlock_t lock;
650
651                // List of blocked threads
652                dlist( info_thread(L) ) blocked_threads;
653
654                // Count of current blocked threads
655                int count;
656        };
657
658
659        void  ?{}( condition_variable(L) & this );
660        void ^?{}( condition_variable(L) & this );
661
662        bool notify_one( condition_variable(L) & this );
663        bool notify_all( condition_variable(L) & this );
664
665        uintptr_t front( condition_variable(L) & this );
666
667        bool empty  ( condition_variable(L) & this );
668        int  counter( condition_variable(L) & this );
669
670        void wait( condition_variable(L) & this );
671        void wait( condition_variable(L) & this, uintptr_t info );
672        bool wait( condition_variable(L) & this, Duration duration );
673        bool wait( condition_variable(L) & this, uintptr_t info, Duration duration );
674
675        void wait( condition_variable(L) & this, L & l );
676        void wait( condition_variable(L) & this, L & l, uintptr_t info );
677        bool wait( condition_variable(L) & this, L & l, Duration duration );
678        bool wait( condition_variable(L) & this, L & l, uintptr_t info, Duration duration );
679
680        //-----------------------------------------------------------------------------
681        // fast_cond_var
682
683        // The trimmed and slim condition variable
684        // - no internal lock so you must hold a lock while using this cond var
685        // - signalling without holding branded lock is UNSAFE!
686        // - only allows usage of one lock, cond var is branded after usage
687
688        struct fast_cond_var {
689                // List of blocked threads
690                dlist( info_thread(L) ) blocked_threads;
691                #ifdef __CFA_DEBUG__
692                L * lock_used;
693                #endif
694        };
695
696        void  ?{}( fast_cond_var(L) & this );
697        void ^?{}( fast_cond_var(L) & this );
698
699        bool notify_one( fast_cond_var(L) & this );
700        bool notify_all( fast_cond_var(L) & this );
701
702        uintptr_t front( fast_cond_var(L) & this );
703        bool empty  ( fast_cond_var(L) & this );
704
705        void wait( fast_cond_var(L) & this, L & l );
706        void wait( fast_cond_var(L) & this, L & l, uintptr_t info );
707
708
709        //-----------------------------------------------------------------------------
710        // pthread_cond_var
711        //
712        // - cond var with minimal footprint
713        // - supports operations needed for phthread cond
714
715        struct pthread_cond_var {
716                dlist( info_thread(L) ) blocked_threads;
717                __spinlock_t lock;
718        };
719
720        void  ?{}( pthread_cond_var(L) & this );
721        void ^?{}( pthread_cond_var(L) & this );
722
723        bool notify_one( pthread_cond_var(L) & this );
724        bool notify_all( pthread_cond_var(L) & this );
725
726        uintptr_t front( pthread_cond_var(L) & this );
727        bool empty ( pthread_cond_var(L) & this );
728
729        void wait( pthread_cond_var(L) & this, L & l );
730        void wait( pthread_cond_var(L) & this, L & l, uintptr_t info );
731        bool wait( pthread_cond_var(L) & this, L & l, timespec t );
732        bool wait( pthread_cond_var(L) & this, L & l, uintptr_t info, timespec t );
733}
Note: See TracBrowser for help on using the repository browser.