mirror of
				https://github.com/rust-lang/rust.git
				synced 2025-10-28 11:38:01 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			456 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			456 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| use std::ffi::OsStr;
 | |
| use std::fs;
 | |
| use std::path::PathBuf;
 | |
| use std::process::Command;
 | |
| 
 | |
| use crate::path::{Dirs, RelPath};
 | |
| use crate::prepare::{GitRepo, apply_patches};
 | |
| use crate::rustc_info::get_default_sysroot;
 | |
| use crate::shared_utils::rustflags_from_env;
 | |
| use crate::utils::{CargoProject, Compiler, LogGroup, ensure_empty_dir, spawn_and_wait};
 | |
| use crate::{CodegenBackend, SysrootKind, build_sysroot, config};
 | |
| 
 | |
| static BUILD_EXAMPLE_OUT_DIR: RelPath = RelPath::build("example");
 | |
| 
 | |
| struct TestCase {
 | |
|     config: &'static str,
 | |
|     cmd: TestCaseCmd,
 | |
| }
 | |
| 
 | |
| enum TestCaseCmd {
 | |
|     Custom { func: &'static dyn Fn(&TestRunner<'_>) },
 | |
|     BuildLib { source: &'static str, crate_types: &'static str },
 | |
|     BuildBin { source: &'static str },
 | |
|     BuildBinAndRun { source: &'static str, args: &'static [&'static str] },
 | |
|     JitBin { source: &'static str, args: &'static str },
 | |
| }
 | |
| 
 | |
| impl TestCase {
 | |
|     // FIXME reduce usage of custom test case commands
 | |
|     const fn custom(config: &'static str, func: &'static dyn Fn(&TestRunner<'_>)) -> Self {
 | |
|         Self { config, cmd: TestCaseCmd::Custom { func } }
 | |
|     }
 | |
| 
 | |
|     const fn build_lib(
 | |
|         config: &'static str,
 | |
|         source: &'static str,
 | |
|         crate_types: &'static str,
 | |
|     ) -> Self {
 | |
|         Self { config, cmd: TestCaseCmd::BuildLib { source, crate_types } }
 | |
|     }
 | |
| 
 | |
|     const fn build_bin(config: &'static str, source: &'static str) -> Self {
 | |
|         Self { config, cmd: TestCaseCmd::BuildBin { source } }
 | |
|     }
 | |
| 
 | |
|     const fn build_bin_and_run(
 | |
|         config: &'static str,
 | |
|         source: &'static str,
 | |
|         args: &'static [&'static str],
 | |
|     ) -> Self {
 | |
|         Self { config, cmd: TestCaseCmd::BuildBinAndRun { source, args } }
 | |
|     }
 | |
| 
 | |
|     const fn jit_bin(config: &'static str, source: &'static str, args: &'static str) -> Self {
 | |
|         Self { config, cmd: TestCaseCmd::JitBin { source, args } }
 | |
|     }
 | |
| }
 | |
| 
 | |
| const NO_SYSROOT_SUITE: &[TestCase] = &[
 | |
|     TestCase::build_lib("build.mini_core", "example/mini_core.rs", "lib,dylib"),
 | |
|     TestCase::build_lib("build.example", "example/example.rs", "lib"),
 | |
|     TestCase::jit_bin("jit.mini_core_hello_world", "example/mini_core_hello_world.rs", "abc bcd"),
 | |
|     TestCase::build_bin_and_run(
 | |
|         "aot.mini_core_hello_world",
 | |
|         "example/mini_core_hello_world.rs",
 | |
|         &["abc", "bcd"],
 | |
|     ),
 | |
| ];
 | |
| 
 | |
| const BASE_SYSROOT_SUITE: &[TestCase] = &[
 | |
|     TestCase::build_bin_and_run(
 | |
|         "aot.arbitrary_self_types_pointers_and_wrappers",
 | |
|         "example/arbitrary_self_types_pointers_and_wrappers.rs",
 | |
|         &[],
 | |
|     ),
 | |
|     TestCase::build_lib("build.alloc_system", "example/alloc_system.rs", "lib"),
 | |
|     TestCase::build_bin_and_run("aot.alloc_example", "example/alloc_example.rs", &[]),
 | |
|     TestCase::jit_bin("jit.std_example", "example/std_example.rs", "arg"),
 | |
|     TestCase::build_bin_and_run("aot.std_example", "example/std_example.rs", &["arg"]),
 | |
|     TestCase::build_bin_and_run("aot.dst_field_align", "example/dst-field-align.rs", &[]),
 | |
|     TestCase::build_bin_and_run(
 | |
|         "aot.subslice-patterns-const-eval",
 | |
|         "example/subslice-patterns-const-eval.rs",
 | |
|         &[],
 | |
|     ),
 | |
|     TestCase::build_bin_and_run(
 | |
|         "aot.track-caller-attribute",
 | |
|         "example/track-caller-attribute.rs",
 | |
|         &[],
 | |
|     ),
 | |
|     TestCase::build_bin_and_run("aot.float-minmax-pass", "example/float-minmax-pass.rs", &[]),
 | |
|     TestCase::build_bin_and_run("aot.mod_bench", "example/mod_bench.rs", &[]),
 | |
|     TestCase::build_bin_and_run("aot.issue-72793", "example/issue-72793.rs", &[]),
 | |
|     TestCase::build_bin("aot.issue-59326", "example/issue-59326.rs"),
 | |
|     TestCase::build_bin_and_run("aot.neon", "example/neon.rs", &[]),
 | |
|     TestCase::custom("aot.gen_block_iterate", &|runner| {
 | |
|         runner.run_rustc([
 | |
|             "example/gen_block_iterate.rs",
 | |
|             "--edition",
 | |
|             "2024",
 | |
|             "-Zunstable-options",
 | |
|         ]);
 | |
|         runner.run_out_command("gen_block_iterate", &[]);
 | |
|     }),
 | |
|     TestCase::build_bin_and_run("aot.raw-dylib", "example/raw-dylib.rs", &[]),
 | |
| ];
 | |
| 
 | |
| pub(crate) static RAND_REPO: GitRepo = GitRepo::github(
 | |
|     "rust-random",
 | |
|     "rand",
 | |
|     "1f4507a8e1cf8050e4ceef95eeda8f64645b6719",
 | |
|     "981f8bf489338978",
 | |
|     "rand",
 | |
| );
 | |
| 
 | |
| static RAND: CargoProject = CargoProject::new(&RAND_REPO.source_dir(), "rand_target");
 | |
| 
 | |
| pub(crate) static REGEX_REPO: GitRepo = GitRepo::github(
 | |
|     "rust-lang",
 | |
|     "regex",
 | |
|     "061ee815ef2c44101dba7b0b124600fcb03c1912",
 | |
|     "dc26aefbeeac03ca",
 | |
|     "regex",
 | |
| );
 | |
| 
 | |
| static REGEX: CargoProject = CargoProject::new(®EX_REPO.source_dir(), "regex_target");
 | |
| 
 | |
| static PORTABLE_SIMD_SRC: RelPath = RelPath::build("portable-simd");
 | |
| 
 | |
| static PORTABLE_SIMD: CargoProject = CargoProject::new(&PORTABLE_SIMD_SRC, "portable-simd_target");
 | |
| 
 | |
| static LIBCORE_TESTS_SRC: RelPath = RelPath::build("coretests");
 | |
| 
 | |
| static LIBCORE_TESTS: CargoProject = CargoProject::new(&LIBCORE_TESTS_SRC, "coretests_target");
 | |
| 
 | |
| const EXTENDED_SYSROOT_SUITE: &[TestCase] = &[
 | |
|     TestCase::custom("test.rust-random/rand", &|runner| {
 | |
|         RAND_REPO.patch(&runner.dirs);
 | |
| 
 | |
|         RAND.clean(&runner.dirs);
 | |
| 
 | |
|         if runner.is_native {
 | |
|             let mut test_cmd = RAND.test(&runner.target_compiler, &runner.dirs);
 | |
|             test_cmd.arg("--workspace").arg("--").arg("-q");
 | |
|             spawn_and_wait(test_cmd);
 | |
|         } else {
 | |
|             eprintln!("Cross-Compiling: Not running tests");
 | |
|             let mut build_cmd = RAND.build(&runner.target_compiler, &runner.dirs);
 | |
|             build_cmd.arg("--workspace").arg("--tests");
 | |
|             spawn_and_wait(build_cmd);
 | |
|         }
 | |
|     }),
 | |
|     TestCase::custom("test.libcore", &|runner| {
 | |
|         apply_patches(
 | |
|             &runner.dirs,
 | |
|             "coretests",
 | |
|             &runner.stdlib_source.join("library/core/tests"),
 | |
|             &LIBCORE_TESTS_SRC.to_path(&runner.dirs),
 | |
|         );
 | |
| 
 | |
|         let source_lockfile = runner.dirs.source_dir.join("patches/coretests-lock.toml");
 | |
|         let target_lockfile = LIBCORE_TESTS_SRC.to_path(&runner.dirs).join("Cargo.lock");
 | |
|         fs::copy(source_lockfile, target_lockfile).unwrap();
 | |
| 
 | |
|         LIBCORE_TESTS.clean(&runner.dirs);
 | |
| 
 | |
|         if runner.is_native {
 | |
|             let mut test_cmd = LIBCORE_TESTS.test(&runner.target_compiler, &runner.dirs);
 | |
|             test_cmd.arg("--").arg("-q");
 | |
|             spawn_and_wait(test_cmd);
 | |
|         } else {
 | |
|             eprintln!("Cross-Compiling: Not running tests");
 | |
|             let mut build_cmd = LIBCORE_TESTS.build(&runner.target_compiler, &runner.dirs);
 | |
|             build_cmd.arg("--tests");
 | |
|             spawn_and_wait(build_cmd);
 | |
|         }
 | |
|     }),
 | |
|     TestCase::custom("test.regex", &|runner| {
 | |
|         REGEX_REPO.patch(&runner.dirs);
 | |
| 
 | |
|         REGEX.clean(&runner.dirs);
 | |
| 
 | |
|         if runner.is_native {
 | |
|             let mut run_cmd = REGEX.test(&runner.target_compiler, &runner.dirs);
 | |
|             // regex-capi and regex-debug don't have any tests. Nor do they contain any code
 | |
|             // that is useful to test with cg_clif. Skip building them to reduce test time.
 | |
|             run_cmd.args([
 | |
|                 "-p",
 | |
|                 "regex",
 | |
|                 "-p",
 | |
|                 "regex-syntax",
 | |
|                 "--release",
 | |
|                 "--all-targets",
 | |
|                 "--",
 | |
|                 "-q",
 | |
|             ]);
 | |
|             spawn_and_wait(run_cmd);
 | |
| 
 | |
|             let mut run_cmd = REGEX.test(&runner.target_compiler, &runner.dirs);
 | |
|             // don't run integration tests for regex-autonata. they take like 2min each without
 | |
|             // much extra coverage of simd usage.
 | |
|             run_cmd.args(["-p", "regex-automata", "--release", "--lib", "--", "-q"]);
 | |
|             spawn_and_wait(run_cmd);
 | |
|         } else {
 | |
|             eprintln!("Cross-Compiling: Not running tests");
 | |
|             let mut build_cmd = REGEX.build(&runner.target_compiler, &runner.dirs);
 | |
|             build_cmd.arg("--tests");
 | |
|             spawn_and_wait(build_cmd);
 | |
|         }
 | |
|     }),
 | |
|     TestCase::custom("test.portable-simd", &|runner| {
 | |
|         apply_patches(
 | |
|             &runner.dirs,
 | |
|             "portable-simd",
 | |
|             &runner.stdlib_source.join("library/portable-simd"),
 | |
|             &PORTABLE_SIMD_SRC.to_path(&runner.dirs),
 | |
|         );
 | |
| 
 | |
|         PORTABLE_SIMD.clean(&runner.dirs);
 | |
| 
 | |
|         let mut build_cmd = PORTABLE_SIMD.build(&runner.target_compiler, &runner.dirs);
 | |
|         build_cmd.arg("--all-targets");
 | |
|         spawn_and_wait(build_cmd);
 | |
| 
 | |
|         if runner.is_native {
 | |
|             let mut test_cmd = PORTABLE_SIMD.test(&runner.target_compiler, &runner.dirs);
 | |
|             test_cmd.arg("-q");
 | |
|             spawn_and_wait(test_cmd);
 | |
|         }
 | |
|     }),
 | |
| ];
 | |
| 
 | |
| pub(crate) fn run_tests(
 | |
|     dirs: &Dirs,
 | |
|     sysroot_kind: SysrootKind,
 | |
|     use_unstable_features: bool,
 | |
|     skip_tests: &[&str],
 | |
|     cg_clif_dylib: &CodegenBackend,
 | |
|     bootstrap_host_compiler: &Compiler,
 | |
|     rustup_toolchain_name: Option<&str>,
 | |
|     target_triple: String,
 | |
| ) {
 | |
|     let stdlib_source =
 | |
|         get_default_sysroot(&bootstrap_host_compiler.rustc).join("lib/rustlib/src/rust");
 | |
|     assert!(stdlib_source.exists());
 | |
| 
 | |
|     if config::get_bool("testsuite.no_sysroot") && !skip_tests.contains(&"testsuite.no_sysroot") {
 | |
|         let target_compiler = build_sysroot::build_sysroot(
 | |
|             dirs,
 | |
|             SysrootKind::None,
 | |
|             cg_clif_dylib,
 | |
|             bootstrap_host_compiler,
 | |
|             rustup_toolchain_name,
 | |
|             target_triple.clone(),
 | |
|         );
 | |
| 
 | |
|         let runner = TestRunner::new(
 | |
|             dirs.clone(),
 | |
|             target_compiler,
 | |
|             use_unstable_features,
 | |
|             skip_tests,
 | |
|             bootstrap_host_compiler.triple == target_triple,
 | |
|             stdlib_source.clone(),
 | |
|         );
 | |
| 
 | |
|         let path = BUILD_EXAMPLE_OUT_DIR.to_path(dirs);
 | |
|         ensure_empty_dir(&path);
 | |
| 
 | |
|         runner.run_testsuite(NO_SYSROOT_SUITE);
 | |
|     } else {
 | |
|         eprintln!("[SKIP] no_sysroot tests");
 | |
|     }
 | |
| 
 | |
|     let run_base_sysroot = config::get_bool("testsuite.base_sysroot")
 | |
|         && !skip_tests.contains(&"testsuite.base_sysroot");
 | |
|     let run_extended_sysroot = config::get_bool("testsuite.extended_sysroot")
 | |
|         && !skip_tests.contains(&"testsuite.extended_sysroot");
 | |
| 
 | |
|     if run_base_sysroot || run_extended_sysroot {
 | |
|         let target_compiler = build_sysroot::build_sysroot(
 | |
|             dirs,
 | |
|             sysroot_kind,
 | |
|             cg_clif_dylib,
 | |
|             bootstrap_host_compiler,
 | |
|             rustup_toolchain_name,
 | |
|             target_triple.clone(),
 | |
|         );
 | |
| 
 | |
|         let mut runner = TestRunner::new(
 | |
|             dirs.clone(),
 | |
|             target_compiler,
 | |
|             use_unstable_features,
 | |
|             skip_tests,
 | |
|             bootstrap_host_compiler.triple == target_triple,
 | |
|             stdlib_source,
 | |
|         );
 | |
| 
 | |
|         if run_base_sysroot {
 | |
|             runner.run_testsuite(BASE_SYSROOT_SUITE);
 | |
|         } else {
 | |
|             eprintln!("[SKIP] base_sysroot tests");
 | |
|         }
 | |
| 
 | |
|         if run_extended_sysroot {
 | |
|             // Rust's build system denies a couple of lints that trigger on several of the test
 | |
|             // projects. Changing the code to fix them is not worth it, so just silence all lints.
 | |
|             runner.target_compiler.rustflags.push("--cap-lints=allow".to_owned());
 | |
|             runner.run_testsuite(EXTENDED_SYSROOT_SUITE);
 | |
|         } else {
 | |
|             eprintln!("[SKIP] extended_sysroot tests");
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| struct TestRunner<'a> {
 | |
|     is_native: bool,
 | |
|     jit_supported: bool,
 | |
|     skip_tests: &'a [&'a str],
 | |
|     dirs: Dirs,
 | |
|     target_compiler: Compiler,
 | |
|     stdlib_source: PathBuf,
 | |
| }
 | |
| 
 | |
| impl<'a> TestRunner<'a> {
 | |
|     fn new(
 | |
|         dirs: Dirs,
 | |
|         mut target_compiler: Compiler,
 | |
|         use_unstable_features: bool,
 | |
|         skip_tests: &'a [&'a str],
 | |
|         is_native: bool,
 | |
|         stdlib_source: PathBuf,
 | |
|     ) -> Self {
 | |
|         target_compiler.rustflags.extend(rustflags_from_env("RUSTFLAGS"));
 | |
|         target_compiler.rustdocflags.extend(rustflags_from_env("RUSTDOCFLAGS"));
 | |
| 
 | |
|         // FIXME fix `#[linkage = "extern_weak"]` without this
 | |
|         if target_compiler.triple.contains("darwin") {
 | |
|             target_compiler.rustflags.extend([
 | |
|                 "-Clink-arg=-undefined".to_owned(),
 | |
|                 "-Clink-arg=dynamic_lookup".to_owned(),
 | |
|             ]);
 | |
|         }
 | |
| 
 | |
|         let jit_supported = use_unstable_features
 | |
|             && is_native
 | |
|             && target_compiler.triple.contains("x86_64")
 | |
|             && !target_compiler.triple.contains("windows");
 | |
| 
 | |
|         Self { is_native, jit_supported, skip_tests, dirs, target_compiler, stdlib_source }
 | |
|     }
 | |
| 
 | |
|     fn run_testsuite(&self, tests: &[TestCase]) {
 | |
|         for TestCase { config, cmd } in tests {
 | |
|             let (tag, testname) = config.split_once('.').unwrap();
 | |
|             let tag = tag.to_uppercase();
 | |
|             let is_jit_test = tag == "JIT";
 | |
| 
 | |
|             let _guard = if !config::get_bool(config)
 | |
|                 || (is_jit_test && !self.jit_supported)
 | |
|                 || self.skip_tests.contains(&config)
 | |
|             {
 | |
|                 eprintln!("[{tag}] {testname} (skipped)");
 | |
|                 continue;
 | |
|             } else {
 | |
|                 let guard = LogGroup::guard(&format!("[{tag}] {testname}"));
 | |
|                 eprintln!("[{tag}] {testname}");
 | |
|                 guard
 | |
|             };
 | |
| 
 | |
|             match *cmd {
 | |
|                 TestCaseCmd::Custom { func } => func(self),
 | |
|                 TestCaseCmd::BuildLib { source, crate_types } => {
 | |
|                     self.run_rustc([source, "--crate-type", crate_types]);
 | |
|                 }
 | |
|                 TestCaseCmd::BuildBin { source } => {
 | |
|                     self.run_rustc([source]);
 | |
|                 }
 | |
|                 TestCaseCmd::BuildBinAndRun { source, args } => {
 | |
|                     self.run_rustc([source]);
 | |
|                     self.run_out_command(
 | |
|                         source.split('/').last().unwrap().split('.').next().unwrap(),
 | |
|                         args,
 | |
|                     );
 | |
|                 }
 | |
|                 TestCaseCmd::JitBin { source, args } => {
 | |
|                     let mut jit_cmd = self.rustc_command([
 | |
|                         "-Zunstable-options",
 | |
|                         "-Cllvm-args=mode=jit",
 | |
|                         "-Cprefer-dynamic",
 | |
|                         source,
 | |
|                         "--cfg",
 | |
|                         "jit",
 | |
|                     ]);
 | |
|                     if !args.is_empty() {
 | |
|                         jit_cmd.env("CG_CLIF_JIT_ARGS", args);
 | |
|                     }
 | |
|                     spawn_and_wait(jit_cmd);
 | |
| 
 | |
|                     eprintln!("[JIT-lazy] {testname}");
 | |
|                     let mut jit_cmd = self.rustc_command([
 | |
|                         "-Zunstable-options",
 | |
|                         "-Cllvm-args=mode=jit-lazy",
 | |
|                         "-Cprefer-dynamic",
 | |
|                         source,
 | |
|                         "--cfg",
 | |
|                         "jit",
 | |
|                     ]);
 | |
|                     if !args.is_empty() {
 | |
|                         jit_cmd.env("CG_CLIF_JIT_ARGS", args);
 | |
|                     }
 | |
|                     spawn_and_wait(jit_cmd);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #[must_use]
 | |
|     fn rustc_command<I, S>(&self, args: I) -> Command
 | |
|     where
 | |
|         I: IntoIterator<Item = S>,
 | |
|         S: AsRef<OsStr>,
 | |
|     {
 | |
|         let mut cmd = Command::new(&self.target_compiler.rustc);
 | |
|         cmd.args(&self.target_compiler.rustflags);
 | |
|         cmd.arg("-L");
 | |
|         cmd.arg(format!("crate={}", BUILD_EXAMPLE_OUT_DIR.to_path(&self.dirs).display()));
 | |
|         cmd.arg("--out-dir");
 | |
|         cmd.arg(BUILD_EXAMPLE_OUT_DIR.to_path(&self.dirs));
 | |
|         cmd.arg("-Cdebuginfo=2");
 | |
|         cmd.arg("--target");
 | |
|         cmd.arg(&self.target_compiler.triple);
 | |
|         cmd.arg("-Cpanic=abort");
 | |
|         cmd.arg("--check-cfg=cfg(jit)");
 | |
|         cmd.args(args);
 | |
|         cmd
 | |
|     }
 | |
| 
 | |
|     fn run_rustc<I, S>(&self, args: I)
 | |
|     where
 | |
|         I: IntoIterator<Item = S>,
 | |
|         S: AsRef<OsStr>,
 | |
|     {
 | |
|         spawn_and_wait(self.rustc_command(args));
 | |
|     }
 | |
| 
 | |
|     fn run_out_command(&self, name: &str, args: &[&str]) {
 | |
|         let mut cmd = self
 | |
|             .target_compiler
 | |
|             .run_with_runner(BUILD_EXAMPLE_OUT_DIR.to_path(&self.dirs).join(name));
 | |
| 
 | |
|         cmd.args(args);
 | |
| 
 | |
|         spawn_and_wait(cmd);
 | |
|     }
 | |
| }
 | 
