2022-01-18 19:38:17 -03:00
|
|
|
//@ run-pass
|
|
|
|
//@ needs-unwind
|
2023-01-24 16:06:35 +08:00
|
|
|
//@ revisions: edition2021 edition2024
|
|
|
|
//@ [edition2021] edition: 2021
|
|
|
|
//@ [edition2024] compile-flags: -Z unstable-options
|
|
|
|
//@ [edition2024] edition: 2024
|
2022-01-18 19:38:17 -03:00
|
|
|
|
|
|
|
// See `mir_drop_order.rs` for more information
|
|
|
|
|
2022-08-20 20:40:08 +02:00
|
|
|
#![feature(let_chains)]
|
2023-01-24 16:06:35 +08:00
|
|
|
#![cfg_attr(edition2024, feature(if_let_rescope))]
|
2022-03-15 03:48:53 +01:00
|
|
|
#![allow(irrefutable_let_patterns)]
|
2022-01-18 19:38:17 -03:00
|
|
|
|
|
|
|
use std::cell::RefCell;
|
|
|
|
use std::panic;
|
|
|
|
|
|
|
|
pub struct DropLogger<'a, T> {
|
|
|
|
extra: T,
|
|
|
|
id: usize,
|
2022-10-13 20:13:52 -07:00
|
|
|
log: &'a panic::AssertUnwindSafe<RefCell<Vec<usize>>>,
|
2022-01-18 19:38:17 -03:00
|
|
|
}
|
|
|
|
|
|
|
|
impl<'a, T> Drop for DropLogger<'a, T> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
self.log.0.borrow_mut().push(self.id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
struct InjectedFailure;
|
|
|
|
|
|
|
|
#[allow(unreachable_code)]
|
|
|
|
fn main() {
|
|
|
|
let log = panic::AssertUnwindSafe(RefCell::new(vec![]));
|
|
|
|
let d = |id, extra| DropLogger { extra, id: id, log: &log };
|
|
|
|
let get = || -> Vec<_> {
|
|
|
|
let mut m = log.0.borrow_mut();
|
|
|
|
let n = m.drain(..);
|
|
|
|
n.collect()
|
|
|
|
};
|
|
|
|
|
|
|
|
{
|
|
|
|
let _x = (
|
|
|
|
d(
|
|
|
|
0,
|
|
|
|
d(
|
|
|
|
1,
|
2023-01-24 16:06:35 +08:00
|
|
|
if let Some(_) = d(2, Some(true)).extra
|
|
|
|
&& let DropLogger { .. } = d(3, None)
|
|
|
|
{
|
2022-01-18 19:38:17 -03:00
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(true)
|
2023-01-24 16:06:35 +08:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.extra,
|
2022-01-18 19:38:17 -03:00
|
|
|
),
|
|
|
|
d(4, None),
|
|
|
|
&d(5, None),
|
|
|
|
d(6, None),
|
2023-01-24 16:06:35 +08:00
|
|
|
if let DropLogger { .. } = d(7, None)
|
|
|
|
&& let DropLogger { .. } = d(8, None)
|
|
|
|
{
|
2022-01-18 19:38:17 -03:00
|
|
|
d(9, None)
|
2023-01-24 16:06:35 +08:00
|
|
|
} else {
|
2022-01-18 19:38:17 -03:00
|
|
|
// 10 is not constructed
|
|
|
|
d(10, None)
|
2022-10-13 20:13:52 -07:00
|
|
|
},
|
2022-01-18 19:38:17 -03:00
|
|
|
);
|
2023-01-24 16:06:35 +08:00
|
|
|
#[cfg(edition2021)]
|
2022-10-13 20:13:52 -07:00
|
|
|
assert_eq!(get(), vec![8, 7, 1, 3, 2]);
|
2023-01-24 16:06:35 +08:00
|
|
|
#[cfg(edition2024)]
|
|
|
|
assert_eq!(get(), vec![3, 2, 8, 7, 1]);
|
2022-01-18 19:38:17 -03:00
|
|
|
}
|
|
|
|
assert_eq!(get(), vec![0, 4, 6, 9, 5]);
|
|
|
|
|
|
|
|
let _ = std::panic::catch_unwind(|| {
|
|
|
|
(
|
|
|
|
d(
|
|
|
|
11,
|
|
|
|
d(
|
|
|
|
12,
|
|
|
|
if let Some(_) = d(13, Some(true)).extra
|
|
|
|
&& let DropLogger { .. } = d(14, None)
|
|
|
|
{
|
|
|
|
None
|
|
|
|
} else {
|
|
|
|
Some(true)
|
2023-01-24 16:06:35 +08:00
|
|
|
},
|
|
|
|
)
|
|
|
|
.extra,
|
2022-01-18 19:38:17 -03:00
|
|
|
),
|
|
|
|
d(15, None),
|
|
|
|
&d(16, None),
|
|
|
|
d(17, None),
|
2023-01-24 16:06:35 +08:00
|
|
|
if let DropLogger { .. } = d(18, None)
|
|
|
|
&& let DropLogger { .. } = d(19, None)
|
|
|
|
{
|
2022-01-18 19:38:17 -03:00
|
|
|
d(20, None)
|
2023-01-24 16:06:35 +08:00
|
|
|
} else {
|
2022-01-18 19:38:17 -03:00
|
|
|
// 10 is not constructed
|
|
|
|
d(21, None)
|
|
|
|
},
|
2023-01-24 16:06:35 +08:00
|
|
|
panic::panic_any(InjectedFailure),
|
2022-01-18 19:38:17 -03:00
|
|
|
);
|
|
|
|
});
|
2023-01-24 16:06:35 +08:00
|
|
|
#[cfg(edition2021)]
|
2022-10-13 20:13:52 -07:00
|
|
|
assert_eq!(get(), vec![20, 17, 15, 11, 19, 18, 16, 12, 14, 13]);
|
2023-01-24 16:06:35 +08:00
|
|
|
#[cfg(edition2024)]
|
|
|
|
assert_eq!(get(), vec![14, 13, 19, 18, 20, 17, 15, 11, 16, 12]);
|
2022-01-18 19:38:17 -03:00
|
|
|
}
|