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}