mirror of
https://github.com/rust-lang/rust.git
synced 2025-10-13 15:51:22 +00:00
136 lines
5.0 KiB
Rust
136 lines
5.0 KiB
Rust
use std::collections::HashMap;
|
|
use std::ffi::OsStr;
|
|
#[cfg(unix)]
|
|
use std::os::unix::process::CommandExt;
|
|
use std::path::PathBuf;
|
|
|
|
use crate::config::ConfigInfo;
|
|
use crate::utils::{get_toolchain, rustc_toolchain_version_info, rustc_version_info};
|
|
|
|
fn args(command: &str) -> Result<Option<Vec<String>>, String> {
|
|
// We skip the binary and the "cargo"/"rustc" option.
|
|
if let Some("--help") = std::env::args().nth(2).as_deref() {
|
|
usage(command);
|
|
return Ok(None);
|
|
}
|
|
let args = std::env::args().skip(2).collect::<Vec<_>>();
|
|
if args.is_empty() {
|
|
return Err(format!(
|
|
"Expected at least one argument for `{command}` subcommand, found none"
|
|
));
|
|
}
|
|
Ok(Some(args))
|
|
}
|
|
|
|
fn usage(command: &str) {
|
|
println!(
|
|
r#"
|
|
`{command}` command help:
|
|
|
|
[args] : Arguments to be passed to the cargo command
|
|
--help : Show this help
|
|
"#,
|
|
)
|
|
}
|
|
|
|
struct RustcTools {
|
|
env: HashMap<String, String>,
|
|
args: Vec<String>,
|
|
toolchain: String,
|
|
config: ConfigInfo,
|
|
}
|
|
|
|
impl RustcTools {
|
|
fn new(command: &str) -> Result<Option<Self>, String> {
|
|
let Some(args) = args(command)? else { return Ok(None) };
|
|
|
|
// We first need to go to the original location to ensure that the config setup will go as
|
|
// expected.
|
|
let current_dir = std::env::current_dir()
|
|
.and_then(|path| path.canonicalize())
|
|
.map_err(|error| format!("Failed to get current directory path: {error:?}"))?;
|
|
let current_exe = std::env::current_exe()
|
|
.and_then(|path| path.canonicalize())
|
|
.map_err(|error| format!("Failed to get current exe path: {error:?}"))?;
|
|
let mut parent_dir =
|
|
current_exe.components().map(|comp| comp.as_os_str()).collect::<Vec<_>>();
|
|
// We run this script from "build_system/target/release/y", so we need to remove these elements.
|
|
for to_remove in &["y", "release", "target", "build_system"] {
|
|
if parent_dir.last().map(|part| part == to_remove).unwrap_or(false) {
|
|
parent_dir.pop();
|
|
} else {
|
|
return Err(format!(
|
|
"Build script not executed from `build_system/target/release/y` (in path {})",
|
|
current_exe.display(),
|
|
));
|
|
}
|
|
}
|
|
let parent_dir = PathBuf::from(parent_dir.join(OsStr::new("/")));
|
|
std::env::set_current_dir(&parent_dir).map_err(|error| {
|
|
format!("Failed to go to `{}` folder: {:?}", parent_dir.display(), error)
|
|
})?;
|
|
|
|
let mut env: HashMap<String, String> = std::env::vars().collect();
|
|
let mut config = ConfigInfo::default();
|
|
config.setup(&mut env, false)?;
|
|
let toolchain = get_toolchain()?;
|
|
|
|
let toolchain_version = rustc_toolchain_version_info(&toolchain)?;
|
|
let default_version = rustc_version_info(None)?;
|
|
if toolchain_version != default_version {
|
|
println!(
|
|
"rustc_codegen_gcc is built for {} but the default rustc version is {}.",
|
|
toolchain_version.short, default_version.short,
|
|
);
|
|
println!("Using {}.", toolchain_version.short);
|
|
}
|
|
|
|
// We go back to the original folder since we now have set up everything we needed.
|
|
std::env::set_current_dir(¤t_dir).map_err(|error| {
|
|
format!("Failed to go back to `{}` folder: {:?}", current_dir.display(), error)
|
|
})?;
|
|
let toolchain = format!("+{toolchain}");
|
|
Ok(Some(Self { toolchain, args, env, config }))
|
|
}
|
|
}
|
|
|
|
fn exec(input: &[&dyn AsRef<OsStr>], env: &HashMap<String, String>) -> Result<(), String> {
|
|
#[cfg(unix)]
|
|
{
|
|
// We use `exec` to call the `execvp` syscall instead of creating a new process where the
|
|
// command will be executed because very few signals can actually kill a current process,
|
|
// so if segmentation fault (SIGSEGV signal) happens and we raise to the current process,
|
|
// it will simply do nothing and we won't have the nice error message for the shell.
|
|
let error = crate::utils::get_command_inner(input, None, Some(env)).exec();
|
|
eprintln!("execvp syscall failed: {error:?}");
|
|
std::process::exit(1);
|
|
}
|
|
#[cfg(not(unix))]
|
|
{
|
|
if crate::utils::run_command_with_output_and_env_no_err(input, None, Some(env)).is_err() {
|
|
std::process::exit(1);
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
pub fn run_cargo() -> Result<(), String> {
|
|
let Some(mut tools) = RustcTools::new("cargo")? else { return Ok(()) };
|
|
let rustflags = tools.env.get("RUSTFLAGS").cloned().unwrap_or_default();
|
|
tools.env.insert("RUSTDOCFLAGS".to_string(), rustflags);
|
|
let mut command: Vec<&dyn AsRef<OsStr>> = vec![&"cargo", &tools.toolchain];
|
|
for arg in &tools.args {
|
|
command.push(arg);
|
|
}
|
|
exec(&command, &tools.env)
|
|
}
|
|
|
|
pub fn run_rustc() -> Result<(), String> {
|
|
let Some(tools) = RustcTools::new("rustc")? else { return Ok(()) };
|
|
let mut command = tools.config.rustc_command_vec();
|
|
for arg in &tools.args {
|
|
command.push(arg);
|
|
}
|
|
exec(&command, &tools.env)
|
|
}
|