2019-11-17 12:47:06 -06:00
|
|
|
// ignore-windows: Unwind panicking does not currently work on Windows
|
2019-11-29 04:19:15 -06:00
|
|
|
#![allow(const_err)]
|
|
|
|
|
2019-11-24 03:24:41 -06:00
|
|
|
use std::panic::{catch_unwind, AssertUnwindSafe};
|
2019-04-14 20:02:55 -05:00
|
|
|
use std::cell::Cell;
|
|
|
|
|
|
|
|
thread_local! {
|
|
|
|
static MY_COUNTER: Cell<usize> = Cell::new(0);
|
|
|
|
static DROPPED: Cell<bool> = Cell::new(false);
|
|
|
|
static HOOK_CALLED: Cell<bool> = Cell::new(false);
|
|
|
|
}
|
|
|
|
|
|
|
|
struct DropTester;
|
|
|
|
|
|
|
|
impl Drop for DropTester {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
DROPPED.with(|c| {
|
|
|
|
c.set(true);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-24 03:24:41 -06:00
|
|
|
fn do_panic_counter(do_panic: impl FnOnce(usize) -> !) {
|
2019-04-14 20:02:55 -05:00
|
|
|
// If this gets leaked, it will be easy to spot
|
|
|
|
// in Miri's leak report
|
|
|
|
let _string = "LEAKED FROM do_panic_counter".to_string();
|
|
|
|
|
|
|
|
// When we panic, this should get dropped during unwinding
|
|
|
|
let _drop_tester = DropTester;
|
|
|
|
|
|
|
|
// Check for bugs in Miri's panic implementation.
|
|
|
|
// If do_panic_counter() somehow gets called more than once,
|
2019-11-24 03:24:41 -06:00
|
|
|
// we'll generate a different panic message and stderr will differ.
|
2019-04-14 20:02:55 -05:00
|
|
|
let old_val = MY_COUNTER.with(|c| {
|
|
|
|
let val = c.get();
|
|
|
|
c.set(val + 1);
|
|
|
|
val
|
|
|
|
});
|
2019-11-24 03:24:41 -06:00
|
|
|
do_panic(old_val);
|
2019-04-14 20:02:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
2019-11-24 03:24:41 -06:00
|
|
|
let prev = std::panic::take_hook();
|
|
|
|
std::panic::set_hook(Box::new(move |panic_info| {
|
2019-04-14 20:02:55 -05:00
|
|
|
HOOK_CALLED.with(|h| h.set(true));
|
2019-11-24 03:24:41 -06:00
|
|
|
prev(panic_info)
|
2019-04-14 20:02:55 -05:00
|
|
|
}));
|
2019-11-24 03:24:41 -06:00
|
|
|
|
2019-11-29 04:19:15 -06:00
|
|
|
// Std panics
|
2019-11-28 12:17:43 -06:00
|
|
|
test(|_old_val| std::panic!("Hello from panic: std"));
|
|
|
|
test(|old_val| std::panic!(format!("Hello from panic: {:?}", old_val)));
|
|
|
|
test(|old_val| std::panic!("Hello from panic: {:?}", old_val));
|
|
|
|
test(|_old_val| std::panic!(1337));
|
|
|
|
|
2019-11-29 04:19:15 -06:00
|
|
|
// Core panics
|
2019-11-28 12:17:43 -06:00
|
|
|
test(|_old_val| core::panic!("Hello from panic: core"));
|
|
|
|
test(|old_val| core::panic!(&format!("Hello from panic: {:?}", old_val)));
|
|
|
|
test(|old_val| core::panic!("Hello from panic: {:?}", old_val));
|
2019-11-24 03:24:41 -06:00
|
|
|
|
2019-11-29 04:19:15 -06:00
|
|
|
// Built-in panics
|
|
|
|
test(|_old_val| { let _val = [0, 1, 2][4]; loop {} });
|
|
|
|
test(|_old_val| { let _val = 1/0; loop {} });
|
|
|
|
|
2019-11-24 03:24:41 -06:00
|
|
|
// Cleanup: reset to default hook.
|
|
|
|
drop(std::panic::take_hook());
|
|
|
|
|
|
|
|
eprintln!("Success!"); // Make sure we get this in stderr
|
|
|
|
}
|
|
|
|
|
|
|
|
fn test(do_panic: impl FnOnce(usize) -> !) {
|
|
|
|
// Reset test flags.
|
|
|
|
DROPPED.with(|c| c.set(false));
|
|
|
|
HOOK_CALLED.with(|c| c.set(false));
|
|
|
|
|
|
|
|
// Cause and catch a panic.
|
|
|
|
let res = catch_unwind(AssertUnwindSafe(|| {
|
2019-04-14 20:02:55 -05:00
|
|
|
let _string = "LEAKED FROM CLOSURE".to_string();
|
2019-11-24 03:24:41 -06:00
|
|
|
do_panic_counter(do_panic)
|
|
|
|
})).expect_err("do_panic() did not panic!");
|
2019-04-14 20:02:55 -05:00
|
|
|
|
2019-12-02 02:11:33 -06:00
|
|
|
// See if we can extract the panic message.
|
|
|
|
if let Some(s) = res.downcast_ref::<String>() {
|
|
|
|
eprintln!("Caught panic message (String): {}", s);
|
|
|
|
} else if let Some(s) = res.downcast_ref::<&str>() {
|
|
|
|
eprintln!("Caught panic message (&str): {}", s);
|
|
|
|
} else {
|
|
|
|
eprintln!("Failed get caught panic message.");
|
2019-11-24 03:24:41 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// Test flags.
|
|
|
|
assert!(DROPPED.with(|c| c.get()));
|
|
|
|
assert!(HOOK_CALLED.with(|c| c.get()));
|
2019-04-14 20:02:55 -05:00
|
|
|
}
|
|
|
|
|