Add futex timeout

This commit is contained in:
Jeremy Soller
2017-11-06 22:05:20 -07:00
committed by Jeremy Soller
parent 12e6b53744
commit 2f68b870b5
2 changed files with 41 additions and 25 deletions

View File

@@ -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]

View File

@@ -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> {