2018-08-30 14:18:55 +02:00
|
|
|
// run-pass
|
2018-09-25 23:51:35 +02:00
|
|
|
#![allow(dead_code)]
|
2017-06-19 15:44:29 +03:00
|
|
|
// check that we don't have linear stack usage with multiple calls to `push`
|
|
|
|
|
|
|
|
#![feature(test)]
|
|
|
|
|
|
|
|
extern crate test;
|
|
|
|
use std::mem;
|
|
|
|
|
|
|
|
fn meal() -> Big {
|
|
|
|
if test::black_box(false) {
|
|
|
|
panic!()
|
|
|
|
}
|
|
|
|
Big { drop_me: [
|
|
|
|
None, None, None, None, None, None, None, None,
|
|
|
|
None, None, None, None, None, None, None, None,
|
|
|
|
None, None, None, None, None, None, None, None,
|
|
|
|
None, None, None, None, None, None, None, None,
|
|
|
|
None, None, None, None, None, None, None, None,
|
|
|
|
None, None, None, None, None, None, None, None,
|
|
|
|
]}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub struct Big {
|
|
|
|
drop_me: [Option<Box<u8>>; 48],
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline]
|
|
|
|
fn push(out: &mut Vec<Big>) {
|
|
|
|
out.push(meal());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(never)]
|
|
|
|
pub fn supersize_me(out: &mut Vec<Big>) {
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out); // 16 calls to `push`
|
|
|
|
|
|
|
|
verify_stack_usage(out);
|
|
|
|
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out);
|
|
|
|
push(out); // 16 calls to `push`
|
|
|
|
}
|
|
|
|
|
|
|
|
#[inline(never)]
|
|
|
|
fn verify_stack_usage(before_ptr: *mut Vec<Big>) {
|
2020-03-18 13:35:46 +01:00
|
|
|
// To check stack usage, create locals before and after
|
2017-06-19 15:44:29 +03:00
|
|
|
// and check the difference in addresses between them.
|
|
|
|
let mut stack_var: Vec<Big> = vec![];
|
|
|
|
test::black_box(&mut stack_var);
|
|
|
|
let stack_usage = isize::abs(
|
|
|
|
(&mut stack_var as *mut _ as isize) -
|
|
|
|
(before_ptr as isize)) as usize;
|
2020-03-18 13:35:46 +01:00
|
|
|
// Give space for 2 copies of `Big` + 272 "misc" bytes
|
|
|
|
// (value observed on x86_64-pc-windows-gnu).
|
|
|
|
if stack_usage > mem::size_of::<Big>() * 2 + 272 {
|
2017-06-19 15:44:29 +03:00
|
|
|
panic!("used {} bytes of stack, but `struct Big` is only {} bytes",
|
|
|
|
stack_usage, mem::size_of::<Big>());
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn main() {
|
|
|
|
let mut v = vec![];
|
|
|
|
test::black_box(&mut v);
|
|
|
|
supersize_me(&mut v);
|
|
|
|
}
|