2013-05-19 14:39:46 -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.
|
|
|
|
|
2013-05-19 14:57:47 -07:00
|
|
|
//! Access to a single thread-local pointer.
|
|
|
|
//!
|
|
|
|
//! The runtime will use this for storing ~Task.
|
|
|
|
//!
|
|
|
|
//! XXX: Add runtime checks for usage of inconsistent pointer types.
|
|
|
|
//! and for overwriting an existing pointer.
|
2013-05-19 14:39:46 -07:00
|
|
|
|
|
|
|
use libc::c_void;
|
|
|
|
use cast;
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-05-19 14:57:47 -07:00
|
|
|
use ptr;
|
|
|
|
use cell::Cell;
|
2013-05-19 14:39:46 -07:00
|
|
|
use option::{Option, Some, None};
|
2013-05-19 14:57:47 -07:00
|
|
|
use unstable::finally::Finally;
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-11-14 00:21:43 -08:00
|
|
|
use unstable::mutex::{Mutex, MUTEX_INIT};
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-05-19 14:39:46 -07:00
|
|
|
use tls = rt::thread_local_storage;
|
|
|
|
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(not(stage0), not(windows), test)]
|
|
|
|
#[thread_local]
|
|
|
|
pub use realstd::rt::shouldnt_be_public::RT_TLS_PTR;
|
|
|
|
|
|
|
|
#[cfg(not(stage0), not(windows), not(test))]
|
|
|
|
#[thread_local]
|
|
|
|
pub static mut RT_TLS_PTR: *mut c_void = 0 as *mut c_void;
|
|
|
|
|
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-08-17 02:12:08 -07:00
|
|
|
static mut RT_TLS_KEY: tls::Key = -1;
|
2013-11-25 18:27:27 -08:00
|
|
|
static mut tls_lock: Mutex = MUTEX_INIT;
|
|
|
|
static mut tls_initialized: bool = false;
|
2013-08-17 02:12:08 -07:00
|
|
|
|
2013-05-19 14:39:46 -07:00
|
|
|
/// Initialize the TLS key. Other ops will fail if this isn't executed first.
|
2013-11-06 01:17:04 -05:00
|
|
|
#[inline(never)]
|
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-05-19 14:39:46 -07:00
|
|
|
pub fn init_tls_key() {
|
|
|
|
unsafe {
|
2013-11-25 18:27:27 -08:00
|
|
|
tls_lock.lock();
|
|
|
|
if !tls_initialized {
|
2013-11-14 00:21:43 -08:00
|
|
|
tls::create(&mut RT_TLS_KEY);
|
2013-11-25 18:27:27 -08:00
|
|
|
tls_initialized = true;
|
2013-05-19 14:39:46 -07:00
|
|
|
}
|
2013-11-25 18:27:27 -08:00
|
|
|
tls_lock.unlock();
|
2013-05-19 14:39:46 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(not(stage0), not(windows))]
|
|
|
|
pub fn init_tls_key() {}
|
|
|
|
|
2013-11-25 18:27:27 -08:00
|
|
|
#[cfg(windows)]
|
|
|
|
pub unsafe fn cleanup() {
|
|
|
|
// No real use to acquiring a lock around these operations. All we're
|
|
|
|
// going to do is destroy the lock anyway which races locking itself. This
|
|
|
|
// is why the whole function is labeled as 'unsafe'
|
|
|
|
assert!(tls_initialized);
|
|
|
|
tls::destroy(RT_TLS_KEY);
|
|
|
|
tls_lock.destroy();
|
|
|
|
tls_initialized = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(windows))]
|
|
|
|
pub unsafe fn cleanup() {
|
|
|
|
assert!(tls_initialized);
|
|
|
|
tls_lock.destroy();
|
|
|
|
tls_initialized = false;
|
|
|
|
}
|
|
|
|
|
2013-05-19 14:57:47 -07:00
|
|
|
/// Give a pointer to thread-local storage.
|
|
|
|
///
|
|
|
|
/// # Safety note
|
|
|
|
///
|
|
|
|
/// Does not validate the pointer type.
|
2013-08-17 01:24:29 -07:00
|
|
|
#[inline]
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-05-19 14:57:47 -07:00
|
|
|
pub unsafe fn put<T>(sched: ~T) {
|
|
|
|
let key = tls_key();
|
|
|
|
let void_ptr: *mut c_void = cast::transmute(sched);
|
|
|
|
tls::set(key, void_ptr);
|
|
|
|
}
|
|
|
|
|
2013-11-06 01:17:04 -05:00
|
|
|
/// Give a pointer to thread-local storage.
|
|
|
|
///
|
|
|
|
/// # Safety note
|
|
|
|
///
|
|
|
|
/// Does not validate the pointer type.
|
|
|
|
#[inline]
|
|
|
|
#[cfg(not(stage0), not(windows))]
|
|
|
|
pub unsafe fn put<T>(sched: ~T) {
|
|
|
|
RT_TLS_PTR = cast::transmute(sched)
|
|
|
|
}
|
|
|
|
|
2013-05-19 14:57:47 -07:00
|
|
|
/// Take ownership of a pointer from thread-local storage.
|
|
|
|
///
|
|
|
|
/// # Safety note
|
|
|
|
///
|
|
|
|
/// Does not validate the pointer type.
|
2013-08-17 01:24:29 -07:00
|
|
|
#[inline]
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-05-19 14:57:47 -07:00
|
|
|
pub unsafe fn take<T>() -> ~T {
|
|
|
|
let key = tls_key();
|
|
|
|
let void_ptr: *mut c_void = tls::get(key);
|
2013-08-03 17:39:59 -07:00
|
|
|
if void_ptr.is_null() {
|
|
|
|
rtabort!("thread-local pointer is null. bogus!");
|
|
|
|
}
|
2013-05-19 14:57:47 -07:00
|
|
|
let ptr: ~T = cast::transmute(void_ptr);
|
|
|
|
tls::set(key, ptr::mut_null());
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2013-11-06 01:17:04 -05:00
|
|
|
/// Take ownership of a pointer from thread-local storage.
|
|
|
|
///
|
|
|
|
/// # Safety note
|
|
|
|
///
|
|
|
|
/// Does not validate the pointer type.
|
|
|
|
#[inline]
|
|
|
|
#[cfg(not(stage0), not(windows))]
|
|
|
|
pub unsafe fn take<T>() -> ~T {
|
|
|
|
let ptr: ~T = cast::transmute(RT_TLS_PTR);
|
|
|
|
RT_TLS_PTR = cast::transmute(0); // can't use `as`, due to type not matching with `cfg(test)`
|
|
|
|
ptr
|
|
|
|
}
|
|
|
|
|
2013-08-17 17:40:38 -07:00
|
|
|
/// Take ownership of a pointer from thread-local storage.
|
|
|
|
///
|
|
|
|
/// # Safety note
|
|
|
|
///
|
|
|
|
/// Does not validate the pointer type.
|
|
|
|
/// Leaves the old pointer in TLS for speed.
|
|
|
|
#[inline]
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-08-17 17:40:38 -07:00
|
|
|
pub unsafe fn unsafe_take<T>() -> ~T {
|
|
|
|
let key = tls_key();
|
|
|
|
let void_ptr: *mut c_void = tls::get(key);
|
|
|
|
if void_ptr.is_null() {
|
|
|
|
rtabort!("thread-local pointer is null. bogus!");
|
|
|
|
}
|
|
|
|
let ptr: ~T = cast::transmute(void_ptr);
|
|
|
|
return ptr;
|
|
|
|
}
|
|
|
|
|
2013-11-06 01:17:04 -05:00
|
|
|
/// Take ownership of a pointer from thread-local storage.
|
|
|
|
///
|
|
|
|
/// # Safety note
|
|
|
|
///
|
|
|
|
/// Does not validate the pointer type.
|
|
|
|
/// Leaves the old pointer in TLS for speed.
|
|
|
|
#[inline]
|
|
|
|
#[cfg(not(stage0), not(windows))]
|
|
|
|
pub unsafe fn unsafe_take<T>() -> ~T {
|
|
|
|
cast::transmute(RT_TLS_PTR)
|
|
|
|
}
|
|
|
|
|
2013-05-19 14:57:47 -07:00
|
|
|
/// Check whether there is a thread-local pointer installed.
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-05-19 14:57:47 -07:00
|
|
|
pub fn exists() -> bool {
|
|
|
|
unsafe {
|
|
|
|
match maybe_tls_key() {
|
|
|
|
Some(key) => tls::get(key).is_not_null(),
|
|
|
|
None => false
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 01:17:04 -05:00
|
|
|
/// Check whether there is a thread-local pointer installed.
|
|
|
|
#[cfg(not(stage0), not(windows))]
|
|
|
|
pub fn exists() -> bool {
|
|
|
|
unsafe {
|
|
|
|
RT_TLS_PTR.is_not_null()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-03 17:39:59 -07:00
|
|
|
/// Borrow the thread-local value from thread-local storage.
|
|
|
|
/// While the value is borrowed it is not available in TLS.
|
2013-05-19 14:57:47 -07:00
|
|
|
///
|
|
|
|
/// # Safety note
|
|
|
|
///
|
|
|
|
/// Does not validate the pointer type.
|
2013-11-18 21:15:42 -08:00
|
|
|
pub unsafe fn borrow<T>(f: |&mut T|) {
|
2013-05-19 14:57:47 -07:00
|
|
|
let mut value = take();
|
|
|
|
|
|
|
|
// XXX: Need a different abstraction from 'finally' here to avoid unsafety
|
|
|
|
let unsafe_ptr = cast::transmute_mut_region(&mut *value);
|
2013-06-04 12:03:58 +02:00
|
|
|
let value_cell = Cell::new(value);
|
2013-05-19 14:57:47 -07:00
|
|
|
|
2013-11-20 14:17:12 -08:00
|
|
|
(|| f(unsafe_ptr)).finally(|| put(value_cell.take()));
|
2013-05-19 14:57:47 -07:00
|
|
|
}
|
|
|
|
|
2013-08-03 17:39:59 -07:00
|
|
|
/// Borrow a mutable reference to the thread-local value
|
2013-05-19 14:57:47 -07:00
|
|
|
///
|
|
|
|
/// # Safety Note
|
|
|
|
///
|
2013-08-03 17:39:59 -07:00
|
|
|
/// Because this leaves the value in thread-local storage it is possible
|
2013-05-19 14:57:47 -07:00
|
|
|
/// For the Scheduler pointer to be aliased
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-05-19 14:57:47 -07:00
|
|
|
pub unsafe fn unsafe_borrow<T>() -> *mut T {
|
2013-08-17 01:24:29 -07:00
|
|
|
let key = tls_key();
|
2013-08-29 22:28:49 +10:00
|
|
|
let void_ptr = tls::get(key);
|
2013-08-17 01:24:29 -07:00
|
|
|
if void_ptr.is_null() {
|
|
|
|
rtabort!("thread-local pointer is null. bogus!");
|
2013-08-12 19:09:46 -07:00
|
|
|
}
|
2013-08-29 22:28:49 +10:00
|
|
|
void_ptr as *mut T
|
2013-08-12 19:09:46 -07:00
|
|
|
}
|
|
|
|
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(not(stage0), not(windows))]
|
|
|
|
pub unsafe fn unsafe_borrow<T>() -> *mut T {
|
|
|
|
if RT_TLS_PTR.is_null() {
|
|
|
|
rtabort!("thread-local pointer is null. bogus!");
|
|
|
|
}
|
|
|
|
RT_TLS_PTR as *mut T
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-08-12 19:09:46 -07:00
|
|
|
pub unsafe fn try_unsafe_borrow<T>() -> Option<*mut T> {
|
2013-09-25 00:56:18 -07:00
|
|
|
match maybe_tls_key() {
|
|
|
|
Some(key) => {
|
|
|
|
let void_ptr = tls::get(key);
|
|
|
|
if void_ptr.is_null() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(void_ptr as *mut T)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
None => None
|
2013-05-19 14:57:47 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(not(stage0), not(windows))]
|
|
|
|
pub unsafe fn try_unsafe_borrow<T>() -> Option<*mut T> {
|
|
|
|
if RT_TLS_PTR.is_null() {
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(RT_TLS_PTR as *mut T)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-17 01:24:29 -07:00
|
|
|
#[inline]
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(stage0)]
|
|
|
|
#[cfg(windows)]
|
2013-05-19 14:57:47 -07:00
|
|
|
fn tls_key() -> tls::Key {
|
2013-05-19 14:39:46 -07:00
|
|
|
match maybe_tls_key() {
|
|
|
|
Some(key) => key,
|
2013-06-17 23:24:50 -07:00
|
|
|
None => rtabort!("runtime tls key not initialized")
|
2013-05-19 14:39:46 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-08-17 02:12:08 -07:00
|
|
|
#[inline]
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(not(test), stage0)]
|
|
|
|
#[cfg(not(test), windows)]
|
2013-08-17 02:12:08 -07:00
|
|
|
pub fn maybe_tls_key() -> Option<tls::Key> {
|
2013-05-19 14:39:46 -07:00
|
|
|
unsafe {
|
|
|
|
// 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.
|
2013-08-17 02:12:08 -07:00
|
|
|
if RT_TLS_KEY != -1 {
|
|
|
|
return Some(RT_TLS_KEY);
|
2013-05-19 14:39:46 -07:00
|
|
|
} else {
|
|
|
|
return None;
|
|
|
|
}
|
|
|
|
}
|
2013-08-17 02:12:08 -07:00
|
|
|
}
|
2013-05-19 14:39:46 -07:00
|
|
|
|
2013-08-17 02:12:08 -07:00
|
|
|
#[inline]
|
2013-11-06 01:17:04 -05:00
|
|
|
#[cfg(test, stage0)]
|
|
|
|
#[cfg(test, windows)]
|
2013-08-17 02:12:08 -07:00
|
|
|
pub fn maybe_tls_key() -> Option<tls::Key> {
|
2013-10-09 10:34:27 -07:00
|
|
|
unsafe { ::cast::transmute(::realstd::rt::shouldnt_be_public::maybe_tls_key()) }
|
2013-05-19 14:39:46 -07:00
|
|
|
}
|