1use std::collections::{HashMap, HashSet};
12use std::ffi::{OsStr, OsString};
13use std::path::PathBuf;
14use std::{env, fs};
15
16#[cfg(not(test))]
17use crate::builder::Builder;
18use crate::builder::Kind;
19#[cfg(not(test))]
20use crate::core::build_steps::tool;
21use crate::core::config::Target;
22use crate::utils::exec::command;
23use crate::{Build, Subcommand};
24
25pub struct Finder {
26 cache: HashMap<OsString, Option<PathBuf>>,
27 path: OsString,
28}
29
30const STAGE0_MISSING_TARGETS: &[&str] = &[
36 "loongarch32-unknown-none",
38 "loongarch32-unknown-none-softfloat",
39];
40
41#[cfg(not(test))]
44const LIBSTDCXX_MIN_VERSION_THRESHOLD: usize = 8;
45
46impl Finder {
47 pub fn new() -> Self {
48 Self { cache: HashMap::new(), path: env::var_os("PATH").unwrap_or_default() }
49 }
50
51 pub fn maybe_have<S: Into<OsString>>(&mut self, cmd: S) -> Option<PathBuf> {
52 let cmd: OsString = cmd.into();
53 let path = &self.path;
54 self.cache
55 .entry(cmd.clone())
56 .or_insert_with(|| {
57 for path in env::split_paths(path) {
58 let target = path.join(&cmd);
59 let mut cmd_exe = cmd.clone();
60 cmd_exe.push(".exe");
61
62 if target.is_file() || path.join(&cmd_exe).exists() || target.join(&cmd_exe).exists()
65 {
67 return Some(target);
68 }
69 }
70 None
71 })
72 .clone()
73 }
74
75 pub fn must_have<S: AsRef<OsStr>>(&mut self, cmd: S) -> PathBuf {
76 self.maybe_have(&cmd).unwrap_or_else(|| {
77 panic!("\n\ncouldn't find required command: {:?}\n\n", cmd.as_ref());
78 })
79 }
80}
81
82pub fn check(build: &mut Build) {
83 let mut skip_target_sanity =
84 env::var_os("BOOTSTRAP_SKIP_TARGET_SANITY").is_some_and(|s| s == "1" || s == "true");
85
86 skip_target_sanity |= build.config.cmd.kind() == Kind::Check;
87
88 let skipped_paths = [OsStr::new("mir-opt"), OsStr::new("miri")];
90 skip_target_sanity |= build.config.paths.iter().any(|path| {
91 path.components().any(|component| skipped_paths.contains(&component.as_os_str()))
92 });
93
94 let path = env::var_os("PATH").unwrap_or_default();
95 if cfg!(windows) && path.to_string_lossy().contains('\"') {
100 panic!("PATH contains invalid character '\"'");
101 }
102
103 let mut cmd_finder = Finder::new();
104 if build.rust_info().is_managed_git_subrepository() {
107 cmd_finder.must_have("git");
108 }
109
110 #[cfg(not(test))]
112 if !build.config.dry_run() && !build.host_target.is_msvc() && build.config.llvm_from_ci {
113 let builder = Builder::new(build);
114 let libcxx_version = builder.ensure(tool::LibcxxVersionTool { target: build.host_target });
115
116 match libcxx_version {
117 tool::LibcxxVersion::Gnu(version) => {
118 if LIBSTDCXX_MIN_VERSION_THRESHOLD > version {
119 eprintln!(
120 "\nYour system's libstdc++ version is too old for the `llvm.download-ci-llvm` option."
121 );
122 eprintln!("Current version detected: '{version}'");
123 eprintln!("Minimum required version: '{LIBSTDCXX_MIN_VERSION_THRESHOLD}'");
124 eprintln!(
125 "Consider upgrading libstdc++ or disabling the `llvm.download-ci-llvm` option."
126 );
127 eprintln!(
128 "If you choose to upgrade libstdc++, run `x clean` or delete `build/host/libcxx-version` manually after the upgrade."
129 );
130 }
131 }
132 tool::LibcxxVersion::Llvm(_) => {
133 }
135 }
136 }
137
138 let building_llvm = !build.config.llvm_from_ci
140 && build.hosts.iter().any(|host| {
141 build.config.llvm_enabled(*host)
142 && build
143 .config
144 .target_config
145 .get(host)
146 .map(|config| config.llvm_config.is_none())
147 .unwrap_or(true)
148 });
149
150 let need_cmake = building_llvm || build.config.any_sanitizers_to_build();
151 if need_cmake && cmd_finder.maybe_have("cmake").is_none() {
152 eprintln!(
153 "
154Couldn't find required command: cmake
155
156You should install cmake, or set `download-ci-llvm = true` in the
157`[llvm]` section of `bootstrap.toml` to download LLVM rather
158than building it.
159"
160 );
161 crate::exit!(1);
162 }
163
164 build.config.python = build
165 .config
166 .python
167 .take()
168 .map(|p| cmd_finder.must_have(p))
169 .or_else(|| env::var_os("BOOTSTRAP_PYTHON").map(PathBuf::from)) .or_else(|| cmd_finder.maybe_have("python"))
171 .or_else(|| cmd_finder.maybe_have("python3"))
172 .or_else(|| cmd_finder.maybe_have("python2"));
173
174 build.config.nodejs = build
175 .config
176 .nodejs
177 .take()
178 .map(|p| cmd_finder.must_have(p))
179 .or_else(|| cmd_finder.maybe_have("node"))
180 .or_else(|| cmd_finder.maybe_have("nodejs"));
181
182 build.config.npm = build
183 .config
184 .npm
185 .take()
186 .map(|p| cmd_finder.must_have(p))
187 .or_else(|| cmd_finder.maybe_have("npm"));
188
189 build.config.gdb = build
190 .config
191 .gdb
192 .take()
193 .map(|p| cmd_finder.must_have(p))
194 .or_else(|| cmd_finder.maybe_have("gdb"));
195
196 build.config.reuse = build
197 .config
198 .reuse
199 .take()
200 .map(|p| cmd_finder.must_have(p))
201 .or_else(|| cmd_finder.maybe_have("reuse"));
202
203 let stage0_supported_target_list: HashSet<String> = command(&build.config.initial_rustc)
204 .args(["--print", "target-list"])
205 .run_always()
206 .run_capture_stdout(&build)
207 .stdout()
208 .lines()
209 .map(|s| s.to_string())
210 .collect();
211
212 let skip_tools_checks = build.config.dry_run()
217 || matches!(
218 build.config.cmd,
219 Subcommand::Clean { .. }
220 | Subcommand::Check { .. }
221 | Subcommand::Suggest { .. }
222 | Subcommand::Format { .. }
223 | Subcommand::Setup { .. }
224 );
225
226 for target in &build.targets {
229 if target.contains("emscripten") {
233 continue;
234 }
235
236 if target.contains("wasm32") {
238 continue;
239 }
240
241 if skip_target_sanity && target != &build.host_target {
243 continue;
244 }
245
246 if cfg!(not(test)) && !skip_target_sanity && !build.local_rebuild {
248 let mut has_target = false;
249 let target_str = target.to_string();
250
251 let missing_targets_hashset: HashSet<_> =
252 STAGE0_MISSING_TARGETS.iter().map(|t| t.to_string()).collect();
253 let duplicated_targets: Vec<_> =
254 stage0_supported_target_list.intersection(&missing_targets_hashset).collect();
255
256 if !duplicated_targets.is_empty() {
257 println!(
258 "Following targets supported from the stage0 compiler, please remove them from STAGE0_MISSING_TARGETS list."
259 );
260 for duplicated_target in duplicated_targets {
261 println!(" {duplicated_target}");
262 }
263 std::process::exit(1);
264 }
265
266 has_target |= stage0_supported_target_list.contains(&target_str);
268 has_target |= STAGE0_MISSING_TARGETS.contains(&target_str.as_str());
269
270 if !has_target {
271 if target.filepath().is_some_and(|p| p.exists()) {
273 has_target = true;
274 } else if let Some(custom_target_path) = env::var_os("RUST_TARGET_PATH") {
275 let mut target_filename = OsString::from(&target_str);
276 target_filename.push(".json");
278
279 let walker = walkdir::WalkDir::new(custom_target_path).into_iter();
281 for entry in walker.filter_map(|e| e.ok()) {
282 has_target |= entry.file_name() == target_filename;
283 }
284 }
285 }
286
287 if !has_target {
288 panic!(
289 "No such target exists in the target list,\n\
290 make sure to correctly specify the location \
291 of the JSON specification file \
292 for custom targets!\n\
293 Use BOOTSTRAP_SKIP_TARGET_SANITY=1 to \
294 bypass this check."
295 );
296 }
297 }
298
299 if !skip_tools_checks {
300 cmd_finder.must_have(build.cc(*target));
301 if let Some(ar) = build.ar(*target) {
302 cmd_finder.must_have(ar);
303 }
304 }
305 }
306
307 if !skip_tools_checks {
308 for host in &build.hosts {
309 cmd_finder.must_have(build.cxx(*host).unwrap());
310
311 if build.config.llvm_enabled(*host) {
312 let filecheck = build.llvm_filecheck(build.host_target);
314 if !filecheck.starts_with(&build.out)
315 && !filecheck.exists()
316 && build.config.codegen_tests
317 {
318 panic!("FileCheck executable {filecheck:?} does not exist");
319 }
320 }
321 }
322 }
323
324 for target in &build.targets {
325 build
326 .config
327 .target_config
328 .entry(*target)
329 .or_insert_with(|| Target::from_triple(&target.triple));
330
331 if (target.contains("-none-") || target.contains("nvptx"))
332 && build.no_std(*target) == Some(false)
333 {
334 panic!("All the *-none-* and nvptx* targets are no-std targets")
335 }
336
337 if skip_target_sanity && target != &build.host_target {
339 continue;
340 }
341
342 if target.contains("musl") && !target.contains("unikraft") {
344 if build.musl_root(*target).is_none() && build.config.is_host_target(*target) {
347 let target = build.config.target_config.entry(*target).or_default();
348 target.musl_root = Some("/usr".into());
349 }
350 match build.musl_libdir(*target) {
351 Some(libdir) => {
352 if fs::metadata(libdir.join("libc.a")).is_err() {
353 panic!("couldn't find libc.a in musl libdir: {}", libdir.display());
354 }
355 }
356 None => panic!(
357 "when targeting MUSL either the rust.musl-root \
358 option or the target.$TARGET.musl-root option must \
359 be specified in bootstrap.toml"
360 ),
361 }
362 }
363
364 if need_cmake && target.is_msvc() {
365 let out =
369 command("cmake").arg("--help").run_always().run_capture_stdout(&build).stdout();
370 if !out.contains("Visual Studio") {
371 panic!(
372 "
373cmake does not support Visual Studio generators.
374
375This is likely due to it being an msys/cygwin build of cmake,
376rather than the required windows version, built using MinGW
377or Visual Studio.
378
379If you are building under msys2 try installing the mingw-w64-x86_64-cmake
380package instead of cmake:
381
382$ pacman -R cmake && pacman -S mingw-w64-x86_64-cmake
383"
384 );
385 }
386 }
387 }
388
389 if let Some(ref s) = build.config.ccache {
390 cmd_finder.must_have(s);
391 }
392}