Skip to content

Commit c4b0265

Browse files
committed
Enable some timeouts in SGX platform
This would partially resolve fortanix/rust-sgx#31
1 parent 50c0192 commit c4b0265

File tree

8 files changed

+179
-31
lines changed

8 files changed

+179
-31
lines changed

src/libstd/sync/condvar.rs

-4
Original file line numberDiff line numberDiff line change
@@ -695,7 +695,6 @@ mod tests {
695695

696696
#[test]
697697
#[cfg_attr(target_os = "emscripten", ignore)]
698-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
699698
fn wait_timeout_wait() {
700699
let m = Arc::new(Mutex::new(()));
701700
let c = Arc::new(Condvar::new());
@@ -715,7 +714,6 @@ mod tests {
715714

716715
#[test]
717716
#[cfg_attr(target_os = "emscripten", ignore)]
718-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
719717
fn wait_timeout_while_wait() {
720718
let m = Arc::new(Mutex::new(()));
721719
let c = Arc::new(Condvar::new());
@@ -740,7 +738,6 @@ mod tests {
740738

741739
#[test]
742740
#[cfg_attr(target_os = "emscripten", ignore)]
743-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
744741
fn wait_timeout_while_wake() {
745742
let pair = Arc::new((Mutex::new(false), Condvar::new()));
746743
let pair_copy = pair.clone();
@@ -764,7 +761,6 @@ mod tests {
764761

765762
#[test]
766763
#[cfg_attr(target_os = "emscripten", ignore)]
767-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
768764
fn wait_timeout_wake() {
769765
let m = Arc::new(Mutex::new(()));
770766
let c = Arc::new(Condvar::new());

src/libstd/sync/mpsc/mod.rs

-9
Original file line numberDiff line numberDiff line change
@@ -2088,7 +2088,6 @@ mod tests {
20882088
}
20892089

20902090
#[test]
2091-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
20922091
fn oneshot_single_thread_recv_timeout() {
20932092
let (tx, rx) = channel();
20942093
tx.send(()).unwrap();
@@ -2099,7 +2098,6 @@ mod tests {
20992098
}
21002099

21012100
#[test]
2102-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
21032101
fn stress_recv_timeout_two_threads() {
21042102
let (tx, rx) = channel();
21052103
let stress = stress_factor() + 100;
@@ -2130,7 +2128,6 @@ mod tests {
21302128
}
21312129

21322130
#[test]
2133-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
21342131
fn recv_timeout_upgrade() {
21352132
let (tx, rx) = channel::<()>();
21362133
let timeout = Duration::from_millis(1);
@@ -2142,7 +2139,6 @@ mod tests {
21422139
}
21432140

21442141
#[test]
2145-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
21462142
fn stress_recv_timeout_shared() {
21472143
let (tx, rx) = channel();
21482144
let stress = stress_factor() + 100;
@@ -2173,7 +2169,6 @@ mod tests {
21732169
}
21742170

21752171
#[test]
2176-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
21772172
fn very_long_recv_timeout_wont_panic() {
21782173
let (tx, rx) = channel::<()>();
21792174
let join_handle =
@@ -2196,7 +2191,6 @@ mod tests {
21962191
}
21972192

21982193
#[test]
2199-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
22002194
fn shared_recv_timeout() {
22012195
let (tx, rx) = channel();
22022196
let total = 5;
@@ -2426,7 +2420,6 @@ mod sync_tests {
24262420
}
24272421

24282422
#[test]
2429-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
24302423
fn recv_timeout() {
24312424
let (tx, rx) = sync_channel::<i32>(1);
24322425
assert_eq!(rx.recv_timeout(Duration::from_millis(1)), Err(RecvTimeoutError::Timeout));
@@ -2518,7 +2511,6 @@ mod sync_tests {
25182511
}
25192512

25202513
#[test]
2521-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
25222514
fn stress_recv_timeout_two_threads() {
25232515
let (tx, rx) = sync_channel::<i32>(0);
25242516

@@ -2544,7 +2536,6 @@ mod sync_tests {
25442536
}
25452537

25462538
#[test]
2547-
#[cfg_attr(target_env = "sgx", ignore)] // FIXME: https://github.com/fortanix/rust-sgx/issues/31
25482539
fn stress_recv_timeout_shared() {
25492540
const AMT: u32 = 1000;
25502541
const NTHREADS: u32 = 8;

src/libstd/sys/sgx/abi/usercalls/mod.rs

+23-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,6 @@
11
use crate::cmp;
22
use crate::io::{Error as IoError, IoSlice, IoSliceMut, Result as IoResult};
3+
use crate::sys::rand::rdrand64;
34
use crate::time::Duration;
45

56
pub(crate) mod alloc;
@@ -149,7 +150,28 @@ pub fn exit(panic: bool) -> ! {
149150

150151
/// Usercall `wait`. See the ABI documentation for more information.
151152
#[unstable(feature = "sgx_platform", issue = "56975")]
152-
pub fn wait(event_mask: u64, timeout: u64) -> IoResult<u64> {
153+
pub fn wait(event_mask: u64, mut timeout: u64) -> IoResult<u64> {
154+
if timeout != WAIT_NO && timeout != WAIT_INDEFINITE {
155+
// We don't want people to rely on accuracy of timeouts to make
156+
// security decisions in an SGX enclave. That's why we add a random
157+
// amount not exceeding +/- 10% to the timeout value to discourage
158+
// people from relying on accuracy of timeouts while providing a way
159+
// to make things work in other cases. Note that in the SGX threat
160+
// model the enclave runner which is serving the wait usercall is not
161+
// trusted to ensure accurate timeouts.
162+
let base = cmp::max(1, timeout / 10) * 2 + 1;
163+
let zero = base / 2;
164+
match rdrand64() % base {
165+
jitter if jitter > zero => {
166+
timeout = timeout.checked_add(jitter - zero).unwrap_or(timeout)
167+
}
168+
jitter if jitter < zero => {
169+
timeout = timeout.checked_sub(zero - jitter).unwrap_or(timeout)
170+
}
171+
_ => {}
172+
};
173+
timeout = cmp::min(u64::MAX - 1, cmp::max(1, timeout));
174+
}
153175
unsafe { raw::wait(event_mask, timeout).from_sgx_result() }
154176
}
155177

src/libstd/sys/sgx/condvar.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -31,8 +31,10 @@ impl Condvar {
3131
mutex.lock()
3232
}
3333

34-
pub unsafe fn wait_timeout(&self, _mutex: &Mutex, _dur: Duration) -> bool {
35-
rtabort!("timeout not supported in SGX");
34+
pub unsafe fn wait_timeout(&self, mutex: &Mutex, dur: Duration) -> bool {
35+
let success = WaitQueue::wait_timeout(&self.inner, dur, || mutex.unlock());
36+
mutex.lock();
37+
success
3638
}
3739

3840
#[inline]

src/libstd/sys/sgx/mod.rs

+42-3
Original file line numberDiff line numberDiff line change
@@ -110,6 +110,42 @@ pub fn decode_error_kind(code: i32) -> ErrorKind {
110110
}
111111
}
112112

113+
// This function makes an effort to sleep at least as long as `duration`.
114+
// Note that in general there is no guarantee about accuracy of time and
115+
// timeouts in SGX model. The enclave runner serving usercalls may lie about
116+
// current time and/or ignore timeout values.
117+
//
118+
// FIXME: note these caveats in documentation of all public types that use this
119+
// function in their execution path.
120+
pub fn wait_timeout_sgx(event_mask: u64, duration: crate::time::Duration) {
121+
use self::abi::usercalls;
122+
use crate::cmp;
123+
use crate::io::ErrorKind;
124+
use crate::time::Instant;
125+
126+
let start = Instant::now();
127+
let mut remaining = duration;
128+
loop {
129+
let timeout = cmp::min((u64::MAX - 1) as u128, remaining.as_nanos()) as u64;
130+
match usercalls::wait(event_mask, timeout) {
131+
Ok(eventset) => {
132+
if event_mask != 0 {
133+
rtassert!(eventset & event_mask == event_mask);
134+
return;
135+
}
136+
rtabort!("expected usercalls::wait() to return Err, found Ok.");
137+
}
138+
Err(e) => {
139+
rtassert!(e.kind() == ErrorKind::TimedOut || e.kind() == ErrorKind::WouldBlock)
140+
}
141+
}
142+
remaining = match duration.checked_sub(start.elapsed()) {
143+
Some(remaining) => remaining,
144+
None => break,
145+
}
146+
}
147+
}
148+
113149
// This enum is used as the storage for a bunch of types which can't actually
114150
// exist.
115151
#[derive(Copy, Clone, PartialEq, Eq, PartialOrd, Ord, Debug, Hash)]
@@ -137,8 +173,8 @@ pub extern "C" fn __rust_abort() {
137173
abort_internal();
138174
}
139175

140-
pub fn hashmap_random_keys() -> (u64, u64) {
141-
fn rdrand64() -> u64 {
176+
pub mod rand {
177+
pub fn rdrand64() -> u64 {
142178
unsafe {
143179
let mut ret: u64 = 0;
144180
for _ in 0..10 {
@@ -149,7 +185,10 @@ pub fn hashmap_random_keys() -> (u64, u64) {
149185
rtabort!("Failed to obtain random data");
150186
}
151187
}
152-
(rdrand64(), rdrand64())
188+
}
189+
190+
pub fn hashmap_random_keys() -> (u64, u64) {
191+
(self::rand::rdrand64(), self::rand::rdrand64())
153192
}
154193

155194
pub use crate::sys_common::{AsInner, FromInner, IntoInner};

src/libstd/sys/sgx/thread.rs

+4-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
#![cfg_attr(test, allow(dead_code))] // why is this necessary?
2+
23
use crate::ffi::CStr;
34
use crate::io;
5+
use crate::sys::wait_timeout_sgx;
46
use crate::time::Duration;
57

68
use super::abi::usercalls;
@@ -73,8 +75,8 @@ impl Thread {
7375
// FIXME: could store this pointer in TLS somewhere
7476
}
7577

76-
pub fn sleep(_dur: Duration) {
77-
rtabort!("can't sleep"); // FIXME
78+
pub fn sleep(dur: Duration) {
79+
wait_timeout_sgx(0, dur);
7880
}
7981

8082
pub fn join(self) {

src/libstd/sys/sgx/waitqueue.rs

+106-7
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,3 @@
1-
use crate::num::NonZeroUsize;
21
/// A simple queue implementation for synchronization primitives.
32
///
43
/// This queue is used to implement condition variable and mutexes.
@@ -10,7 +9,10 @@ use crate::num::NonZeroUsize;
109
/// Since userspace may send spurious wake-ups, the wakeup event state is
1110
/// recorded in the enclave. The wakeup event state is protected by a spinlock.
1211
/// The queue and associated wait state are stored in a `WaitVariable`.
12+
use crate::num::NonZeroUsize;
1313
use crate::ops::{Deref, DerefMut};
14+
use crate::sys::wait_timeout_sgx;
15+
use crate::time::Duration;
1416

1517
use super::abi::thread;
1618
use super::abi::usercalls;
@@ -158,6 +160,37 @@ impl WaitQueue {
158160
}
159161
}
160162

163+
/// Adds the calling thread to the `WaitVariable`'s wait queue, then wait
164+
/// until a wakeup event or timeout. If event was observed, returns true.
165+
/// If not, it will remove the calling thread from the wait queue.
166+
pub fn wait_timeout<T, F: FnOnce()>(
167+
lock: &SpinMutex<WaitVariable<T>>,
168+
timeout: Duration,
169+
before_wait: F,
170+
) -> bool {
171+
// very unsafe: check requirements of UnsafeList::push
172+
unsafe {
173+
let mut entry = UnsafeListEntry::new(SpinMutex::new(WaitEntry {
174+
tcs: thread::current(),
175+
wake: false,
176+
}));
177+
let entry_lock = lock.lock().queue.inner.push(&mut entry);
178+
before_wait();
179+
// don't panic, this would invalidate `entry` during unwinding
180+
wait_timeout_sgx(EV_UNPARK, timeout);
181+
// acquire the wait queue's lock first to avoid deadlock.
182+
let mut guard = lock.lock();
183+
let entry_guard = entry_lock.lock();
184+
let success = entry_guard.wake;
185+
if !success {
186+
// nobody is waking us up, so remove the entry from the wait queue.
187+
drop(entry_guard);
188+
guard.queue.inner.remove(&mut entry);
189+
}
190+
success
191+
}
192+
}
193+
161194
/// Either find the next waiter on the wait queue, or return the mutex
162195
/// guard unchanged.
163196
///
@@ -325,6 +358,31 @@ mod unsafe_list {
325358
Some((*first.as_ptr()).value.as_ref().unwrap())
326359
}
327360
}
361+
362+
/// Removes an entry from the list.
363+
///
364+
/// # Safety
365+
///
366+
/// The caller must ensure that entry has been pushed prior to this
367+
/// call and has not moved since push.
368+
pub unsafe fn remove(&mut self, entry: &mut UnsafeListEntry<T>) {
369+
rtassert!(!self.is_empty());
370+
// BEFORE:
371+
// /----\ next ---> /-----\ next ---> /----\
372+
// ... |prev| |entry| |next| ...
373+
// \----/ <--- prev \-----/ <--- prev \----/
374+
//
375+
// AFTER:
376+
// /----\ next ---> /----\
377+
// ... |prev| |next| ...
378+
// \----/ <--- prev \----/
379+
let mut prev = entry.prev;
380+
let mut next = entry.next;
381+
prev.as_mut().next = next;
382+
next.as_mut().prev = prev;
383+
entry.next = NonNull::dangling();
384+
entry.prev = NonNull::dangling();
385+
}
328386
}
329387

330388
#[cfg(test)]
@@ -354,6 +412,51 @@ mod unsafe_list {
354412
}
355413
}
356414

415+
#[test]
416+
fn push_remove() {
417+
unsafe {
418+
let mut node = UnsafeListEntry::new(1234);
419+
let mut list = UnsafeList::new();
420+
assert_eq!(list.push(&mut node), &1234);
421+
list.remove(&mut node);
422+
assert_empty(&mut list);
423+
}
424+
}
425+
426+
#[test]
427+
fn push_remove_pop() {
428+
unsafe {
429+
let mut node1 = UnsafeListEntry::new(11);
430+
let mut node2 = UnsafeListEntry::new(12);
431+
let mut node3 = UnsafeListEntry::new(13);
432+
let mut node4 = UnsafeListEntry::new(14);
433+
let mut node5 = UnsafeListEntry::new(15);
434+
let mut list = UnsafeList::new();
435+
assert_eq!(list.push(&mut node1), &11);
436+
assert_eq!(list.push(&mut node2), &12);
437+
assert_eq!(list.push(&mut node3), &13);
438+
assert_eq!(list.push(&mut node4), &14);
439+
assert_eq!(list.push(&mut node5), &15);
440+
441+
list.remove(&mut node1);
442+
assert_eq!(list.pop().unwrap(), &12);
443+
list.remove(&mut node3);
444+
assert_eq!(list.pop().unwrap(), &14);
445+
list.remove(&mut node5);
446+
assert_empty(&mut list);
447+
448+
assert_eq!(list.push(&mut node1), &11);
449+
assert_eq!(list.pop().unwrap(), &11);
450+
assert_empty(&mut list);
451+
452+
assert_eq!(list.push(&mut node3), &13);
453+
assert_eq!(list.push(&mut node4), &14);
454+
list.remove(&mut node3);
455+
list.remove(&mut node4);
456+
assert_empty(&mut list);
457+
}
458+
}
459+
357460
#[test]
358461
fn complex_pushes_pops() {
359462
unsafe {
@@ -474,7 +577,7 @@ mod spin_mutex {
474577
use super::*;
475578
use crate::sync::Arc;
476579
use crate::thread;
477-
use crate::time::{Duration, SystemTime};
580+
use crate::time::Duration;
478581

479582
#[test]
480583
fn sleep() {
@@ -485,11 +588,7 @@ mod spin_mutex {
485588
*mutex2.lock() = 1;
486589
});
487590

488-
// "sleep" for 50ms
489-
// FIXME: https://github.com/fortanix/rust-sgx/issues/31
490-
let start = SystemTime::now();
491-
let max = Duration::from_millis(50);
492-
while start.elapsed().unwrap() < max {}
591+
thread::sleep(Duration::from_millis(50));
493592

494593
assert_eq!(*guard, 0);
495594
drop(guard);

0 commit comments

Comments
 (0)