2019-07-26 21:54:25 +00:00
|
|
|
// run-pass
|
2019-05-22 22:17:38 +00:00
|
|
|
// Regression test for incorrect DropAndReplace behavior introduced in #60840
|
|
|
|
// and fixed in #61373. When combined with the optimization implemented in
|
2023-10-19 21:46:28 +00:00
|
|
|
// #60187, this produced incorrect code for coroutines when a saved local was
|
2019-05-22 22:17:38 +00:00
|
|
|
// re-assigned.
|
|
|
|
|
2023-10-19 21:46:28 +00:00
|
|
|
#![feature(coroutines, coroutine_trait)]
|
2019-05-22 22:17:38 +00:00
|
|
|
|
2023-10-19 16:06:43 +00:00
|
|
|
use std::ops::{Coroutine, CoroutineState};
|
2019-05-22 22:17:38 +00:00
|
|
|
use std::pin::Pin;
|
|
|
|
|
|
|
|
#[derive(Debug, PartialEq)]
|
|
|
|
struct Foo(i32);
|
|
|
|
|
|
|
|
impl Drop for Foo {
|
2023-10-19 16:06:43 +00:00
|
|
|
fn drop(&mut self) {}
|
2019-05-22 22:17:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let mut a = || {
|
|
|
|
let mut x = Foo(4);
|
|
|
|
yield;
|
|
|
|
assert_eq!(x.0, 4);
|
|
|
|
|
|
|
|
// At one point this tricked our dataflow analysis into thinking `x` was
|
|
|
|
// StorageDead after the assignment.
|
|
|
|
x = Foo(5);
|
|
|
|
assert_eq!(x.0, 5);
|
|
|
|
|
|
|
|
{
|
|
|
|
let y = Foo(6);
|
|
|
|
yield;
|
|
|
|
assert_eq!(y.0, 6);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert_eq!(x.0, 5);
|
|
|
|
};
|
|
|
|
|
|
|
|
loop {
|
2020-01-25 19:03:10 +00:00
|
|
|
match Pin::new(&mut a).resume(()) {
|
2023-10-19 16:06:43 +00:00
|
|
|
CoroutineState::Complete(()) => break,
|
2019-05-22 22:17:38 +00:00
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|