2014-12-12 23:39:27 +00:00
|
|
|
// Copyright 2014-2015 The Rust Project Developers. See the COPYRIGHT
|
2014-11-14 22:20:57 +00:00
|
|
|
// 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.
|
|
|
|
|
|
|
|
//! Thread local storage
|
2015-03-20 07:46:13 +00:00
|
|
|
|
2015-08-13 17:12:38 +00:00
|
|
|
#![unstable(feature = "thread_local_internals", issue = "0")]
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
use cell::UnsafeCell;
|
2016-11-25 18:21:49 +00:00
|
|
|
use fmt;
|
2015-12-08 15:23:37 +00:00
|
|
|
use mem;
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
/// A thread local storage key which owns its contents.
|
|
|
|
///
|
|
|
|
/// This key uses the fastest possible implementation available to it for the
|
2017-05-14 18:06:54 +00:00
|
|
|
/// target platform. It is instantiated with the [`thread_local!`] macro and the
|
|
|
|
/// primary method is the [`with`] method.
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
2017-05-14 18:06:54 +00:00
|
|
|
/// The [`with`] method yields a reference to the contained value which cannot be
|
2015-05-08 15:12:29 +00:00
|
|
|
/// sent across threads or escape the given closure.
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
|
|
|
/// # Initialization and Destruction
|
|
|
|
///
|
2017-05-14 18:06:54 +00:00
|
|
|
/// Initialization is dynamically performed on the first call to [`with`]
|
|
|
|
/// within a thread, and values that implement [`Drop`] get destructed when a
|
2017-02-15 22:26:29 +00:00
|
|
|
/// thread exits. Some caveats apply, which are explained below.
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
2017-09-07 19:57:08 +00:00
|
|
|
/// A `LocalKey`'s initializer cannot recursively depend on itself, and using
|
|
|
|
/// a `LocalKey` in this way will cause the initializer to infinitely recurse
|
|
|
|
/// on the first call to `with`.
|
|
|
|
///
|
2015-03-12 01:11:40 +00:00
|
|
|
/// # Examples
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// use std::cell::RefCell;
|
2015-02-17 23:10:25 +00:00
|
|
|
/// use std::thread;
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
2015-02-23 16:24:50 +00:00
|
|
|
/// thread_local!(static FOO: RefCell<u32> = RefCell::new(1));
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
|
|
|
/// FOO.with(|f| {
|
|
|
|
/// assert_eq!(*f.borrow(), 1);
|
|
|
|
/// *f.borrow_mut() = 2;
|
|
|
|
/// });
|
|
|
|
///
|
|
|
|
/// // each thread starts out with the initial value of 1
|
2015-02-17 23:10:25 +00:00
|
|
|
/// thread::spawn(move|| {
|
2014-11-14 22:20:57 +00:00
|
|
|
/// FOO.with(|f| {
|
|
|
|
/// assert_eq!(*f.borrow(), 1);
|
|
|
|
/// *f.borrow_mut() = 3;
|
|
|
|
/// });
|
2015-01-06 05:59:45 +00:00
|
|
|
/// });
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
|
|
|
/// // we retain our original value of 2 despite the child thread
|
|
|
|
/// FOO.with(|f| {
|
|
|
|
/// assert_eq!(*f.borrow(), 2);
|
|
|
|
/// });
|
|
|
|
/// ```
|
2016-01-29 21:46:47 +00:00
|
|
|
///
|
|
|
|
/// # Platform-specific behavior
|
|
|
|
///
|
|
|
|
/// Note that a "best effort" is made to ensure that destructors for types
|
2016-04-25 20:01:19 +00:00
|
|
|
/// stored in thread local storage are run, but not all platforms can guarantee
|
2016-01-29 21:46:47 +00:00
|
|
|
/// that destructors will be run for all types in thread local storage. For
|
|
|
|
/// example, there are a number of known caveats where destructors are not run:
|
|
|
|
///
|
|
|
|
/// 1. On Unix systems when pthread-based TLS is being used, destructors will
|
|
|
|
/// not be run for TLS values on the main thread when it exits. Note that the
|
|
|
|
/// application will exit immediately after the main thread exits as well.
|
|
|
|
/// 2. On all platforms it's possible for TLS to re-initialize other TLS slots
|
|
|
|
/// during destruction. Some platforms ensure that this cannot happen
|
|
|
|
/// infinitely by preventing re-initialization of any slot that has been
|
|
|
|
/// destroyed, but not all platforms have this guard. Those platforms that do
|
|
|
|
/// not guard typically have a synthetic limit after which point no more
|
|
|
|
/// destructors are run.
|
2017-03-12 18:13:35 +00:00
|
|
|
/// 3. On macOS, initializing TLS during destruction of other TLS slots can
|
2016-01-29 21:46:47 +00:00
|
|
|
/// sometimes cancel *all* destructors for the current thread, whether or not
|
|
|
|
/// the slots have already had their destructors run or not.
|
2017-05-14 18:06:54 +00:00
|
|
|
///
|
|
|
|
/// [`with`]: ../../std/thread/struct.LocalKey.html#method.with
|
|
|
|
/// [`thread_local!`]: ../../std/macro.thread_local.html
|
|
|
|
/// [`Drop`]: ../../std/ops/trait.Drop.html
|
2015-01-24 05:48:20 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-12-11 20:42:29 +00:00
|
|
|
pub struct LocalKey<T: 'static> {
|
|
|
|
// This outer `LocalKey<T>` type is what's going to be stored in statics,
|
|
|
|
// but actual data inside will sometimes be tagged with #[thread_local].
|
|
|
|
// It's not valid for a true static to reference a #[thread_local] static,
|
|
|
|
// so we get around that by exposing an accessor through a layer of function
|
|
|
|
// indirection (this thunk).
|
|
|
|
//
|
|
|
|
// Note that the thunk is itself unsafe because the returned lifetime of the
|
|
|
|
// slot where data lives, `'static`, is not actually valid. The lifetime
|
2017-08-08 15:22:51 +00:00
|
|
|
// here is actually slightly shorter than the currently running thread!
|
2014-11-14 22:20:57 +00:00
|
|
|
//
|
2015-12-11 20:42:29 +00:00
|
|
|
// Although this is an extra layer of indirection, it should in theory be
|
|
|
|
// trivially devirtualizable by LLVM because the value of `inner` never
|
|
|
|
// changes and the constant should be readonly within a crate. This mainly
|
|
|
|
// only runs into problems when TLS statics are exported across crates.
|
2017-08-08 15:22:51 +00:00
|
|
|
inner: unsafe fn() -> Option<&'static UnsafeCell<Option<T>>>,
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
// initialization routine to invoke to create a value
|
2015-05-27 08:18:36 +00:00
|
|
|
init: fn() -> T,
|
2014-11-14 17:18:10 +00:00
|
|
|
}
|
2014-11-14 22:20:57 +00:00
|
|
|
|
2017-01-29 13:31:47 +00:00
|
|
|
#[stable(feature = "std_debug", since = "1.16.0")]
|
2016-11-25 18:21:49 +00:00
|
|
|
impl<T: 'static> fmt::Debug for LocalKey<T> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
f.pad("LocalKey { .. }")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-05-14 18:06:54 +00:00
|
|
|
/// Declare a new thread local storage key of type [`std::thread::LocalKey`].
|
2015-05-27 08:18:36 +00:00
|
|
|
///
|
2016-06-04 19:19:22 +00:00
|
|
|
/// # Syntax
|
|
|
|
///
|
|
|
|
/// The macro wraps any number of static declarations and makes them thread local.
|
2017-04-01 03:06:34 +00:00
|
|
|
/// Publicity and attributes for each static are allowed. Example:
|
2016-06-04 19:19:22 +00:00
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// use std::cell::RefCell;
|
|
|
|
/// thread_local! {
|
|
|
|
/// pub static FOO: RefCell<u32> = RefCell::new(1);
|
|
|
|
///
|
|
|
|
/// #[allow(unused)]
|
|
|
|
/// static BAR: RefCell<f32> = RefCell::new(1.0);
|
|
|
|
/// }
|
|
|
|
/// # fn main() {}
|
|
|
|
/// ```
|
|
|
|
///
|
2017-05-14 18:06:54 +00:00
|
|
|
/// See [LocalKey documentation][`std::thread::LocalKey`] for more
|
2015-05-28 06:24:27 +00:00
|
|
|
/// information.
|
2017-05-14 18:06:54 +00:00
|
|
|
///
|
|
|
|
/// [`std::thread::LocalKey`]: ../std/thread/struct.LocalKey.html
|
2014-11-14 22:20:57 +00:00
|
|
|
#[macro_export]
|
2015-05-27 08:18:36 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2015-03-01 03:09:42 +00:00
|
|
|
#[allow_internal_unstable]
|
2015-05-28 06:24:27 +00:00
|
|
|
macro_rules! thread_local {
|
2017-04-01 03:06:34 +00:00
|
|
|
// empty (base case for the recursion)
|
2016-06-04 19:19:22 +00:00
|
|
|
() => {};
|
|
|
|
|
2017-04-19 02:29:40 +00:00
|
|
|
// process multiple declarations
|
|
|
|
($(#[$attr:meta])* $vis:vis static $name:ident: $t:ty = $init:expr; $($rest:tt)*) => (
|
|
|
|
__thread_local_inner!($(#[$attr])* $vis $name, $t, $init);
|
2016-06-04 19:19:22 +00:00
|
|
|
thread_local!($($rest)*);
|
|
|
|
);
|
|
|
|
|
2017-04-19 02:29:40 +00:00
|
|
|
// handle a single declaration
|
|
|
|
($(#[$attr:meta])* $vis:vis static $name:ident: $t:ty = $init:expr) => (
|
|
|
|
__thread_local_inner!($(#[$attr])* $vis $name, $t, $init);
|
2015-05-28 06:24:27 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[doc(hidden)]
|
|
|
|
#[unstable(feature = "thread_local_internals",
|
2015-08-13 17:12:38 +00:00
|
|
|
reason = "should not be necessary",
|
|
|
|
issue = "0")]
|
2015-05-28 06:24:27 +00:00
|
|
|
#[macro_export]
|
|
|
|
#[allow_internal_unstable]
|
2017-08-25 15:39:02 +00:00
|
|
|
#[allow_internal_unsafe]
|
2015-05-28 06:24:27 +00:00
|
|
|
macro_rules! __thread_local_inner {
|
2017-08-13 13:42:10 +00:00
|
|
|
(@key $(#[$attr:meta])* $vis:vis $name:ident, $t:ty, $init:expr) => {
|
|
|
|
{
|
|
|
|
#[inline]
|
2017-04-01 03:06:34 +00:00
|
|
|
fn __init() -> $t { $init }
|
|
|
|
|
2017-08-08 15:22:51 +00:00
|
|
|
unsafe fn __getit() -> $crate::option::Option<
|
2017-04-01 03:06:34 +00:00
|
|
|
&'static $crate::cell::UnsafeCell<
|
|
|
|
$crate::option::Option<$t>>>
|
|
|
|
{
|
|
|
|
#[thread_local]
|
|
|
|
#[cfg(target_thread_local)]
|
|
|
|
static __KEY: $crate::thread::__FastLocalKeyInner<$t> =
|
|
|
|
$crate::thread::__FastLocalKeyInner::new();
|
|
|
|
|
|
|
|
#[cfg(not(target_thread_local))]
|
|
|
|
static __KEY: $crate::thread::__OsLocalKeyInner<$t> =
|
|
|
|
$crate::thread::__OsLocalKeyInner::new();
|
|
|
|
|
|
|
|
__KEY.get()
|
|
|
|
}
|
|
|
|
|
2017-08-08 15:22:51 +00:00
|
|
|
unsafe {
|
|
|
|
$crate::thread::LocalKey::new(__getit, __init)
|
|
|
|
}
|
2017-08-13 13:42:10 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
($(#[$attr:meta])* $vis:vis $name:ident, $t:ty, $init:expr) => {
|
|
|
|
$(#[$attr])* $vis const $name: $crate::thread::LocalKey<$t> =
|
|
|
|
__thread_local_inner!(@key $(#[$attr])* $vis $name, $t, $init);
|
2017-04-01 03:06:34 +00:00
|
|
|
}
|
2015-05-28 06:24:27 +00:00
|
|
|
}
|
|
|
|
|
2014-12-30 23:20:47 +00:00
|
|
|
/// Indicator of the state of a thread local storage key.
|
2015-06-10 01:15:22 +00:00
|
|
|
#[unstable(feature = "thread_local_state",
|
2015-08-13 17:12:38 +00:00
|
|
|
reason = "state querying was recently added",
|
|
|
|
issue = "27716")]
|
2016-11-25 18:21:49 +00:00
|
|
|
#[derive(Debug, Eq, PartialEq, Copy, Clone)]
|
2015-03-20 07:46:13 +00:00
|
|
|
pub enum LocalKeyState {
|
2014-12-30 23:20:47 +00:00
|
|
|
/// All keys are in this state whenever a thread starts. Keys will
|
2017-05-14 18:06:54 +00:00
|
|
|
/// transition to the `Valid` state once the first call to [`with`] happens
|
2014-12-30 23:20:47 +00:00
|
|
|
/// and the initialization expression succeeds.
|
|
|
|
///
|
|
|
|
/// Keys in the `Uninitialized` state will yield a reference to the closure
|
2017-05-14 18:06:54 +00:00
|
|
|
/// passed to [`with`] so long as the initialization routine does not panic.
|
|
|
|
///
|
|
|
|
/// [`with`]: ../../std/thread/struct.LocalKey.html#method.with
|
2014-12-30 23:20:47 +00:00
|
|
|
Uninitialized,
|
|
|
|
|
|
|
|
/// Once a key has been accessed successfully, it will enter the `Valid`
|
|
|
|
/// state. Keys in the `Valid` state will remain so until the thread exits,
|
|
|
|
/// at which point the destructor will be run and the key will enter the
|
|
|
|
/// `Destroyed` state.
|
|
|
|
///
|
|
|
|
/// Keys in the `Valid` state will be guaranteed to yield a reference to the
|
2017-05-14 18:06:54 +00:00
|
|
|
/// closure passed to [`with`].
|
|
|
|
///
|
|
|
|
/// [`with`]: ../../std/thread/struct.LocalKey.html#method.with
|
2014-12-30 23:20:47 +00:00
|
|
|
Valid,
|
|
|
|
|
|
|
|
/// When a thread exits, the destructors for keys will be run (if
|
|
|
|
/// necessary). While a destructor is running, and possibly after a
|
|
|
|
/// destructor has run, a key is in the `Destroyed` state.
|
|
|
|
///
|
|
|
|
/// Keys in the `Destroyed` states will trigger a panic when accessed via
|
2017-05-14 18:06:54 +00:00
|
|
|
/// [`with`].
|
|
|
|
///
|
|
|
|
/// [`with`]: ../../std/thread/struct.LocalKey.html#method.with
|
2014-12-30 23:20:47 +00:00
|
|
|
Destroyed,
|
|
|
|
}
|
|
|
|
|
2017-07-10 23:26:11 +00:00
|
|
|
/// An error returned by [`LocalKey::try_with`](struct.LocalKey.html#method.try_with).
|
|
|
|
#[unstable(feature = "thread_local_state",
|
|
|
|
reason = "state querying was recently added",
|
|
|
|
issue = "27716")]
|
|
|
|
pub struct AccessError {
|
|
|
|
_private: (),
|
|
|
|
}
|
|
|
|
|
|
|
|
#[unstable(feature = "thread_local_state",
|
|
|
|
reason = "state querying was recently added",
|
|
|
|
issue = "27716")]
|
|
|
|
impl fmt::Debug for AccessError {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
f.debug_struct("AccessError").finish()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[unstable(feature = "thread_local_state",
|
|
|
|
reason = "state querying was recently added",
|
|
|
|
issue = "27716")]
|
|
|
|
impl fmt::Display for AccessError {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
fmt::Display::fmt("already destroyed", f)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-20 07:46:13 +00:00
|
|
|
impl<T: 'static> LocalKey<T> {
|
2015-05-27 08:18:36 +00:00
|
|
|
#[doc(hidden)]
|
2015-05-28 06:24:27 +00:00
|
|
|
#[unstable(feature = "thread_local_internals",
|
2015-08-13 17:12:38 +00:00
|
|
|
reason = "recently added to create a key",
|
|
|
|
issue = "0")]
|
2017-08-08 15:22:51 +00:00
|
|
|
pub const unsafe fn new(inner: unsafe fn() -> Option<&'static UnsafeCell<Option<T>>>,
|
|
|
|
init: fn() -> T) -> LocalKey<T> {
|
2015-05-27 08:18:36 +00:00
|
|
|
LocalKey {
|
2017-08-07 05:54:09 +00:00
|
|
|
inner,
|
|
|
|
init,
|
2015-05-27 08:18:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-13 14:21:32 +00:00
|
|
|
/// Acquires a reference to the value in this TLS key.
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
|
|
|
/// This will lazily initialize the value if this thread has not referenced
|
|
|
|
/// this key yet.
|
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
|
|
|
/// This function will `panic!()` if the key currently has its
|
|
|
|
/// destructor running, and it **may** panic if the destructor has
|
|
|
|
/// previously been run for this thread.
|
2015-01-24 05:48:20 +00:00
|
|
|
#[stable(feature = "rust1", since = "1.0.0")]
|
2014-12-10 15:49:45 +00:00
|
|
|
pub fn with<F, R>(&'static self, f: F) -> R
|
|
|
|
where F: FnOnce(&T) -> R {
|
2017-07-12 16:55:39 +00:00
|
|
|
self.try_with(f).expect("cannot access a TLS value during or \
|
|
|
|
after it is destroyed")
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
|
2014-12-09 04:20:03 +00:00
|
|
|
unsafe fn init(&self, slot: &UnsafeCell<Option<T>>) -> &T {
|
2014-12-30 23:20:47 +00:00
|
|
|
// Execute the initialization up front, *then* move it into our slot,
|
|
|
|
// just in case initialization fails.
|
|
|
|
let value = (self.init)();
|
|
|
|
let ptr = slot.get();
|
2015-12-08 15:23:37 +00:00
|
|
|
|
|
|
|
// note that this can in theory just be `*ptr = Some(value)`, but due to
|
|
|
|
// the compiler will currently codegen that pattern with something like:
|
|
|
|
//
|
|
|
|
// ptr::drop_in_place(ptr)
|
|
|
|
// ptr::write(ptr, Some(value))
|
|
|
|
//
|
|
|
|
// Due to this pattern it's possible for the destructor of the value in
|
|
|
|
// `ptr` (e.g. if this is being recursively initialized) to re-access
|
|
|
|
// TLS, in which case there will be a `&` and `&mut` pointer to the same
|
|
|
|
// value (an aliasing violation). To avoid setting the "I'm running a
|
|
|
|
// destructor" flag we just use `mem::replace` which should sequence the
|
|
|
|
// operations a little differently and make this safe to call.
|
|
|
|
mem::replace(&mut *ptr, Some(value));
|
|
|
|
|
2014-12-30 23:20:47 +00:00
|
|
|
(*ptr).as_ref().unwrap()
|
2014-12-09 04:20:03 +00:00
|
|
|
}
|
|
|
|
|
2014-12-30 23:20:47 +00:00
|
|
|
/// Query the current state of this key.
|
|
|
|
///
|
|
|
|
/// A key is initially in the `Uninitialized` state whenever a thread
|
2017-05-14 18:06:54 +00:00
|
|
|
/// starts. It will remain in this state up until the first call to [`with`]
|
2014-12-30 23:20:47 +00:00
|
|
|
/// within a thread has run the initialization expression successfully.
|
|
|
|
///
|
|
|
|
/// Once the initialization expression succeeds, the key transitions to the
|
2017-05-14 18:06:54 +00:00
|
|
|
/// `Valid` state which will guarantee that future calls to [`with`] will
|
2017-10-17 09:02:25 +00:00
|
|
|
/// succeed within the thread. Some keys might skip the `Uninitialized`
|
|
|
|
/// state altogether and start in the `Valid` state as an optimization
|
|
|
|
/// (e.g. keys initialized with a constant expression), but no guarantees
|
|
|
|
/// are made.
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
2014-12-30 23:20:47 +00:00
|
|
|
/// When a thread exits, each key will be destroyed in turn, and as keys are
|
|
|
|
/// destroyed they will enter the `Destroyed` state just before the
|
|
|
|
/// destructor starts to run. Keys may remain in the `Destroyed` state after
|
|
|
|
/// destruction has completed. Keys without destructors (e.g. with types
|
2017-05-14 18:06:54 +00:00
|
|
|
/// that are [`Copy`]), may never enter the `Destroyed` state.
|
2014-11-14 22:20:57 +00:00
|
|
|
///
|
2016-04-25 20:01:19 +00:00
|
|
|
/// Keys in the `Uninitialized` state can be accessed so long as the
|
2014-12-30 23:20:47 +00:00
|
|
|
/// initialization does not panic. Keys in the `Valid` state are guaranteed
|
|
|
|
/// to be able to be accessed. Keys in the `Destroyed` state will panic on
|
2017-05-14 18:06:54 +00:00
|
|
|
/// any call to [`with`].
|
|
|
|
///
|
|
|
|
/// [`with`]: ../../std/thread/struct.LocalKey.html#method.with
|
|
|
|
/// [`Copy`]: ../../std/marker/trait.Copy.html
|
2015-06-10 01:15:22 +00:00
|
|
|
#[unstable(feature = "thread_local_state",
|
2015-08-13 17:12:38 +00:00
|
|
|
reason = "state querying was recently added",
|
|
|
|
issue = "27716")]
|
2015-03-20 07:46:13 +00:00
|
|
|
pub fn state(&'static self) -> LocalKeyState {
|
2014-12-30 23:20:47 +00:00
|
|
|
unsafe {
|
2015-12-11 20:42:29 +00:00
|
|
|
match (self.inner)() {
|
2014-12-30 23:20:47 +00:00
|
|
|
Some(cell) => {
|
|
|
|
match *cell.get() {
|
2015-03-20 07:46:13 +00:00
|
|
|
Some(..) => LocalKeyState::Valid,
|
|
|
|
None => LocalKeyState::Uninitialized,
|
2014-12-30 23:20:47 +00:00
|
|
|
}
|
|
|
|
}
|
2015-03-20 07:46:13 +00:00
|
|
|
None => LocalKeyState::Destroyed,
|
2014-12-30 23:20:47 +00:00
|
|
|
}
|
|
|
|
}
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
2017-07-10 23:26:11 +00:00
|
|
|
|
|
|
|
/// Acquires a reference to the value in this TLS key.
|
|
|
|
///
|
|
|
|
/// This will lazily initialize the value if this thread has not referenced
|
|
|
|
/// this key yet. If the key has been destroyed (which may happen if this is called
|
|
|
|
/// in a destructor), this function will return a ThreadLocalError.
|
|
|
|
///
|
|
|
|
/// # Panics
|
|
|
|
///
|
|
|
|
/// This function will still `panic!()` if the key is uninitialized and the
|
|
|
|
/// key's initializer panics.
|
|
|
|
#[unstable(feature = "thread_local_state",
|
|
|
|
reason = "state querying was recently added",
|
|
|
|
issue = "27716")]
|
|
|
|
pub fn try_with<F, R>(&'static self, f: F) -> Result<R, AccessError>
|
|
|
|
where F: FnOnce(&T) -> R {
|
|
|
|
unsafe {
|
|
|
|
let slot = (self.inner)().ok_or(AccessError {
|
|
|
|
_private: (),
|
|
|
|
})?;
|
|
|
|
Ok(f(match *slot.get() {
|
|
|
|
Some(ref inner) => inner,
|
|
|
|
None => self.init(slot),
|
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
|
2016-02-17 07:07:09 +00:00
|
|
|
#[doc(hidden)]
|
|
|
|
#[cfg(target_thread_local)]
|
|
|
|
pub mod fast {
|
|
|
|
use cell::{Cell, UnsafeCell};
|
|
|
|
use fmt;
|
|
|
|
use mem;
|
|
|
|
use ptr;
|
|
|
|
use sys::fast_thread_local::{register_dtor, requires_move_before_drop};
|
|
|
|
|
|
|
|
pub struct Key<T> {
|
|
|
|
inner: UnsafeCell<Option<T>>,
|
|
|
|
|
|
|
|
// Metadata to keep track of the state of the destructor. Remember that
|
|
|
|
// these variables are thread-local, not global.
|
|
|
|
dtor_registered: Cell<bool>,
|
|
|
|
dtor_running: Cell<bool>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> fmt::Debug for Key<T> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
f.pad("Key { .. }")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl<T> Key<T> {
|
|
|
|
pub const fn new() -> Key<T> {
|
|
|
|
Key {
|
|
|
|
inner: UnsafeCell::new(None),
|
|
|
|
dtor_registered: Cell::new(false),
|
|
|
|
dtor_running: Cell::new(false)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-08-08 15:22:51 +00:00
|
|
|
pub unsafe fn get(&self) -> Option<&'static UnsafeCell<Option<T>>> {
|
|
|
|
if mem::needs_drop::<T>() && self.dtor_running.get() {
|
|
|
|
return None
|
2016-02-17 07:07:09 +00:00
|
|
|
}
|
2017-08-08 15:22:51 +00:00
|
|
|
self.register_dtor();
|
|
|
|
Some(&*(&self.inner as *const _))
|
2016-02-17 07:07:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
unsafe fn register_dtor(&self) {
|
|
|
|
if !mem::needs_drop::<T>() || self.dtor_registered.get() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
register_dtor(self as *const _ as *mut u8,
|
|
|
|
destroy_value::<T>);
|
|
|
|
self.dtor_registered.set(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
unsafe extern fn destroy_value<T>(ptr: *mut u8) {
|
|
|
|
let ptr = ptr as *mut Key<T>;
|
|
|
|
// Right before we run the user destructor be sure to flag the
|
|
|
|
// destructor as running for this thread so calls to `get` will return
|
|
|
|
// `None`.
|
|
|
|
(*ptr).dtor_running.set(true);
|
|
|
|
|
|
|
|
// Some implementations may require us to move the value before we drop
|
|
|
|
// it as it could get re-initialized in-place during destruction.
|
|
|
|
//
|
|
|
|
// Hence, we use `ptr::read` on those platforms (to move to a "safe"
|
|
|
|
// location) instead of drop_in_place.
|
|
|
|
if requires_move_before_drop() {
|
|
|
|
ptr::read((*ptr).inner.get());
|
|
|
|
} else {
|
|
|
|
ptr::drop_in_place((*ptr).inner.get());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-04-15 19:26:27 +00:00
|
|
|
#[doc(hidden)]
|
2015-12-11 20:42:29 +00:00
|
|
|
pub mod os {
|
2015-05-27 08:18:36 +00:00
|
|
|
use cell::{Cell, UnsafeCell};
|
2016-11-25 18:21:49 +00:00
|
|
|
use fmt;
|
2015-05-27 08:18:36 +00:00
|
|
|
use marker;
|
2015-01-21 17:23:27 +00:00
|
|
|
use ptr;
|
2014-11-14 22:20:57 +00:00
|
|
|
use sys_common::thread_local::StaticKey as OsStaticKey;
|
|
|
|
|
|
|
|
pub struct Key<T> {
|
|
|
|
// OS-TLS key that we'll use to key off.
|
2015-05-27 08:18:36 +00:00
|
|
|
os: OsStaticKey,
|
|
|
|
marker: marker::PhantomData<Cell<T>>,
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
|
2016-11-25 18:21:49 +00:00
|
|
|
impl<T> fmt::Debug for Key<T> {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
|
|
f.pad("Key { .. }")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-01-06 22:33:42 +00:00
|
|
|
unsafe impl<T> ::marker::Sync for Key<T> { }
|
2014-12-06 16:39:25 +00:00
|
|
|
|
2014-11-14 22:20:57 +00:00
|
|
|
struct Value<T: 'static> {
|
|
|
|
key: &'static Key<T>,
|
2015-05-27 08:18:36 +00:00
|
|
|
value: UnsafeCell<Option<T>>,
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
|
2015-05-27 08:18:36 +00:00
|
|
|
impl<T: 'static> Key<T> {
|
|
|
|
pub const fn new() -> Key<T> {
|
|
|
|
Key {
|
|
|
|
os: OsStaticKey::new(Some(destroy_value::<T>)),
|
|
|
|
marker: marker::PhantomData
|
|
|
|
}
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
|
2017-08-08 15:22:51 +00:00
|
|
|
pub unsafe fn get(&'static self) -> Option<&'static UnsafeCell<Option<T>>> {
|
|
|
|
let ptr = self.os.get() as *mut Value<T>;
|
|
|
|
if !ptr.is_null() {
|
|
|
|
if ptr as usize == 1 {
|
|
|
|
return None
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
2017-08-08 15:22:51 +00:00
|
|
|
return Some(&(*ptr).value);
|
2016-02-04 12:20:20 +00:00
|
|
|
}
|
2017-08-08 15:22:51 +00:00
|
|
|
|
|
|
|
// If the lookup returned null, we haven't initialized our own
|
|
|
|
// local copy, so do that now.
|
|
|
|
let ptr: Box<Value<T>> = box Value {
|
|
|
|
key: self,
|
|
|
|
value: UnsafeCell::new(None),
|
|
|
|
};
|
|
|
|
let ptr = Box::into_raw(ptr);
|
|
|
|
self.os.set(ptr as *mut u8);
|
|
|
|
Some(&(*ptr).value)
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-17 07:07:09 +00:00
|
|
|
unsafe extern fn destroy_value<T: 'static>(ptr: *mut u8) {
|
2014-11-14 22:20:57 +00:00
|
|
|
// The OS TLS ensures that this key contains a NULL value when this
|
|
|
|
// destructor starts to run. We set it back to a sentinel value of 1 to
|
|
|
|
// ensure that any future calls to `get` for this thread will return
|
|
|
|
// `None`.
|
|
|
|
//
|
|
|
|
// Note that to prevent an infinite loop we reset it back to null right
|
|
|
|
// before we return from the destructor ourselves.
|
2015-04-15 19:26:27 +00:00
|
|
|
let ptr = Box::from_raw(ptr as *mut Value<T>);
|
2014-11-14 22:20:57 +00:00
|
|
|
let key = ptr.key;
|
|
|
|
key.os.set(1 as *mut u8);
|
|
|
|
drop(ptr);
|
2015-01-19 05:27:09 +00:00
|
|
|
key.os.set(ptr::null_mut());
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-09-22 20:04:48 +00:00
|
|
|
#[cfg(all(test, not(target_os = "emscripten")))]
|
2014-11-14 22:20:57 +00:00
|
|
|
mod tests {
|
2014-12-23 19:53:35 +00:00
|
|
|
use sync::mpsc::{channel, Sender};
|
2015-05-27 08:18:36 +00:00
|
|
|
use cell::{Cell, UnsafeCell};
|
2015-03-20 07:46:13 +00:00
|
|
|
use super::LocalKeyState;
|
2015-02-17 23:10:25 +00:00
|
|
|
use thread;
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
struct Foo(Sender<()>);
|
|
|
|
|
|
|
|
impl Drop for Foo {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
let Foo(ref s) = *self;
|
2014-12-23 19:53:35 +00:00
|
|
|
s.send(()).unwrap();
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn smoke_no_dtor() {
|
2015-05-27 08:18:36 +00:00
|
|
|
thread_local!(static FOO: Cell<i32> = Cell::new(1));
|
2014-11-14 22:20:57 +00:00
|
|
|
|
2015-05-27 08:18:36 +00:00
|
|
|
FOO.with(|f| {
|
|
|
|
assert_eq!(f.get(), 1);
|
|
|
|
f.set(2);
|
2014-11-14 22:20:57 +00:00
|
|
|
});
|
|
|
|
let (tx, rx) = channel();
|
2015-02-17 23:10:25 +00:00
|
|
|
let _t = thread::spawn(move|| {
|
2015-05-27 08:18:36 +00:00
|
|
|
FOO.with(|f| {
|
|
|
|
assert_eq!(f.get(), 1);
|
2014-11-14 22:20:57 +00:00
|
|
|
});
|
2014-12-23 19:53:35 +00:00
|
|
|
tx.send(()).unwrap();
|
2014-11-14 22:20:57 +00:00
|
|
|
});
|
2014-12-23 19:53:35 +00:00
|
|
|
rx.recv().unwrap();
|
2014-11-14 22:20:57 +00:00
|
|
|
|
2015-05-27 08:18:36 +00:00
|
|
|
FOO.with(|f| {
|
|
|
|
assert_eq!(f.get(), 2);
|
2014-11-14 22:20:57 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-12-30 23:20:47 +00:00
|
|
|
#[test]
|
|
|
|
fn states() {
|
|
|
|
struct Foo;
|
|
|
|
impl Drop for Foo {
|
|
|
|
fn drop(&mut self) {
|
2015-03-20 07:46:13 +00:00
|
|
|
assert!(FOO.state() == LocalKeyState::Destroyed);
|
2014-12-30 23:20:47 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
fn foo() -> Foo {
|
2015-03-20 07:46:13 +00:00
|
|
|
assert!(FOO.state() == LocalKeyState::Uninitialized);
|
2014-12-30 23:20:47 +00:00
|
|
|
Foo
|
|
|
|
}
|
|
|
|
thread_local!(static FOO: Foo = foo());
|
|
|
|
|
2015-02-17 23:10:25 +00:00
|
|
|
thread::spawn(|| {
|
2015-03-20 07:46:13 +00:00
|
|
|
assert!(FOO.state() == LocalKeyState::Uninitialized);
|
2014-12-30 23:20:47 +00:00
|
|
|
FOO.with(|_| {
|
2015-03-20 07:46:13 +00:00
|
|
|
assert!(FOO.state() == LocalKeyState::Valid);
|
2014-12-30 23:20:47 +00:00
|
|
|
});
|
2015-03-20 07:46:13 +00:00
|
|
|
assert!(FOO.state() == LocalKeyState::Valid);
|
2014-12-30 23:20:47 +00:00
|
|
|
}).join().ok().unwrap();
|
|
|
|
}
|
|
|
|
|
2014-11-14 22:20:57 +00:00
|
|
|
#[test]
|
|
|
|
fn smoke_dtor() {
|
2015-05-27 08:18:36 +00:00
|
|
|
thread_local!(static FOO: UnsafeCell<Option<Foo>> = UnsafeCell::new(None));
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
let (tx, rx) = channel();
|
2015-02-17 23:10:25 +00:00
|
|
|
let _t = thread::spawn(move|| unsafe {
|
2014-11-14 22:20:57 +00:00
|
|
|
let mut tx = Some(tx);
|
|
|
|
FOO.with(|f| {
|
|
|
|
*f.get() = Some(Foo(tx.take().unwrap()));
|
|
|
|
});
|
|
|
|
});
|
2014-12-23 19:53:35 +00:00
|
|
|
rx.recv().unwrap();
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn circular() {
|
|
|
|
struct S1;
|
|
|
|
struct S2;
|
2015-05-27 08:18:36 +00:00
|
|
|
thread_local!(static K1: UnsafeCell<Option<S1>> = UnsafeCell::new(None));
|
|
|
|
thread_local!(static K2: UnsafeCell<Option<S2>> = UnsafeCell::new(None));
|
2015-02-23 16:24:50 +00:00
|
|
|
static mut HITS: u32 = 0;
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
impl Drop for S1 {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
unsafe {
|
|
|
|
HITS += 1;
|
2015-03-20 07:46:13 +00:00
|
|
|
if K2.state() == LocalKeyState::Destroyed {
|
2014-11-14 22:20:57 +00:00
|
|
|
assert_eq!(HITS, 3);
|
|
|
|
} else {
|
|
|
|
if HITS == 1 {
|
|
|
|
K2.with(|s| *s.get() = Some(S2));
|
|
|
|
} else {
|
|
|
|
assert_eq!(HITS, 3);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
impl Drop for S2 {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
unsafe {
|
|
|
|
HITS += 1;
|
2015-03-20 07:46:13 +00:00
|
|
|
assert!(K1.state() != LocalKeyState::Destroyed);
|
2014-11-14 22:20:57 +00:00
|
|
|
assert_eq!(HITS, 2);
|
|
|
|
K1.with(|s| *s.get() = Some(S1));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 23:10:25 +00:00
|
|
|
thread::spawn(move|| {
|
2014-11-14 22:20:57 +00:00
|
|
|
drop(S1);
|
2014-12-22 17:04:23 +00:00
|
|
|
}).join().ok().unwrap();
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn self_referential() {
|
|
|
|
struct S1;
|
2015-05-27 08:18:36 +00:00
|
|
|
thread_local!(static K1: UnsafeCell<Option<S1>> = UnsafeCell::new(None));
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
impl Drop for S1 {
|
|
|
|
fn drop(&mut self) {
|
2015-03-20 07:46:13 +00:00
|
|
|
assert!(K1.state() == LocalKeyState::Destroyed);
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-02-17 23:10:25 +00:00
|
|
|
thread::spawn(move|| unsafe {
|
2014-11-14 22:20:57 +00:00
|
|
|
K1.with(|s| *s.get() = Some(S1));
|
2014-12-22 17:04:23 +00:00
|
|
|
}).join().ok().unwrap();
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
|
2016-01-29 21:46:47 +00:00
|
|
|
// Note that this test will deadlock if TLS destructors aren't run (this
|
2017-03-12 18:13:35 +00:00
|
|
|
// requires the destructor to be run to pass the test). macOS has a known bug
|
2016-01-29 21:46:47 +00:00
|
|
|
// where dtors-in-dtors may cancel other destructors, so we just ignore this
|
2017-03-12 18:13:35 +00:00
|
|
|
// test on macOS.
|
2014-11-14 22:20:57 +00:00
|
|
|
#[test]
|
2016-01-29 21:46:47 +00:00
|
|
|
#[cfg_attr(target_os = "macos", ignore)]
|
2014-11-14 22:20:57 +00:00
|
|
|
fn dtors_in_dtors_in_dtors() {
|
|
|
|
struct S1(Sender<()>);
|
2015-05-27 08:18:36 +00:00
|
|
|
thread_local!(static K1: UnsafeCell<Option<S1>> = UnsafeCell::new(None));
|
|
|
|
thread_local!(static K2: UnsafeCell<Option<Foo>> = UnsafeCell::new(None));
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
impl Drop for S1 {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
let S1(ref tx) = *self;
|
|
|
|
unsafe {
|
2015-03-20 07:46:13 +00:00
|
|
|
if K2.state() != LocalKeyState::Destroyed {
|
2014-11-14 22:20:57 +00:00
|
|
|
K2.with(|s| *s.get() = Some(Foo(tx.clone())));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
let (tx, rx) = channel();
|
2015-02-17 23:10:25 +00:00
|
|
|
let _t = thread::spawn(move|| unsafe {
|
2014-11-14 22:20:57 +00:00
|
|
|
let mut tx = Some(tx);
|
|
|
|
K1.with(|s| *s.get() = Some(S1(tx.take().unwrap())));
|
|
|
|
});
|
2014-12-23 19:53:35 +00:00
|
|
|
rx.recv().unwrap();
|
2014-11-14 22:20:57 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod dynamic_tests {
|
|
|
|
use cell::RefCell;
|
|
|
|
use collections::HashMap;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn smoke() {
|
2015-02-23 16:24:50 +00:00
|
|
|
fn square(i: i32) -> i32 { i * i }
|
|
|
|
thread_local!(static FOO: i32 = square(3));
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
FOO.with(|f| {
|
|
|
|
assert_eq!(*f, 9);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn hashmap() {
|
2015-02-23 16:24:50 +00:00
|
|
|
fn map() -> RefCell<HashMap<i32, i32>> {
|
2014-11-14 22:20:57 +00:00
|
|
|
let mut m = HashMap::new();
|
|
|
|
m.insert(1, 2);
|
|
|
|
RefCell::new(m)
|
|
|
|
}
|
2015-02-23 16:24:50 +00:00
|
|
|
thread_local!(static FOO: RefCell<HashMap<i32, i32>> = map());
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
FOO.with(|map| {
|
2015-03-22 01:15:47 +00:00
|
|
|
assert_eq!(map.borrow()[&1], 2);
|
2014-11-14 22:20:57 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn refcell_vec() {
|
2015-02-23 16:24:50 +00:00
|
|
|
thread_local!(static FOO: RefCell<Vec<u32>> = RefCell::new(vec![1, 2, 3]));
|
2014-11-14 22:20:57 +00:00
|
|
|
|
|
|
|
FOO.with(|vec| {
|
|
|
|
assert_eq!(vec.borrow().len(), 3);
|
|
|
|
vec.borrow_mut().push(4);
|
|
|
|
assert_eq!(vec.borrow()[3], 4);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|