2022-03-17 13:49:10 +00:00
|
|
|
use colored::*;
|
|
|
|
use regex::Regex;
|
2022-07-14 10:03:08 +00:00
|
|
|
use std::path::{Path, PathBuf};
|
|
|
|
use std::{env, ffi::OsString};
|
|
|
|
use ui_test::{color_eyre::Result, Config, DependencyBuilder, Mode, OutputConflictHandling};
|
2017-06-21 14:28:13 -07:00
|
|
|
|
2017-08-08 10:28:05 +02:00
|
|
|
fn miri_path() -> PathBuf {
|
2020-05-21 14:24:41 +02:00
|
|
|
PathBuf::from(option_env!("MIRI").unwrap_or(env!("CARGO_BIN_EXE_miri")))
|
2017-08-08 10:28:05 +02:00
|
|
|
}
|
2017-08-02 18:28:12 +02:00
|
|
|
|
2022-07-06 09:09:53 +00:00
|
|
|
fn run_tests(mode: Mode, path: &str, target: Option<String>) -> Result<()> {
|
2020-05-11 10:37:35 +02:00
|
|
|
let in_rustc_test_suite = option_env!("RUSTC_STAGE").is_some();
|
2022-03-17 13:49:10 +00:00
|
|
|
|
2019-07-31 13:48:15 +02:00
|
|
|
// Add some flags we always want.
|
2022-07-14 10:03:08 +00:00
|
|
|
let mut flags: Vec<OsString> = Vec::new();
|
|
|
|
flags.push("--edition".into());
|
|
|
|
flags.push("2018".into());
|
2019-08-21 09:07:27 +02:00
|
|
|
if in_rustc_test_suite {
|
|
|
|
// Less aggressive warnings to make the rustc toolstate management less painful.
|
|
|
|
// (We often get warnings when e.g. a feature gets stabilized or some lint gets added/improved.)
|
2022-07-14 10:03:08 +00:00
|
|
|
flags.push("-Astable-features".into());
|
|
|
|
flags.push("-Aunused".into());
|
2019-08-21 09:07:27 +02:00
|
|
|
} else {
|
2022-07-14 10:03:08 +00:00
|
|
|
flags.push("-Dwarnings".into());
|
|
|
|
flags.push("-Dunused".into());
|
2019-07-31 13:44:55 +02:00
|
|
|
}
|
2022-07-14 10:03:08 +00:00
|
|
|
if let Some(sysroot) = env::var_os("MIRI_SYSROOT") {
|
|
|
|
flags.push("--sysroot".into());
|
2022-03-17 13:49:10 +00:00
|
|
|
flags.push(sysroot);
|
2019-06-09 17:10:04 +02:00
|
|
|
}
|
2020-10-22 10:36:01 +02:00
|
|
|
if let Ok(extra_flags) = env::var("MIRIFLAGS") {
|
2022-03-17 13:49:10 +00:00
|
|
|
for flag in extra_flags.split_whitespace() {
|
2022-07-14 10:03:08 +00:00
|
|
|
flags.push(flag.into());
|
2022-03-17 13:49:10 +00:00
|
|
|
}
|
|
|
|
}
|
2022-07-14 10:03:08 +00:00
|
|
|
flags.push("-Zui-testing".into());
|
2022-03-17 13:49:10 +00:00
|
|
|
if let Some(target) = &target {
|
2022-07-14 10:03:08 +00:00
|
|
|
flags.push("--target".into());
|
|
|
|
flags.push(target.into());
|
2020-03-18 11:19:01 +01:00
|
|
|
}
|
|
|
|
|
2022-06-02 21:09:10 -04:00
|
|
|
let skip_ui_checks = env::var_os("MIRI_SKIP_UI_CHECKS").is_some();
|
2019-06-09 17:10:04 +02:00
|
|
|
|
2022-03-17 13:49:10 +00:00
|
|
|
let output_conflict_handling = match (env::var_os("MIRI_BLESS").is_some(), skip_ui_checks) {
|
|
|
|
(false, false) => OutputConflictHandling::Error,
|
|
|
|
(true, false) => OutputConflictHandling::Bless,
|
|
|
|
(false, true) => OutputConflictHandling::Ignore,
|
|
|
|
(true, true) => panic!("cannot use MIRI_BLESS and MIRI_SKIP_UI_CHECKS at the same time"),
|
|
|
|
};
|
2018-11-09 11:48:10 +01:00
|
|
|
|
2022-05-30 10:27:41 +02:00
|
|
|
// Pass on all arguments as filters.
|
|
|
|
let path_filter = std::env::args().skip(1);
|
2022-05-27 11:43:14 +00:00
|
|
|
|
2022-07-14 10:03:08 +00:00
|
|
|
let use_std = env::var_os("MIRI_NO_STD").is_none();
|
|
|
|
|
2022-03-17 13:49:10 +00:00
|
|
|
let config = Config {
|
|
|
|
args: flags,
|
|
|
|
target,
|
|
|
|
stderr_filters: STDERR.clone(),
|
|
|
|
stdout_filters: STDOUT.clone(),
|
|
|
|
root_dir: PathBuf::from(path),
|
|
|
|
mode,
|
2022-05-30 10:27:41 +02:00
|
|
|
path_filter: path_filter.collect(),
|
2022-03-17 13:49:10 +00:00
|
|
|
program: miri_path(),
|
|
|
|
output_conflict_handling,
|
2022-07-14 10:03:08 +00:00
|
|
|
dependencies_crate_manifest_path: use_std
|
|
|
|
.then(|| Path::new("test_dependencies").join("Cargo.toml")),
|
|
|
|
dependency_builder: Some(DependencyBuilder {
|
|
|
|
program: std::env::var_os("CARGO").unwrap().into(),
|
|
|
|
args: vec![
|
|
|
|
"run".into(),
|
|
|
|
"--manifest-path".into(),
|
|
|
|
"cargo-miri/Cargo.toml".into(),
|
|
|
|
"--".into(),
|
|
|
|
"miri".into(),
|
|
|
|
],
|
|
|
|
envs: vec![],
|
|
|
|
}),
|
2022-03-17 13:49:10 +00:00
|
|
|
};
|
|
|
|
ui_test::run_tests(config)
|
|
|
|
}
|
2018-10-23 13:09:17 +02:00
|
|
|
|
2022-03-17 13:49:10 +00:00
|
|
|
macro_rules! regexes {
|
|
|
|
($name:ident: $($regex:expr => $replacement:expr,)*) => {lazy_static::lazy_static! {
|
|
|
|
static ref $name: Vec<(Regex, &'static str)> = vec![
|
|
|
|
$((Regex::new($regex).unwrap(), $replacement),)*
|
|
|
|
];
|
|
|
|
}};
|
2016-06-15 13:18:35 +02:00
|
|
|
}
|
|
|
|
|
2022-03-17 13:49:10 +00:00
|
|
|
regexes! {
|
|
|
|
STDOUT:
|
|
|
|
// Windows file paths
|
|
|
|
r"\\" => "/",
|
|
|
|
}
|
2018-10-23 13:09:17 +02:00
|
|
|
|
2022-03-17 13:49:10 +00:00
|
|
|
regexes! {
|
|
|
|
STDERR:
|
|
|
|
// erase line and column info
|
2022-05-30 19:21:22 -04:00
|
|
|
r"\.rs:[0-9]+:[0-9]+(: [0-9]+:[0-9]+)?" => ".rs:LL:CC",
|
2022-03-17 13:49:10 +00:00
|
|
|
// erase alloc ids
|
|
|
|
"alloc[0-9]+" => "ALLOC",
|
|
|
|
// erase Stacked Borrows tags
|
|
|
|
"<[0-9]+>" => "<TAG>",
|
|
|
|
// erase whitespace that differs between platforms
|
|
|
|
r" +at (.*\.rs)" => " at $1",
|
|
|
|
// erase generics in backtraces
|
|
|
|
"([0-9]+: .*)::<.*>" => "$1",
|
|
|
|
// erase addresses in backtraces
|
|
|
|
"([0-9]+: ) +0x[0-9a-f]+ - (.*)" => "$1$2",
|
|
|
|
// erase long hexadecimals
|
|
|
|
r"0x[0-9a-fA-F]+[0-9a-fA-F]{2,2}" => "$$HEX",
|
|
|
|
// erase specific alignments
|
|
|
|
"alignment [0-9]+" => "alignment ALIGN",
|
|
|
|
// erase thread caller ids
|
2022-07-11 20:54:31 -04:00
|
|
|
r"call [0-9]+" => "call ID",
|
2022-03-17 13:49:10 +00:00
|
|
|
// erase platform module paths
|
|
|
|
"sys::[a-z]+::" => "sys::PLATFORM::",
|
|
|
|
// Windows file paths
|
|
|
|
r"\\" => "/",
|
2022-05-31 18:23:47 -04:00
|
|
|
// erase Rust stdlib path
|
|
|
|
"[^ `]*/(rust[^/]*|checkout)/library/" => "RUSTLIB/",
|
2022-03-17 13:49:10 +00:00
|
|
|
// erase platform file paths
|
|
|
|
"sys/[a-z]+/" => "sys/PLATFORM/",
|
2022-07-14 10:03:08 +00:00
|
|
|
// erase paths into the crate registry
|
|
|
|
r"[^ ]*/\.cargo/registry/.*/(.*\.rs)" => "CARGO_REGISTRY/$1",
|
2016-12-16 17:10:16 -08:00
|
|
|
}
|
|
|
|
|
2022-07-06 09:09:53 +00:00
|
|
|
fn ui(mode: Mode, path: &str) -> Result<()> {
|
2022-03-17 13:49:10 +00:00
|
|
|
let target = get_target();
|
|
|
|
|
2022-05-25 18:12:54 +02:00
|
|
|
let msg = format!(
|
|
|
|
"## Running ui tests in {path} against miri for {}",
|
|
|
|
target.as_deref().unwrap_or("host")
|
|
|
|
);
|
|
|
|
eprintln!("{}", msg.green().bold());
|
2022-03-17 13:49:10 +00:00
|
|
|
|
2022-07-06 09:09:53 +00:00
|
|
|
run_tests(mode, path, target)
|
2017-07-19 12:52:20 -07:00
|
|
|
}
|
2017-02-07 12:32:39 +01:00
|
|
|
|
2022-03-17 13:49:10 +00:00
|
|
|
fn get_target() -> Option<String> {
|
|
|
|
env::var("MIRI_TEST_TARGET").ok()
|
2018-12-12 11:03:42 +01:00
|
|
|
}
|
2017-07-19 12:52:20 -07:00
|
|
|
|
2022-07-06 09:09:53 +00:00
|
|
|
fn main() -> Result<()> {
|
|
|
|
ui_test::color_eyre::install()?;
|
|
|
|
|
2020-03-18 11:19:01 +01:00
|
|
|
// Add a test env var to do environment communication tests.
|
2020-10-22 10:36:01 +02:00
|
|
|
env::set_var("MIRI_ENV_VAR_TEST", "0");
|
2020-03-23 23:42:03 +01:00
|
|
|
// Let the tests know where to store temp files (they might run for a different target, which can make this hard to find).
|
2020-10-22 10:36:01 +02:00
|
|
|
env::set_var("MIRI_TEMP", env::temp_dir());
|
2019-08-13 12:10:24 -05:00
|
|
|
|
2022-07-06 09:09:53 +00:00
|
|
|
ui(Mode::Pass, "tests/pass")?;
|
|
|
|
ui(Mode::Panic, "tests/panic")?;
|
|
|
|
ui(Mode::Fail, "tests/fail")?;
|
|
|
|
|
|
|
|
Ok(())
|
2018-07-12 15:01:10 +02:00
|
|
|
}
|