1use std::borrow::Cow;
10use std::collections::{BTreeMap, HashMap, HashSet};
11use std::ffi::OsStr;
12use std::io::BufReader;
13use std::io::prelude::*;
14use std::path::{Path, PathBuf};
15use std::time::SystemTime;
16use std::{env, fs, str};
17
18use serde_derive::Deserialize;
19#[cfg(feature = "tracing")]
20use tracing::span;
21
22use crate::core::build_steps::gcc::{Gcc, GccOutput, GccTargetPair};
23use crate::core::build_steps::tool::{RustcPrivateCompilers, SourceType, copy_lld_artifacts};
24use crate::core::build_steps::{dist, llvm};
25use crate::core::builder;
26use crate::core::builder::{
27 Builder, Cargo, Kind, RunConfig, ShouldRun, Step, StepMetadata, crate_description,
28};
29use crate::core::config::toml::target::DefaultLinuxLinkerOverride;
30use crate::core::config::{
31 CompilerBuiltins, DebuginfoLevel, LlvmLibunwind, RustcLto, TargetSelection,
32};
33use crate::utils::build_stamp;
34use crate::utils::build_stamp::BuildStamp;
35use crate::utils::exec::command;
36use crate::utils::helpers::{
37 exe, get_clang_cl_resource_dir, is_debug_info, is_dylib, symlink_dir, t, up_to_date,
38};
39use crate::{
40 CLang, CodegenBackendKind, Compiler, DependencyType, FileType, GitRepo, LLVM_TOOLS, Mode,
41 debug, trace,
42};
43
44#[derive(Debug, Clone, PartialEq, Eq, Hash)]
46pub struct Std {
47 pub target: TargetSelection,
48 pub build_compiler: Compiler,
50 crates: Vec<String>,
54 force_recompile: bool,
57 extra_rust_args: &'static [&'static str],
58 is_for_mir_opt_tests: bool,
59}
60
61impl Std {
62 pub fn new(build_compiler: Compiler, target: TargetSelection) -> Self {
63 Self {
64 target,
65 build_compiler,
66 crates: Default::default(),
67 force_recompile: false,
68 extra_rust_args: &[],
69 is_for_mir_opt_tests: false,
70 }
71 }
72
73 pub fn force_recompile(mut self, force_recompile: bool) -> Self {
74 self.force_recompile = force_recompile;
75 self
76 }
77
78 #[expect(clippy::wrong_self_convention)]
79 pub fn is_for_mir_opt_tests(mut self, is_for_mir_opt_tests: bool) -> Self {
80 self.is_for_mir_opt_tests = is_for_mir_opt_tests;
81 self
82 }
83
84 pub fn extra_rust_args(mut self, extra_rust_args: &'static [&'static str]) -> Self {
85 self.extra_rust_args = extra_rust_args;
86 self
87 }
88
89 fn copy_extra_objects(
90 &self,
91 builder: &Builder<'_>,
92 compiler: &Compiler,
93 target: TargetSelection,
94 ) -> Vec<(PathBuf, DependencyType)> {
95 let mut deps = Vec::new();
96 if !self.is_for_mir_opt_tests {
97 deps.extend(copy_third_party_objects(builder, compiler, target));
98 deps.extend(copy_self_contained_objects(builder, compiler, target));
99 }
100 deps
101 }
102
103 pub fn should_be_uplifted_from_stage_1(builder: &Builder<'_>, stage: u32) -> bool {
108 stage > 1 && !builder.config.full_bootstrap
109 }
110}
111
112impl Step for Std {
113 type Output = Option<BuildStamp>;
115
116 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
117 run.crate_or_deps("sysroot").path("library")
118 }
119
120 fn is_default_step(_builder: &Builder<'_>) -> bool {
121 true
122 }
123
124 fn make_run(run: RunConfig<'_>) {
125 let crates = std_crates_for_run_make(&run);
126 let builder = run.builder;
127
128 let force_recompile = builder.rust_info().is_managed_git_subrepository()
132 && builder.download_rustc()
133 && builder.config.has_changes_from_upstream(&["library"]);
134
135 trace!("is managed git repo: {}", builder.rust_info().is_managed_git_subrepository());
136 trace!("download_rustc: {}", builder.download_rustc());
137 trace!(force_recompile);
138
139 run.builder.ensure(Std {
140 build_compiler: run.builder.compiler(run.builder.top_stage, builder.host_target),
143 target: run.target,
144 crates,
145 force_recompile,
146 extra_rust_args: &[],
147 is_for_mir_opt_tests: false,
148 });
149 }
150
151 fn run(self, builder: &Builder<'_>) -> Self::Output {
157 let target = self.target;
158
159 if self.build_compiler.stage == 0
164 && !(builder.local_rebuild && target != builder.host_target)
165 {
166 let compiler = self.build_compiler;
167 builder.ensure(StdLink::from_std(self, compiler));
168
169 return None;
170 }
171
172 let build_compiler = if builder.download_rustc() && self.force_recompile {
173 builder
176 .compiler(self.build_compiler.stage.saturating_sub(1), builder.config.host_target)
177 } else {
178 self.build_compiler
179 };
180
181 if builder.download_rustc()
184 && builder.config.is_host_target(target)
185 && !self.force_recompile
186 {
187 let sysroot =
188 builder.ensure(Sysroot { compiler: build_compiler, force_recompile: false });
189 cp_rustc_component_to_ci_sysroot(
190 builder,
191 &sysroot,
192 builder.config.ci_rust_std_contents(),
193 );
194 return None;
195 }
196
197 if builder.config.keep_stage.contains(&build_compiler.stage)
198 || builder.config.keep_stage_std.contains(&build_compiler.stage)
199 {
200 trace!(keep_stage = ?builder.config.keep_stage);
201 trace!(keep_stage_std = ?builder.config.keep_stage_std);
202
203 builder.info("WARNING: Using a potentially old libstd. This may not behave well.");
204
205 builder.ensure(StartupObjects { compiler: build_compiler, target });
206
207 self.copy_extra_objects(builder, &build_compiler, target);
208
209 builder.ensure(StdLink::from_std(self, build_compiler));
210 return Some(build_stamp::libstd_stamp(builder, build_compiler, target));
211 }
212
213 let mut target_deps = builder.ensure(StartupObjects { compiler: build_compiler, target });
214
215 let stage = build_compiler.stage;
217
218 if Self::should_be_uplifted_from_stage_1(builder, build_compiler.stage) {
219 let build_compiler_for_std_to_uplift = builder.compiler(1, builder.host_target);
220 let stage_1_stamp = builder.std(build_compiler_for_std_to_uplift, target);
221
222 let msg = if build_compiler_for_std_to_uplift.host == target {
223 format!(
224 "Uplifting library (stage{} -> stage{stage})",
225 build_compiler_for_std_to_uplift.stage
226 )
227 } else {
228 format!(
229 "Uplifting library (stage{}:{} -> stage{stage}:{target})",
230 build_compiler_for_std_to_uplift.stage, build_compiler_for_std_to_uplift.host,
231 )
232 };
233
234 builder.info(&msg);
235
236 self.copy_extra_objects(builder, &build_compiler, target);
239
240 builder.ensure(StdLink::from_std(self, build_compiler_for_std_to_uplift));
241 return stage_1_stamp;
242 }
243
244 target_deps.extend(self.copy_extra_objects(builder, &build_compiler, target));
245
246 let mut cargo = if self.is_for_mir_opt_tests {
250 trace!("building special sysroot for mir-opt tests");
251 let mut cargo = builder::Cargo::new_for_mir_opt_tests(
252 builder,
253 build_compiler,
254 Mode::Std,
255 SourceType::InTree,
256 target,
257 Kind::Check,
258 );
259 cargo.rustflag("-Zalways-encode-mir");
260 cargo.arg("--manifest-path").arg(builder.src.join("library/sysroot/Cargo.toml"));
261 cargo
262 } else {
263 trace!("building regular sysroot");
264 let mut cargo = builder::Cargo::new(
265 builder,
266 build_compiler,
267 Mode::Std,
268 SourceType::InTree,
269 target,
270 Kind::Build,
271 );
272 std_cargo(builder, target, &mut cargo, &self.crates);
273 cargo
274 };
275
276 if target.is_synthetic() {
278 cargo.env("RUSTC_BOOTSTRAP_SYNTHETIC_TARGET", "1");
279 }
280 for rustflag in self.extra_rust_args.iter() {
281 cargo.rustflag(rustflag);
282 }
283
284 let _guard = builder.msg(
285 Kind::Build,
286 format_args!("library artifacts{}", crate_description(&self.crates)),
287 Mode::Std,
288 build_compiler,
289 target,
290 );
291
292 let stamp = build_stamp::libstd_stamp(builder, build_compiler, target);
293 run_cargo(
294 builder,
295 cargo,
296 vec![],
297 &stamp,
298 target_deps,
299 if self.is_for_mir_opt_tests {
300 ArtifactKeepMode::OnlyRmeta
301 } else {
302 ArtifactKeepMode::BothRlibAndRmeta
304 },
305 );
306
307 builder.ensure(StdLink::from_std(
308 self,
309 builder.compiler(build_compiler.stage, builder.config.host_target),
310 ));
311 Some(stamp)
312 }
313
314 fn metadata(&self) -> Option<StepMetadata> {
315 Some(StepMetadata::build("std", self.target).built_by(self.build_compiler))
316 }
317}
318
319fn copy_and_stamp(
320 builder: &Builder<'_>,
321 libdir: &Path,
322 sourcedir: &Path,
323 name: &str,
324 target_deps: &mut Vec<(PathBuf, DependencyType)>,
325 dependency_type: DependencyType,
326) {
327 let target = libdir.join(name);
328 builder.copy_link(&sourcedir.join(name), &target, FileType::Regular);
329
330 target_deps.push((target, dependency_type));
331}
332
333fn copy_llvm_libunwind(builder: &Builder<'_>, target: TargetSelection, libdir: &Path) -> PathBuf {
334 let libunwind_path = builder.ensure(llvm::Libunwind { target });
335 let libunwind_source = libunwind_path.join("libunwind.a");
336 let libunwind_target = libdir.join("libunwind.a");
337 builder.copy_link(&libunwind_source, &libunwind_target, FileType::NativeLibrary);
338 libunwind_target
339}
340
341fn copy_third_party_objects(
343 builder: &Builder<'_>,
344 compiler: &Compiler,
345 target: TargetSelection,
346) -> Vec<(PathBuf, DependencyType)> {
347 let mut target_deps = vec![];
348
349 if builder.config.needs_sanitizer_runtime_built(target) && compiler.stage != 0 {
350 target_deps.extend(
353 copy_sanitizers(builder, compiler, target)
354 .into_iter()
355 .map(|d| (d, DependencyType::Target)),
356 );
357 }
358
359 if target == "x86_64-fortanix-unknown-sgx"
360 || builder.config.llvm_libunwind(target) == LlvmLibunwind::InTree
361 && (target.contains("linux")
362 || target.contains("fuchsia")
363 || target.contains("aix")
364 || target.contains("hexagon"))
365 {
366 let libunwind_path =
367 copy_llvm_libunwind(builder, target, &builder.sysroot_target_libdir(*compiler, target));
368 target_deps.push((libunwind_path, DependencyType::Target));
369 }
370
371 target_deps
372}
373
374fn copy_self_contained_objects(
376 builder: &Builder<'_>,
377 compiler: &Compiler,
378 target: TargetSelection,
379) -> Vec<(PathBuf, DependencyType)> {
380 let libdir_self_contained =
381 builder.sysroot_target_libdir(*compiler, target).join("self-contained");
382 t!(fs::create_dir_all(&libdir_self_contained));
383 let mut target_deps = vec![];
384
385 if target.needs_crt_begin_end() {
393 let srcdir = builder.musl_libdir(target).unwrap_or_else(|| {
394 panic!("Target {:?} does not have a \"musl-libdir\" key", target.triple)
395 });
396 if !target.starts_with("wasm32") {
397 for &obj in &["libc.a", "crt1.o", "Scrt1.o", "rcrt1.o", "crti.o", "crtn.o"] {
398 copy_and_stamp(
399 builder,
400 &libdir_self_contained,
401 &srcdir,
402 obj,
403 &mut target_deps,
404 DependencyType::TargetSelfContained,
405 );
406 }
407 let crt_path = builder.ensure(llvm::CrtBeginEnd { target });
408 for &obj in &["crtbegin.o", "crtbeginS.o", "crtend.o", "crtendS.o"] {
409 let src = crt_path.join(obj);
410 let target = libdir_self_contained.join(obj);
411 builder.copy_link(&src, &target, FileType::NativeLibrary);
412 target_deps.push((target, DependencyType::TargetSelfContained));
413 }
414 } else {
415 for &obj in &["libc.a", "crt1-command.o"] {
418 copy_and_stamp(
419 builder,
420 &libdir_self_contained,
421 &srcdir,
422 obj,
423 &mut target_deps,
424 DependencyType::TargetSelfContained,
425 );
426 }
427 }
428 if !target.starts_with("s390x") {
429 let libunwind_path = copy_llvm_libunwind(builder, target, &libdir_self_contained);
430 target_deps.push((libunwind_path, DependencyType::TargetSelfContained));
431 }
432 } else if target.contains("-wasi") {
433 let srcdir = builder.wasi_libdir(target).unwrap_or_else(|| {
434 panic!(
435 "Target {:?} does not have a \"wasi-root\" key in bootstrap.toml \
436 or `$WASI_SDK_PATH` set",
437 target.triple
438 )
439 });
440
441 let srcdir = if target == "wasm32-wasip3" {
445 assert!(!srcdir.exists(), "wasip3 support is in wasi-libc, this should be updated now");
446 builder.wasi_libdir(TargetSelection::from_user("wasm32-wasip2")).unwrap()
447 } else {
448 srcdir
449 };
450 for &obj in &["libc.a", "crt1-command.o", "crt1-reactor.o"] {
451 copy_and_stamp(
452 builder,
453 &libdir_self_contained,
454 &srcdir,
455 obj,
456 &mut target_deps,
457 DependencyType::TargetSelfContained,
458 );
459 }
460 if srcdir.join("eh").exists() {
461 copy_and_stamp(
462 builder,
463 &libdir_self_contained,
464 &srcdir.join("eh"),
465 "libunwind.a",
466 &mut target_deps,
467 DependencyType::TargetSelfContained,
468 );
469 }
470 } else if target.is_windows_gnu() || target.is_windows_gnullvm() {
471 for obj in ["crt2.o", "dllcrt2.o"].iter() {
472 let src = compiler_file(builder, &builder.cc(target), target, CLang::C, obj);
473 let dst = libdir_self_contained.join(obj);
474 builder.copy_link(&src, &dst, FileType::NativeLibrary);
475 target_deps.push((dst, DependencyType::TargetSelfContained));
476 }
477 }
478
479 target_deps
480}
481
482pub fn std_crates_for_run_make(run: &RunConfig<'_>) -> Vec<String> {
485 let mut crates = run.make_run_crates(builder::Alias::Library);
486
487 let target_is_no_std = run.builder.no_std(run.target).unwrap_or(false);
496 if target_is_no_std {
497 crates.retain(|c| c == "core" || c == "alloc");
498 }
499 crates
500}
501
502fn compiler_rt_for_profiler(builder: &Builder<'_>) -> PathBuf {
508 if builder.config.llvm_from_ci {
510 builder.config.maybe_download_ci_llvm();
512 let ci_llvm_compiler_rt = builder.config.ci_llvm_root().join("compiler-rt");
513 if ci_llvm_compiler_rt.exists() {
514 return ci_llvm_compiler_rt;
515 }
516 }
517
518 builder.require_submodule("src/llvm-project", {
520 Some("The `build.profiler` config option requires `compiler-rt` sources from LLVM.")
521 });
522 builder.src.join("src/llvm-project/compiler-rt")
523}
524
525pub fn std_cargo(
528 builder: &Builder<'_>,
529 target: TargetSelection,
530 cargo: &mut Cargo,
531 crates: &[String],
532) {
533 if target.contains("apple") && !builder.config.dry_run() {
551 let mut cmd = builder.rustc_cmd(cargo.compiler());
555 cmd.arg("--target").arg(target.rustc_target_arg());
556 cmd.arg("-Zunstable-options").env("RUSTC_BOOTSTRAP", "1");
559 cmd.arg("--print=deployment-target");
560 let output = cmd.run_capture_stdout(builder).stdout();
561
562 let (env_var, value) = output.split_once('=').unwrap();
563 cargo.env(env_var.trim(), value.trim());
566
567 if let Some(target) = env::var_os("MACOSX_STD_DEPLOYMENT_TARGET") {
577 cargo.env("MACOSX_DEPLOYMENT_TARGET", target);
578 }
579 }
580
581 if let Some(path) = builder.config.profiler_path(target) {
583 cargo.env("LLVM_PROFILER_RT_LIB", path);
584 } else if builder.config.profiler_enabled(target) {
585 let compiler_rt = compiler_rt_for_profiler(builder);
586 cargo.env("RUST_COMPILER_RT_FOR_PROFILER", compiler_rt);
590 }
591
592 let compiler_builtins_c_feature = match builder.config.optimized_compiler_builtins(target) {
606 CompilerBuiltins::LinkLLVMBuiltinsLib(path) => {
607 cargo.env("LLVM_COMPILER_RT_LIB", path);
608 " compiler-builtins-c"
609 }
610 CompilerBuiltins::BuildLLVMFuncs => {
611 builder.require_submodule(
621 "src/llvm-project",
622 Some(
623 "The `build.optimized-compiler-builtins` config option \
624 requires `compiler-rt` sources from LLVM.",
625 ),
626 );
627 let compiler_builtins_root = builder.src.join("src/llvm-project/compiler-rt");
628 if !builder.config.dry_run() {
629 assert!(compiler_builtins_root.exists());
632 }
633
634 cargo.env("RUST_COMPILER_RT_ROOT", &compiler_builtins_root);
637 " compiler-builtins-c"
638 }
639 CompilerBuiltins::BuildRustOnly => "",
640 };
641
642 for krate in crates {
643 cargo.args(["-p", krate]);
644 }
645
646 let mut features = String::new();
647
648 if builder.no_std(target) == Some(true) {
649 features += " compiler-builtins-mem";
650 if !target.starts_with("bpf") {
651 features.push_str(compiler_builtins_c_feature);
652 }
653
654 if crates.is_empty() {
656 cargo.args(["-p", "alloc"]);
657 }
658 cargo
659 .arg("--manifest-path")
660 .arg(builder.src.join("library/alloc/Cargo.toml"))
661 .arg("--features")
662 .arg(features);
663 } else {
664 features += &builder.std_features(target);
665 features.push_str(compiler_builtins_c_feature);
666
667 cargo
668 .arg("--features")
669 .arg(features)
670 .arg("--manifest-path")
671 .arg(builder.src.join("library/sysroot/Cargo.toml"));
672
673 if target.contains("musl")
676 && let Some(p) = builder.musl_libdir(target)
677 {
678 let root = format!("native={}", p.to_str().unwrap());
679 cargo.rustflag("-L").rustflag(&root);
680 }
681
682 if target.contains("-wasi")
683 && let Some(dir) = builder.wasi_libdir(target)
684 {
685 let root = format!("native={}", dir.to_str().unwrap());
686 cargo.rustflag("-L").rustflag(&root);
687 }
688 }
689
690 if builder.config.rust_lto == RustcLto::Off {
691 cargo.rustflag("-Clto=off");
692 }
693
694 if target.contains("riscv") {
701 cargo.rustflag("-Cforce-unwind-tables=yes");
702 }
703
704 let html_root =
705 format!("-Zcrate-attr=doc(html_root_url=\"{}/\")", builder.doc_rust_lang_org_channel(),);
706 cargo.rustflag(&html_root);
707 cargo.rustdocflag(&html_root);
708
709 cargo.rustdocflag("-Zcrate-attr=warn(rust_2018_idioms)");
710}
711
712#[derive(Debug, Clone, PartialEq, Eq, Hash)]
721pub struct StdLink {
722 pub compiler: Compiler,
723 pub target_compiler: Compiler,
724 pub target: TargetSelection,
725 crates: Vec<String>,
727 force_recompile: bool,
729}
730
731impl StdLink {
732 pub fn from_std(std: Std, host_compiler: Compiler) -> Self {
733 Self {
734 compiler: host_compiler,
735 target_compiler: std.build_compiler,
736 target: std.target,
737 crates: std.crates,
738 force_recompile: std.force_recompile,
739 }
740 }
741}
742
743impl Step for StdLink {
744 type Output = ();
745
746 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
747 run.never()
748 }
749
750 fn run(self, builder: &Builder<'_>) {
759 let compiler = self.compiler;
760 let target_compiler = self.target_compiler;
761 let target = self.target;
762
763 let (libdir, hostdir) = if !self.force_recompile && builder.download_rustc() {
765 let lib = builder.sysroot_libdir_relative(self.compiler);
767 let sysroot = builder.ensure(crate::core::build_steps::compile::Sysroot {
768 compiler: self.compiler,
769 force_recompile: self.force_recompile,
770 });
771 let libdir = sysroot.join(lib).join("rustlib").join(target).join("lib");
772 let hostdir = sysroot.join(lib).join("rustlib").join(compiler.host).join("lib");
773 (libdir, hostdir)
774 } else {
775 let libdir = builder.sysroot_target_libdir(target_compiler, target);
776 let hostdir = builder.sysroot_target_libdir(target_compiler, compiler.host);
777 (libdir, hostdir)
778 };
779
780 let is_downloaded_beta_stage0 = builder
781 .build
782 .config
783 .initial_rustc
784 .starts_with(builder.out.join(compiler.host).join("stage0/bin"));
785
786 if compiler.stage == 0 && is_downloaded_beta_stage0 {
790 let sysroot = builder.out.join(compiler.host).join("stage0-sysroot");
792
793 let host = compiler.host;
794 let stage0_bin_dir = builder.out.join(host).join("stage0/bin");
795 let sysroot_bin_dir = sysroot.join("bin");
796 t!(fs::create_dir_all(&sysroot_bin_dir));
797 builder.cp_link_r(&stage0_bin_dir, &sysroot_bin_dir);
798
799 let stage0_lib_dir = builder.out.join(host).join("stage0/lib");
800 t!(fs::create_dir_all(sysroot.join("lib")));
801 builder.cp_link_r(&stage0_lib_dir, &sysroot.join("lib"));
802
803 let sysroot_codegen_backends = builder.sysroot_codegen_backends(compiler);
805 t!(fs::create_dir_all(&sysroot_codegen_backends));
806 let stage0_codegen_backends = builder
807 .out
808 .join(host)
809 .join("stage0/lib/rustlib")
810 .join(host)
811 .join("codegen-backends");
812 if stage0_codegen_backends.exists() {
813 builder.cp_link_r(&stage0_codegen_backends, &sysroot_codegen_backends);
814 }
815 } else if compiler.stage == 0 {
816 let sysroot = builder.out.join(compiler.host.triple).join("stage0-sysroot");
817
818 if builder.local_rebuild {
819 let _ = fs::remove_dir_all(sysroot.join("lib/rustlib/src/rust"));
823 }
824
825 builder.cp_link_r(&builder.initial_sysroot.join("lib"), &sysroot.join("lib"));
826 } else {
827 if builder.download_rustc() {
828 let _ = fs::remove_dir_all(&libdir);
830 let _ = fs::remove_dir_all(&hostdir);
831 }
832
833 add_to_sysroot(
834 builder,
835 &libdir,
836 &hostdir,
837 &build_stamp::libstd_stamp(builder, compiler, target),
838 );
839 }
840 }
841}
842
843fn copy_sanitizers(
845 builder: &Builder<'_>,
846 compiler: &Compiler,
847 target: TargetSelection,
848) -> Vec<PathBuf> {
849 let runtimes: Vec<llvm::SanitizerRuntime> = builder.ensure(llvm::Sanitizers { target });
850
851 if builder.config.dry_run() {
852 return Vec::new();
853 }
854
855 let mut target_deps = Vec::new();
856 let libdir = builder.sysroot_target_libdir(*compiler, target);
857
858 for runtime in &runtimes {
859 let dst = libdir.join(&runtime.name);
860 builder.copy_link(&runtime.path, &dst, FileType::NativeLibrary);
861
862 if target == "x86_64-apple-darwin"
866 || target == "aarch64-apple-darwin"
867 || target == "aarch64-apple-ios"
868 || target == "aarch64-apple-ios-sim"
869 || target == "x86_64-apple-ios"
870 {
871 apple_darwin_update_library_name(builder, &dst, &format!("@rpath/{}", runtime.name));
873 apple_darwin_sign_file(builder, &dst);
876 }
877
878 target_deps.push(dst);
879 }
880
881 target_deps
882}
883
884fn apple_darwin_update_library_name(builder: &Builder<'_>, library_path: &Path, new_name: &str) {
885 command("install_name_tool").arg("-id").arg(new_name).arg(library_path).run(builder);
886}
887
888fn apple_darwin_sign_file(builder: &Builder<'_>, file_path: &Path) {
889 command("codesign")
890 .arg("-f") .arg("-s")
892 .arg("-")
893 .arg(file_path)
894 .run(builder);
895}
896
897#[derive(Debug, Clone, PartialEq, Eq, Hash)]
898pub struct StartupObjects {
899 pub compiler: Compiler,
900 pub target: TargetSelection,
901}
902
903impl Step for StartupObjects {
904 type Output = Vec<(PathBuf, DependencyType)>;
905
906 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
907 run.path("library/rtstartup")
908 }
909
910 fn make_run(run: RunConfig<'_>) {
911 run.builder.ensure(StartupObjects {
912 compiler: run.builder.compiler(run.builder.top_stage, run.build_triple()),
913 target: run.target,
914 });
915 }
916
917 fn run(self, builder: &Builder<'_>) -> Vec<(PathBuf, DependencyType)> {
924 let for_compiler = self.compiler;
925 let target = self.target;
926 if !target.is_windows_gnu() {
929 return vec![];
930 }
931
932 let mut target_deps = vec![];
933
934 let src_dir = &builder.src.join("library").join("rtstartup");
935 let dst_dir = &builder.native_dir(target).join("rtstartup");
936 let sysroot_dir = &builder.sysroot_target_libdir(for_compiler, target);
937 t!(fs::create_dir_all(dst_dir));
938
939 for file in &["rsbegin", "rsend"] {
940 let src_file = &src_dir.join(file.to_string() + ".rs");
941 let dst_file = &dst_dir.join(file.to_string() + ".o");
942 if !up_to_date(src_file, dst_file) {
943 let mut cmd = command(&builder.initial_rustc);
944 cmd.env("RUSTC_BOOTSTRAP", "1");
945 if !builder.local_rebuild {
946 cmd.arg("--cfg").arg("bootstrap");
948 }
949 cmd.arg("--target")
950 .arg(target.rustc_target_arg())
951 .arg("--emit=obj")
952 .arg("-o")
953 .arg(dst_file)
954 .arg(src_file)
955 .run(builder);
956 }
957
958 let obj = sysroot_dir.join((*file).to_string() + ".o");
959 builder.copy_link(dst_file, &obj, FileType::NativeLibrary);
960 target_deps.push((obj, DependencyType::Target));
961 }
962
963 target_deps
964 }
965}
966
967fn cp_rustc_component_to_ci_sysroot(builder: &Builder<'_>, sysroot: &Path, contents: Vec<String>) {
968 let ci_rustc_dir = builder.config.ci_rustc_dir();
969
970 for file in contents {
971 let src = ci_rustc_dir.join(&file);
972 let dst = sysroot.join(file);
973 if src.is_dir() {
974 t!(fs::create_dir_all(dst));
975 } else {
976 builder.copy_link(&src, &dst, FileType::Regular);
977 }
978 }
979}
980
981#[derive(Clone, Debug)]
983pub struct BuiltRustc {
984 pub build_compiler: Compiler,
988}
989
990#[derive(Debug, Clone, PartialEq, Eq, Hash)]
997pub struct Rustc {
998 pub target: TargetSelection,
1000 pub build_compiler: Compiler,
1002 crates: Vec<String>,
1008}
1009
1010impl Rustc {
1011 pub fn new(build_compiler: Compiler, target: TargetSelection) -> Self {
1012 Self { target, build_compiler, crates: Default::default() }
1013 }
1014}
1015
1016impl Step for Rustc {
1017 type Output = BuiltRustc;
1018 const IS_HOST: bool = true;
1019
1020 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
1021 let mut crates = run.builder.in_tree_crates("rustc-main", None);
1022 for (i, krate) in crates.iter().enumerate() {
1023 if krate.name == "rustc-main" {
1026 crates.swap_remove(i);
1027 break;
1028 }
1029 }
1030 run.crates(crates)
1031 }
1032
1033 fn is_default_step(_builder: &Builder<'_>) -> bool {
1034 false
1035 }
1036
1037 fn make_run(run: RunConfig<'_>) {
1038 if run.builder.paths == vec![PathBuf::from("compiler")] {
1041 return;
1042 }
1043
1044 let crates = run.cargo_crates_in_set();
1045 run.builder.ensure(Rustc {
1046 build_compiler: run
1047 .builder
1048 .compiler(run.builder.top_stage.saturating_sub(1), run.build_triple()),
1049 target: run.target,
1050 crates,
1051 });
1052 }
1053
1054 fn run(self, builder: &Builder<'_>) -> Self::Output {
1060 let build_compiler = self.build_compiler;
1061 let target = self.target;
1062
1063 if builder.download_rustc() && build_compiler.stage != 0 {
1066 trace!(stage = build_compiler.stage, "`download_rustc` requested");
1067
1068 let sysroot =
1069 builder.ensure(Sysroot { compiler: build_compiler, force_recompile: false });
1070 cp_rustc_component_to_ci_sysroot(
1071 builder,
1072 &sysroot,
1073 builder.config.ci_rustc_dev_contents(),
1074 );
1075 return BuiltRustc { build_compiler };
1076 }
1077
1078 builder.std(build_compiler, target);
1081
1082 if builder.config.keep_stage.contains(&build_compiler.stage) {
1083 trace!(stage = build_compiler.stage, "`keep-stage` requested");
1084
1085 builder.info("WARNING: Using a potentially old librustc. This may not behave well.");
1086 builder.info("WARNING: Use `--keep-stage-std` if you want to rebuild the compiler when it changes");
1087 builder.ensure(RustcLink::from_rustc(self));
1088
1089 return BuiltRustc { build_compiler };
1090 }
1091
1092 let stage = build_compiler.stage + 1;
1094
1095 if build_compiler.stage >= 2
1100 && !builder.config.full_bootstrap
1101 && target == builder.host_target
1102 {
1103 let uplift_build_compiler = builder.compiler(1, build_compiler.host);
1107
1108 let msg = format!("Uplifting rustc from stage2 to stage{stage})");
1109 builder.info(&msg);
1110
1111 builder.ensure(RustcLink::from_build_compiler_and_sysroot(
1115 uplift_build_compiler,
1117 build_compiler,
1119 target,
1120 self.crates,
1121 ));
1122
1123 return BuiltRustc { build_compiler: uplift_build_compiler };
1126 }
1127
1128 builder.std(
1134 builder.compiler(self.build_compiler.stage, builder.config.host_target),
1135 builder.config.host_target,
1136 );
1137
1138 let mut cargo = builder::Cargo::new(
1139 builder,
1140 build_compiler,
1141 Mode::Rustc,
1142 SourceType::InTree,
1143 target,
1144 Kind::Build,
1145 );
1146
1147 rustc_cargo(builder, &mut cargo, target, &build_compiler, &self.crates);
1148
1149 for krate in &*self.crates {
1153 cargo.arg("-p").arg(krate);
1154 }
1155
1156 if builder.build.config.enable_bolt_settings && build_compiler.stage == 1 {
1157 cargo.env("RUSTC_BOLT_LINK_FLAGS", "1");
1159 }
1160
1161 let _guard = builder.msg(
1162 Kind::Build,
1163 format_args!("compiler artifacts{}", crate_description(&self.crates)),
1164 Mode::Rustc,
1165 build_compiler,
1166 target,
1167 );
1168 let stamp = build_stamp::librustc_stamp(builder, build_compiler, target);
1169
1170 run_cargo(
1171 builder,
1172 cargo,
1173 vec![],
1174 &stamp,
1175 vec![],
1176 ArtifactKeepMode::Custom(Box::new(|filename| {
1177 if filename.contains("jemalloc_sys")
1178 || filename.contains("rustc_public_bridge")
1179 || filename.contains("rustc_public")
1180 {
1181 filename.ends_with(".rlib")
1184 } else {
1185 filename.ends_with(".rmeta")
1189 }
1190 })),
1191 );
1192
1193 let target_root_dir = stamp.path().parent().unwrap();
1194 if builder.config.rust_debuginfo_level_rustc == DebuginfoLevel::None
1200 && builder.config.rust_debuginfo_level_tools == DebuginfoLevel::None
1201 {
1202 let rustc_driver = target_root_dir.join("librustc_driver.so");
1203 strip_debug(builder, target, &rustc_driver);
1204 }
1205
1206 if builder.config.rust_debuginfo_level_rustc == DebuginfoLevel::None {
1207 strip_debug(builder, target, &target_root_dir.join("rustc-main"));
1210 }
1211
1212 builder.ensure(RustcLink::from_rustc(self));
1213 BuiltRustc { build_compiler }
1214 }
1215
1216 fn metadata(&self) -> Option<StepMetadata> {
1217 Some(StepMetadata::build("rustc", self.target).built_by(self.build_compiler))
1218 }
1219}
1220
1221pub fn rustc_cargo(
1222 builder: &Builder<'_>,
1223 cargo: &mut Cargo,
1224 target: TargetSelection,
1225 build_compiler: &Compiler,
1226 crates: &[String],
1227) {
1228 cargo
1229 .arg("--features")
1230 .arg(builder.rustc_features(builder.kind, target, crates))
1231 .arg("--manifest-path")
1232 .arg(builder.src.join("compiler/rustc/Cargo.toml"));
1233
1234 cargo.rustdocflag("-Zcrate-attr=warn(rust_2018_idioms)");
1235
1236 cargo.rustflag("-Zon-broken-pipe=kill");
1250
1251 if builder.build.config.bootstrap_override_lld.is_used() {
1256 cargo.rustflag("-Zdefault-visibility=protected");
1257 }
1258
1259 if is_lto_stage(build_compiler) {
1260 match builder.config.rust_lto {
1261 RustcLto::Thin | RustcLto::Fat => {
1262 cargo.rustflag("-Zdylib-lto");
1265 let lto_type = match builder.config.rust_lto {
1269 RustcLto::Thin => "thin",
1270 RustcLto::Fat => "fat",
1271 _ => unreachable!(),
1272 };
1273 cargo.rustflag(&format!("-Clto={lto_type}"));
1274 cargo.rustflag("-Cembed-bitcode=yes");
1275 }
1276 RustcLto::ThinLocal => { }
1277 RustcLto::Off => {
1278 cargo.rustflag("-Clto=off");
1279 }
1280 }
1281 } else if builder.config.rust_lto == RustcLto::Off {
1282 cargo.rustflag("-Clto=off");
1283 }
1284
1285 if builder.config.bootstrap_override_lld.is_used() && !build_compiler.host.is_msvc() {
1293 cargo.rustflag("-Clink-args=-Wl,--icf=all");
1294 }
1295
1296 if builder.config.rust_profile_use.is_some() && builder.config.rust_profile_generate.is_some() {
1297 panic!("Cannot use and generate PGO profiles at the same time");
1298 }
1299 let is_collecting = if let Some(path) = &builder.config.rust_profile_generate {
1300 if build_compiler.stage == 1 {
1301 cargo.rustflag(&format!("-Cprofile-generate={path}"));
1302 cargo.rustflag("-Cllvm-args=-vp-counters-per-site=4");
1305 true
1306 } else {
1307 false
1308 }
1309 } else if let Some(path) = &builder.config.rust_profile_use {
1310 if build_compiler.stage == 1 {
1311 cargo.rustflag(&format!("-Cprofile-use={path}"));
1312 if builder.is_verbose() {
1313 cargo.rustflag("-Cllvm-args=-pgo-warn-missing-function");
1314 }
1315 true
1316 } else {
1317 false
1318 }
1319 } else {
1320 false
1321 };
1322 if is_collecting {
1323 cargo.rustflag(&format!(
1325 "-Cllvm-args=-static-func-strip-dirname-prefix={}",
1326 builder.config.src.components().count()
1327 ));
1328 }
1329
1330 if let Some(ref ccache) = builder.config.ccache
1335 && build_compiler.stage == 0
1336 && !builder.config.incremental
1337 {
1338 cargo.env("RUSTC_WRAPPER", ccache);
1339 }
1340
1341 rustc_cargo_env(builder, cargo, target);
1342}
1343
1344pub fn rustc_cargo_env(builder: &Builder<'_>, cargo: &mut Cargo, target: TargetSelection) {
1345 cargo
1348 .env("CFG_RELEASE", builder.rust_release())
1349 .env("CFG_RELEASE_CHANNEL", &builder.config.channel)
1350 .env("CFG_VERSION", builder.rust_version());
1351
1352 if builder.config.omit_git_hash {
1356 cargo.env("CFG_OMIT_GIT_HASH", "1");
1357 }
1358
1359 cargo.env("CFG_DEFAULT_CODEGEN_BACKEND", builder.config.default_codegen_backend(target).name());
1360
1361 let libdir_relative = builder.config.libdir_relative().unwrap_or_else(|| Path::new("lib"));
1362 let target_config = builder.config.target_config.get(&target);
1363
1364 cargo.env("CFG_LIBDIR_RELATIVE", libdir_relative);
1365
1366 if let Some(ref ver_date) = builder.rust_info().commit_date() {
1367 cargo.env("CFG_VER_DATE", ver_date);
1368 }
1369 if let Some(ref ver_hash) = builder.rust_info().sha() {
1370 cargo.env("CFG_VER_HASH", ver_hash);
1371 }
1372 if !builder.unstable_features() {
1373 cargo.env("CFG_DISABLE_UNSTABLE_FEATURES", "1");
1374 }
1375
1376 if let Some(s) = target_config.and_then(|c| c.default_linker.as_ref()) {
1379 cargo.env("CFG_DEFAULT_LINKER", s);
1380 } else if let Some(ref s) = builder.config.rustc_default_linker {
1381 cargo.env("CFG_DEFAULT_LINKER", s);
1382 }
1383
1384 if let Some(linker) = target_config.map(|c| c.default_linker_linux_override) {
1386 match linker {
1387 DefaultLinuxLinkerOverride::Off => {}
1388 DefaultLinuxLinkerOverride::SelfContainedLldCc => {
1389 cargo.env("CFG_DEFAULT_LINKER_SELF_CONTAINED_LLD_CC", "1");
1390 }
1391 }
1392 }
1393
1394 cargo.env("CFG_COMPILER_HOST_TRIPLE", target.triple);
1396
1397 if builder.config.rust_verify_llvm_ir {
1398 cargo.env("RUSTC_VERIFY_LLVM_IR", "1");
1399 }
1400
1401 if builder.config.llvm_enabled(target) {
1413 let building_llvm_is_expensive =
1414 crate::core::build_steps::llvm::prebuilt_llvm_config(builder, target, false)
1415 .should_build();
1416
1417 let skip_llvm = (builder.kind == Kind::Check) && building_llvm_is_expensive;
1418 if !skip_llvm {
1419 rustc_llvm_env(builder, cargo, target)
1420 }
1421 }
1422
1423 if builder.config.jemalloc(target) && env::var_os("JEMALLOC_SYS_WITH_LG_PAGE").is_none() {
1425 if target.starts_with("aarch64") {
1428 cargo.env("JEMALLOC_SYS_WITH_LG_PAGE", "16");
1429 }
1430 else if target.starts_with("loongarch") {
1432 cargo.env("JEMALLOC_SYS_WITH_LG_PAGE", "14");
1433 }
1434 }
1435}
1436
1437fn rustc_llvm_env(builder: &Builder<'_>, cargo: &mut Cargo, target: TargetSelection) {
1443 if builder.config.is_rust_llvm(target) {
1444 cargo.env("LLVM_RUSTLLVM", "1");
1445 }
1446 if builder.config.llvm_enzyme {
1447 cargo.env("LLVM_ENZYME", "1");
1448 }
1449 let llvm::LlvmResult { host_llvm_config, .. } = builder.ensure(llvm::Llvm { target });
1450 if builder.config.llvm_offload {
1451 builder.ensure(llvm::OmpOffload { target });
1452 cargo.env("LLVM_OFFLOAD", "1");
1453 }
1454
1455 cargo.env("LLVM_CONFIG", &host_llvm_config);
1456
1457 let mut llvm_linker_flags = String::new();
1467 if builder.config.llvm_profile_generate
1468 && target.is_msvc()
1469 && let Some(ref clang_cl_path) = builder.config.llvm_clang_cl
1470 {
1471 let clang_rt_dir = get_clang_cl_resource_dir(builder, clang_cl_path);
1473 llvm_linker_flags.push_str(&format!("-L{}", clang_rt_dir.display()));
1474 }
1475
1476 if let Some(ref s) = builder.config.llvm_ldflags {
1478 if !llvm_linker_flags.is_empty() {
1479 llvm_linker_flags.push(' ');
1480 }
1481 llvm_linker_flags.push_str(s);
1482 }
1483
1484 if !llvm_linker_flags.is_empty() {
1486 cargo.env("LLVM_LINKER_FLAGS", llvm_linker_flags);
1487 }
1488
1489 if builder.config.llvm_static_stdcpp
1492 && !target.contains("freebsd")
1493 && !target.is_msvc()
1494 && !target.contains("apple")
1495 && !target.contains("solaris")
1496 {
1497 let libstdcxx_name =
1498 if target.contains("windows-gnullvm") { "libc++.a" } else { "libstdc++.a" };
1499 let file = compiler_file(
1500 builder,
1501 &builder.cxx(target).unwrap(),
1502 target,
1503 CLang::Cxx,
1504 libstdcxx_name,
1505 );
1506 cargo.env("LLVM_STATIC_STDCPP", file);
1507 }
1508 if builder.llvm_link_shared() {
1509 cargo.env("LLVM_LINK_SHARED", "1");
1510 }
1511 if builder.config.llvm_use_libcxx {
1512 cargo.env("LLVM_USE_LIBCXX", "1");
1513 }
1514 if builder.config.llvm_assertions {
1515 cargo.env("LLVM_ASSERTIONS", "1");
1516 }
1517}
1518
1519#[derive(Debug, Clone, PartialEq, Eq, Hash)]
1532struct RustcLink {
1533 build_compiler: Compiler,
1535 sysroot_compiler: Compiler,
1538 target: TargetSelection,
1539 crates: Vec<String>,
1541}
1542
1543impl RustcLink {
1544 fn from_rustc(rustc: Rustc) -> Self {
1547 Self {
1548 build_compiler: rustc.build_compiler,
1549 sysroot_compiler: rustc.build_compiler,
1550 target: rustc.target,
1551 crates: rustc.crates,
1552 }
1553 }
1554
1555 fn from_build_compiler_and_sysroot(
1557 build_compiler: Compiler,
1558 sysroot_compiler: Compiler,
1559 target: TargetSelection,
1560 crates: Vec<String>,
1561 ) -> Self {
1562 Self { build_compiler, sysroot_compiler, target, crates }
1563 }
1564}
1565
1566impl Step for RustcLink {
1567 type Output = ();
1568
1569 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
1570 run.never()
1571 }
1572
1573 fn run(self, builder: &Builder<'_>) {
1575 let build_compiler = self.build_compiler;
1576 let sysroot_compiler = self.sysroot_compiler;
1577 let target = self.target;
1578 add_to_sysroot(
1579 builder,
1580 &builder.sysroot_target_libdir(sysroot_compiler, target),
1581 &builder.sysroot_target_libdir(sysroot_compiler, sysroot_compiler.host),
1582 &build_stamp::librustc_stamp(builder, build_compiler, target),
1583 );
1584 }
1585}
1586
1587#[derive(Clone)]
1593pub struct GccDylibSet {
1594 dylibs: BTreeMap<GccTargetPair, GccOutput>,
1595}
1596
1597impl GccDylibSet {
1598 pub fn build(
1601 builder: &Builder<'_>,
1602 host: TargetSelection,
1603 targets: Vec<TargetSelection>,
1604 ) -> Self {
1605 let dylibs = targets
1606 .iter()
1607 .map(|t| GccTargetPair::for_target_pair(host, *t))
1608 .map(|target_pair| (target_pair, builder.ensure(Gcc { target_pair })))
1609 .collect();
1610 Self { dylibs }
1611 }
1612
1613 pub fn install_to(&self, builder: &Builder<'_>, compiler: Compiler) {
1617 if builder.config.dry_run() {
1618 return;
1619 }
1620
1621 let cg_sysroot = builder.sysroot_codegen_backends(compiler);
1623
1624 for (target_pair, libgccjit) in &self.dylibs {
1625 assert_eq!(
1626 target_pair.host(),
1627 compiler.host,
1628 "Trying to install libgccjit ({target_pair}) to a compiler with a different host ({})",
1629 compiler.host
1630 );
1631 let libgccjit_path = libgccjit.libgccjit();
1632
1633 let libgccjit_path = t!(
1637 libgccjit_path.canonicalize(),
1638 format!("Cannot find libgccjit at {}", libgccjit_path.display())
1639 );
1640
1641 let dst = cg_sysroot.join(libgccjit_path_relative_to_cg_dir(target_pair, libgccjit));
1642 t!(std::fs::create_dir_all(dst.parent().unwrap()));
1643 builder.copy_link(&libgccjit_path, &dst, FileType::NativeLibrary);
1644 }
1645 }
1646}
1647
1648pub fn libgccjit_path_relative_to_cg_dir(
1651 target_pair: &GccTargetPair,
1652 libgccjit: &GccOutput,
1653) -> PathBuf {
1654 let target_filename = libgccjit.libgccjit().file_name().unwrap().to_str().unwrap();
1655
1656 Path::new("lib").join(target_pair.target()).join(target_filename)
1658}
1659
1660#[derive(Clone)]
1664pub struct GccCodegenBackendOutput {
1665 stamp: BuildStamp,
1666}
1667
1668impl GccCodegenBackendOutput {
1669 pub fn stamp(&self) -> &BuildStamp {
1670 &self.stamp
1671 }
1672}
1673
1674#[derive(Debug, Clone, PartialEq, Eq, Hash)]
1681pub struct GccCodegenBackend {
1682 compilers: RustcPrivateCompilers,
1683 target: TargetSelection,
1684}
1685
1686impl GccCodegenBackend {
1687 pub fn for_target(compilers: RustcPrivateCompilers, target: TargetSelection) -> Self {
1689 Self { compilers, target }
1690 }
1691}
1692
1693impl Step for GccCodegenBackend {
1694 type Output = GccCodegenBackendOutput;
1695
1696 const IS_HOST: bool = true;
1697
1698 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
1699 run.alias("rustc_codegen_gcc").alias("cg_gcc")
1700 }
1701
1702 fn make_run(run: RunConfig<'_>) {
1703 let compilers = RustcPrivateCompilers::new(run.builder, run.builder.top_stage, run.target);
1704 run.builder.ensure(GccCodegenBackend::for_target(compilers, run.target));
1705 }
1706
1707 fn run(self, builder: &Builder<'_>) -> Self::Output {
1708 let host = self.compilers.target();
1709 let build_compiler = self.compilers.build_compiler();
1710
1711 let stamp = build_stamp::codegen_backend_stamp(
1712 builder,
1713 build_compiler,
1714 host,
1715 &CodegenBackendKind::Gcc,
1716 );
1717
1718 if builder.config.keep_stage.contains(&build_compiler.stage) && stamp.path().exists() {
1719 trace!("`keep-stage` requested");
1720 builder.info(
1721 "WARNING: Using a potentially old codegen backend. \
1722 This may not behave well.",
1723 );
1724 return GccCodegenBackendOutput { stamp };
1727 }
1728
1729 let mut cargo = builder::Cargo::new(
1730 builder,
1731 build_compiler,
1732 Mode::Codegen,
1733 SourceType::InTree,
1734 host,
1735 Kind::Build,
1736 );
1737 cargo.arg("--manifest-path").arg(builder.src.join("compiler/rustc_codegen_gcc/Cargo.toml"));
1738 rustc_cargo_env(builder, &mut cargo, host);
1739
1740 let _guard =
1741 builder.msg(Kind::Build, "codegen backend gcc", Mode::Codegen, build_compiler, host);
1742 let files = run_cargo(builder, cargo, vec![], &stamp, vec![], ArtifactKeepMode::OnlyRlib);
1743
1744 GccCodegenBackendOutput {
1745 stamp: write_codegen_backend_stamp(stamp, files, builder.config.dry_run()),
1746 }
1747 }
1748
1749 fn metadata(&self) -> Option<StepMetadata> {
1750 Some(
1751 StepMetadata::build("rustc_codegen_gcc", self.compilers.target())
1752 .built_by(self.compilers.build_compiler()),
1753 )
1754 }
1755}
1756
1757#[derive(Debug, Clone, PartialEq, Eq, Hash)]
1758pub struct CraneliftCodegenBackend {
1759 pub compilers: RustcPrivateCompilers,
1760}
1761
1762impl Step for CraneliftCodegenBackend {
1763 type Output = BuildStamp;
1764 const IS_HOST: bool = true;
1765
1766 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
1767 run.alias("rustc_codegen_cranelift").alias("cg_clif")
1768 }
1769
1770 fn make_run(run: RunConfig<'_>) {
1771 run.builder.ensure(CraneliftCodegenBackend {
1772 compilers: RustcPrivateCompilers::new(run.builder, run.builder.top_stage, run.target),
1773 });
1774 }
1775
1776 fn run(self, builder: &Builder<'_>) -> Self::Output {
1777 let target = self.compilers.target();
1778 let build_compiler = self.compilers.build_compiler();
1779
1780 let stamp = build_stamp::codegen_backend_stamp(
1781 builder,
1782 build_compiler,
1783 target,
1784 &CodegenBackendKind::Cranelift,
1785 );
1786
1787 if builder.config.keep_stage.contains(&build_compiler.stage) {
1788 trace!("`keep-stage` requested");
1789 builder.info(
1790 "WARNING: Using a potentially old codegen backend. \
1791 This may not behave well.",
1792 );
1793 return stamp;
1796 }
1797
1798 let mut cargo = builder::Cargo::new(
1799 builder,
1800 build_compiler,
1801 Mode::Codegen,
1802 SourceType::InTree,
1803 target,
1804 Kind::Build,
1805 );
1806 cargo
1807 .arg("--manifest-path")
1808 .arg(builder.src.join("compiler/rustc_codegen_cranelift/Cargo.toml"));
1809 rustc_cargo_env(builder, &mut cargo, target);
1810
1811 let _guard = builder.msg(
1812 Kind::Build,
1813 "codegen backend cranelift",
1814 Mode::Codegen,
1815 build_compiler,
1816 target,
1817 );
1818 let files = run_cargo(builder, cargo, vec![], &stamp, vec![], ArtifactKeepMode::OnlyRlib);
1819 write_codegen_backend_stamp(stamp, files, builder.config.dry_run())
1820 }
1821
1822 fn metadata(&self) -> Option<StepMetadata> {
1823 Some(
1824 StepMetadata::build("rustc_codegen_cranelift", self.compilers.target())
1825 .built_by(self.compilers.build_compiler()),
1826 )
1827 }
1828}
1829
1830fn write_codegen_backend_stamp(
1832 mut stamp: BuildStamp,
1833 files: Vec<PathBuf>,
1834 dry_run: bool,
1835) -> BuildStamp {
1836 if dry_run {
1837 return stamp;
1838 }
1839
1840 let mut files = files.into_iter().filter(|f| {
1841 let filename = f.file_name().unwrap().to_str().unwrap();
1842 is_dylib(f) && filename.contains("rustc_codegen_")
1843 });
1844 let codegen_backend = match files.next() {
1845 Some(f) => f,
1846 None => panic!("no dylibs built for codegen backend?"),
1847 };
1848 if let Some(f) = files.next() {
1849 panic!("codegen backend built two dylibs:\n{}\n{}", codegen_backend.display(), f.display());
1850 }
1851
1852 let codegen_backend = codegen_backend.to_str().unwrap();
1853 stamp = stamp.add_stamp(codegen_backend);
1854 t!(stamp.write());
1855 stamp
1856}
1857
1858fn copy_codegen_backends_to_sysroot(
1865 builder: &Builder<'_>,
1866 stamp: BuildStamp,
1867 target_compiler: Compiler,
1868) {
1869 let dst = builder.sysroot_codegen_backends(target_compiler);
1878 t!(fs::create_dir_all(&dst), dst);
1879
1880 if builder.config.dry_run() {
1881 return;
1882 }
1883
1884 if stamp.path().exists() {
1885 let file = get_codegen_backend_file(&stamp);
1886 builder.copy_link(
1887 &file,
1888 &dst.join(normalize_codegen_backend_name(builder, &file)),
1889 FileType::NativeLibrary,
1890 );
1891 }
1892}
1893
1894pub fn get_codegen_backend_file(stamp: &BuildStamp) -> PathBuf {
1896 PathBuf::from(t!(fs::read_to_string(stamp.path())))
1897}
1898
1899pub fn normalize_codegen_backend_name(builder: &Builder<'_>, path: &Path) -> String {
1901 let filename = path.file_name().unwrap().to_str().unwrap();
1902 let dash = filename.find('-').unwrap();
1905 let dot = filename.find('.').unwrap();
1906 format!("{}-{}{}", &filename[..dash], builder.rust_release(), &filename[dot..])
1907}
1908
1909pub fn compiler_file(
1910 builder: &Builder<'_>,
1911 compiler: &Path,
1912 target: TargetSelection,
1913 c: CLang,
1914 file: &str,
1915) -> PathBuf {
1916 if builder.config.dry_run() {
1917 return PathBuf::new();
1918 }
1919 let mut cmd = command(compiler);
1920 cmd.args(builder.cc_handled_clags(target, c));
1921 cmd.args(builder.cc_unhandled_cflags(target, GitRepo::Rustc, c));
1922 cmd.arg(format!("-print-file-name={file}"));
1923 let out = cmd.run_capture_stdout(builder).stdout();
1924 PathBuf::from(out.trim())
1925}
1926
1927#[derive(Debug, Clone, PartialEq, Eq, Hash)]
1928pub struct Sysroot {
1929 pub compiler: Compiler,
1930 force_recompile: bool,
1932}
1933
1934impl Sysroot {
1935 pub(crate) fn new(compiler: Compiler) -> Self {
1936 Sysroot { compiler, force_recompile: false }
1937 }
1938}
1939
1940impl Step for Sysroot {
1941 type Output = PathBuf;
1942
1943 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
1944 run.never()
1945 }
1946
1947 fn run(self, builder: &Builder<'_>) -> PathBuf {
1951 let compiler = self.compiler;
1952 let host_dir = builder.out.join(compiler.host);
1953
1954 let sysroot_dir = |stage| {
1955 if stage == 0 {
1956 host_dir.join("stage0-sysroot")
1957 } else if self.force_recompile && stage == compiler.stage {
1958 host_dir.join(format!("stage{stage}-test-sysroot"))
1959 } else if builder.download_rustc() && compiler.stage != builder.top_stage {
1960 host_dir.join("ci-rustc-sysroot")
1961 } else {
1962 host_dir.join(format!("stage{stage}"))
1963 }
1964 };
1965 let sysroot = sysroot_dir(compiler.stage);
1966 trace!(stage = ?compiler.stage, ?sysroot);
1967
1968 builder.do_if_verbose(|| {
1969 println!("Removing sysroot {} to avoid caching bugs", sysroot.display())
1970 });
1971 let _ = fs::remove_dir_all(&sysroot);
1972 t!(fs::create_dir_all(&sysroot));
1973
1974 if compiler.stage == 0 {
1981 dist::maybe_install_llvm_target(builder, compiler.host, &sysroot);
1982 }
1983
1984 if builder.download_rustc() && compiler.stage != 0 {
1986 assert_eq!(
1987 builder.config.host_target, compiler.host,
1988 "Cross-compiling is not yet supported with `download-rustc`",
1989 );
1990
1991 for stage in 0..=2 {
1993 if stage != compiler.stage {
1994 let dir = sysroot_dir(stage);
1995 if !dir.ends_with("ci-rustc-sysroot") {
1996 let _ = fs::remove_dir_all(dir);
1997 }
1998 }
1999 }
2000
2001 let mut filtered_files = Vec::new();
2015 let mut add_filtered_files = |suffix, contents| {
2016 for path in contents {
2017 let path = Path::new(&path);
2018 if path.parent().is_some_and(|parent| parent.ends_with(suffix)) {
2019 filtered_files.push(path.file_name().unwrap().to_owned());
2020 }
2021 }
2022 };
2023 let suffix = format!("lib/rustlib/{}/lib", compiler.host);
2024 add_filtered_files(suffix.as_str(), builder.config.ci_rustc_dev_contents());
2025 add_filtered_files("lib", builder.config.ci_rust_std_contents());
2028
2029 let filtered_extensions = [
2030 OsStr::new("rmeta"),
2031 OsStr::new("rlib"),
2032 OsStr::new(std::env::consts::DLL_EXTENSION),
2034 ];
2035 let ci_rustc_dir = builder.config.ci_rustc_dir();
2036 builder.cp_link_filtered(&ci_rustc_dir, &sysroot, &|path| {
2037 if path.extension().is_none_or(|ext| !filtered_extensions.contains(&ext)) {
2038 return true;
2039 }
2040 if !path.parent().is_none_or(|p| p.ends_with(&suffix)) {
2041 return true;
2042 }
2043 filtered_files.iter().all(|f| f != path.file_name().unwrap())
2044 });
2045 }
2046
2047 if compiler.stage != 0 {
2053 let sysroot_lib_rustlib_src = sysroot.join("lib/rustlib/src");
2054 t!(fs::create_dir_all(&sysroot_lib_rustlib_src));
2055 let sysroot_lib_rustlib_src_rust = sysroot_lib_rustlib_src.join("rust");
2056 if let Err(e) =
2057 symlink_dir(&builder.config, &builder.src, &sysroot_lib_rustlib_src_rust)
2058 {
2059 eprintln!(
2060 "ERROR: creating symbolic link `{}` to `{}` failed with {}",
2061 sysroot_lib_rustlib_src_rust.display(),
2062 builder.src.display(),
2063 e,
2064 );
2065 if builder.config.rust_remap_debuginfo {
2066 eprintln!(
2067 "ERROR: some `tests/ui` tests will fail when lacking `{}`",
2068 sysroot_lib_rustlib_src_rust.display(),
2069 );
2070 }
2071 build_helper::exit!(1);
2072 }
2073 }
2074
2075 if !builder.download_rustc() {
2077 let sysroot_lib_rustlib_rustcsrc = sysroot.join("lib/rustlib/rustc-src");
2078 t!(fs::create_dir_all(&sysroot_lib_rustlib_rustcsrc));
2079 let sysroot_lib_rustlib_rustcsrc_rust = sysroot_lib_rustlib_rustcsrc.join("rust");
2080 if let Err(e) =
2081 symlink_dir(&builder.config, &builder.src, &sysroot_lib_rustlib_rustcsrc_rust)
2082 {
2083 eprintln!(
2084 "ERROR: creating symbolic link `{}` to `{}` failed with {}",
2085 sysroot_lib_rustlib_rustcsrc_rust.display(),
2086 builder.src.display(),
2087 e,
2088 );
2089 build_helper::exit!(1);
2090 }
2091 }
2092
2093 sysroot
2094 }
2095}
2096
2097#[derive(Debug, Clone, PartialEq, Eq, Hash)]
2104pub struct Assemble {
2105 pub target_compiler: Compiler,
2110}
2111
2112impl Step for Assemble {
2113 type Output = Compiler;
2114 const IS_HOST: bool = true;
2115
2116 fn should_run(run: ShouldRun<'_>) -> ShouldRun<'_> {
2117 run.path("compiler/rustc").path("compiler")
2118 }
2119
2120 fn make_run(run: RunConfig<'_>) {
2121 run.builder.ensure(Assemble {
2122 target_compiler: run.builder.compiler(run.builder.top_stage, run.target),
2123 });
2124 }
2125
2126 fn run(self, builder: &Builder<'_>) -> Compiler {
2127 let target_compiler = self.target_compiler;
2128
2129 if target_compiler.stage == 0 {
2130 trace!("stage 0 build compiler is always available, simply returning");
2131 assert_eq!(
2132 builder.config.host_target, target_compiler.host,
2133 "Cannot obtain compiler for non-native build triple at stage 0"
2134 );
2135 return target_compiler;
2137 }
2138
2139 let libdir = builder.sysroot_target_libdir(target_compiler, target_compiler.host);
2142 let libdir_bin = libdir.parent().unwrap().join("bin");
2143 t!(fs::create_dir_all(&libdir_bin));
2144
2145 if builder.config.llvm_enabled(target_compiler.host) {
2146 trace!("target_compiler.host" = ?target_compiler.host, "LLVM enabled");
2147
2148 let target = target_compiler.host;
2149 let llvm::LlvmResult { host_llvm_config, .. } = builder.ensure(llvm::Llvm { target });
2150 if !builder.config.dry_run() && builder.config.llvm_tools_enabled {
2151 trace!("LLVM tools enabled");
2152
2153 let host_llvm_bin_dir = command(&host_llvm_config)
2154 .arg("--bindir")
2155 .cached()
2156 .run_capture_stdout(builder)
2157 .stdout()
2158 .trim()
2159 .to_string();
2160
2161 let llvm_bin_dir = if target == builder.host_target {
2162 PathBuf::from(host_llvm_bin_dir)
2163 } else {
2164 let external_llvm_config = builder
2167 .config
2168 .target_config
2169 .get(&target)
2170 .and_then(|t| t.llvm_config.clone());
2171 if let Some(external_llvm_config) = external_llvm_config {
2172 external_llvm_config.parent().unwrap().to_path_buf()
2175 } else {
2176 let host_llvm_out = builder.llvm_out(builder.host_target);
2180 let target_llvm_out = builder.llvm_out(target);
2181 if let Ok(relative_path) =
2182 Path::new(&host_llvm_bin_dir).strip_prefix(host_llvm_out)
2183 {
2184 target_llvm_out.join(relative_path)
2185 } else {
2186 PathBuf::from(
2189 host_llvm_bin_dir
2190 .replace(&*builder.host_target.triple, &target.triple),
2191 )
2192 }
2193 }
2194 };
2195
2196 #[cfg(feature = "tracing")]
2203 let _llvm_tools_span =
2204 span!(tracing::Level::TRACE, "installing llvm tools to sysroot", ?libdir_bin)
2205 .entered();
2206 for tool in LLVM_TOOLS {
2207 trace!("installing `{tool}`");
2208 let tool_exe = exe(tool, target_compiler.host);
2209 let src_path = llvm_bin_dir.join(&tool_exe);
2210
2211 if !src_path.exists() && builder.config.llvm_from_ci {
2213 eprintln!("{} does not exist; skipping copy", src_path.display());
2214 continue;
2215 }
2216
2217 builder.resolve_symlink_and_copy(&src_path, &libdir_bin.join(&tool_exe));
2224 }
2225 }
2226 }
2227
2228 let maybe_install_llvm_bitcode_linker = || {
2229 if builder.config.llvm_bitcode_linker_enabled {
2230 trace!("llvm-bitcode-linker enabled, installing");
2231 let llvm_bitcode_linker = builder.ensure(
2232 crate::core::build_steps::tool::LlvmBitcodeLinker::from_target_compiler(
2233 builder,
2234 target_compiler,
2235 ),
2236 );
2237
2238 let bindir_self_contained = builder
2240 .sysroot(target_compiler)
2241 .join(format!("lib/rustlib/{}/bin/self-contained", target_compiler.host));
2242 let tool_exe = exe("llvm-bitcode-linker", target_compiler.host);
2243
2244 t!(fs::create_dir_all(&bindir_self_contained));
2245 builder.copy_link(
2246 &llvm_bitcode_linker.tool_path,
2247 &bindir_self_contained.join(tool_exe),
2248 FileType::Executable,
2249 );
2250 }
2251 };
2252
2253 if builder.download_rustc() {
2255 trace!("`download-rustc` requested, reusing CI compiler for stage > 0");
2256
2257 builder.std(target_compiler, target_compiler.host);
2258 let sysroot =
2259 builder.ensure(Sysroot { compiler: target_compiler, force_recompile: false });
2260 dist::maybe_install_llvm_target(builder, target_compiler.host, &sysroot);
2263 if target_compiler.stage == builder.top_stage {
2265 builder.info(&format!("Creating a sysroot for stage{stage} compiler (use `rustup toolchain link 'name' build/host/stage{stage}`)", stage = target_compiler.stage));
2266 }
2267
2268 maybe_install_llvm_bitcode_linker();
2271
2272 return target_compiler;
2273 }
2274
2275 debug!(
2289 "ensuring build compiler is available: compiler(stage = {}, host = {:?})",
2290 target_compiler.stage - 1,
2291 builder.config.host_target,
2292 );
2293 let build_compiler =
2294 builder.compiler(target_compiler.stage - 1, builder.config.host_target);
2295
2296 if builder.config.llvm_enzyme {
2298 debug!("`llvm_enzyme` requested");
2299 let enzyme = builder.ensure(llvm::Enzyme { target: build_compiler.host });
2300 let target_libdir =
2301 builder.sysroot_target_libdir(target_compiler, target_compiler.host);
2302 let target_dst_lib = target_libdir.join(enzyme.enzyme_filename());
2303 builder.copy_link(&enzyme.enzyme_path(), &target_dst_lib, FileType::NativeLibrary);
2304 }
2305
2306 if builder.config.llvm_offload && !builder.config.dry_run() {
2307 debug!("`llvm_offload` requested");
2308 let offload_install = builder.ensure(llvm::OmpOffload { target: build_compiler.host });
2309 if let Some(_llvm_config) = builder.llvm_config(builder.config.host_target) {
2310 let target_libdir =
2311 builder.sysroot_target_libdir(target_compiler, target_compiler.host);
2312 for p in offload_install.offload_paths() {
2313 let libname = p.file_name().unwrap();
2314 let dst_lib = target_libdir.join(libname);
2315 builder.resolve_symlink_and_copy(&p, &dst_lib);
2316 }
2317 }
2322 }
2323
2324 debug!(
2327 ?build_compiler,
2328 "target_compiler.host" = ?target_compiler.host,
2329 "building compiler libraries to link to"
2330 );
2331
2332 let BuiltRustc { build_compiler } =
2334 builder.ensure(Rustc::new(build_compiler, target_compiler.host));
2335
2336 let stage = target_compiler.stage;
2337 let host = target_compiler.host;
2338 let (host_info, dir_name) = if build_compiler.host == host {
2339 ("".into(), "host".into())
2340 } else {
2341 (format!(" ({host})"), host.to_string())
2342 };
2343 let msg = format!(
2348 "Creating a sysroot for stage{stage} compiler{host_info} (use `rustup toolchain link 'name' build/{dir_name}/stage{stage}`)"
2349 );
2350 builder.info(&msg);
2351
2352 let stamp = build_stamp::librustc_stamp(builder, build_compiler, target_compiler.host);
2354 let proc_macros = builder
2355 .read_stamp_file(&stamp)
2356 .into_iter()
2357 .filter_map(|(path, dependency_type)| {
2358 if dependency_type == DependencyType::Host {
2359 Some(path.file_name().unwrap().to_owned().into_string().unwrap())
2360 } else {
2361 None
2362 }
2363 })
2364 .collect::<HashSet<_>>();
2365
2366 let sysroot = builder.sysroot(target_compiler);
2367 let rustc_libdir = builder.rustc_libdir(target_compiler);
2368 t!(fs::create_dir_all(&rustc_libdir));
2369 let src_libdir = builder.sysroot_target_libdir(build_compiler, host);
2370 for f in builder.read_dir(&src_libdir) {
2371 let filename = f.file_name().into_string().unwrap();
2372
2373 let is_proc_macro = proc_macros.contains(&filename);
2374 let is_dylib_or_debug = is_dylib(&f.path()) || is_debug_info(&filename);
2375
2376 let can_be_rustc_dynamic_dep = if builder
2380 .link_std_into_rustc_driver(target_compiler.host)
2381 && !target_compiler.host.is_windows()
2382 {
2383 let is_std = filename.starts_with("std-") || filename.starts_with("libstd-");
2384 !is_std
2385 } else {
2386 true
2387 };
2388
2389 if is_dylib_or_debug && can_be_rustc_dynamic_dep && !is_proc_macro {
2390 builder.copy_link(&f.path(), &rustc_libdir.join(&filename), FileType::Regular);
2391 }
2392 }
2393
2394 {
2395 #[cfg(feature = "tracing")]
2396 let _codegen_backend_span =
2397 span!(tracing::Level::DEBUG, "building requested codegen backends").entered();
2398
2399 for backend in builder.config.enabled_codegen_backends(target_compiler.host) {
2400 if builder.kind == Kind::Check && builder.top_stage == 1 {
2417 continue;
2418 }
2419
2420 let prepare_compilers = || {
2421 RustcPrivateCompilers::from_build_and_target_compiler(
2422 build_compiler,
2423 target_compiler,
2424 )
2425 };
2426
2427 match backend {
2428 CodegenBackendKind::Cranelift => {
2429 let stamp = builder
2430 .ensure(CraneliftCodegenBackend { compilers: prepare_compilers() });
2431 copy_codegen_backends_to_sysroot(builder, stamp, target_compiler);
2432 }
2433 CodegenBackendKind::Gcc => {
2434 let compilers = prepare_compilers();
2467 let cg_gcc = builder
2468 .ensure(GccCodegenBackend::for_target(compilers, target_compiler.host));
2469 copy_codegen_backends_to_sysroot(builder, cg_gcc.stamp, target_compiler);
2470
2471 let mut targets = HashSet::new();
2478 for target in &builder.hosts {
2481 targets.insert(*target);
2482 }
2483 for target in &builder.targets {
2485 targets.insert(*target);
2486 }
2487 targets.insert(compilers.target_compiler().host);
2490
2491 let dylib_set = GccDylibSet::build(
2493 builder,
2494 compilers.target_compiler().host,
2495 targets.into_iter().collect(),
2496 );
2497
2498 dylib_set.install_to(builder, target_compiler);
2501 }
2502 CodegenBackendKind::Llvm | CodegenBackendKind::Custom(_) => continue,
2503 }
2504 }
2505 }
2506
2507 if builder.config.lld_enabled {
2508 let lld_wrapper =
2509 builder.ensure(crate::core::build_steps::tool::LldWrapper::for_use_by_compiler(
2510 builder,
2511 target_compiler,
2512 ));
2513 copy_lld_artifacts(builder, lld_wrapper, target_compiler);
2514 }
2515
2516 if builder.config.llvm_enabled(target_compiler.host) && builder.config.llvm_tools_enabled {
2517 debug!(
2518 "llvm and llvm tools enabled; copying `llvm-objcopy` as `rust-objcopy` to \
2519 workaround faulty homebrew `strip`s"
2520 );
2521
2522 let src_exe = exe("llvm-objcopy", target_compiler.host);
2529 let dst_exe = exe("rust-objcopy", target_compiler.host);
2530 builder.copy_link(
2531 &libdir_bin.join(src_exe),
2532 &libdir_bin.join(dst_exe),
2533 FileType::Executable,
2534 );
2535 }
2536
2537 if builder.tool_enabled("wasm-component-ld") {
2540 let wasm_component = builder.ensure(
2541 crate::core::build_steps::tool::WasmComponentLd::for_use_by_compiler(
2542 builder,
2543 target_compiler,
2544 ),
2545 );
2546 builder.copy_link(
2547 &wasm_component.tool_path,
2548 &libdir_bin.join(wasm_component.tool_path.file_name().unwrap()),
2549 FileType::Executable,
2550 );
2551 }
2552
2553 maybe_install_llvm_bitcode_linker();
2554
2555 debug!(
2558 "target_compiler.host" = ?target_compiler.host,
2559 ?sysroot,
2560 "ensuring availability of `libLLVM.so` in compiler directory"
2561 );
2562 dist::maybe_install_llvm_runtime(builder, target_compiler.host, &sysroot);
2563 dist::maybe_install_llvm_target(builder, target_compiler.host, &sysroot);
2564
2565 let out_dir = builder.cargo_out(build_compiler, Mode::Rustc, host);
2567 let rustc = out_dir.join(exe("rustc-main", host));
2568 let bindir = sysroot.join("bin");
2569 t!(fs::create_dir_all(bindir));
2570 let compiler = builder.rustc(target_compiler);
2571 debug!(src = ?rustc, dst = ?compiler, "linking compiler binary itself");
2572 builder.copy_link(&rustc, &compiler, FileType::Executable);
2573
2574 target_compiler
2575 }
2576}
2577
2578#[track_caller]
2583pub fn add_to_sysroot(
2584 builder: &Builder<'_>,
2585 sysroot_dst: &Path,
2586 sysroot_host_dst: &Path,
2587 stamp: &BuildStamp,
2588) {
2589 let self_contained_dst = &sysroot_dst.join("self-contained");
2590 t!(fs::create_dir_all(sysroot_dst));
2591 t!(fs::create_dir_all(sysroot_host_dst));
2592 t!(fs::create_dir_all(self_contained_dst));
2593
2594 let mut crates = HashMap::new();
2595 for (path, dependency_type) in builder.read_stamp_file(stamp) {
2596 let filename = path.file_name().unwrap().to_str().unwrap();
2597 let dst = match dependency_type {
2598 DependencyType::Host => {
2599 if sysroot_dst == sysroot_host_dst {
2600 crates.insert(filename.split_once('.').unwrap().0.to_owned(), path.clone());
2603 }
2604
2605 sysroot_host_dst
2606 }
2607 DependencyType::Target => {
2608 crates.insert(filename.split_once('.').unwrap().0.to_owned(), path.clone());
2611
2612 sysroot_dst
2613 }
2614 DependencyType::TargetSelfContained => self_contained_dst,
2615 };
2616 builder.copy_link(&path, &dst.join(filename), FileType::Regular);
2617 }
2618
2619 let mut seen_crates = HashMap::new();
2625 for (filestem, path) in crates {
2626 if !filestem.contains("rustc_") || filestem.contains("rustc_hash") {
2627 continue;
2628 }
2629 if let Some(other_path) =
2630 seen_crates.insert(filestem.split_once('-').unwrap().0.to_owned(), path.clone())
2631 {
2632 panic!(
2633 "duplicate rustc crate {}\n- first copy at {}\n- second copy at {}",
2634 filestem.split_once('-').unwrap().0.to_owned(),
2635 other_path.display(),
2636 path.display(),
2637 );
2638 }
2639 }
2640}
2641
2642pub enum ArtifactKeepMode {
2646 OnlyRlib,
2648 OnlyRmeta,
2650 BothRlibAndRmeta,
2654 Custom(Box<dyn Fn(&str) -> bool>),
2657}
2658
2659pub fn run_cargo(
2660 builder: &Builder<'_>,
2661 cargo: Cargo,
2662 tail_args: Vec<String>,
2663 stamp: &BuildStamp,
2664 additional_target_deps: Vec<(PathBuf, DependencyType)>,
2665 artifact_keep_mode: ArtifactKeepMode,
2666) -> Vec<PathBuf> {
2667 let target_root_dir = stamp.path().parent().unwrap();
2669 let target_deps_dir = target_root_dir.join("deps");
2671 let host_root_dir = target_root_dir
2673 .parent()
2674 .unwrap() .parent()
2676 .unwrap() .join(target_root_dir.file_name().unwrap());
2678
2679 let mut deps = Vec::new();
2683 let mut toplevel = Vec::new();
2684 let ok = stream_cargo(builder, cargo, tail_args, &mut |msg| {
2685 let (filenames_vec, crate_types) = match msg {
2686 CargoMessage::CompilerArtifact {
2687 filenames,
2688 target: CargoTarget { crate_types },
2689 ..
2690 } => {
2691 let mut f: Vec<String> = filenames.into_iter().map(|s| s.into_owned()).collect();
2692 f.sort(); (f, crate_types)
2694 }
2695 _ => return,
2696 };
2697 for filename in filenames_vec {
2698 let keep = if filename.ends_with(".lib")
2700 || filename.ends_with(".a")
2701 || is_debug_info(&filename)
2702 || is_dylib(Path::new(&*filename))
2703 {
2704 true
2706 } else {
2707 match &artifact_keep_mode {
2708 ArtifactKeepMode::OnlyRlib => filename.ends_with(".rlib"),
2709 ArtifactKeepMode::OnlyRmeta => filename.ends_with(".rmeta"),
2710 ArtifactKeepMode::BothRlibAndRmeta => {
2711 filename.ends_with(".rmeta") || filename.ends_with(".rlib")
2712 }
2713 ArtifactKeepMode::Custom(func) => func(&filename),
2714 }
2715 };
2716
2717 if !keep {
2718 continue;
2719 }
2720
2721 let filename = Path::new(&*filename);
2722
2723 if filename.starts_with(&host_root_dir) {
2726 if crate_types.iter().any(|t| t == "proc-macro") {
2728 if filename.file_name().unwrap().to_str().unwrap().contains("-") {
2733 deps.push((filename.to_path_buf(), DependencyType::Host));
2734 }
2735 }
2736 continue;
2737 }
2738
2739 if filename.starts_with(&target_deps_dir) {
2742 deps.push((filename.to_path_buf(), DependencyType::Target));
2743 continue;
2744 }
2745
2746 let expected_len = t!(filename.metadata()).len();
2757 let filename = filename.file_name().unwrap().to_str().unwrap();
2758 let mut parts = filename.splitn(2, '.');
2759 let file_stem = parts.next().unwrap().to_owned();
2760 let extension = parts.next().unwrap().to_owned();
2761
2762 toplevel.push((file_stem, extension, expected_len));
2763 }
2764 });
2765
2766 if !ok {
2767 crate::exit!(1);
2768 }
2769
2770 if builder.config.dry_run() {
2771 return Vec::new();
2772 }
2773
2774 let contents = target_deps_dir
2778 .read_dir()
2779 .unwrap_or_else(|e| panic!("Couldn't read {}: {}", target_deps_dir.display(), e))
2780 .map(|e| t!(e))
2781 .map(|e| (e.path(), e.file_name().into_string().unwrap(), t!(e.metadata())))
2782 .collect::<Vec<_>>();
2783 for (prefix, extension, expected_len) in toplevel {
2784 let candidates = contents.iter().filter(|&(_, filename, meta)| {
2785 meta.len() == expected_len
2786 && filename
2787 .strip_prefix(&prefix[..])
2788 .map(|s| s.starts_with('-') && s.ends_with(&extension[..]))
2789 .unwrap_or(false)
2790 });
2791 let max = candidates.max_by_key(|&(_, _, metadata)| {
2792 metadata.modified().expect("mtime should be available on all relevant OSes")
2793 });
2794 let path_to_add = match max {
2795 Some(triple) => triple.0.to_str().unwrap(),
2796 None => panic!("no output generated for {prefix:?} {extension:?}"),
2797 };
2798 if is_dylib(Path::new(path_to_add)) {
2799 let candidate = format!("{path_to_add}.lib");
2800 let candidate = PathBuf::from(candidate);
2801 if candidate.exists() {
2802 deps.push((candidate, DependencyType::Target));
2803 }
2804 }
2805 deps.push((path_to_add.into(), DependencyType::Target));
2806 }
2807
2808 deps.extend(additional_target_deps);
2809 deps.sort();
2810 let mut new_contents = Vec::new();
2811 for (dep, dependency_type) in deps.iter() {
2812 new_contents.extend(match *dependency_type {
2813 DependencyType::Host => b"h",
2814 DependencyType::Target => b"t",
2815 DependencyType::TargetSelfContained => b"s",
2816 });
2817 new_contents.extend(dep.to_str().unwrap().as_bytes());
2818 new_contents.extend(b"\0");
2819 }
2820 t!(fs::write(stamp.path(), &new_contents));
2821 deps.into_iter().map(|(d, _)| d).collect()
2822}
2823
2824pub fn stream_cargo(
2825 builder: &Builder<'_>,
2826 cargo: Cargo,
2827 tail_args: Vec<String>,
2828 cb: &mut dyn FnMut(CargoMessage<'_>),
2829) -> bool {
2830 let mut cmd = cargo.into_cmd();
2831
2832 let mut message_format = if builder.config.json_output {
2835 String::from("json")
2836 } else {
2837 String::from("json-render-diagnostics")
2838 };
2839 if let Some(s) = &builder.config.rustc_error_format {
2840 message_format.push_str(",json-diagnostic-");
2841 message_format.push_str(s);
2842 }
2843 cmd.arg("--message-format").arg(message_format);
2844
2845 for arg in tail_args {
2846 cmd.arg(arg);
2847 }
2848
2849 builder.do_if_verbose(|| println!("running: {cmd:?}"));
2850
2851 let streaming_command = cmd.stream_capture_stdout(&builder.config.exec_ctx);
2852
2853 let Some(mut streaming_command) = streaming_command else {
2854 return true;
2855 };
2856
2857 let stdout = BufReader::new(streaming_command.stdout.take().unwrap());
2861 for line in stdout.lines() {
2862 let line = t!(line);
2863 match serde_json::from_str::<CargoMessage<'_>>(&line) {
2864 Ok(msg) => {
2865 if builder.config.json_output {
2866 println!("{line}");
2868 }
2869 cb(msg)
2870 }
2871 Err(_) => println!("{line}"),
2873 }
2874 }
2875
2876 let status = t!(streaming_command.wait(&builder.config.exec_ctx));
2878 if builder.is_verbose() && !status.success() {
2879 eprintln!(
2880 "command did not execute successfully: {cmd:?}\n\
2881 expected success, got: {status}"
2882 );
2883 }
2884
2885 status.success()
2886}
2887
2888#[derive(Deserialize)]
2889pub struct CargoTarget<'a> {
2890 crate_types: Vec<Cow<'a, str>>,
2891}
2892
2893#[derive(Deserialize)]
2894#[serde(tag = "reason", rename_all = "kebab-case")]
2895pub enum CargoMessage<'a> {
2896 CompilerArtifact { filenames: Vec<Cow<'a, str>>, target: CargoTarget<'a> },
2897 BuildScriptExecuted,
2898 BuildFinished,
2899}
2900
2901pub fn strip_debug(builder: &Builder<'_>, target: TargetSelection, path: &Path) {
2902 if target != "x86_64-unknown-linux-gnu"
2906 || !builder.config.is_host_target(target)
2907 || !path.exists()
2908 {
2909 return;
2910 }
2911
2912 let previous_mtime = t!(t!(path.metadata()).modified());
2913 let stamp = BuildStamp::new(path.parent().unwrap())
2914 .with_prefix(path.file_name().unwrap().to_str().unwrap())
2915 .with_prefix("strip")
2916 .add_stamp(previous_mtime.duration_since(SystemTime::UNIX_EPOCH).unwrap().as_nanos());
2917
2918 if !stamp.is_up_to_date() {
2921 command("strip").arg("--strip-debug").arg(path).run_capture(builder);
2922 }
2923 t!(stamp.write());
2924
2925 let file = t!(fs::File::open(path));
2926
2927 t!(file.set_modified(previous_mtime));
2940}
2941
2942pub fn is_lto_stage(build_compiler: &Compiler) -> bool {
2944 build_compiler.stage != 0
2945}