@@ -4,49 +4,82 @@ use crate::sys::dur2timeout;
4
4
use core:: ffi:: c_void;
5
5
use core:: mem;
6
6
use core:: ptr;
7
+ use core:: sync:: atomic:: {
8
+ AtomicBool , AtomicI16 , AtomicI32 , AtomicI64 , AtomicI8 , AtomicIsize , AtomicPtr , AtomicU16 ,
9
+ AtomicU32 , AtomicU64 , AtomicU8 , AtomicUsize ,
10
+ } ;
7
11
use core:: time:: Duration ;
8
12
9
- #[ inline( always) ]
10
- pub fn wait_on_address < T , U > ( address : & T , compare : U , timeout : Option < Duration > ) -> bool {
11
- assert_eq ! ( mem:: size_of:: <T >( ) , mem:: size_of:: <U >( ) ) ;
13
+ pub unsafe trait Waitable {
14
+ type Atomic ;
15
+ }
16
+ macro_rules! unsafe_waitable_int {
17
+ ( $( ( $int: ty, $atomic: ty) ) ,* $( , ) ?) => {
18
+ $(
19
+ unsafe impl Waitable for $int {
20
+ type Atomic = $atomic;
21
+ }
22
+ ) *
23
+ } ;
24
+ }
25
+ unsafe_waitable_int ! {
26
+ ( bool , AtomicBool ) ,
27
+ ( i8 , AtomicI8 ) ,
28
+ ( i16 , AtomicI16 ) ,
29
+ ( i32 , AtomicI32 ) ,
30
+ ( i64 , AtomicI64 ) ,
31
+ ( isize , AtomicIsize ) ,
32
+ ( u8 , AtomicU8 ) ,
33
+ ( u16 , AtomicU16 ) ,
34
+ ( u32 , AtomicU32 ) ,
35
+ ( u64 , AtomicU64 ) ,
36
+ ( usize , AtomicUsize ) ,
37
+ }
38
+ unsafe impl < T > Waitable for * const T {
39
+ type Atomic = AtomicPtr < T > ;
40
+ }
41
+ unsafe impl < T > Waitable for * mut T {
42
+ type Atomic = AtomicPtr < T > ;
43
+ }
44
+
45
+ pub fn wait_on_address < W : Waitable > (
46
+ address : & W :: Atomic ,
47
+ compare : W ,
48
+ timeout : Option < Duration > ,
49
+ ) -> bool {
12
50
unsafe {
13
51
let addr = ptr:: from_ref ( address) . cast :: < c_void > ( ) ;
14
- let size = mem:: size_of :: < T > ( ) ;
52
+ let size = mem:: size_of :: < W > ( ) ;
15
53
let compare_addr = ptr:: addr_of!( compare) . cast :: < c_void > ( ) ;
16
54
let timeout = timeout. map ( dur2timeout) . unwrap_or ( c:: INFINITE ) ;
17
55
c:: WaitOnAddress ( addr, compare_addr, size, timeout) == c:: TRUE
18
56
}
19
57
}
20
58
21
- #[ inline( always) ]
22
59
pub fn wake_by_address_single < T > ( address : & T ) {
23
60
unsafe {
24
61
let addr = ptr:: from_ref ( address) . cast :: < c_void > ( ) ;
25
62
c:: WakeByAddressSingle ( addr) ;
26
63
}
27
64
}
28
65
29
- #[ inline( always) ]
30
66
pub fn wake_by_address_all < T > ( address : & T ) {
31
67
unsafe {
32
68
let addr = ptr:: from_ref ( address) . cast :: < c_void > ( ) ;
33
69
c:: WakeByAddressAll ( addr) ;
34
70
}
35
71
}
36
72
37
- #[ inline( always) ]
38
- pub fn futex_wait < T , U > ( futex : & T , expected : U , timeout : Option < Duration > ) -> bool {
73
+ pub fn futex_wait < W : Waitable > ( futex : & W :: Atomic , expected : W , timeout : Option < Duration > ) -> bool {
39
74
// return false only on timeout
40
75
wait_on_address ( futex, expected, timeout) || api:: get_last_error ( ) . code != c:: ERROR_TIMEOUT
41
76
}
42
77
43
- #[ inline( always) ]
44
78
pub fn futex_wake < T > ( futex : & T ) -> bool {
45
79
wake_by_address_single ( futex) ;
46
80
false
47
81
}
48
82
49
- #[ inline( always) ]
50
83
pub fn futex_wake_all < T > ( futex : & T ) {
51
84
wake_by_address_all ( futex)
52
85
}
0 commit comments