Skip to content

Commit cf83d83

Browse files
committed
Add Waitable trait
1 parent 2a09857 commit cf83d83

File tree

1 file changed

+43
-10
lines changed

1 file changed

+43
-10
lines changed

library/std/src/sys/pal/windows/futex.rs

+43-10
Original file line numberDiff line numberDiff line change
@@ -4,49 +4,82 @@ use crate::sys::dur2timeout;
44
use core::ffi::c_void;
55
use core::mem;
66
use core::ptr;
7+
use core::sync::atomic::{
8+
AtomicBool, AtomicI16, AtomicI32, AtomicI64, AtomicI8, AtomicIsize, AtomicPtr, AtomicU16,
9+
AtomicU32, AtomicU64, AtomicU8, AtomicUsize,
10+
};
711
use core::time::Duration;
812

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 {
1250
unsafe {
1351
let addr = ptr::from_ref(address).cast::<c_void>();
14-
let size = mem::size_of::<T>();
52+
let size = mem::size_of::<W>();
1553
let compare_addr = ptr::addr_of!(compare).cast::<c_void>();
1654
let timeout = timeout.map(dur2timeout).unwrap_or(c::INFINITE);
1755
c::WaitOnAddress(addr, compare_addr, size, timeout) == c::TRUE
1856
}
1957
}
2058

21-
#[inline(always)]
2259
pub fn wake_by_address_single<T>(address: &T) {
2360
unsafe {
2461
let addr = ptr::from_ref(address).cast::<c_void>();
2562
c::WakeByAddressSingle(addr);
2663
}
2764
}
2865

29-
#[inline(always)]
3066
pub fn wake_by_address_all<T>(address: &T) {
3167
unsafe {
3268
let addr = ptr::from_ref(address).cast::<c_void>();
3369
c::WakeByAddressAll(addr);
3470
}
3571
}
3672

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 {
3974
// return false only on timeout
4075
wait_on_address(futex, expected, timeout) || api::get_last_error().code != c::ERROR_TIMEOUT
4176
}
4277

43-
#[inline(always)]
4478
pub fn futex_wake<T>(futex: &T) -> bool {
4579
wake_by_address_single(futex);
4680
false
4781
}
4882

49-
#[inline(always)]
5083
pub fn futex_wake_all<T>(futex: &T) {
5184
wake_by_address_all(futex)
5285
}

0 commit comments

Comments
 (0)