// Copyright 2014-2015 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 or the MIT license // , at your // option. This file may not be copied, modified, or distributed // except according to those terms. //! Scoped thread-local storage //! //! This module provides the ability to generate *scoped* thread-local //! variables. In this sense, scoped indicates that thread local storage //! actually stores a reference to a value, and this reference is only placed //! in storage for a scoped amount of time. //! //! There are no restrictions on what types can be placed into a scoped //! variable, but all scoped variables are initialized to the equivalent of //! null. Scoped thread local storage is useful when a value is present for a known //! period of time and it is not required to relinquish ownership of the //! contents. //! //! # Examples //! //! ``` //! #[macro_use] //! extern crate scoped_tls; //! //! scoped_thread_local!(static FOO: u32); //! //! # fn main() { //! // Initially each scoped slot is empty. //! assert!(!FOO.is_set()); //! //! // When inserting a value, the value is only in place for the duration //! // of the closure specified. //! FOO.set(&1, || { //! FOO.with(|slot| { //! assert_eq!(*slot, 1); //! }); //! }); //! # } //! ``` #![deny(missing_docs, warnings)] use std::cell::Cell; use std::marker; use std::thread::LocalKey; /// The macro. See the module level documentation for the description and examples. #[macro_export] macro_rules! scoped_thread_local { ($(#[$attrs:meta])* $vis:vis static $name:ident: $ty:ty) => ( $(#[$attrs])* $vis static $name: $crate::ScopedKey<$ty> = $crate::ScopedKey { inner: { thread_local!(static FOO: ::std::cell::Cell = { ::std::cell::Cell::new(0) }); &FOO }, _marker: ::std::marker::PhantomData, }; ) } /// Type representing a thread local storage key corresponding to a reference /// to the type parameter `T`. /// /// Keys are statically allocated and can contain a reference to an instance of /// type `T` scoped to a particular lifetime. Keys provides two methods, `set` /// and `with`, both of which currently use closures to control the scope of /// their contents. pub struct ScopedKey { #[doc(hidden)] pub inner: &'static LocalKey>, #[doc(hidden)] pub _marker: marker::PhantomData, } unsafe impl Sync for ScopedKey {} impl ScopedKey { /// Inserts a value into this scoped thread local storage slot for a /// duration of a closure. /// /// While `cb` is running, the value `t` will be returned by `get` unless /// this function is called recursively inside of `cb`. /// /// Upon return, this function will restore the previous value, if any /// was available. /// /// # Examples /// /// ``` /// #[macro_use] /// extern crate scoped_tls; /// /// scoped_thread_local!(static FOO: u32); /// /// # fn main() { /// FOO.set(&100, || { /// let val = FOO.with(|v| *v); /// assert_eq!(val, 100); /// /// // set can be called recursively /// FOO.set(&101, || { /// // ... /// }); /// /// // Recursive calls restore the previous value. /// let val = FOO.with(|v| *v); /// assert_eq!(val, 100); /// }); /// # } /// ``` pub fn set(&'static self, t: &T, f: F) -> R where F: FnOnce() -> R { struct Reset { key: &'static LocalKey>, val: usize, } impl Drop for Reset { fn drop(&mut self) { self.key.with(|c| c.set(self.val)); } } let prev = self.inner.with(|c| { let prev = c.get(); c.set(t as *const T as usize); prev }); let _reset = Reset { key: self.inner, val: prev }; f() } /// Gets a value out of this scoped variable. /// /// This function takes a closure which receives the value of this /// variable. /// /// # Panics /// /// This function will panic if `set` has not previously been called. /// /// # Examples /// /// ```no_run /// #[macro_use] /// extern crate scoped_tls; /// /// scoped_thread_local!(static FOO: u32); /// /// # fn main() { /// FOO.with(|slot| { /// // work with `slot` /// # drop(slot); /// }); /// # } /// ``` pub fn with(&'static self, f: F) -> R where F: FnOnce(&T) -> R { let val = self.inner.with(|c| c.get()); assert!(val != 0, "cannot access a scoped thread local \ variable without calling `set` first"); unsafe { f(&*(val as *const T)) } } /// Test whether this TLS key has been `set` for the current thread. pub fn is_set(&'static self) -> bool { self.inner.with(|c| c.get() != 0) } } #[cfg(test)] mod tests { use std::cell::Cell; use std::sync::mpsc::{channel, Sender}; use std::thread; scoped_thread_local!(static FOO: u32); #[test] fn smoke() { scoped_thread_local!(static BAR: u32); assert!(!BAR.is_set()); BAR.set(&1, || { assert!(BAR.is_set()); BAR.with(|slot| { assert_eq!(*slot, 1); }); }); assert!(!BAR.is_set()); } #[test] fn cell_allowed() { scoped_thread_local!(static BAR: Cell); BAR.set(&Cell::new(1), || { BAR.with(|slot| { assert_eq!(slot.get(), 1); }); }); } #[test] fn scope_item_allowed() { assert!(!FOO.is_set()); FOO.set(&1, || { assert!(FOO.is_set()); FOO.with(|slot| { assert_eq!(*slot, 1); }); }); assert!(!FOO.is_set()); } #[test] fn panic_resets() { struct Check(Sender); impl Drop for Check { fn drop(&mut self) { FOO.with(|r| { self.0.send(*r).unwrap(); }) } } let (tx, rx) = channel(); let t = thread::spawn(|| { FOO.set(&1, || { let _r = Check(tx); FOO.set(&2, || { panic!() }); }); }); assert_eq!(rx.recv().unwrap(), 1); assert!(t.join().is_err()); } #[test] fn attrs_allowed() { scoped_thread_local!( /// Docs static BAZ: u32 ); scoped_thread_local!( #[allow(non_upper_case_globals)] static quux: u32 ); let _ = BAZ; let _ = quux; } }