bootstrap/core/build_steps/
run.rs1use std::path::PathBuf;
7
8use build_helper::exit;
9use build_helper::git::get_git_untracked_files;
10use clap_complete::{Generator, shells};
11
12use crate::core::build_steps::dist::distdir;
13use crate::core::build_steps::test;
14use crate::core::build_steps::tool::{self, RustcPrivateCompilers, SourceType, Tool};
15use crate::core::build_steps::vendor::{Vendor, default_paths_to_vendor};
16use crate::core::builder::{Builder, Kind, RunConfig, ShouldRun, Step, StepMetadata};
17use crate::core::config::TargetSelection;
18use crate::core::config::flags::{get_completion, top_level_help};
19use crate::utils::exec::command;
20use crate::{Mode, t};
21
22#[derive(Debug, Clone, Hash, PartialEq, Eq)]
23pub struct BuildManifest;
24
25impl Step for BuildManifest {
26 type Output = ();
27 const IS_HOST: bool = true;
28
29 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
30 run.path("src/tools/build-manifest")
31 }
32
33 fn make_run(run: RunConfig<'_>) {
34 run.builder.ensure(BuildManifest);
35 }
36
37 fn run(self, builder: &Builder<'_>) {
38 let mut cmd = command(
41 builder.ensure(tool::BuildManifest::new(builder, builder.config.host_target)).tool_path,
42 );
43 let sign = builder.config.dist_sign_folder.as_ref().unwrap_or_else(|| {
44 panic!("\n\nfailed to specify `dist.sign-folder` in `bootstrap.toml`\n\n")
45 });
46 let addr = builder.config.dist_upload_addr.as_ref().unwrap_or_else(|| {
47 panic!("\n\nfailed to specify `dist.upload-addr` in `bootstrap.toml`\n\n")
48 });
49
50 let today = command("date").arg("+%Y-%m-%d").run_capture_stdout(builder).stdout();
51
52 cmd.arg(sign);
53 cmd.arg(distdir(builder));
54 cmd.arg(today.trim());
55 cmd.arg(addr);
56 cmd.arg(&builder.config.channel);
57
58 builder.create_dir(&distdir(builder));
59 cmd.run(builder);
60 }
61}
62
63#[derive(Debug, Clone, Hash, PartialEq, Eq)]
64pub struct BumpStage0;
65
66impl Step for BumpStage0 {
67 type Output = ();
68 const IS_HOST: bool = true;
69
70 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
71 run.path("src/tools/bump-stage0")
72 }
73
74 fn make_run(run: RunConfig<'_>) {
75 run.builder.ensure(BumpStage0);
76 }
77
78 fn run(self, builder: &Builder<'_>) -> Self::Output {
79 let mut cmd = builder.tool_cmd(Tool::BumpStage0);
80 cmd.args(builder.config.args());
81 cmd.run(builder);
82 }
83}
84
85#[derive(Debug, Clone, Hash, PartialEq, Eq)]
86pub struct ReplaceVersionPlaceholder;
87
88impl Step for ReplaceVersionPlaceholder {
89 type Output = ();
90 const IS_HOST: bool = true;
91
92 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
93 run.path("src/tools/replace-version-placeholder")
94 }
95
96 fn make_run(run: RunConfig<'_>) {
97 run.builder.ensure(ReplaceVersionPlaceholder);
98 }
99
100 fn run(self, builder: &Builder<'_>) -> Self::Output {
101 let mut cmd = builder.tool_cmd(Tool::ReplaceVersionPlaceholder);
102 cmd.arg(&builder.src);
103 cmd.run(builder);
104 }
105}
106
107#[derive(Debug, Clone, PartialEq, Eq, Hash)]
114pub struct Miri {
115 compilers: RustcPrivateCompilers,
117 target: TargetSelection,
119}
120
121impl Step for Miri {
122 type Output = ();
123
124 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
125 run.path("src/tools/miri")
126 }
127
128 fn make_run(run: RunConfig<'_>) {
129 let builder = run.builder;
130
131 let stage = if builder.config.is_explicit_stage() || builder.top_stage >= 1 {
134 builder.top_stage
135 } else {
136 1
137 };
138
139 if stage == 0 {
140 eprintln!("ERROR: miri cannot be run at stage 0");
141 exit!(1);
142 }
143
144 let compilers = RustcPrivateCompilers::new(builder, stage, builder.host_target);
146
147 run.builder.ensure(Miri { compilers, target: run.target });
148 }
149
150 fn run(self, builder: &Builder<'_>) {
151 let host = builder.build.host_target;
152 let compilers = self.compilers;
153 let target = self.target;
154
155 builder.ensure(tool::Miri::from_compilers(compilers));
156
157 let miri_sysroot =
159 test::Miri::build_miri_sysroot(builder, compilers.target_compiler(), target);
160
161 let mut miri = tool::prepare_tool_cargo(
165 builder,
166 compilers.build_compiler(),
167 Mode::ToolRustcPrivate,
168 host,
169 Kind::Run,
170 "src/tools/miri",
171 SourceType::InTree,
172 &[],
173 );
174 miri.add_rustc_lib_path(builder);
175 miri.arg("--").arg("--target").arg(target.rustc_target_arg());
176
177 miri.arg("--sysroot").arg(miri_sysroot);
179
180 miri.args(builder.config.args());
183 if !builder.config.args().iter().any(|arg| arg.starts_with("--edition")) {
185 miri.arg("--edition=2021");
186 }
187
188 miri.into_cmd().run(builder);
189 }
190
191 fn metadata(&self) -> Option<StepMetadata> {
192 Some(StepMetadata::run("miri", self.target).built_by(self.compilers.build_compiler()))
193 }
194}
195
196#[derive(Debug, Clone, Hash, PartialEq, Eq)]
197pub struct CollectLicenseMetadata;
198
199impl Step for CollectLicenseMetadata {
200 type Output = PathBuf;
201 const IS_HOST: bool = true;
202
203 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
204 run.path("src/tools/collect-license-metadata")
205 }
206
207 fn make_run(run: RunConfig<'_>) {
208 run.builder.ensure(CollectLicenseMetadata);
209 }
210
211 fn run(self, builder: &Builder<'_>) -> Self::Output {
212 let Some(reuse) = &builder.config.reuse else {
213 panic!("REUSE is required to collect the license metadata");
214 };
215
216 let dest = builder.src.join("license-metadata.json");
217
218 if !builder.config.dry_run() {
219 builder.require_and_update_all_submodules();
220 if let Ok(Some(untracked)) = get_git_untracked_files(None) {
221 eprintln!(
222 "Warning: {} untracked files may cause the license report to be incorrect.",
223 untracked.len()
224 );
225 }
226 }
227
228 let mut cmd = builder.tool_cmd(Tool::CollectLicenseMetadata);
229 cmd.env("REUSE_EXE", reuse);
230 cmd.env("DEST", &dest);
231 cmd.run(builder);
232
233 dest
234 }
235}
236
237#[derive(Debug, Clone, Hash, PartialEq, Eq)]
238pub struct GenerateCopyright;
239
240impl Step for GenerateCopyright {
241 type Output = Vec<PathBuf>;
242 const IS_HOST: bool = true;
243
244 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
245 run.path("src/tools/generate-copyright")
246 }
247
248 fn make_run(run: RunConfig<'_>) {
249 run.builder.ensure(GenerateCopyright);
250 }
251
252 fn run(self, builder: &Builder<'_>) -> Self::Output {
253 let license_metadata = builder.src.join("license-metadata.json");
254 let dest = builder.out.join("COPYRIGHT.html");
255 let dest_libstd = builder.out.join("COPYRIGHT-library.html");
256
257 let paths_to_vendor = default_paths_to_vendor(builder);
258 for (_, submodules) in &paths_to_vendor {
259 for submodule in submodules {
260 builder.build.require_submodule(submodule, None);
261 }
262 }
263 let cargo_manifests = paths_to_vendor
264 .into_iter()
265 .map(|(path, _submodules)| path.to_str().unwrap().to_string())
266 .inspect(|path| assert!(!path.contains(','), "{path} contains a comma in its name"))
267 .collect::<Vec<_>>()
268 .join(",");
269
270 let vendored_sources = if let Some(path) = builder.vendored_crates_path() {
271 path
272 } else {
273 let cache_dir = builder.out.join("tmp").join("generate-copyright-vendor");
274 builder.ensure(Vendor {
275 sync_args: Vec::new(),
276 versioned_dirs: true,
277 root_dir: builder.src.clone(),
278 output_dir: cache_dir.clone(),
279 });
280 cache_dir
281 };
282
283 let _guard = builder.group("generate-copyright");
284
285 let mut cmd = builder.tool_cmd(Tool::GenerateCopyright);
286 cmd.env("CARGO_MANIFESTS", &cargo_manifests);
287 cmd.env("LICENSE_METADATA", &license_metadata);
288 cmd.env("DEST", &dest);
289 cmd.env("DEST_LIBSTD", &dest_libstd);
290 cmd.env("SRC_DIR", &builder.src);
291 cmd.env("VENDOR_DIR", &vendored_sources);
292 cmd.env("CARGO", &builder.initial_cargo);
293 cmd.env("CARGO_HOME", t!(home::cargo_home()));
294 cmd.current_dir(&builder.src);
297 cmd.run(builder);
298
299 vec![dest, dest_libstd]
300 }
301}
302
303#[derive(Debug, Clone, Hash, PartialEq, Eq)]
304pub struct GenerateWindowsSys;
305
306impl Step for GenerateWindowsSys {
307 type Output = ();
308 const IS_HOST: bool = true;
309
310 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
311 run.path("src/tools/generate-windows-sys")
312 }
313
314 fn make_run(run: RunConfig<'_>) {
315 run.builder.ensure(GenerateWindowsSys);
316 }
317
318 fn run(self, builder: &Builder<'_>) {
319 let mut cmd = builder.tool_cmd(Tool::GenerateWindowsSys);
320 cmd.arg(&builder.src);
321 cmd.run(builder);
322 }
323}
324
325pub fn get_completion_paths(builder: &Builder<'_>) -> Vec<(&'static dyn Generator, PathBuf)> {
327 vec![
328 (&shells::Bash as &'static dyn Generator, builder.src.join("src/etc/completions/x.py.sh")),
329 (&shells::Zsh, builder.src.join("src/etc/completions/x.py.zsh")),
330 (&shells::Fish, builder.src.join("src/etc/completions/x.py.fish")),
331 (&shells::PowerShell, builder.src.join("src/etc/completions/x.py.ps1")),
332 (&shells::Bash, builder.src.join("src/etc/completions/x.sh")),
333 (&shells::Zsh, builder.src.join("src/etc/completions/x.zsh")),
334 (&shells::Fish, builder.src.join("src/etc/completions/x.fish")),
335 (&shells::PowerShell, builder.src.join("src/etc/completions/x.ps1")),
336 ]
337}
338
339#[derive(Debug, Clone, PartialEq, Eq, Hash)]
340pub struct GenerateCompletions;
341
342impl Step for GenerateCompletions {
343 type Output = ();
344
345 fn run(self, builder: &Builder<'_>) {
347 for (shell, path) in get_completion_paths(builder) {
348 if let Some(comp) = get_completion(shell, &path) {
349 std::fs::write(&path, comp).unwrap_or_else(|e| {
350 panic!("writing completion into {} failed: {e:?}", path.display())
351 });
352 }
353 }
354 }
355
356 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
357 run.alias("generate-completions")
358 }
359
360 fn make_run(run: RunConfig<'_>) {
361 run.builder.ensure(GenerateCompletions);
362 }
363}
364
365#[derive(Debug, Clone, Hash, PartialEq, Eq)]
368pub struct UnicodeTableGenerator;
369
370impl Step for UnicodeTableGenerator {
371 type Output = ();
372 const IS_HOST: bool = true;
373
374 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
375 run.path("src/tools/unicode-table-generator")
376 }
377
378 fn make_run(run: RunConfig<'_>) {
379 run.builder.ensure(UnicodeTableGenerator);
380 }
381
382 fn run(self, builder: &Builder<'_>) {
383 let mut cmd = builder.tool_cmd(Tool::UnicodeTableGenerator);
384 cmd.arg(builder.src.join("library/core/src/unicode/unicode_data.rs"));
386 cmd.arg(builder.src.join("library/coretests/tests/unicode/test_data.rs"));
387 cmd.run(builder);
388 }
389}
390
391#[derive(Debug, Clone, Hash, PartialEq, Eq)]
392pub struct FeaturesStatusDump;
393
394impl Step for FeaturesStatusDump {
395 type Output = ();
396 const IS_HOST: bool = true;
397
398 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
399 run.path("src/tools/features-status-dump")
400 }
401
402 fn make_run(run: RunConfig<'_>) {
403 run.builder.ensure(FeaturesStatusDump);
404 }
405
406 fn run(self, builder: &Builder<'_>) {
407 let mut cmd = builder.tool_cmd(Tool::FeaturesStatusDump);
408
409 cmd.arg("--library-path");
410 cmd.arg(builder.src.join("library"));
411
412 cmd.arg("--compiler-path");
413 cmd.arg(builder.src.join("compiler"));
414
415 cmd.arg("--output-path");
416 cmd.arg(builder.out.join("features-status-dump.json"));
417
418 cmd.run(builder);
419 }
420}
421
422#[derive(Clone, Debug, PartialEq, Eq, Hash)]
425pub struct CyclicStep {
426 n: u32,
427}
428
429impl Step for CyclicStep {
430 type Output = ();
431
432 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
433 run.alias("cyclic-step")
434 }
435
436 fn make_run(run: RunConfig<'_>) {
437 run.builder.ensure(CyclicStep { n: 2 })
439 }
440
441 fn run(self, builder: &Builder<'_>) -> Self::Output {
442 builder.ensure(CyclicStep { n: self.n.saturating_sub(1) })
444 }
445}
446
447#[derive(Debug, Clone, Hash, PartialEq, Eq)]
452pub struct CoverageDump;
453
454impl Step for CoverageDump {
455 type Output = ();
456 const IS_HOST: bool = true;
457
458 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
459 run.path("src/tools/coverage-dump")
460 }
461
462 fn is_default_step(_builder: &Builder<'_>) -> bool {
463 false
464 }
465
466 fn make_run(run: RunConfig<'_>) {
467 run.builder.ensure(Self {});
468 }
469
470 fn run(self, builder: &Builder<'_>) {
471 let mut cmd = builder.tool_cmd(Tool::CoverageDump);
472 cmd.args(&builder.config.free_args);
473 cmd.run(builder);
474 }
475}
476
477#[derive(Debug, Clone, PartialEq, Eq, Hash)]
478pub struct Rustfmt;
479
480impl Step for Rustfmt {
481 type Output = ();
482 const IS_HOST: bool = true;
483
484 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
485 run.path("src/tools/rustfmt")
486 }
487
488 fn make_run(run: RunConfig<'_>) {
489 run.builder.ensure(Rustfmt);
490 }
491
492 fn run(self, builder: &Builder<'_>) {
493 let host = builder.build.host_target;
494
495 let stage = if builder.config.is_explicit_stage() || builder.top_stage >= 1 {
498 builder.top_stage
499 } else {
500 1
501 };
502
503 if stage == 0 {
504 eprintln!("rustfmt cannot be run at stage 0");
505 eprintln!("HELP: Use `x fmt` to use stage 0 rustfmt.");
506 std::process::exit(1);
507 }
508
509 let compilers = RustcPrivateCompilers::new(builder, stage, host);
510 let rustfmt_build = builder.ensure(tool::Rustfmt::from_compilers(compilers));
511
512 let mut rustfmt = tool::prepare_tool_cargo(
513 builder,
514 rustfmt_build.build_compiler,
515 Mode::ToolRustcPrivate,
516 host,
517 Kind::Run,
518 "src/tools/rustfmt",
519 SourceType::InTree,
520 &[],
521 );
522
523 rustfmt.args(["--bin", "rustfmt", "--"]);
524 rustfmt.args(builder.config.args());
525
526 rustfmt.into_cmd().run(builder);
527 }
528}
529
530pub fn get_help_path(builder: &Builder<'_>) -> PathBuf {
532 builder.src.join("src/etc/xhelp")
533}
534
535#[derive(Debug, Clone, PartialEq, Eq, Hash)]
536pub struct GenerateHelp;
537
538impl Step for GenerateHelp {
539 type Output = ();
540
541 fn run(self, builder: &Builder<'_>) {
542 let help = top_level_help();
543 let path = get_help_path(builder);
544 std::fs::write(&path, help)
545 .unwrap_or_else(|e| panic!("writing help into {} failed: {e:?}", path.display()));
546 }
547
548 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
549 run.alias("generate-help")
550 }
551
552 fn make_run(run: RunConfig<'_>) {
553 run.builder.ensure(GenerateHelp)
554 }
555}