1
1
use crate :: sync:: atomic:: {
2
- AtomicU32 ,
2
+ self ,
3
3
Ordering :: { Acquire , Relaxed , Release } ,
4
4
} ;
5
5
use crate :: sys:: futex:: { futex_wait, futex_wake} ;
6
6
7
+ cfg_if:: cfg_if! {
8
+ if #[ cfg( windows) ] {
9
+ // On Windows we can have a smol futex
10
+ type Atomic = atomic:: AtomicU8 ;
11
+ type State = u8 ;
12
+ } else {
13
+ type Atomic = atomic:: AtomicU32 ;
14
+ type State = u32 ;
15
+ }
16
+ }
17
+
7
18
pub struct Mutex {
8
- /// 0: unlocked
9
- /// 1: locked, no other threads waiting
10
- /// 2: locked, and other threads waiting (contended)
11
- futex : AtomicU32 ,
19
+ futex : Atomic ,
12
20
}
13
21
22
+ const UNLOCKED : State = 0 ;
23
+ const LOCKED : State = 1 ; // locked, no other threads waiting
24
+ const CONTENDED : State = 2 ; // locked, and other threads waiting (contended)
25
+
14
26
impl Mutex {
15
27
#[ inline]
16
28
pub const fn new ( ) -> Self {
17
- Self { futex : AtomicU32 :: new ( 0 ) }
29
+ Self { futex : Atomic :: new ( UNLOCKED ) }
18
30
}
19
31
20
32
#[ inline]
21
33
pub fn try_lock ( & self ) -> bool {
22
- self . futex . compare_exchange ( 0 , 1 , Acquire , Relaxed ) . is_ok ( )
34
+ self . futex . compare_exchange ( UNLOCKED , LOCKED , Acquire , Relaxed ) . is_ok ( )
23
35
}
24
36
25
37
#[ inline]
26
38
pub fn lock ( & self ) {
27
- if self . futex . compare_exchange ( 0 , 1 , Acquire , Relaxed ) . is_err ( ) {
39
+ if self . futex . compare_exchange ( UNLOCKED , LOCKED , Acquire , Relaxed ) . is_err ( ) {
28
40
self . lock_contended ( ) ;
29
41
}
30
42
}
@@ -36,40 +48,40 @@ impl Mutex {
36
48
37
49
// If it's unlocked now, attempt to take the lock
38
50
// without marking it as contended.
39
- if state == 0 {
40
- match self . futex . compare_exchange ( 0 , 1 , Acquire , Relaxed ) {
51
+ if state == UNLOCKED {
52
+ match self . futex . compare_exchange ( UNLOCKED , LOCKED , Acquire , Relaxed ) {
41
53
Ok ( _) => return , // Locked!
42
54
Err ( s) => state = s,
43
55
}
44
56
}
45
57
46
58
loop {
47
59
// Put the lock in contended state.
48
- // We avoid an unnecessary write if it as already set to 2 ,
60
+ // We avoid an unnecessary write if it as already set to CONTENDED ,
49
61
// to be friendlier for the caches.
50
- if state != 2 && self . futex . swap ( 2 , Acquire ) == 0 {
51
- // We changed it from 0 to 2 , so we just successfully locked it.
62
+ if state != CONTENDED && self . futex . swap ( CONTENDED , Acquire ) == UNLOCKED {
63
+ // We changed it from UNLOCKED to CONTENDED , so we just successfully locked it.
52
64
return ;
53
65
}
54
66
55
- // Wait for the futex to change state, assuming it is still 2 .
56
- futex_wait ( & self . futex , 2 , None ) ;
67
+ // Wait for the futex to change state, assuming it is still CONTENDED .
68
+ futex_wait ( & self . futex , CONTENDED , None ) ;
57
69
58
70
// Spin again after waking up.
59
71
state = self . spin ( ) ;
60
72
}
61
73
}
62
74
63
- fn spin ( & self ) -> u32 {
75
+ fn spin ( & self ) -> State {
64
76
let mut spin = 100 ;
65
77
loop {
66
78
// We only use `load` (and not `swap` or `compare_exchange`)
67
79
// while spinning, to be easier on the caches.
68
80
let state = self . futex . load ( Relaxed ) ;
69
81
70
- // We stop spinning when the mutex is unlocked (0) ,
71
- // but also when it's contended (2) .
72
- if state != 1 || spin == 0 {
82
+ // We stop spinning when the mutex is UNLOCKED ,
83
+ // but also when it's CONTENDED .
84
+ if state != LOCKED || spin == 0 {
73
85
return state;
74
86
}
75
87
@@ -80,9 +92,9 @@ impl Mutex {
80
92
81
93
#[ inline]
82
94
pub unsafe fn unlock ( & self ) {
83
- if self . futex . swap ( 0 , Release ) == 2 {
95
+ if self . futex . swap ( UNLOCKED , Release ) == CONTENDED {
84
96
// We only wake up one thread. When that thread locks the mutex, it
85
- // will mark the mutex as contended (2) (see lock_contended above),
97
+ // will mark the mutex as CONTENDED (see lock_contended above),
86
98
// which makes sure that any other waiting threads will also be
87
99
// woken up eventually.
88
100
self . wake ( ) ;
0 commit comments