Skip to content

Commit 7743aa8

Browse files
committed
Auto merge of #100581 - joboet:sync_rwlock_everywhere, r=thomcc
std: use `sync::RwLock` for internal statics Since `sync::RwLock` is now `const`-constructible, it can be used for internal statics, removing the need for `sys_common::StaticRwLock`. This adds some extra allocations on platforms which need to box their locks (currently SGX and some UNIX), but these will become unnecessary with the lock improvements tracked in #93740.
2 parents 432abd8 + be09a4a commit 7743aa8

File tree

8 files changed

+62
-151
lines changed

8 files changed

+62
-151
lines changed

library/std/src/panicking.rs

+50-79
Original file line numberDiff line numberDiff line change
@@ -18,9 +18,9 @@ use crate::intrinsics;
1818
use crate::mem::{self, ManuallyDrop};
1919
use crate::process;
2020
use crate::sync::atomic::{AtomicBool, Ordering};
21+
use crate::sync::{PoisonError, RwLock};
2122
use crate::sys::stdio::panic_output;
2223
use crate::sys_common::backtrace;
23-
use crate::sys_common::rwlock::StaticRwLock;
2424
use crate::sys_common::thread_info;
2525
use crate::thread;
2626

@@ -71,20 +71,29 @@ extern "C" fn __rust_foreign_exception() -> ! {
7171
rtabort!("Rust cannot catch foreign exceptions");
7272
}
7373

74-
#[derive(Copy, Clone)]
7574
enum Hook {
7675
Default,
77-
Custom(*mut (dyn Fn(&PanicInfo<'_>) + 'static + Sync + Send)),
76+
Custom(Box<dyn Fn(&PanicInfo<'_>) + 'static + Sync + Send>),
7877
}
7978

8079
impl Hook {
81-
fn custom(f: impl Fn(&PanicInfo<'_>) + 'static + Sync + Send) -> Self {
82-
Self::Custom(Box::into_raw(Box::new(f)))
80+
#[inline]
81+
fn into_box(self) -> Box<dyn Fn(&PanicInfo<'_>) + 'static + Sync + Send> {
82+
match self {
83+
Hook::Default => Box::new(default_hook),
84+
Hook::Custom(hook) => hook,
85+
}
8386
}
8487
}
8588

86-
static HOOK_LOCK: StaticRwLock = StaticRwLock::new();
87-
static mut HOOK: Hook = Hook::Default;
89+
impl Default for Hook {
90+
#[inline]
91+
fn default() -> Hook {
92+
Hook::Default
93+
}
94+
}
95+
96+
static HOOK: RwLock<Hook> = RwLock::new(Hook::Default);
8897

8998
/// Registers a custom panic hook, replacing any that was previously registered.
9099
///
@@ -125,24 +134,13 @@ pub fn set_hook(hook: Box<dyn Fn(&PanicInfo<'_>) + 'static + Sync + Send>) {
125134
panic!("cannot modify the panic hook from a panicking thread");
126135
}
127136

128-
// SAFETY:
129-
//
130-
// - `HOOK` can only be modified while holding write access to `HOOK_LOCK`.
131-
// - The argument of `Box::from_raw` is always a valid pointer that was created using
132-
// `Box::into_raw`.
133-
unsafe {
134-
let guard = HOOK_LOCK.write();
135-
let old_hook = HOOK;
136-
HOOK = Hook::Custom(Box::into_raw(hook));
137-
drop(guard);
138-
139-
if let Hook::Custom(ptr) = old_hook {
140-
#[allow(unused_must_use)]
141-
{
142-
Box::from_raw(ptr);
143-
}
144-
}
145-
}
137+
let new = Hook::Custom(hook);
138+
let mut hook = HOOK.write().unwrap_or_else(PoisonError::into_inner);
139+
let old = mem::replace(&mut *hook, new);
140+
drop(hook);
141+
// Only drop the old hook after releasing the lock to avoid deadlocking
142+
// if its destructor panics.
143+
drop(old);
146144
}
147145

148146
/// Unregisters the current panic hook, returning it.
@@ -179,22 +177,11 @@ pub fn take_hook() -> Box<dyn Fn(&PanicInfo<'_>) + 'static + Sync + Send> {
179177
panic!("cannot modify the panic hook from a panicking thread");
180178
}
181179

182-
// SAFETY:
183-
//
184-
// - `HOOK` can only be modified while holding write access to `HOOK_LOCK`.
185-
// - The argument of `Box::from_raw` is always a valid pointer that was created using
186-
// `Box::into_raw`.
187-
unsafe {
188-
let guard = HOOK_LOCK.write();
189-
let hook = HOOK;
190-
HOOK = Hook::Default;
191-
drop(guard);
180+
let mut hook = HOOK.write().unwrap_or_else(PoisonError::into_inner);
181+
let old_hook = mem::take(&mut *hook);
182+
drop(hook);
192183

193-
match hook {
194-
Hook::Default => Box::new(default_hook),
195-
Hook::Custom(ptr) => Box::from_raw(ptr),
196-
}
197-
}
184+
old_hook.into_box()
198185
}
199186

200187
/// Atomic combination of [`take_hook`] and [`set_hook`]. Use this to replace the panic handler with
@@ -240,24 +227,9 @@ where
240227
panic!("cannot modify the panic hook from a panicking thread");
241228
}
242229

243-
// SAFETY:
244-
//
245-
// - `HOOK` can only be modified while holding write access to `HOOK_LOCK`.
246-
// - The argument of `Box::from_raw` is always a valid pointer that was created using
247-
// `Box::into_raw`.
248-
unsafe {
249-
let guard = HOOK_LOCK.write();
250-
let old_hook = HOOK;
251-
HOOK = Hook::Default;
252-
253-
let prev = match old_hook {
254-
Hook::Default => Box::new(default_hook),
255-
Hook::Custom(ptr) => Box::from_raw(ptr),
256-
};
257-
258-
HOOK = Hook::custom(move |info| hook_fn(&prev, info));
259-
drop(guard);
260-
}
230+
let mut hook = HOOK.write().unwrap_or_else(PoisonError::into_inner);
231+
let prev = mem::take(&mut *hook).into_box();
232+
*hook = Hook::Custom(Box::new(move |info| hook_fn(&prev, info)));
261233
}
262234

263235
fn default_hook(info: &PanicInfo<'_>) {
@@ -682,27 +654,26 @@ fn rust_panic_with_hook(
682654
crate::sys::abort_internal();
683655
}
684656

685-
unsafe {
686-
let mut info = PanicInfo::internal_constructor(message, location, can_unwind);
687-
let _guard = HOOK_LOCK.read();
688-
match HOOK {
689-
// Some platforms (like wasm) know that printing to stderr won't ever actually
690-
// print anything, and if that's the case we can skip the default
691-
// hook. Since string formatting happens lazily when calling `payload`
692-
// methods, this means we avoid formatting the string at all!
693-
// (The panic runtime might still call `payload.take_box()` though and trigger
694-
// formatting.)
695-
Hook::Default if panic_output().is_none() => {}
696-
Hook::Default => {
697-
info.set_payload(payload.get());
698-
default_hook(&info);
699-
}
700-
Hook::Custom(ptr) => {
701-
info.set_payload(payload.get());
702-
(*ptr)(&info);
703-
}
704-
};
705-
}
657+
let mut info = PanicInfo::internal_constructor(message, location, can_unwind);
658+
let hook = HOOK.read().unwrap_or_else(PoisonError::into_inner);
659+
match *hook {
660+
// Some platforms (like wasm) know that printing to stderr won't ever actually
661+
// print anything, and if that's the case we can skip the default
662+
// hook. Since string formatting happens lazily when calling `payload`
663+
// methods, this means we avoid formatting the string at all!
664+
// (The panic runtime might still call `payload.take_box()` though and trigger
665+
// formatting.)
666+
Hook::Default if panic_output().is_none() => {}
667+
Hook::Default => {
668+
info.set_payload(payload.get());
669+
default_hook(&info);
670+
}
671+
Hook::Custom(ref hook) => {
672+
info.set_payload(payload.get());
673+
hook(&info);
674+
}
675+
};
676+
drop(hook);
706677

707678
if panics > 1 || !can_unwind {
708679
// If a thread panics while it's already unwinding then we

library/std/src/sys/solid/os.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use crate::os::{
88
solid::ffi::{OsStrExt, OsStringExt},
99
};
1010
use crate::path::{self, PathBuf};
11-
use crate::sys_common::rwlock::StaticRwLock;
11+
use crate::sync::RwLock;
1212
use crate::vec;
1313

1414
use super::{error, itron, memchr};
@@ -78,7 +78,7 @@ pub fn current_exe() -> io::Result<PathBuf> {
7878
unsupported()
7979
}
8080

81-
static ENV_LOCK: StaticRwLock = StaticRwLock::new();
81+
static ENV_LOCK: RwLock<()> = RwLock::new(());
8282

8383
pub struct Env {
8484
iter: vec::IntoIter<(OsString, OsString)>,

library/std/src/sys/unix/locks/mod.rs

+3-3
Original file line numberDiff line numberDiff line change
@@ -11,21 +11,21 @@ cfg_if::cfg_if! {
1111
mod futex_rwlock;
1212
mod futex_condvar;
1313
pub(crate) use futex_mutex::{Mutex, MovableMutex};
14-
pub(crate) use futex_rwlock::{RwLock, MovableRwLock};
14+
pub(crate) use futex_rwlock::MovableRwLock;
1515
pub(crate) use futex_condvar::MovableCondvar;
1616
} else if #[cfg(target_os = "fuchsia")] {
1717
mod fuchsia_mutex;
1818
mod futex_rwlock;
1919
mod futex_condvar;
2020
pub(crate) use fuchsia_mutex::{Mutex, MovableMutex};
21-
pub(crate) use futex_rwlock::{RwLock, MovableRwLock};
21+
pub(crate) use futex_rwlock::MovableRwLock;
2222
pub(crate) use futex_condvar::MovableCondvar;
2323
} else {
2424
mod pthread_mutex;
2525
mod pthread_rwlock;
2626
mod pthread_condvar;
2727
pub(crate) use pthread_mutex::{Mutex, MovableMutex};
28-
pub(crate) use pthread_rwlock::{RwLock, MovableRwLock};
28+
pub(crate) use pthread_rwlock::MovableRwLock;
2929
pub(crate) use pthread_condvar::MovableCondvar;
3030
}
3131
}

library/std/src/sys/unix/os.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -17,10 +17,10 @@ use crate::path::{self, PathBuf};
1717
use crate::ptr;
1818
use crate::slice;
1919
use crate::str;
20+
use crate::sync::{PoisonError, RwLock};
2021
use crate::sys::cvt;
2122
use crate::sys::fd;
2223
use crate::sys::memchr;
23-
use crate::sys_common::rwlock::{StaticRwLock, StaticRwLockReadGuard};
2424
use crate::vec;
2525

2626
#[cfg(all(target_env = "gnu", not(target_os = "vxworks")))]
@@ -501,10 +501,10 @@ pub unsafe fn environ() -> *mut *const *const c_char {
501501
ptr::addr_of_mut!(environ)
502502
}
503503

504-
static ENV_LOCK: StaticRwLock = StaticRwLock::new();
504+
static ENV_LOCK: RwLock<()> = RwLock::new(());
505505

506-
pub fn env_read_lock() -> StaticRwLockReadGuard {
507-
ENV_LOCK.read()
506+
pub fn env_read_lock() -> impl Drop {
507+
ENV_LOCK.read().unwrap_or_else(PoisonError::into_inner)
508508
}
509509

510510
/// Returns a vector of (variable, value) byte-vector pairs for all the

library/std/src/sys/unsupported/locks/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,4 +3,4 @@ mod mutex;
33
mod rwlock;
44
pub use condvar::{Condvar, MovableCondvar};
55
pub use mutex::{MovableMutex, Mutex};
6-
pub use rwlock::{MovableRwLock, RwLock};
6+
pub use rwlock::MovableRwLock;

library/std/src/sys/wasm/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -57,7 +57,7 @@ cfg_if::cfg_if! {
5757
mod futex_rwlock;
5858
pub(crate) use futex_condvar::{Condvar, MovableCondvar};
5959
pub(crate) use futex_mutex::{Mutex, MovableMutex};
60-
pub(crate) use futex_rwlock::{RwLock, MovableRwLock};
60+
pub(crate) use futex_rwlock::MovableRwLock;
6161
}
6262
#[path = "atomics/futex.rs"]
6363
pub mod futex;

library/std/src/sys/windows/locks/mod.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -3,4 +3,4 @@ mod mutex;
33
mod rwlock;
44
pub use condvar::{Condvar, MovableCondvar};
55
pub use mutex::{MovableMutex, Mutex};
6-
pub use rwlock::{MovableRwLock, RwLock};
6+
pub use rwlock::MovableRwLock;

library/std/src/sys_common/rwlock.rs

-60
Original file line numberDiff line numberDiff line change
@@ -1,65 +1,5 @@
11
use crate::sys::locks as imp;
22

3-
/// An OS-based reader-writer lock, meant for use in static variables.
4-
///
5-
/// This rwlock does not implement poisoning.
6-
///
7-
/// This rwlock has a const constructor ([`StaticRwLock::new`]), does not
8-
/// implement `Drop` to cleanup resources.
9-
pub struct StaticRwLock(imp::RwLock);
10-
11-
impl StaticRwLock {
12-
/// Creates a new rwlock for use.
13-
#[inline]
14-
pub const fn new() -> Self {
15-
Self(imp::RwLock::new())
16-
}
17-
18-
/// Acquires shared access to the underlying lock, blocking the current
19-
/// thread to do so.
20-
///
21-
/// The lock is automatically unlocked when the returned guard is dropped.
22-
#[inline]
23-
pub fn read(&'static self) -> StaticRwLockReadGuard {
24-
unsafe { self.0.read() };
25-
StaticRwLockReadGuard(&self.0)
26-
}
27-
28-
/// Acquires write access to the underlying lock, blocking the current thread
29-
/// to do so.
30-
///
31-
/// The lock is automatically unlocked when the returned guard is dropped.
32-
#[inline]
33-
pub fn write(&'static self) -> StaticRwLockWriteGuard {
34-
unsafe { self.0.write() };
35-
StaticRwLockWriteGuard(&self.0)
36-
}
37-
}
38-
39-
#[must_use]
40-
pub struct StaticRwLockReadGuard(&'static imp::RwLock);
41-
42-
impl Drop for StaticRwLockReadGuard {
43-
#[inline]
44-
fn drop(&mut self) {
45-
unsafe {
46-
self.0.read_unlock();
47-
}
48-
}
49-
}
50-
51-
#[must_use]
52-
pub struct StaticRwLockWriteGuard(&'static imp::RwLock);
53-
54-
impl Drop for StaticRwLockWriteGuard {
55-
#[inline]
56-
fn drop(&mut self) {
57-
unsafe {
58-
self.0.write_unlock();
59-
}
60-
}
61-
}
62-
633
/// An OS-based reader-writer lock.
644
///
655
/// This rwlock cleans up its resources in its `Drop` implementation and may

0 commit comments

Comments
 (0)