2023-07-29 14:57:53 -07:00
|
|
|
// revisions: x32 x64
|
2019-07-27 00:54:25 +03:00
|
|
|
// run-pass
|
2023-07-29 14:57:53 -07:00
|
|
|
//[x32] only-x86
|
|
|
|
//[x64] only-x86_64
|
2017-10-17 18:45:42 -07:00
|
|
|
// ignore-emscripten no processes
|
2019-04-24 09:26:33 -07:00
|
|
|
// ignore-sgx no processes
|
2022-09-19 21:02:42 +00:00
|
|
|
// ignore-fuchsia no exception handler registered for segfault
|
2023-01-10 10:44:05 +01:00
|
|
|
// ignore-nto Crash analysis impossible at SIGSEGV in QNX Neutrino
|
2017-06-21 12:08:18 -07:00
|
|
|
|
2020-09-01 17:12:52 -04:00
|
|
|
use std::env;
|
2019-12-21 21:43:13 -05:00
|
|
|
use std::mem::MaybeUninit;
|
2017-06-21 12:08:18 -07:00
|
|
|
use std::process::Command;
|
|
|
|
use std::thread;
|
|
|
|
|
|
|
|
#[link(name = "rust_test_helpers", kind = "static")]
|
2020-09-01 17:12:52 -04:00
|
|
|
extern "C" {
|
2017-06-21 12:08:18 -07:00
|
|
|
#[link_name = "rust_dbg_extern_identity_u64"]
|
|
|
|
fn black_box(u: u64);
|
|
|
|
}
|
|
|
|
|
|
|
|
fn main() {
|
|
|
|
let args = env::args().skip(1).collect::<Vec<_>>();
|
|
|
|
if args.len() > 0 {
|
|
|
|
match &args[0][..] {
|
2022-09-26 13:40:24 -07:00
|
|
|
"main-recurse" => overflow_recurse(),
|
|
|
|
"child-recurse" => thread::spawn(overflow_recurse).join().unwrap(),
|
|
|
|
"child-frame" => overflow_frame(),
|
2017-06-21 12:08:18 -07:00
|
|
|
_ => panic!(),
|
|
|
|
}
|
2020-09-01 17:12:52 -04:00
|
|
|
return;
|
2017-06-21 12:08:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let me = env::current_exe().unwrap();
|
|
|
|
|
|
|
|
// The linux kernel has some different behavior for the main thread because
|
|
|
|
// the main thread's stack can typically grow. We can't always guarantee
|
|
|
|
// that we report stack overflow on the main thread, see #43052 for some
|
|
|
|
// details
|
|
|
|
if cfg!(not(target_os = "linux")) {
|
2022-09-26 13:40:24 -07:00
|
|
|
assert_overflow(Command::new(&me).arg("main-recurse"));
|
2017-06-21 12:08:18 -07:00
|
|
|
}
|
2022-09-26 13:40:24 -07:00
|
|
|
assert_overflow(Command::new(&me).arg("child-recurse"));
|
|
|
|
assert_overflow(Command::new(&me).arg("child-frame"));
|
2017-06-21 12:08:18 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
#[allow(unconditional_recursion)]
|
2019-12-21 21:43:13 -05:00
|
|
|
fn recurse(array: &MaybeUninit<[u64; 1024]>) {
|
|
|
|
unsafe {
|
|
|
|
black_box(array.as_ptr() as u64);
|
|
|
|
}
|
|
|
|
let local: MaybeUninit<[u64; 1024]> = MaybeUninit::uninit();
|
2017-06-21 12:08:18 -07:00
|
|
|
recurse(&local);
|
|
|
|
}
|
|
|
|
|
2022-09-26 13:40:24 -07:00
|
|
|
#[inline(never)]
|
|
|
|
fn overflow_recurse() {
|
|
|
|
recurse(&MaybeUninit::uninit());
|
|
|
|
}
|
|
|
|
|
|
|
|
fn overflow_frame() {
|
|
|
|
// By using a 1MiB stack frame with only 512KiB stack, we'll jump over any
|
|
|
|
// guard page, even with 64K pages -- but stack probes should catch it.
|
|
|
|
const STACK_SIZE: usize = 512 * 1024;
|
|
|
|
thread::Builder::new().stack_size(STACK_SIZE).spawn(|| {
|
|
|
|
let local: MaybeUninit<[u8; 2 * STACK_SIZE]> = MaybeUninit::uninit();
|
|
|
|
unsafe {
|
|
|
|
black_box(local.as_ptr() as u64);
|
|
|
|
}
|
|
|
|
}).unwrap().join().unwrap();
|
|
|
|
}
|
|
|
|
|
2017-06-21 12:08:18 -07:00
|
|
|
fn assert_overflow(cmd: &mut Command) {
|
|
|
|
let output = cmd.output().unwrap();
|
|
|
|
assert!(!output.status.success());
|
|
|
|
let stdout = String::from_utf8_lossy(&output.stdout);
|
|
|
|
let stderr = String::from_utf8_lossy(&output.stderr);
|
|
|
|
println!("status: {}", output.status);
|
|
|
|
println!("stdout: {}", stdout);
|
|
|
|
println!("stderr: {}", stderr);
|
|
|
|
assert!(stdout.is_empty());
|
|
|
|
assert!(stderr.contains("has overflowed its stack\n"));
|
|
|
|
}
|