Skip to content

Commit b4307a5

Browse files
committed
Auto merge of rust-lang#128321 - BatmanAoD:catch-unwind-doc-update, r=Mark-Simulacrum
Update `catch_unwind` doc comments for `c_unwind` Updates `catch_unwind` doc comments to indicate that catching a foreign exception _will no longer_ be UB. Instead, there are two possible behaviors, though it is not specified which one an implementation will choose. Nominated for t-lang to confirm that they are okay with making such a promise based on t-opsem FCP, or whether they would like to be included in the FCP. Related: rust-lang#74990, rust-lang#115285, rust-lang/reference#1226
2 parents 7311aa8 + 35edcb6 commit b4307a5

File tree

3 files changed

+70
-30
lines changed

3 files changed

+70
-30
lines changed

core/src/intrinsics.rs

+10-5
Original file line numberDiff line numberDiff line change
@@ -2659,12 +2659,17 @@ extern "rust-intrinsic" {
26592659
///
26602660
/// `catch_fn` must not unwind.
26612661
///
2662-
/// The third argument is a function called if an unwind occurs (both Rust unwinds and foreign
2663-
/// unwinds). This function takes the data pointer and a pointer to the target-specific
2664-
/// exception object that was caught. For more information, see the compiler's source as well as
2665-
/// std's `catch_unwind` implementation.
2662+
/// The third argument is a function called if an unwind occurs (both Rust `panic` and foreign
2663+
/// unwinds). This function takes the data pointer and a pointer to the target- and
2664+
/// runtime-specific exception object that was caught.
26662665
///
2667-
/// The stable version of this intrinsic is `std::panic::catch_unwind`.
2666+
/// Note that in the case of a foreign unwinding operation, the exception object data may not be
2667+
/// safely usable from Rust, and should not be directly exposed via the standard library. To
2668+
/// prevent unsafe access, the library implementation may either abort the process or present an
2669+
/// opaque error type to the user.
2670+
///
2671+
/// For more information, see the compiler's source, as well as the documentation for the stable
2672+
/// version of this intrinsic, `std::panic::catch_unwind`.
26682673
#[rustc_nounwind]
26692674
pub fn catch_unwind(try_fn: fn(*mut u8), data: *mut u8, catch_fn: fn(*mut u8, *mut u8)) -> i32;
26702675

std/src/panic.rs

+34-24
Original file line numberDiff line numberDiff line change
@@ -288,45 +288,55 @@ pub use core::panic::abort_unwind;
288288

289289
/// Invokes a closure, capturing the cause of an unwinding panic if one occurs.
290290
///
291-
/// This function will return `Ok` with the closure's result if the closure
292-
/// does not panic, and will return `Err(cause)` if the closure panics. The
293-
/// `cause` returned is the object with which panic was originally invoked.
291+
/// This function will return `Ok` with the closure's result if the closure does
292+
/// not panic, and will return `Err(cause)` if the closure panics. The `cause`
293+
/// returned is the object with which panic was originally invoked.
294294
///
295-
/// It is currently undefined behavior to unwind from Rust code into foreign
296-
/// code, so this function is particularly useful when Rust is called from
297-
/// another language (normally C). This can run arbitrary Rust code, capturing a
298-
/// panic and allowing a graceful handling of the error.
295+
/// Rust functions that are expected to be called from foreign code that does
296+
/// not support unwinding (such as C compiled with `-fno-exceptions`) should be
297+
/// defined using `extern "C"`, which ensures that if the Rust code panics, it
298+
/// is automatically caught and the process is aborted. If this is the desired
299+
/// behavior, it is not necessary to use `catch_unwind` explicitly. This
300+
/// function should instead be used when more graceful error-handling is needed.
299301
///
300302
/// It is **not** recommended to use this function for a general try/catch
301303
/// mechanism. The [`Result`] type is more appropriate to use for functions that
302304
/// can fail on a regular basis. Additionally, this function is not guaranteed
303305
/// to catch all panics, see the "Notes" section below.
304306
///
305-
/// The closure provided is required to adhere to the [`UnwindSafe`] trait to ensure
306-
/// that all captured variables are safe to cross this boundary. The purpose of
307-
/// this bound is to encode the concept of [exception safety][rfc] in the type
308-
/// system. Most usage of this function should not need to worry about this
309-
/// bound as programs are naturally unwind safe without `unsafe` code. If it
310-
/// becomes a problem the [`AssertUnwindSafe`] wrapper struct can be used to quickly
311-
/// assert that the usage here is indeed unwind safe.
307+
/// The closure provided is required to adhere to the [`UnwindSafe`] trait to
308+
/// ensure that all captured variables are safe to cross this boundary. The
309+
/// purpose of this bound is to encode the concept of [exception safety][rfc] in
310+
/// the type system. Most usage of this function should not need to worry about
311+
/// this bound as programs are naturally unwind safe without `unsafe` code. If
312+
/// it becomes a problem the [`AssertUnwindSafe`] wrapper struct can be used to
313+
/// quickly assert that the usage here is indeed unwind safe.
312314
///
313315
/// [rfc]: https://github.com/rust-lang/rfcs/blob/master/text/1236-stabilize-catch-panic.md
314316
///
315317
/// # Notes
316318
///
317-
/// Note that this function **might not catch all panics** in Rust. A panic in
318-
/// Rust is not always implemented via unwinding, but can be implemented by
319-
/// aborting the process as well. This function *only* catches unwinding panics,
320-
/// not those that abort the process.
319+
/// This function **might not catch all Rust panics**. A Rust panic is not
320+
/// always implemented via unwinding, but can be implemented by aborting the
321+
/// process as well. This function *only* catches unwinding panics, not those
322+
/// that abort the process.
321323
///
322-
/// Note that if a custom panic hook has been set, it will be invoked before
323-
/// the panic is caught, before unwinding.
324+
/// If a custom panic hook has been set, it will be invoked before the panic is
325+
/// caught, before unwinding.
324326
///
325-
/// Also note that unwinding into Rust code with a foreign exception (e.g.
326-
/// an exception thrown from C++ code) is undefined behavior.
327+
/// Although unwinding into Rust code with a foreign exception (e.g. an
328+
/// exception thrown from C++ code, or a `panic!` in Rust code compiled or
329+
/// linked with a different runtime) via an appropriate ABI (e.g. `"C-unwind"`)
330+
/// is permitted, catching such an exception using this function will have one
331+
/// of two behaviors, and it is unspecified which will occur:
327332
///
328-
/// Finally, be **careful in how you drop the result of this function**.
329-
/// If it is `Err`, it contains the panic payload, and dropping that may in turn panic!
333+
/// * The process aborts, after executing all destructors of `f` and the
334+
/// functions it called.
335+
/// * The function returns a `Result::Err` containing an opaque type.
336+
///
337+
/// Finally, be **careful in how you drop the result of this function**. If it
338+
/// is `Err`, it contains the panic payload, and dropping that may in turn
339+
/// panic!
330340
///
331341
/// # Examples
332342
///

std/src/thread/mod.rs

+26-1
Original file line numberDiff line numberDiff line change
@@ -664,6 +664,19 @@ impl Builder {
664664
/// println!("{result}");
665665
/// ```
666666
///
667+
/// # Notes
668+
///
669+
/// This function has the same minimal guarantee regarding "foreign" unwinding operations (e.g.
670+
/// an exception thrown from C++ code, or a `panic!` in Rust code compiled or linked with a
671+
/// different runtime) as [`catch_unwind`]; namely, if the thread created with `thread::spawn`
672+
/// unwinds all the way to the root with such an exception, one of two behaviors are possible,
673+
/// and it is unspecified which will occur:
674+
///
675+
/// * The process aborts.
676+
/// * The process does not abort, and [`join`] will return a `Result::Err`
677+
/// containing an opaque type.
678+
///
679+
/// [`catch_unwind`]: ../../std/panic/fn.catch_unwind.html
667680
/// [`channels`]: crate::sync::mpsc
668681
/// [`join`]: JoinHandle::join
669682
/// [`Err`]: crate::result::Result::Err
@@ -1784,7 +1797,7 @@ impl<T> JoinHandle<T> {
17841797
/// operations that happen after `join` returns.
17851798
///
17861799
/// If the associated thread panics, [`Err`] is returned with the parameter given
1787-
/// to [`panic!`].
1800+
/// to [`panic!`] (though see the Notes below).
17881801
///
17891802
/// [`Err`]: crate::result::Result::Err
17901803
/// [atomic memory orderings]: crate::sync::atomic
@@ -1806,6 +1819,18 @@ impl<T> JoinHandle<T> {
18061819
/// }).unwrap();
18071820
/// join_handle.join().expect("Couldn't join on the associated thread");
18081821
/// ```
1822+
///
1823+
/// # Notes
1824+
///
1825+
/// If a "foreign" unwinding operation (e.g. an exception thrown from C++
1826+
/// code, or a `panic!` in Rust code compiled or linked with a different
1827+
/// runtime) unwinds all the way to the thread root, the process may be
1828+
/// aborted; see the Notes on [`thread::spawn`]. If the process is not
1829+
/// aborted, this function will return a `Result::Err` containing an opaque
1830+
/// type.
1831+
///
1832+
/// [`catch_unwind`]: ../../std/panic/fn.catch_unwind.html
1833+
/// [`thread::spawn`]: spawn
18091834
#[stable(feature = "rust1", since = "1.0.0")]
18101835
pub fn join(self) -> Result<T> {
18111836
self.0.join()

0 commit comments

Comments
 (0)