source: libcfa/src/bits/locks.hfa @ b7d6a36

ADTarm-ehast-experimentalenumforall-pointer-decayjacob/cs343-translationnew-astnew-ast-unique-exprpthread-emulationqualifiedEnum
Last change on this file since b7d6a36 was e3fea42, checked in by Peter A. Buhr <pabuhr@…>, 5 years ago

change "const char *" to "const char []"

  • Property mode set to 100644
File size: 3.8 KB
RevLine 
[ea7d2b0]1//
2// Cforall Version 1.0.0 Copyright (C) 2016 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//
[73abe95]7// bits/locks.hfa -- Fast internal locks.
[ea7d2b0]8//
9// Author           : Thierry Delisle
10// Created On       : Tue Oct 31 15:14:38 2017
[b158d8f]11// Last Modified By : Peter A. Buhr
[e3fea42]12// Last Modified On : Tue Feb  4 13:03:19 2020
13// Update Count     : 11
[ea7d2b0]14//
15
16#pragma once
17
[73abe95]18#include "bits/debug.hfa"
19#include "bits/defs.hfa"
[ea8b2f7]20#include <assert.h>
21
22#ifdef __cforall
23        extern "C" {
24                #include <pthread.h>
25        }
26#endif
[ea7d2b0]27
28// pause to prevent excess processor bus usage
29#if defined( __sparc )
30        #define Pause() __asm__ __volatile__ ( "rd %ccr,%g0" )
31#elif defined( __i386 ) || defined( __x86_64 )
32        #define Pause() __asm__ __volatile__ ( "pause" : : : )
[b158d8f]33#elif defined( __ARM_ARCH )
34        #define Pause() __asm__ __volatile__ ( "nop" : : : )
[ea7d2b0]35#else
36        #error unsupported architecture
37#endif
38
39struct __spinlock_t {
[13073be]40        // Wrap in struct to prevent false sharing with debug info
[3aeee3c]41        volatile bool lock;
[ea7d2b0]42        #ifdef __CFA_DEBUG__
[13073be]43                // previous function to acquire the lock
[ea7d2b0]44                const char * prev_name;
[13073be]45                // previous thread to acquire the lock
[ea7d2b0]46                void* prev_thrd;
47        #endif
[3aeee3c]48};
[ea7d2b0]49
[0cf5b79]50#ifdef __cforall
[dbe9b08]51        extern "C" {
[2026bb6]52                extern void disable_interrupts() OPTIONAL_THREAD;
53                extern void enable_interrupts_noPoll() OPTIONAL_THREAD;
[1997b4e]54
55                #ifdef __CFA_DEBUG__
[e3fea42]56                        void __cfaabi_dbg_record(__spinlock_t & this, const char prev_name[]);
[1997b4e]57                #else
58                        #define __cfaabi_dbg_record(x, y)
59                #endif
[dbe9b08]60        }
61
[ea7d2b0]62        extern void yield( unsigned int );
63
64        static inline void ?{}( __spinlock_t & this ) {
65                this.lock = 0;
66        }
67
68        // Lock the spinlock, return false if already acquired
[93c2e0a]69        static inline bool try_lock  ( __spinlock_t & this __cfaabi_dbg_ctx_param2 ) {
70                bool result = (this.lock == 0) && (__atomic_test_and_set( &this.lock, __ATOMIC_ACQUIRE ) == 0);
[dbe9b08]71                if( result ) {
72                        disable_interrupts();
[9181f1d]73                        __cfaabi_dbg_record( this, caller );
[dbe9b08]74                }
[ea7d2b0]75                return result;
76        }
77
78        // Lock the spinlock, spin if already acquired
[36982fc]79        static inline void lock( __spinlock_t & this __cfaabi_dbg_ctx_param2 ) {
[ea7d2b0]80                #ifndef NOEXPBACK
81                        enum { SPIN_START = 4, SPIN_END = 64 * 1024, };
82                        unsigned int spin = SPIN_START;
83                #endif
84
85                for ( unsigned int i = 1;; i += 1 ) {
[13073be]86                        if ( (this.lock == 0) && (__atomic_test_and_set( &this.lock, __ATOMIC_ACQUIRE ) == 0) ) break;
[ea7d2b0]87                        #ifndef NOEXPBACK
88                                // exponential spin
89                                for ( volatile unsigned int s = 0; s < spin; s += 1 ) Pause();
90
91                                // slowly increase by powers of 2
92                                if ( i % 64 == 0 ) spin += spin;
93
94                                // prevent overflow
95                                if ( spin > SPIN_END ) spin = SPIN_START;
96                        #else
97                                Pause();
98                        #endif
99                }
[dbe9b08]100                disable_interrupts();
[9181f1d]101                __cfaabi_dbg_record( this, caller );
[ea7d2b0]102        }
103
104        static inline void unlock( __spinlock_t & this ) {
[dbe9b08]105                enable_interrupts_noPoll();
[13073be]106                __atomic_clear( &this.lock, __ATOMIC_RELEASE );
[ea7d2b0]107        }
[ea8b2f7]108
109
110        #ifdef __CFA_WITH_VERIFY__
111                extern bool __cfaabi_dbg_in_kernel();
112        #endif
113
114        struct __bin_sem_t {
[85b1deb]115                bool                    signaled;
116                pthread_mutex_t         lock;
117                pthread_cond_t          cond;
[ea8b2f7]118        };
119
120        static inline void ?{}(__bin_sem_t & this) with( this ) {
[85b1deb]121                signaled = false;
[ea8b2f7]122                pthread_mutex_init(&lock, NULL);
123                pthread_cond_init (&cond, NULL);
124        }
125
126        static inline void ^?{}(__bin_sem_t & this) with( this ) {
127                pthread_mutex_destroy(&lock);
128                pthread_cond_destroy (&cond);
129        }
130
131        static inline void wait(__bin_sem_t & this) with( this ) {
132                verify(__cfaabi_dbg_in_kernel());
133                pthread_mutex_lock(&lock);
[85b1deb]134                        if(!signaled) {   // this must be a loop, not if!
135                                pthread_cond_wait(&cond, &lock);
136                        }
137                        signaled = false;
[ea8b2f7]138                pthread_mutex_unlock(&lock);
139        }
140
141        static inline void post(__bin_sem_t & this) with( this ) {
142                verify(__cfaabi_dbg_in_kernel());
[85b1deb]143
[ea8b2f7]144                pthread_mutex_lock(&lock);
[85b1deb]145                        bool needs_signal = !signaled;
146                        signaled = true;
[ea8b2f7]147                pthread_mutex_unlock(&lock);
[85b1deb]148
149                if (needs_signal)
[ea8b2f7]150                        pthread_cond_signal(&cond);
[85b1deb]151        }
[b158d8f]152#endif
Note: See TracBrowser for help on using the repository browser.