Skip to content

Commit 7420c36

Browse files
authored
Rollup merge of rust-lang#131233 - joboet:stdout-before-main, r=tgross35
std: fix stdout-before-main Fixes rust-lang#130210. Since rust-lang#124881, `ReentrantLock` uses `ThreadId` to identify threads. This has the unfortunate consequence of breaking uses of `Stdout` before main: Locking the `ReentrantLock` that synchronizes the output will initialize the thread ID before the handle for the main thread is set in `rt::init`. But since that would overwrite the current thread ID, `thread::set_current` triggers an abort. This PR fixes the problem by using the already initialized thread ID for constructing the main thread handle and allowing `set_current` calls that do not change the thread's ID.
2 parents 08e7188 + 3febfb3 commit 7420c36

File tree

3 files changed

+42
-23
lines changed

3 files changed

+42
-23
lines changed

std/src/rt.rs

+18-3
Original file line numberDiff line numberDiff line change
@@ -102,9 +102,24 @@ unsafe fn init(argc: isize, argv: *const *const u8, sigpipe: u8) {
102102
sys::init(argc, argv, sigpipe)
103103
};
104104

105-
// Set up the current thread to give it the right name.
106-
let thread = Thread::new_main();
107-
thread::set_current(thread);
105+
// Set up the current thread handle to give it the right name.
106+
//
107+
// When code running before main uses `ReentrantLock` (for example by
108+
// using `println!`), the thread ID can become initialized before we
109+
// create this handle. Since `set_current` fails when the ID of the
110+
// handle does not match the current ID, we should attempt to use the
111+
// current thread ID here instead of unconditionally creating a new
112+
// one. Also see #130210.
113+
let thread = Thread::new_main(thread::current_id());
114+
if let Err(_thread) = thread::set_current(thread) {
115+
// `thread::current` will create a new handle if none has been set yet.
116+
// Thus, if someone uses it before main, this call will fail. That's a
117+
// bad idea though, as we then cannot set the main thread name here.
118+
//
119+
// FIXME: detect the main thread in `thread::current` and use the
120+
// correct name there.
121+
rtabort!("code running before main must not use thread::current");
122+
}
108123
}
109124

110125
/// Clean up the thread-local runtime state. This *should* be run after all other

std/src/thread/current.rs

+11-9
Original file line numberDiff line numberDiff line change
@@ -110,22 +110,24 @@ mod id {
110110
}
111111
}
112112

113-
/// Sets the thread handle for the current thread.
114-
///
115-
/// Aborts if the handle or the ID has been set already.
116-
pub(crate) fn set_current(thread: Thread) {
117-
if CURRENT.get() != NONE || id::get().is_some() {
118-
// Using `panic` here can add ~3kB to the binary size. We have complete
119-
// control over where this is called, so just abort if there is a bug.
120-
rtabort!("thread::set_current should only be called once per thread");
113+
/// Tries to set the thread handle for the current thread. Fails if a handle was
114+
/// already set or if the thread ID of `thread` would change an already-set ID.
115+
pub(crate) fn set_current(thread: Thread) -> Result<(), Thread> {
116+
if CURRENT.get() != NONE {
117+
return Err(thread);
121118
}
122119

123-
id::set(thread.id());
120+
match id::get() {
121+
Some(id) if id == thread.id() => {}
122+
None => id::set(thread.id()),
123+
_ => return Err(thread),
124+
}
124125

125126
// Make sure that `crate::rt::thread_cleanup` will be run, which will
126127
// call `drop_current`.
127128
crate::sys::thread_local::guard::enable();
128129
CURRENT.set(thread.into_raw().cast_mut());
130+
Ok(())
129131
}
130132

131133
/// Gets the id of the thread that invokes it.

std/src/thread/mod.rs

+13-11
Original file line numberDiff line numberDiff line change
@@ -519,9 +519,14 @@ impl Builder {
519519

520520
let f = MaybeDangling::new(f);
521521
let main = move || {
522-
// Immediately store the thread handle to avoid setting it or its ID
523-
// twice, which would cause an abort.
524-
set_current(their_thread.clone());
522+
if let Err(_thread) = set_current(their_thread.clone()) {
523+
// Both the current thread handle and the ID should not be
524+
// initialized yet. Since only the C runtime and some of our
525+
// platform code run before this, this point shouldn't be
526+
// reachable. Use an abort to save binary size (see #123356).
527+
rtabort!("something here is badly broken!");
528+
}
529+
525530
if let Some(name) = their_thread.cname() {
526531
imp::Thread::set_name(name);
527532
}
@@ -1159,9 +1164,6 @@ pub fn park_timeout(dur: Duration) {
11591164
pub struct ThreadId(NonZero<u64>);
11601165

11611166
impl ThreadId {
1162-
// DO NOT rely on this value.
1163-
const MAIN_THREAD: ThreadId = ThreadId(unsafe { NonZero::new_unchecked(1) });
1164-
11651167
// Generate a new unique thread ID.
11661168
pub(crate) fn new() -> ThreadId {
11671169
#[cold]
@@ -1173,7 +1175,7 @@ impl ThreadId {
11731175
if #[cfg(target_has_atomic = "64")] {
11741176
use crate::sync::atomic::AtomicU64;
11751177

1176-
static COUNTER: AtomicU64 = AtomicU64::new(1);
1178+
static COUNTER: AtomicU64 = AtomicU64::new(0);
11771179

11781180
let mut last = COUNTER.load(Ordering::Relaxed);
11791181
loop {
@@ -1189,7 +1191,7 @@ impl ThreadId {
11891191
} else {
11901192
use crate::sync::{Mutex, PoisonError};
11911193

1192-
static COUNTER: Mutex<u64> = Mutex::new(1);
1194+
static COUNTER: Mutex<u64> = Mutex::new(0);
11931195

11941196
let mut counter = COUNTER.lock().unwrap_or_else(PoisonError::into_inner);
11951197
let Some(id) = counter.checked_add(1) else {
@@ -1326,9 +1328,9 @@ impl Thread {
13261328
Self::new_inner(id, ThreadName::Unnamed)
13271329
}
13281330

1329-
// Used in runtime to construct main thread
1330-
pub(crate) fn new_main() -> Thread {
1331-
Self::new_inner(ThreadId::MAIN_THREAD, ThreadName::Main)
1331+
/// Constructs the thread handle for the main thread.
1332+
pub(crate) fn new_main(id: ThreadId) -> Thread {
1333+
Self::new_inner(id, ThreadName::Main)
13321334
}
13331335

13341336
fn new_inner(id: ThreadId, name: ThreadName) -> Thread {

0 commit comments

Comments
 (0)