std/sys/thread_local/
mod.rs

1//! Implementation of the `thread_local` macro.
2//!
3//! There are three different thread-local implementations:
4//! * Some targets lack threading support, and hence have only one thread, so
5//!   the TLS data is stored in a normal `static`.
6//! * Some targets support TLS natively via the dynamic linker and C runtime.
7//! * On some targets, the OS provides a library-based TLS implementation. The
8//!   TLS data is heap-allocated and referenced using a TLS key.
9//!
10//! Each implementation provides a macro which generates the `LocalKey` `const`
11//! used to reference the TLS variable, along with the necessary helper structs
12//! to track the initialization/destruction state of the variable.
13//!
14//! Additionally, this module contains abstractions for the OS interfaces used
15//! for these implementations.
16
17#![cfg_attr(test, allow(unused))]
18#![doc(hidden)]
19#![forbid(unsafe_op_in_unsafe_fn)]
20#![unstable(
21    feature = "thread_local_internals",
22    reason = "internal details of the thread_local macro",
23    issue = "none"
24)]
25
26cfg_if::cfg_if! {
27    if #[cfg(any(
28        all(target_family = "wasm", not(target_feature = "atomics")),
29        target_os = "uefi",
30        target_os = "zkvm",
31    ))] {
32        mod statik;
33        pub use statik::{EagerStorage, LazyStorage, thread_local_inner};
34        pub(crate) use statik::{LocalPointer, local_pointer};
35    } else if #[cfg(target_thread_local)] {
36        mod native;
37        pub use native::{EagerStorage, LazyStorage, thread_local_inner};
38        pub(crate) use native::{LocalPointer, local_pointer};
39    } else {
40        mod os;
41        pub use os::{Storage, thread_local_inner};
42        pub(crate) use os::{LocalPointer, local_pointer};
43    }
44}
45
46/// The native TLS implementation needs a way to register destructors for its data.
47/// This module contains platform-specific implementations of that register.
48///
49/// It turns out however that most platforms don't have a way to register a
50/// destructor for each variable. On these platforms, we keep track of the
51/// destructors ourselves and register (through the [`guard`] module) only a
52/// single callback that runs all of the destructors in the list.
53#[cfg(all(target_thread_local, not(all(target_family = "wasm", not(target_feature = "atomics")))))]
54pub(crate) mod destructors {
55    cfg_if::cfg_if! {
56        if #[cfg(any(
57            target_os = "linux",
58            target_os = "android",
59            target_os = "fuchsia",
60            target_os = "redox",
61            target_os = "hurd",
62            target_os = "netbsd",
63            target_os = "dragonfly"
64        ))] {
65            mod linux_like;
66            mod list;
67            pub(super) use linux_like::register;
68            pub(super) use list::run;
69        } else {
70            mod list;
71            pub(super) use list::register;
72            pub(crate) use list::run;
73        }
74    }
75}
76
77/// This module provides a way to schedule the execution of the destructor list
78/// and the [runtime cleanup](crate::rt::thread_cleanup) function. Calling `enable`
79/// should ensure that these functions are called at the right times.
80pub(crate) mod guard {
81    cfg_if::cfg_if! {
82        if #[cfg(all(target_thread_local, target_vendor = "apple"))] {
83            mod apple;
84            pub(crate) use apple::enable;
85        } else if #[cfg(target_os = "windows")] {
86            mod windows;
87            pub(crate) use windows::enable;
88        } else if #[cfg(any(
89            all(target_family = "wasm", not(
90                all(target_os = "wasi", target_env = "p1", target_feature = "atomics")
91            )),
92            target_os = "uefi",
93            target_os = "zkvm",
94        ))] {
95            pub(crate) fn enable() {
96                // FIXME: Right now there is no concept of "thread exit" on
97                // wasm, but this is likely going to show up at some point in
98                // the form of an exported symbol that the wasm runtime is going
99                // to be expected to call. For now we just leak everything, but
100                // if such a function starts to exist it will probably need to
101                // iterate the destructor list with these functions:
102                #[cfg(all(target_family = "wasm", target_feature = "atomics"))]
103                #[allow(unused)]
104                use super::destructors::run;
105                #[allow(unused)]
106                use crate::rt::thread_cleanup;
107            }
108        } else if #[cfg(any(
109            target_os = "hermit",
110            target_os = "xous",
111        ))] {
112            // `std` is the only runtime, so it just calls the destructor functions
113            // itself when the time comes.
114            pub(crate) fn enable() {}
115        } else if #[cfg(target_os = "solid_asp3")] {
116            mod solid;
117            pub(crate) use solid::enable;
118        } else {
119            mod key;
120            pub(crate) use key::enable;
121        }
122    }
123}
124
125/// `const`-creatable TLS keys.
126///
127/// Most OSs without native TLS will provide a library-based way to create TLS
128/// storage. For each TLS variable, we create a key, which can then be used to
129/// reference an entry in a thread-local table. This then associates each key
130/// with a pointer which we can get and set to store our data.
131pub(crate) mod key {
132    cfg_if::cfg_if! {
133        if #[cfg(any(
134            all(
135                not(target_vendor = "apple"),
136                not(target_family = "wasm"),
137                target_family = "unix",
138            ),
139            target_os = "teeos",
140            all(target_os = "wasi", target_env = "p1", target_feature = "atomics"),
141        ))] {
142            mod racy;
143            mod unix;
144            #[cfg(test)]
145            mod tests;
146            pub(super) use racy::LazyKey;
147            pub(super) use unix::{Key, set};
148            #[cfg(any(not(target_thread_local), test))]
149            pub(super) use unix::get;
150            use unix::{create, destroy};
151        } else if #[cfg(all(not(target_thread_local), target_os = "windows"))] {
152            #[cfg(test)]
153            mod tests;
154            mod windows;
155            pub(super) use windows::{Key, LazyKey, get, run_dtors, set};
156        } else if #[cfg(all(target_vendor = "fortanix", target_env = "sgx"))] {
157            mod racy;
158            mod sgx;
159            #[cfg(test)]
160            mod tests;
161            pub(super) use racy::LazyKey;
162            pub(super) use sgx::{Key, get, set};
163            use sgx::{create, destroy};
164        } else if #[cfg(target_os = "xous")] {
165            mod racy;
166            #[cfg(test)]
167            mod tests;
168            mod xous;
169            pub(super) use racy::LazyKey;
170            pub(crate) use xous::destroy_tls;
171            pub(super) use xous::{Key, get, set};
172            use xous::{create, destroy};
173        }
174    }
175}
176
177/// Run a callback in a scenario which must not unwind (such as a `extern "C"
178/// fn` declared in a user crate). If the callback unwinds anyway, then
179/// `rtabort` with a message about thread local panicking on drop.
180#[inline]
181#[allow(dead_code)]
182fn abort_on_dtor_unwind(f: impl FnOnce()) {
183    // Using a guard like this is lower cost.
184    let guard = DtorUnwindGuard;
185    f();
186    core::mem::forget(guard);
187
188    struct DtorUnwindGuard;
189    impl Drop for DtorUnwindGuard {
190        #[inline]
191        fn drop(&mut self) {
192            // This is not terribly descriptive, but it doesn't need to be as we'll
193            // already have printed a panic message at this point.
194            rtabort!("thread local panicked on drop");
195        }
196    }
197}