2019-07-27 00:54:25 +03:00
|
|
|
// run-pass
|
2019-06-20 18:06:40 -07:00
|
|
|
// Test that we don't duplicate storage for a variable that is moved to another
|
|
|
|
// binding. This used to happen in the presence of unwind and drop edges (see
|
|
|
|
// `complex` below.)
|
|
|
|
//
|
|
|
|
// The exact sizes here can change (we'd like to know when they do). What we
|
|
|
|
// don't want to see is the `complex` generator size being upwards of 2048 bytes
|
|
|
|
// (which would indicate it is reserving space for two copies of Foo.)
|
|
|
|
//
|
|
|
|
// See issue #59123 for a full explanation.
|
|
|
|
|
|
|
|
// edition:2018
|
2019-07-19 14:03:22 +03:00
|
|
|
// ignore-wasm32 issue #62807
|
2019-08-16 22:08:01 -07:00
|
|
|
// ignore-asmjs issue #62807
|
2022-09-20 19:46:27 +00:00
|
|
|
// needs-unwind Size of Closures change on panic=abort
|
2019-06-20 18:06:40 -07:00
|
|
|
|
|
|
|
#![feature(generators, generator_trait)]
|
|
|
|
|
|
|
|
use std::ops::Generator;
|
|
|
|
|
|
|
|
const FOO_SIZE: usize = 1024;
|
2022-07-25 22:36:03 +02:00
|
|
|
struct Foo(#[allow(unused_tuple_struct_fields)] [u8; FOO_SIZE]);
|
2019-06-20 18:06:40 -07:00
|
|
|
|
|
|
|
impl Drop for Foo {
|
|
|
|
fn drop(&mut self) {}
|
|
|
|
}
|
|
|
|
|
2019-06-28 18:41:53 -07:00
|
|
|
fn move_before_yield() -> impl Generator<Yield = (), Return = ()> {
|
2019-06-20 18:06:40 -07:00
|
|
|
static || {
|
|
|
|
let first = Foo([0; FOO_SIZE]);
|
|
|
|
let _second = first;
|
|
|
|
yield;
|
|
|
|
// _second dropped here
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn noop() {}
|
|
|
|
|
2019-06-28 18:41:53 -07:00
|
|
|
fn move_before_yield_with_noop() -> impl Generator<Yield = (), Return = ()> {
|
2019-06-20 18:06:40 -07:00
|
|
|
static || {
|
|
|
|
let first = Foo([0; FOO_SIZE]);
|
|
|
|
noop();
|
|
|
|
let _second = first;
|
|
|
|
yield;
|
|
|
|
// _second dropped here
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-28 18:41:53 -07:00
|
|
|
// Today we don't have NRVO (we allocate space for both `first` and `second`,)
|
|
|
|
// but we can overlap `first` with `_third`.
|
|
|
|
fn overlap_move_points() -> impl Generator<Yield = (), Return = ()> {
|
|
|
|
static || {
|
|
|
|
let first = Foo([0; FOO_SIZE]);
|
|
|
|
yield;
|
|
|
|
let second = first;
|
|
|
|
yield;
|
|
|
|
let _third = second;
|
|
|
|
yield;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-03-10 00:17:33 +01:00
|
|
|
fn overlap_x_and_y() -> impl Generator<Yield = (), Return = ()> {
|
2019-07-16 11:09:29 +09:00
|
|
|
static || {
|
2019-07-25 07:47:57 +09:00
|
|
|
let x = Foo([0; FOO_SIZE]);
|
2019-07-16 11:09:29 +09:00
|
|
|
yield;
|
|
|
|
drop(x);
|
2019-07-25 07:47:57 +09:00
|
|
|
let y = Foo([0; FOO_SIZE]);
|
2019-07-16 11:09:29 +09:00
|
|
|
yield;
|
|
|
|
drop(y);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-20 18:06:40 -07:00
|
|
|
fn main() {
|
2020-03-10 00:17:33 +01:00
|
|
|
assert_eq!(1025, std::mem::size_of_val(&move_before_yield()));
|
|
|
|
assert_eq!(1026, std::mem::size_of_val(&move_before_yield_with_noop()));
|
2020-06-08 14:50:39 -07:00
|
|
|
assert_eq!(2051, std::mem::size_of_val(&overlap_move_points()));
|
2020-03-10 00:17:33 +01:00
|
|
|
assert_eq!(1026, std::mem::size_of_val(&overlap_x_and_y()));
|
2019-06-20 18:06:40 -07:00
|
|
|
}
|