From 5d04234868428fe819fb17d78b8f7468161586b9 Mon Sep 17 00:00:00 2001 From: Brian Anderson Date: Sat, 17 Aug 2013 18:16:30 -0700 Subject: [PATCH] std::rt: Reduce SleeperList contention This makes the lock much less contended. In the test I'm running the number of times it's contended goes from ~100000 down to ~1000. --- src/libstd/rt/sched.rs | 5 +--- src/libstd/rt/sleeper_list.rs | 51 +++++++++++++++++++++++++++++------ 2 files changed, 44 insertions(+), 12 deletions(-) diff --git a/src/libstd/rt/sched.rs b/src/libstd/rt/sched.rs index b161864a74f..91ab87268f3 100644 --- a/src/libstd/rt/sched.rs +++ b/src/libstd/rt/sched.rs @@ -469,10 +469,7 @@ impl Scheduler { // We've made work available. Notify a // sleeping scheduler. - // XXX: perf. Check for a sleeper without - // synchronizing memory. It's not critical - // that we always find it. - match this.sleeper_list.pop() { + match this.sleeper_list.casual_pop() { Some(handle) => { let mut handle = handle; handle.send(Wake) diff --git a/src/libstd/rt/sleeper_list.rs b/src/libstd/rt/sleeper_list.rs index 967fde6f371..7232afd6594 100644 --- a/src/libstd/rt/sleeper_list.rs +++ b/src/libstd/rt/sleeper_list.rs @@ -15,33 +15,68 @@ use container::Container; use vec::OwnedVector; use option::{Option, Some, None}; use cell::Cell; -use unstable::sync::Exclusive; +use unstable::sync::{UnsafeAtomicRcBox, LittleLock}; use rt::sched::SchedHandle; use clone::Clone; pub struct SleeperList { - priv stack: Exclusive<~[SchedHandle]> + priv state: UnsafeAtomicRcBox +} + +struct State { + count: uint, + stack: ~[SchedHandle], + lock: LittleLock } impl SleeperList { pub fn new() -> SleeperList { SleeperList { - stack: Exclusive::new(~[]) + state: UnsafeAtomicRcBox::new(State { + count: 0, + stack: ~[], + lock: LittleLock::new() + }) } } pub fn push(&mut self, handle: SchedHandle) { let handle = Cell::new(handle); unsafe { - self.stack.with(|s| s.push(handle.take())); + let state = self.state.get(); + do (*state).lock.lock { + (*state).count += 1; + (*state).stack.push(handle.take()); + } } } pub fn pop(&mut self) -> Option { unsafe { - do self.stack.with |s| { - if !s.is_empty() { - Some(s.pop()) + let state = self.state.get(); + do (*state).lock.lock { + if !(*state).stack.is_empty() { + (*state).count -= 1; + Some((*state).stack.pop()) + } else { + None + } + } + } + } + + /// A pop that may sometimes miss enqueued elements, but is much faster + /// to give up without doing any synchronization + pub fn casual_pop(&mut self) -> Option { + unsafe { + let state = self.state.get(); + // NB: Unsynchronized check + if (*state).count == 0 { return None; } + do (*state).lock.lock { + if !(*state).stack.is_empty() { + // NB: count is also protected by the lock + (*state).count -= 1; + Some((*state).stack.pop()) } else { None } @@ -53,7 +88,7 @@ impl SleeperList { impl Clone for SleeperList { fn clone(&self) -> SleeperList { SleeperList { - stack: self.stack.clone() + state: self.state.clone() } } }