2013-04-15 13:40:36 -07:00
|
|
|
// Copyright 2013 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
//! Access to the thread-local Scheduler
|
|
|
|
|
2013-03-27 15:24:50 -07:00
|
|
|
use prelude::*;
|
2013-04-15 13:40:36 -07:00
|
|
|
use ptr::mut_null;
|
|
|
|
use libc::c_void;
|
|
|
|
use cast::transmute;
|
|
|
|
|
|
|
|
use super::Scheduler;
|
2013-04-18 19:32:32 -07:00
|
|
|
use super::super::rtio::IoFactoryObject;
|
2013-04-15 13:40:36 -07:00
|
|
|
use tls = super::super::thread_local_storage;
|
|
|
|
#[cfg(test)] use super::super::uvio::UvEventLoop;
|
|
|
|
|
|
|
|
/// Give the Scheduler to thread-local storage
|
|
|
|
pub fn put(sched: ~Scheduler) {
|
|
|
|
unsafe {
|
|
|
|
let key = tls_key();
|
|
|
|
let void_sched: *mut c_void = transmute::<~Scheduler, *mut c_void>(sched);
|
|
|
|
tls::set(key, void_sched);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Take ownership of the Scheduler from thread-local storage
|
|
|
|
pub fn take() -> ~Scheduler {
|
|
|
|
unsafe {
|
|
|
|
let key = tls_key();
|
|
|
|
let void_sched: *mut c_void = tls::get(key);
|
|
|
|
assert!(void_sched.is_not_null());
|
|
|
|
let sched = transmute::<*mut c_void, ~Scheduler>(void_sched);
|
|
|
|
tls::set(key, mut_null());
|
|
|
|
return sched;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-03-27 15:24:50 -07:00
|
|
|
/// Check whether there is a thread-local Scheduler attached to the running thread
|
|
|
|
pub fn exists() -> bool {
|
|
|
|
unsafe {
|
|
|
|
match maybe_tls_key() {
|
|
|
|
Some(key) => tls::get(key).is_not_null(),
|
|
|
|
None => false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-18 19:32:32 -07:00
|
|
|
/// Borrow the thread-local scheduler from thread-local storage.
|
|
|
|
/// While the scheduler is borrowed it is not available in TLS.
|
|
|
|
pub fn borrow(f: &fn(&mut Scheduler)) {
|
|
|
|
let mut sched = take();
|
|
|
|
f(sched);
|
|
|
|
put(sched);
|
|
|
|
}
|
|
|
|
|
2013-04-15 13:40:36 -07:00
|
|
|
/// Borrow a mutable reference to the thread-local Scheduler
|
2013-04-18 19:32:32 -07:00
|
|
|
///
|
2013-04-15 13:40:36 -07:00
|
|
|
/// # Safety Note
|
2013-04-18 19:32:32 -07:00
|
|
|
///
|
2013-04-15 13:40:36 -07:00
|
|
|
/// Because this leaves the Scheduler in thread-local storage it is possible
|
|
|
|
/// For the Scheduler pointer to be aliased
|
2013-04-18 19:32:32 -07:00
|
|
|
pub unsafe fn unsafe_borrow() -> &mut Scheduler {
|
2013-04-15 13:40:36 -07:00
|
|
|
unsafe {
|
|
|
|
let key = tls_key();
|
|
|
|
let mut void_sched: *mut c_void = tls::get(key);
|
|
|
|
assert!(void_sched.is_not_null());
|
|
|
|
{
|
|
|
|
let void_sched_ptr = &mut void_sched;
|
|
|
|
let sched: &mut ~Scheduler = {
|
|
|
|
transmute::<&mut *mut c_void, &mut ~Scheduler>(void_sched_ptr)
|
|
|
|
};
|
|
|
|
let sched: &mut Scheduler = &mut **sched;
|
2013-04-15 16:00:15 -07:00
|
|
|
return sched;
|
2013-04-15 13:40:36 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-18 19:32:32 -07:00
|
|
|
pub unsafe fn unsafe_borrow_io() -> &mut IoFactoryObject {
|
|
|
|
unsafe {
|
|
|
|
let sched = unsafe_borrow();
|
|
|
|
return sched.event_loop.io().unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-15 13:40:36 -07:00
|
|
|
fn tls_key() -> tls::Key {
|
2013-03-27 15:24:50 -07:00
|
|
|
maybe_tls_key().get()
|
|
|
|
}
|
|
|
|
|
|
|
|
fn maybe_tls_key() -> Option<tls::Key> {
|
2013-04-15 13:40:36 -07:00
|
|
|
unsafe {
|
|
|
|
let key: *mut c_void = rust_get_sched_tls_key();
|
|
|
|
let key: &mut tls::Key = transmute(key);
|
2013-03-27 15:24:50 -07:00
|
|
|
let key = *key;
|
|
|
|
// Check that the key has been initialized.
|
|
|
|
|
|
|
|
// NB: This is a little racy because, while the key is
|
|
|
|
// initalized under a mutex and it's assumed to be initalized
|
|
|
|
// in the Scheduler ctor by any thread that needs to use it,
|
|
|
|
// we are not accessing the key under a mutex. Threads that
|
|
|
|
// are not using the new Scheduler but still *want to check*
|
|
|
|
// whether they are running under a new Scheduler may see a 0
|
|
|
|
// value here that is in the process of being initialized in
|
|
|
|
// another thread. I think this is fine since the only action
|
|
|
|
// they could take if it was initialized would be to check the
|
|
|
|
// thread-local value and see that it's not set.
|
|
|
|
if key != 0 {
|
|
|
|
return Some(key);
|
|
|
|
} else {
|
|
|
|
return None;
|
|
|
|
}
|
2013-04-15 13:40:36 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
extern {
|
|
|
|
fn rust_get_sched_tls_key() -> *mut c_void;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn thread_local_scheduler_smoke_test() {
|
|
|
|
let scheduler = ~UvEventLoop::new_scheduler();
|
|
|
|
put(scheduler);
|
|
|
|
let _scheduler = take();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn thread_local_scheduler_two_instances() {
|
|
|
|
let scheduler = ~UvEventLoop::new_scheduler();
|
|
|
|
put(scheduler);
|
|
|
|
let _scheduler = take();
|
|
|
|
let scheduler = ~UvEventLoop::new_scheduler();
|
|
|
|
put(scheduler);
|
|
|
|
let _scheduler = take();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
2013-04-15 15:25:29 -07:00
|
|
|
fn borrow_smoke_test() {
|
2013-04-15 13:40:36 -07:00
|
|
|
let scheduler = ~UvEventLoop::new_scheduler();
|
2013-04-15 15:25:29 -07:00
|
|
|
put(scheduler);
|
|
|
|
unsafe {
|
2013-04-18 19:32:32 -07:00
|
|
|
let _scheduler = unsafe_borrow();
|
2013-04-15 15:25:29 -07:00
|
|
|
}
|
|
|
|
let _scheduler = take();
|
2013-04-15 13:40:36 -07:00
|
|
|
}
|
|
|
|
|