2020-06-11 15:08:53 -07:00
|
|
|
// revisions: stock precise
|
|
|
|
|
|
|
|
#![cfg_attr(precise, feature(const_precise_live_drops))]
|
2019-11-17 21:11:42 -08:00
|
|
|
|
2020-06-11 15:08:53 -07:00
|
|
|
// `x` is *not* always moved into the final value and may be dropped inside the initializer.
|
2019-11-17 21:11:42 -08:00
|
|
|
const _: Option<Vec<i32>> = {
|
|
|
|
let y: Option<Vec<i32>> = None;
|
|
|
|
let x = Some(Vec::new());
|
2022-09-23 14:22:36 +00:00
|
|
|
//[stock,precise]~^ ERROR destructor of
|
2019-11-17 21:11:42 -08:00
|
|
|
|
|
|
|
if true {
|
|
|
|
x
|
|
|
|
} else {
|
|
|
|
y
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// We only clear `NeedsDrop` if a local is moved from in entirely. This is a shortcoming of the
|
|
|
|
// existing analysis.
|
|
|
|
const _: Vec<i32> = {
|
|
|
|
let vec_tuple = (Vec::new(),);
|
2022-09-23 14:22:36 +00:00
|
|
|
//[stock]~^ ERROR destructor of
|
2019-11-17 21:11:42 -08:00
|
|
|
|
|
|
|
vec_tuple.0
|
|
|
|
};
|
|
|
|
|
|
|
|
// This applies to single-field enum variants as well.
|
|
|
|
const _: Vec<i32> = {
|
|
|
|
let x: Result<_, Vec<i32>> = Ok(Vec::new());
|
2022-09-23 14:22:36 +00:00
|
|
|
//[stock]~^ ERROR destructor of
|
2019-11-17 21:11:42 -08:00
|
|
|
|
|
|
|
match x {
|
|
|
|
Ok(x) | Err(x) => x,
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2019-12-10 21:24:46 -08:00
|
|
|
const _: Option<Vec<i32>> = {
|
|
|
|
let mut some = Some(Vec::new());
|
|
|
|
let mut tmp = None;
|
2022-09-23 14:22:36 +00:00
|
|
|
//[stock,precise]~^ ERROR destructor of
|
2019-12-10 21:24:46 -08:00
|
|
|
|
|
|
|
let mut i = 0;
|
|
|
|
while i < 10 {
|
|
|
|
tmp = some;
|
|
|
|
some = None;
|
|
|
|
|
2019-12-11 10:20:50 -08:00
|
|
|
// We can escape the loop with `Some` still in `tmp`,
|
|
|
|
// which would require that it be dropped at the end of the block.
|
2019-12-10 21:24:46 -08:00
|
|
|
if i > 100 {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
some = tmp;
|
|
|
|
tmp = None;
|
|
|
|
|
|
|
|
i += 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
some
|
|
|
|
};
|
|
|
|
|
2019-11-17 21:11:42 -08:00
|
|
|
fn main() {}
|