2022-06-02 14:39:47 +00:00
|
|
|
// run-pass
|
2022-09-04 15:17:43 +00:00
|
|
|
// compile-flags: -Zvalidate-mir
|
2022-06-02 14:39:47 +00:00
|
|
|
|
2022-07-21 15:39:01 +00:00
|
|
|
use std::fmt::Display;
|
2022-07-20 16:35:12 +00:00
|
|
|
use std::rc::Rc;
|
2022-06-02 14:39:47 +00:00
|
|
|
use std::sync::atomic::{AtomicU8, Ordering};
|
|
|
|
|
|
|
|
static TRACKER: AtomicU8 = AtomicU8::new(0);
|
|
|
|
|
|
|
|
#[derive(Default)]
|
|
|
|
struct Droppy {
|
|
|
|
inner: u32,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for Droppy {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
TRACKER.store(1, Ordering::Release);
|
|
|
|
println!("I've been dropped");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-21 15:39:01 +00:00
|
|
|
fn foo<'a>(x: &'a str) -> Result<impl Display + 'a, ()> {
|
|
|
|
Ok(x)
|
|
|
|
}
|
|
|
|
|
2022-06-02 14:39:47 +00:00
|
|
|
fn main() {
|
|
|
|
assert_eq!(TRACKER.load(Ordering::Acquire), 0);
|
|
|
|
let 0 = Droppy::default().inner else { return };
|
|
|
|
assert_eq!(TRACKER.load(Ordering::Acquire), 1);
|
|
|
|
println!("Should have dropped 👆");
|
2022-07-20 16:35:12 +00:00
|
|
|
|
2022-07-22 10:13:17 +00:00
|
|
|
{
|
|
|
|
// cf. https://github.com/rust-lang/rust/pull/99518#issuecomment-1191520030
|
|
|
|
struct Foo<'a>(&'a mut u32);
|
|
|
|
|
|
|
|
impl<'a> Drop for Foo<'a> {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
*self.0 = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let mut foo = 0;
|
|
|
|
let Foo(0) = Foo(&mut foo) else {
|
|
|
|
*&mut foo = 1;
|
|
|
|
todo!()
|
|
|
|
};
|
|
|
|
}
|
2022-07-21 15:39:01 +00:00
|
|
|
{
|
|
|
|
let x = String::from("Hey");
|
|
|
|
|
|
|
|
let Ok(s) = foo(&x) else { panic!() };
|
|
|
|
assert_eq!(s.to_string(), x);
|
|
|
|
}
|
2022-07-20 16:35:12 +00:00
|
|
|
{
|
|
|
|
// test let-else drops temps after statement
|
|
|
|
let rc = Rc::new(0);
|
|
|
|
let 0 = *rc.clone() else { unreachable!() };
|
|
|
|
Rc::try_unwrap(rc).unwrap();
|
|
|
|
}
|
|
|
|
{
|
|
|
|
let mut rc = Rc::new(0);
|
|
|
|
let mut i = 0;
|
|
|
|
loop {
|
|
|
|
if i > 3 {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
let 1 = *rc.clone() else {
|
|
|
|
if let Ok(v) = Rc::try_unwrap(rc) {
|
|
|
|
rc = Rc::new(v);
|
|
|
|
} else {
|
|
|
|
panic!()
|
|
|
|
}
|
|
|
|
i += 1;
|
|
|
|
continue
|
|
|
|
};
|
|
|
|
}
|
|
|
|
}
|
2022-07-31 12:31:53 +00:00
|
|
|
{
|
|
|
|
fn must_pass() {
|
|
|
|
let rc = Rc::new(());
|
|
|
|
let &None = &Some(Rc::clone(&rc)) else {
|
|
|
|
Rc::try_unwrap(rc).unwrap();
|
|
|
|
return;
|
|
|
|
};
|
|
|
|
unreachable!();
|
|
|
|
}
|
|
|
|
must_pass();
|
|
|
|
}
|
2022-07-20 16:35:12 +00:00
|
|
|
{
|
|
|
|
// test let-else drops temps before else block
|
2022-07-22 10:13:17 +00:00
|
|
|
// NOTE: this test has to be the last block in the `main`
|
|
|
|
// body.
|
2022-07-20 16:35:12 +00:00
|
|
|
let rc = Rc::new(0);
|
|
|
|
let 1 = *rc.clone() else {
|
|
|
|
Rc::try_unwrap(rc).unwrap();
|
|
|
|
return;
|
|
|
|
};
|
|
|
|
unreachable!();
|
|
|
|
}
|
2022-06-02 14:39:47 +00:00
|
|
|
}
|