Add futex timeout
This commit is contained in:
committed by
Jeremy Soller
parent
12e6b53744
commit
2f68b870b5
@@ -9,12 +9,12 @@
|
|||||||
// except according to those terms.
|
// except according to those terms.
|
||||||
|
|
||||||
use cell::UnsafeCell;
|
use cell::UnsafeCell;
|
||||||
use intrinsics::{atomic_cxchg, atomic_xadd, atomic_xchg};
|
use intrinsics::{atomic_cxchg, atomic_load, atomic_xadd, atomic_xchg};
|
||||||
use ptr;
|
use ptr;
|
||||||
use time::Duration;
|
use time::Duration;
|
||||||
|
|
||||||
use sys::mutex::{mutex_unlock, Mutex};
|
use sys::mutex::{mutex_unlock, Mutex};
|
||||||
use sys::syscall::{futex, FUTEX_WAIT, FUTEX_WAKE, FUTEX_REQUEUE};
|
use sys::syscall::{futex, TimeSpec, FUTEX_WAIT, FUTEX_WAKE, FUTEX_REQUEUE};
|
||||||
|
|
||||||
pub struct Condvar {
|
pub struct Condvar {
|
||||||
lock: UnsafeCell<*mut i32>,
|
lock: UnsafeCell<*mut i32>,
|
||||||
@@ -63,8 +63,7 @@ impl Condvar {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn wait(&self, mutex: &Mutex) {
|
unsafe fn wait_inner(&self, mutex: &Mutex, timeout_ptr: *const TimeSpec) -> bool {
|
||||||
unsafe {
|
|
||||||
let lock = self.lock.get();
|
let lock = self.lock.get();
|
||||||
let seq = self.seq.get();
|
let seq = self.seq.get();
|
||||||
|
|
||||||
@@ -78,18 +77,36 @@ impl Condvar {
|
|||||||
|
|
||||||
mutex_unlock(*lock);
|
mutex_unlock(*lock);
|
||||||
|
|
||||||
let _ = futex(seq, FUTEX_WAIT, *seq, 0, ptr::null_mut());
|
let seq_before = atomic_load(seq);
|
||||||
|
|
||||||
|
let _ = futex(seq, FUTEX_WAIT, seq_before, timeout_ptr as usize, ptr::null_mut());
|
||||||
|
|
||||||
|
let seq_after = atomic_load(seq);
|
||||||
|
|
||||||
while atomic_xchg(*lock, 2) != 0 {
|
while atomic_xchg(*lock, 2) != 0 {
|
||||||
let _ = futex(*lock, FUTEX_WAIT, 2, 0, ptr::null_mut());
|
let _ = futex(*lock, FUTEX_WAIT, 2, 0, ptr::null_mut());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
seq_before != seq_after
|
||||||
|
}
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn wait(&self, mutex: &Mutex) {
|
||||||
|
unsafe {
|
||||||
|
assert!(self.wait_inner(mutex, ptr::null()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
pub fn wait_timeout(&self, _mutex: &Mutex, _dur: Duration) -> bool {
|
pub fn wait_timeout(&self, mutex: &Mutex, dur: Duration) -> bool {
|
||||||
::sys_common::util::dumb_print(format_args!("condvar wait_timeout\n"));
|
unsafe {
|
||||||
unimplemented!();
|
let timeout = TimeSpec {
|
||||||
|
tv_sec: dur.as_secs() as i64,
|
||||||
|
tv_nsec: dur.subsec_nanos() as i32
|
||||||
|
};
|
||||||
|
|
||||||
|
self.wait_inner(mutex, &timeout as *const TimeSpec)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline]
|
#[inline]
|
||||||
|
|||||||
@@ -437,8 +437,7 @@ pub fn symlink(src: &Path, dst: &Path) -> io::Result<()> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
pub fn link(_src: &Path, _dst: &Path) -> io::Result<()> {
|
pub fn link(_src: &Path, _dst: &Path) -> io::Result<()> {
|
||||||
::sys_common::util::dumb_print(format_args!("Link\n"));
|
Err(Error::from_raw_os_error(syscall::ENOSYS))
|
||||||
unimplemented!();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn stat(p: &Path) -> io::Result<FileAttr> {
|
pub fn stat(p: &Path) -> io::Result<FileAttr> {
|
||||||
|
|||||||
Reference in New Issue
Block a user