1pub mod artifact;
32mod build_config;
33pub(crate) mod build_context;
34pub(crate) mod build_runner;
35mod compilation;
36mod compile_kind;
37mod crate_type;
38mod custom_build;
39pub(crate) mod fingerprint;
40pub mod future_incompat;
41pub(crate) mod job_queue;
42pub(crate) mod layout;
43mod links;
44mod locking;
45mod lto;
46mod output_depinfo;
47mod output_sbom;
48pub mod rustdoc;
49pub mod standard_lib;
50pub mod timings;
51mod unit;
52pub mod unit_dependencies;
53pub mod unit_graph;
54pub mod unused_deps;
55
56use std::borrow::Cow;
57use std::cell::OnceCell;
58use std::collections::{BTreeMap, HashMap, HashSet};
59use std::env;
60use std::ffi::{OsStr, OsString};
61use std::fmt::Display;
62use std::fs::{self, File};
63use std::io::{BufRead, BufWriter, Write};
64use std::ops::Range;
65use std::path::{Path, PathBuf};
66use std::sync::{Arc, LazyLock};
67
68use anyhow::{Context as _, Error};
69use cargo_platform::{Cfg, Platform};
70use cargo_util_terminal::report::{AnnotationKind, Group, Level, Renderer, Snippet};
71use itertools::Itertools;
72use regex::Regex;
73use tracing::{debug, instrument, trace};
74
75pub use self::build_config::UserIntent;
76pub use self::build_config::{BuildConfig, CompileMode, MessageFormat};
77pub use self::build_context::BuildContext;
78pub use self::build_context::DepKindSet;
79pub use self::build_context::FileFlavor;
80pub use self::build_context::FileType;
81pub use self::build_context::RustcTargetData;
82pub use self::build_context::TargetInfo;
83pub use self::build_runner::{BuildRunner, Metadata, UnitHash};
84pub use self::compilation::{Compilation, Doctest, UnitOutput};
85pub use self::compile_kind::{CompileKind, CompileKindFallback, CompileTarget};
86pub use self::crate_type::CrateType;
87pub use self::custom_build::LinkArgTarget;
88pub use self::custom_build::{BuildOutput, BuildScriptOutputs, BuildScripts, LibraryPath};
89pub(crate) use self::fingerprint::DirtyReason;
90pub use self::fingerprint::RustdocFingerprint;
91pub use self::job_queue::Freshness;
92use self::job_queue::{Job, JobQueue, JobState, Work};
93pub(crate) use self::layout::Layout;
94pub use self::lto::Lto;
95use self::output_depinfo::output_depinfo;
96use self::output_sbom::build_sbom;
97use self::unit_graph::UnitDep;
98
99use crate::core::compiler::future_incompat::FutureIncompatReport;
100use crate::core::compiler::locking::LockKey;
101use crate::core::compiler::timings::SectionTiming;
102pub use crate::core::compiler::unit::Unit;
103pub use crate::core::compiler::unit::UnitIndex;
104pub use crate::core::compiler::unit::UnitInterner;
105use crate::core::manifest::TargetSourcePath;
106use crate::core::profiles::{PanicStrategy, Profile, StripInner};
107use crate::core::{Feature, PackageId, Target};
108use crate::diagnostics::get_key_value;
109use crate::util::OnceExt;
110use crate::util::errors::{CargoResult, VerboseError};
111use crate::util::interning::InternedString;
112use crate::util::machine_message::{self, Message};
113use crate::util::{add_path_args, internal, path_args};
114
115use cargo_util::{ProcessBuilder, ProcessError, paths};
116use cargo_util_schemas::manifest::TomlDebugInfo;
117use cargo_util_schemas::manifest::TomlTrimPaths;
118use cargo_util_schemas::manifest::TomlTrimPathsValue;
119use cargo_util_terminal::Verbosity;
120use rustfix::diagnostics::Applicability;
121
122const RUSTDOC_CRATE_VERSION_FLAG: &str = "--crate-version";
123
124pub trait Executor: Send + Sync + 'static {
128 fn init(&self, _build_runner: &BuildRunner<'_, '_>, _unit: &Unit) {}
132
133 fn exec(
136 &self,
137 cmd: &ProcessBuilder,
138 id: PackageId,
139 target: &Target,
140 mode: CompileMode,
141 on_stdout_line: &mut dyn FnMut(&str) -> CargoResult<()>,
142 on_stderr_line: &mut dyn FnMut(&str) -> CargoResult<()>,
143 ) -> CargoResult<()>;
144
145 fn force_rebuild(&self, _unit: &Unit) -> bool {
148 false
149 }
150}
151
152#[derive(Copy, Clone)]
155pub struct DefaultExecutor;
156
157impl Executor for DefaultExecutor {
158 #[instrument(name = "rustc", skip_all, fields(package = id.name().as_str(), process = cmd.to_string()))]
159 fn exec(
160 &self,
161 cmd: &ProcessBuilder,
162 id: PackageId,
163 _target: &Target,
164 _mode: CompileMode,
165 on_stdout_line: &mut dyn FnMut(&str) -> CargoResult<()>,
166 on_stderr_line: &mut dyn FnMut(&str) -> CargoResult<()>,
167 ) -> CargoResult<()> {
168 cmd.exec_with_streaming(on_stdout_line, on_stderr_line, false)
169 .map(drop)
170 }
171}
172
173#[tracing::instrument(skip(build_runner, jobs, exec))]
183fn compile<'gctx>(
184 build_runner: &mut BuildRunner<'_, 'gctx>,
185 jobs: &mut JobQueue<'gctx>,
186 unit: &Unit,
187 exec: &Arc<dyn Executor>,
188 force_rebuild: bool,
189) -> CargoResult<()> {
190 if !build_runner.compiled.insert(unit.clone()) {
191 return Ok(());
192 }
193
194 let lock = if build_runner.bcx.gctx.cli_unstable().fine_grain_locking {
195 Some(build_runner.lock_manager.lock_shared(build_runner, unit)?)
196 } else {
197 None
198 };
199
200 if !unit.skip_non_compile_time_dep {
204 fingerprint::prepare_init(build_runner, unit)?;
207
208 let job = if unit.mode.is_run_custom_build() {
209 custom_build::prepare(build_runner, unit)?
210 } else if unit.mode.is_doc_test() {
211 Job::new_fresh()
213 } else {
214 let force = exec.force_rebuild(unit) || force_rebuild;
215 let mut job = fingerprint::prepare_target(build_runner, unit, force)?;
216 job.before(if job.freshness().is_dirty() {
217 let work = if unit.mode.is_doc() || unit.mode.is_doc_scrape() {
218 rustdoc(build_runner, unit)?
219 } else {
220 rustc(build_runner, unit, exec)?
221 };
222 work.then(link_targets(build_runner, unit, false)?)
223 } else {
224 let output_options = OutputOptions::for_fresh(build_runner, unit);
225 let manifest = ManifestErrorContext::new(build_runner, unit);
226 let work = replay_output_cache(
227 unit.pkg.package_id(),
228 manifest,
229 &unit.target,
230 build_runner.files().message_cache_path(unit),
231 output_options,
232 );
233 work.then(link_targets(build_runner, unit, true)?)
235 });
236
237 if build_runner.bcx.gctx.cli_unstable().fine_grain_locking && job.freshness().is_dirty()
240 {
241 if let Some(lock) = lock {
242 build_runner.lock_manager.unlock(&lock)?;
249 job.before(prebuild_lock_exclusive(lock.clone()));
250 job.after(downgrade_lock_to_shared(lock));
251 }
252 }
253
254 job
255 };
256 jobs.enqueue(build_runner, unit, job)?;
257 }
258
259 let deps = Vec::from(build_runner.unit_deps(unit)); for dep in deps {
262 compile(build_runner, jobs, &dep.unit, exec, false)?;
263 }
264
265 Ok(())
266}
267
268fn make_failed_scrape_diagnostic(
271 build_runner: &BuildRunner<'_, '_>,
272 unit: &Unit,
273 top_line: impl Display,
274) -> String {
275 let manifest_path = unit.pkg.manifest_path();
276 let relative_manifest_path = manifest_path
277 .strip_prefix(build_runner.bcx.ws.root())
278 .unwrap_or(&manifest_path);
279
280 format!(
281 "\
282{top_line}
283 Try running with `--verbose` to see the error message.
284 If an example should not be scanned, then consider adding `doc-scrape-examples = false` to its `[[example]]` definition in {}",
285 relative_manifest_path.display()
286 )
287}
288
289fn rustc(
291 build_runner: &mut BuildRunner<'_, '_>,
292 unit: &Unit,
293 exec: &Arc<dyn Executor>,
294) -> CargoResult<Work> {
295 let mut rustc = prepare_rustc(build_runner, unit)?;
296
297 let name = unit.pkg.name();
298
299 let outputs = build_runner.outputs(unit)?;
300 let root = build_runner.files().output_dir(unit);
301
302 let build_script_outputs = Arc::clone(&build_runner.build_script_outputs);
304 let current_id = unit.pkg.package_id();
305 let manifest = ManifestErrorContext::new(build_runner, unit);
306 let build_scripts = build_runner.build_scripts.get(unit).cloned();
307
308 let pass_l_flag = unit.target.is_lib() || !unit.pkg.targets().iter().any(|t| t.is_lib());
311
312 let dep_info_name =
313 if let Some(c_extra_filename) = build_runner.files().metadata(unit).c_extra_filename() {
314 format!("{}-{}.d", unit.target.crate_name(), c_extra_filename)
315 } else {
316 format!("{}.d", unit.target.crate_name())
317 };
318 let rustc_dep_info_loc = root.join(dep_info_name);
319 let dep_info_loc = fingerprint::dep_info_loc(build_runner, unit);
320
321 let mut output_options = OutputOptions::for_dirty(build_runner, unit);
322 let package_id = unit.pkg.package_id();
323 let target = Target::clone(&unit.target);
324 let mode = unit.mode;
325
326 exec.init(build_runner, unit);
327 let exec = exec.clone();
328
329 let root_output = build_runner.files().host_dest().map(|v| v.to_path_buf());
330 let build_dir = build_runner.bcx.ws.build_dir().into_path_unlocked();
331 let pkg_root = unit.pkg.root().to_path_buf();
332 let cwd = rustc
333 .get_cwd()
334 .unwrap_or_else(|| build_runner.bcx.gctx.cwd())
335 .to_path_buf();
336 let fingerprint_dir = build_runner.files().fingerprint_dir(unit);
337 let script_metadatas = build_runner.find_build_script_metadatas(unit);
338 let is_local = unit.is_local();
339 let artifact = unit.artifact;
340 let sbom_files = build_runner.sbom_output_files(unit)?;
341 let sbom = build_sbom(build_runner, unit)?;
342
343 let hide_diagnostics_for_scrape_unit = build_runner.bcx.unit_can_fail_for_docscraping(unit)
344 && !matches!(
345 build_runner.bcx.gctx.shell().verbosity(),
346 Verbosity::Verbose
347 );
348 let failed_scrape_diagnostic = hide_diagnostics_for_scrape_unit.then(|| {
349 let target_desc = unit.target.description_named();
352 let mut for_scrape_units = build_runner
353 .bcx
354 .scrape_units_have_dep_on(unit)
355 .into_iter()
356 .map(|unit| unit.target.description_named())
357 .collect::<Vec<_>>();
358 for_scrape_units.sort();
359 let for_scrape_units = for_scrape_units.join(", ");
360 make_failed_scrape_diagnostic(build_runner, unit, format_args!("failed to check {target_desc} in package `{name}` as a prerequisite for scraping examples from: {for_scrape_units}"))
361 });
362 if hide_diagnostics_for_scrape_unit {
363 output_options.show_diagnostics = false;
364 }
365 let env_config = Arc::clone(build_runner.bcx.gctx.env_config()?);
366 return Ok(Work::new(move |state| {
367 if artifact.is_true() {
371 paths::create_dir_all(&root)?;
372 }
373
374 if let Some(build_scripts) = build_scripts {
382 let script_outputs = build_script_outputs.lock().unwrap();
383 add_native_deps(
384 &mut rustc,
385 &script_outputs,
386 &build_scripts,
387 pass_l_flag,
388 &target,
389 current_id,
390 mode,
391 )?;
392 if let Some(ref root_output) = root_output {
393 add_plugin_deps(&mut rustc, &script_outputs, &build_scripts, root_output)?;
394 }
395 add_custom_flags(&mut rustc, &script_outputs, script_metadatas)?;
396 }
397
398 for output in outputs.iter() {
399 if output.path.extension() == Some(OsStr::new("rmeta")) {
403 let dst = root.join(&output.path).with_extension("rlib");
404 if dst.exists() {
405 paths::remove_file(&dst)?;
406 }
407 }
408
409 if output.hardlink.is_some() && output.path.exists() {
414 _ = paths::remove_file(&output.path).map_err(|e| {
415 tracing::debug!(
416 "failed to delete previous output file `{:?}`: {e:?}",
417 output.path
418 );
419 });
420 }
421 }
422
423 state.running(&rustc);
424 let timestamp = paths::set_invocation_time(&fingerprint_dir)?;
425 for file in sbom_files {
426 tracing::debug!("writing sbom to {}", file.display());
427 let outfile = BufWriter::new(paths::create(&file)?);
428 serde_json::to_writer(outfile, &sbom)?;
429 }
430
431 let result = exec
432 .exec(
433 &rustc,
434 package_id,
435 &target,
436 mode,
437 &mut |line| on_stdout_line(state, line, package_id, &target),
438 &mut |line| {
439 on_stderr_line(
440 state,
441 line,
442 package_id,
443 &manifest,
444 &target,
445 &mut output_options,
446 )
447 },
448 )
449 .map_err(|e| {
450 if output_options.errors_seen == 0 {
451 e
456 } else {
457 verbose_if_simple_exit_code(e)
458 }
459 })
460 .with_context(|| {
461 let warnings = match output_options.warnings_seen {
463 0 => String::new(),
464 1 => "; 1 warning emitted".to_string(),
465 count => format!("; {} warnings emitted", count),
466 };
467 let errors = match output_options.errors_seen {
468 0 => String::new(),
469 1 => " due to 1 previous error".to_string(),
470 count => format!(" due to {} previous errors", count),
471 };
472 let name = descriptive_pkg_name(&name, &target, &mode);
473 format!("could not compile {name}{errors}{warnings}")
474 });
475
476 if let Err(e) = result {
477 if let Some(diagnostic) = failed_scrape_diagnostic {
478 state.warning(diagnostic);
479 }
480
481 return Err(e);
482 }
483
484 debug_assert_eq!(output_options.errors_seen, 0);
486
487 if rustc_dep_info_loc.exists() {
488 fingerprint::translate_dep_info(
489 &rustc_dep_info_loc,
490 &dep_info_loc,
491 &cwd,
492 &pkg_root,
493 &build_dir,
494 &rustc,
495 is_local,
497 &env_config,
498 )
499 .with_context(|| {
500 internal(format!(
501 "could not parse/generate dep info at: {}",
502 rustc_dep_info_loc.display()
503 ))
504 })?;
505 paths::set_file_time_no_err(dep_info_loc, timestamp);
508 }
509
510 if mode.is_check() {
524 for output in outputs.iter() {
525 paths::set_file_time_no_err(&output.path, timestamp);
526 }
527 }
528
529 Ok(())
530 }));
531
532 fn add_native_deps(
535 rustc: &mut ProcessBuilder,
536 build_script_outputs: &BuildScriptOutputs,
537 build_scripts: &BuildScripts,
538 pass_l_flag: bool,
539 target: &Target,
540 current_id: PackageId,
541 mode: CompileMode,
542 ) -> CargoResult<()> {
543 let mut library_paths = vec![];
544
545 for key in build_scripts.to_link.iter() {
546 let output = build_script_outputs.get(key.1).ok_or_else(|| {
547 internal(format!(
548 "couldn't find build script output for {}/{}",
549 key.0, key.1
550 ))
551 })?;
552 library_paths.extend(output.library_paths.iter());
553 }
554
555 library_paths.sort_by_key(|p| match p {
561 LibraryPath::CargoArtifact(_) => 0,
562 LibraryPath::External(_) => 1,
563 });
564
565 for path in library_paths.iter() {
566 rustc.arg("-L").arg(path.as_ref());
567 }
568
569 for key in build_scripts.to_link.iter() {
570 let output = build_script_outputs.get(key.1).ok_or_else(|| {
571 internal(format!(
572 "couldn't find build script output for {}/{}",
573 key.0, key.1
574 ))
575 })?;
576
577 if key.0 == current_id {
578 if pass_l_flag {
579 for name in output.library_links.iter() {
580 rustc.arg("-l").arg(name);
581 }
582 }
583 }
584
585 for (lt, arg) in &output.linker_args {
586 if lt.applies_to(target, mode)
592 && (key.0 == current_id || *lt == LinkArgTarget::Cdylib)
593 {
594 rustc.arg("-C").arg(format!("link-arg={}", arg));
595 }
596 }
597 }
598 Ok(())
599 }
600}
601
602fn verbose_if_simple_exit_code(err: Error) -> Error {
603 match err
606 .downcast_ref::<ProcessError>()
607 .as_ref()
608 .and_then(|perr| perr.code)
609 {
610 Some(n) if cargo_util::is_simple_exit_code(n) => VerboseError::new(err).into(),
611 _ => err,
612 }
613}
614
615fn prebuild_lock_exclusive(lock: LockKey) -> Work {
616 Work::new(move |state| {
617 state.lock_exclusive(&lock)?;
618 Ok(())
619 })
620}
621
622fn downgrade_lock_to_shared(lock: LockKey) -> Work {
623 Work::new(move |state| {
624 state.downgrade_to_shared(&lock)?;
625 Ok(())
626 })
627}
628
629fn link_targets(
632 build_runner: &mut BuildRunner<'_, '_>,
633 unit: &Unit,
634 fresh: bool,
635) -> CargoResult<Work> {
636 let bcx = build_runner.bcx;
637 let outputs = build_runner.outputs(unit)?;
638 let export_dir = build_runner.files().export_dir();
639 let package_id = unit.pkg.package_id();
640 let manifest_path = PathBuf::from(unit.pkg.manifest_path());
641 let profile = unit.profile.clone();
642 let unit_mode = unit.mode;
643 let features = unit.features.iter().map(|s| s.to_string()).collect();
644 let json_messages = bcx.build_config.emit_json();
645 let executable = build_runner.get_executable(unit)?;
646 let mut target = Target::clone(&unit.target);
647 if let TargetSourcePath::Metabuild = target.src_path() {
648 let path = unit
650 .pkg
651 .manifest()
652 .metabuild_path(build_runner.bcx.ws.build_dir());
653 target.set_src_path(TargetSourcePath::Path(path));
654 }
655
656 Ok(Work::new(move |state| {
657 let mut destinations = vec![];
662 for output in outputs.iter() {
663 let src = &output.path;
664 if !src.exists() {
667 continue;
668 }
669 let Some(dst) = output.hardlink.as_ref() else {
670 destinations.push(src.clone());
671 continue;
672 };
673 destinations.push(dst.clone());
674 paths::link_or_copy(src, dst)?;
675 if let Some(ref path) = output.export_path {
676 let export_dir = export_dir.as_ref().unwrap();
677 paths::create_dir_all(export_dir)?;
678
679 paths::link_or_copy(src, path)?;
680 }
681 }
682
683 if json_messages {
684 let debuginfo = match profile.debuginfo.into_inner() {
685 TomlDebugInfo::None => machine_message::ArtifactDebuginfo::Int(0),
686 TomlDebugInfo::Limited => machine_message::ArtifactDebuginfo::Int(1),
687 TomlDebugInfo::Full => machine_message::ArtifactDebuginfo::Int(2),
688 TomlDebugInfo::LineDirectivesOnly => {
689 machine_message::ArtifactDebuginfo::Named("line-directives-only")
690 }
691 TomlDebugInfo::LineTablesOnly => {
692 machine_message::ArtifactDebuginfo::Named("line-tables-only")
693 }
694 };
695 let art_profile = machine_message::ArtifactProfile {
696 opt_level: profile.opt_level.as_str(),
697 debuginfo: Some(debuginfo),
698 debug_assertions: profile.debug_assertions,
699 overflow_checks: profile.overflow_checks,
700 test: unit_mode.is_any_test(),
701 };
702
703 let msg = machine_message::Artifact {
704 package_id: package_id.to_spec(),
705 manifest_path,
706 target: &target,
707 profile: art_profile,
708 features,
709 filenames: destinations,
710 executable,
711 fresh,
712 }
713 .to_json_string();
714 state.stdout(msg)?;
715 }
716 Ok(())
717 }))
718}
719
720fn add_plugin_deps(
724 rustc: &mut ProcessBuilder,
725 build_script_outputs: &BuildScriptOutputs,
726 build_scripts: &BuildScripts,
727 root_output: &Path,
728) -> CargoResult<()> {
729 let var = paths::dylib_path_envvar();
730 let search_path = rustc.get_env(var).unwrap_or_default();
731 let mut search_path = env::split_paths(&search_path).collect::<Vec<_>>();
732 for (pkg_id, metadata) in &build_scripts.plugins {
733 let output = build_script_outputs
734 .get(*metadata)
735 .ok_or_else(|| internal(format!("couldn't find libs for plugin dep {}", pkg_id)))?;
736 search_path.append(&mut filter_dynamic_search_path(
737 output.library_paths.iter().map(AsRef::as_ref),
738 root_output,
739 ));
740 }
741 let search_path = paths::join_paths(&search_path, var)?;
742 rustc.env(var, &search_path);
743 Ok(())
744}
745
746fn get_dynamic_search_path(path: &Path) -> &Path {
747 match path.to_str().and_then(|s| s.split_once("=")) {
748 Some(("native" | "crate" | "dependency" | "framework" | "all", path)) => Path::new(path),
749 _ => path,
750 }
751}
752
753fn filter_dynamic_search_path<'a, I>(paths: I, root_output: &Path) -> Vec<PathBuf>
759where
760 I: Iterator<Item = &'a PathBuf>,
761{
762 let mut search_path = vec![];
763 for dir in paths {
764 let dir = get_dynamic_search_path(dir);
765 if dir.starts_with(&root_output) {
766 search_path.push(dir.to_path_buf());
767 } else {
768 debug!(
769 "Not including path {} in runtime library search path because it is \
770 outside target root {}",
771 dir.display(),
772 root_output.display()
773 );
774 }
775 }
776 search_path
777}
778
779fn prepare_rustc(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> CargoResult<ProcessBuilder> {
786 let gctx = build_runner.bcx.gctx;
787 let is_primary = build_runner.is_primary_package(unit);
788 let is_workspace = build_runner.bcx.ws.is_member(&unit.pkg);
789
790 let mut base = build_runner
791 .compilation
792 .rustc_process(unit, is_primary, is_workspace)?;
793 build_base_args(build_runner, &mut base, unit)?;
794 if unit.pkg.manifest().is_embedded() {
795 if !gctx.cli_unstable().script {
796 anyhow::bail!(
797 "parsing `{}` requires `-Zscript`",
798 unit.pkg.manifest_path().display()
799 );
800 }
801 base.arg("-Z").arg("crate-attr=feature(frontmatter)");
802 base.arg("-Z").arg("crate-attr=allow(unused_features)");
803 }
804
805 base.inherit_jobserver(&build_runner.jobserver);
806 build_deps_args(&mut base, build_runner, unit)?;
807 add_cap_lints(build_runner.bcx, unit, &mut base);
808 if let Some(args) = build_runner.bcx.extra_args_for(unit) {
809 base.args(args);
810 }
811 base.args(&unit.rustflags);
812 if gctx.cli_unstable().binary_dep_depinfo {
813 base.arg("-Z").arg("binary-dep-depinfo");
814 }
815 if build_runner.bcx.gctx.cli_unstable().checksum_freshness {
816 base.arg("-Z").arg("checksum-hash-algorithm=blake3");
817 }
818 if gctx.shell().verbosity() == Verbosity::Verbose && unit.is_local() {
819 base.arg("--verbose");
820 }
821
822 if is_primary {
823 base.env("CARGO_PRIMARY_PACKAGE", "1");
824 let file_list = build_runner.sbom_output_files(unit)?;
825 if !file_list.is_empty() {
826 let file_list = std::env::join_paths(file_list)?;
827 base.env("CARGO_SBOM_PATH", file_list);
828 }
829 }
830
831 if unit.target.is_test() || unit.target.is_bench() {
832 let tmp = build_runner
833 .files()
834 .layout(unit.kind)
835 .build_dir()
836 .prepare_tmp()?;
837 base.env("CARGO_TARGET_TMPDIR", tmp.display().to_string());
838 }
839
840 if build_runner.bcx.gctx.cli_unstable().cargo_lints {
841 base.arg("--force-warn=unused_crate_dependencies");
844 }
845
846 Ok(base)
847}
848
849fn prepare_rustdoc(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> CargoResult<ProcessBuilder> {
856 let bcx = build_runner.bcx;
857 let mut rustdoc = build_runner.compilation.rustdoc_process(unit, None)?;
859 if unit.pkg.manifest().is_embedded() {
860 if !bcx.gctx.cli_unstable().script {
861 anyhow::bail!(
862 "parsing `{}` requires `-Zscript`",
863 unit.pkg.manifest_path().display()
864 );
865 }
866 rustdoc.arg("-Z").arg("crate-attr=feature(frontmatter)");
867 rustdoc.arg("-Z").arg("crate-attr=allow(unused_features)");
868 }
869 rustdoc.inherit_jobserver(&build_runner.jobserver);
870 let crate_name = unit.target.crate_name();
871 rustdoc.arg("--crate-name").arg(&crate_name);
872 add_path_args(bcx.ws, unit, &mut rustdoc);
873 add_cap_lints(bcx, unit, &mut rustdoc);
874
875 unit.kind.add_target_arg(&mut rustdoc);
876
877 let doc_dir = if build_runner.bcx.build_config.intent.wants_doc_json_output() {
878 build_runner.files().out_dir_new_layout(unit)
882 } else {
883 build_runner.files().output_dir(unit)
884 };
885
886 rustdoc.arg("-o").arg(&doc_dir);
887 rustdoc.args(&features_args(unit));
888 rustdoc.args(&check_cfg_args(unit));
889
890 add_error_format_and_color(build_runner, &mut rustdoc);
891 add_allow_features(build_runner, &mut rustdoc);
892
893 if build_runner.bcx.gctx.cli_unstable().rustdoc_depinfo {
894 let mut arg = if build_runner.bcx.gctx.cli_unstable().rustdoc_mergeable_info {
897 OsString::from("--emit=html-non-static-files,dep-info=")
899 } else {
900 OsString::from("--emit=html-static-files,html-non-static-files,dep-info=")
902 };
903 arg.push(rustdoc_dep_info_loc(build_runner, unit));
904 rustdoc.arg(arg);
905
906 if build_runner.bcx.gctx.cli_unstable().checksum_freshness {
907 rustdoc.arg("-Z").arg("checksum-hash-algorithm=blake3");
908 }
909 } else if build_runner.bcx.gctx.cli_unstable().rustdoc_mergeable_info {
910 rustdoc.arg("--emit=html-non-static-files");
912 }
913
914 if build_runner.bcx.gctx.cli_unstable().rustdoc_mergeable_info {
915 rustdoc.arg("-Zunstable-options");
917 rustdoc.arg("--merge=none");
918 let mut arg = OsString::from("--parts-out-dir=");
919 arg.push(build_runner.files().out_dir_new_layout(unit));
921 rustdoc.arg(arg);
922 }
923
924 if let Some(trim_paths) = unit.profile.trim_paths.as_ref() {
925 trim_paths_args_rustdoc(&mut rustdoc, build_runner, unit, trim_paths)?;
926 }
927
928 rustdoc.args(unit.pkg.manifest().lint_rustflags());
929
930 let metadata = build_runner.metadata_for_doc_units[unit];
931 rustdoc
932 .arg("-C")
933 .arg(format!("metadata={}", metadata.c_metadata()));
934
935 if unit.mode.is_doc_scrape() {
936 debug_assert!(build_runner.bcx.scrape_units.contains(unit));
937
938 if unit.target.is_test() {
939 rustdoc.arg("--scrape-tests");
940 }
941
942 rustdoc.arg("-Zunstable-options");
943
944 rustdoc
945 .arg("--scrape-examples-output-path")
946 .arg(scrape_output_path(build_runner, unit)?);
947
948 for pkg in build_runner.bcx.packages.packages() {
950 let names = pkg
951 .targets()
952 .iter()
953 .map(|target| target.crate_name())
954 .collect::<HashSet<_>>();
955 for name in names {
956 rustdoc.arg("--scrape-examples-target-crate").arg(name);
957 }
958 }
959 }
960
961 if should_include_scrape_units(build_runner.bcx, unit) {
962 rustdoc.arg("-Zunstable-options");
963 }
964
965 build_deps_args(&mut rustdoc, build_runner, unit)?;
966 rustdoc::add_root_urls(build_runner, unit, &mut rustdoc)?;
967
968 rustdoc::add_output_format(build_runner, &mut rustdoc)?;
969
970 if let Some(args) = build_runner.bcx.extra_args_for(unit) {
971 rustdoc.args(args);
972 }
973 rustdoc.args(&unit.rustdocflags);
974
975 if !crate_version_flag_already_present(&rustdoc) {
976 append_crate_version_flag(unit, &mut rustdoc);
977 }
978
979 Ok(rustdoc)
980}
981
982fn rustdoc(build_runner: &mut BuildRunner<'_, '_>, unit: &Unit) -> CargoResult<Work> {
984 let mut rustdoc = prepare_rustdoc(build_runner, unit)?;
985
986 let crate_name = unit.target.crate_name();
987 let is_json_output = build_runner.bcx.build_config.intent.wants_doc_json_output();
988 let doc_dir = build_runner.files().output_dir(unit);
989 paths::create_dir_all(&doc_dir)?;
993
994 let target_desc = unit.target.description_named();
995 let name = unit.pkg.name();
996 let build_script_outputs = Arc::clone(&build_runner.build_script_outputs);
997 let package_id = unit.pkg.package_id();
998 let target = Target::clone(&unit.target);
999 let manifest = ManifestErrorContext::new(build_runner, unit);
1000
1001 let rustdoc_dep_info_loc = rustdoc_dep_info_loc(build_runner, unit);
1002 let dep_info_loc = fingerprint::dep_info_loc(build_runner, unit);
1003 let build_dir = build_runner.bcx.ws.build_dir().into_path_unlocked();
1004 let pkg_root = unit.pkg.root().to_path_buf();
1005 let cwd = rustdoc
1006 .get_cwd()
1007 .unwrap_or_else(|| build_runner.bcx.gctx.cwd())
1008 .to_path_buf();
1009 let fingerprint_dir = build_runner.files().fingerprint_dir(unit);
1010 let is_local = unit.is_local();
1011 let env_config = Arc::clone(build_runner.bcx.gctx.env_config()?);
1012 let rustdoc_depinfo_enabled = build_runner.bcx.gctx.cli_unstable().rustdoc_depinfo;
1013
1014 let mut output_options = OutputOptions::for_dirty(build_runner, unit);
1015 let script_metadatas = build_runner.find_build_script_metadatas(unit);
1016 let scrape_outputs = if should_include_scrape_units(build_runner.bcx, unit) {
1017 Some(
1018 build_runner
1019 .bcx
1020 .scrape_units
1021 .iter()
1022 .map(|unit| {
1023 Ok((
1024 build_runner.files().metadata(unit).unit_id(),
1025 scrape_output_path(build_runner, unit)?,
1026 ))
1027 })
1028 .collect::<CargoResult<HashMap<_, _>>>()?,
1029 )
1030 } else {
1031 None
1032 };
1033
1034 let failed_scrape_units = Arc::clone(&build_runner.failed_scrape_units);
1035 let hide_diagnostics_for_scrape_unit = build_runner.bcx.unit_can_fail_for_docscraping(unit)
1036 && !matches!(
1037 build_runner.bcx.gctx.shell().verbosity(),
1038 Verbosity::Verbose
1039 );
1040 let failed_scrape_diagnostic = hide_diagnostics_for_scrape_unit.then(|| {
1041 make_failed_scrape_diagnostic(
1042 build_runner,
1043 unit,
1044 format_args!("failed to scan {target_desc} in package `{name}` for example code usage"),
1045 )
1046 });
1047 if hide_diagnostics_for_scrape_unit {
1048 output_options.show_diagnostics = false;
1049 }
1050
1051 Ok(Work::new(move |state| {
1052 add_custom_flags(
1053 &mut rustdoc,
1054 &build_script_outputs.lock().unwrap(),
1055 script_metadatas,
1056 )?;
1057
1058 if let Some(scrape_outputs) = scrape_outputs {
1063 let failed_scrape_units = failed_scrape_units.lock().unwrap();
1064 for (metadata, output_path) in &scrape_outputs {
1065 if !failed_scrape_units.contains(metadata) {
1066 rustdoc.arg("--with-examples").arg(output_path);
1067 }
1068 }
1069 }
1070
1071 if !is_json_output {
1072 let crate_dir = doc_dir.join(&crate_name);
1073 if crate_dir.exists() {
1074 debug!("removing pre-existing doc directory {:?}", crate_dir);
1077 paths::remove_dir_all(&crate_dir)?;
1078 }
1079 };
1080 state.running(&rustdoc);
1081 let timestamp = paths::set_invocation_time(&fingerprint_dir)?;
1082
1083 let result = rustdoc
1084 .exec_with_streaming(
1085 &mut |line| on_stdout_line(state, line, package_id, &target),
1086 &mut |line| {
1087 on_stderr_line(
1088 state,
1089 line,
1090 package_id,
1091 &manifest,
1092 &target,
1093 &mut output_options,
1094 )
1095 },
1096 false,
1097 )
1098 .map_err(verbose_if_simple_exit_code)
1099 .with_context(|| format!("could not document `{}`", name));
1100
1101 if let Err(e) = result {
1102 if let Some(diagnostic) = failed_scrape_diagnostic {
1103 state.warning(diagnostic);
1104 }
1105
1106 return Err(e);
1107 }
1108
1109 if rustdoc_depinfo_enabled && rustdoc_dep_info_loc.exists() {
1110 fingerprint::translate_dep_info(
1111 &rustdoc_dep_info_loc,
1112 &dep_info_loc,
1113 &cwd,
1114 &pkg_root,
1115 &build_dir,
1116 &rustdoc,
1117 is_local,
1119 &env_config,
1120 )
1121 .with_context(|| {
1122 internal(format_args!(
1123 "could not parse/generate dep info at: {}",
1124 rustdoc_dep_info_loc.display()
1125 ))
1126 })?;
1127 paths::set_file_time_no_err(dep_info_loc, timestamp);
1130 }
1131
1132 Ok(())
1133 }))
1134}
1135
1136fn crate_version_flag_already_present(rustdoc: &ProcessBuilder) -> bool {
1139 rustdoc.get_args().any(|flag| {
1140 flag.to_str()
1141 .map_or(false, |flag| flag.starts_with(RUSTDOC_CRATE_VERSION_FLAG))
1142 })
1143}
1144
1145fn append_crate_version_flag(unit: &Unit, rustdoc: &mut ProcessBuilder) {
1146 rustdoc
1147 .arg(RUSTDOC_CRATE_VERSION_FLAG)
1148 .arg(unit.pkg.version().to_string());
1149}
1150
1151fn add_cap_lints(bcx: &BuildContext<'_, '_>, unit: &Unit, cmd: &mut ProcessBuilder) {
1155 if !unit.show_warnings(bcx.gctx) {
1158 cmd.arg("--cap-lints").arg("allow");
1159
1160 } else if !unit.is_local() {
1163 cmd.arg("--cap-lints").arg("warn");
1164 }
1165}
1166
1167fn add_allow_features(build_runner: &BuildRunner<'_, '_>, cmd: &mut ProcessBuilder) {
1171 if let Some(allow) = &build_runner.bcx.gctx.cli_unstable().allow_features {
1172 use std::fmt::Write;
1173 let mut arg = String::from("-Zallow-features=");
1174 for f in allow {
1175 let _ = write!(&mut arg, "{f},");
1176 }
1177 cmd.arg(arg.trim_end_matches(','));
1178 }
1179}
1180
1181fn add_error_format_and_color(build_runner: &BuildRunner<'_, '_>, cmd: &mut ProcessBuilder) {
1192 let enable_timings =
1193 build_runner.bcx.gctx.cli_unstable().section_timings && build_runner.bcx.logger.is_some();
1194 if enable_timings {
1195 cmd.arg("-Zunstable-options");
1196 }
1197
1198 cmd.arg("--error-format=json");
1199
1200 let mut json = String::from("--json=diagnostic-rendered-ansi,artifacts,future-incompat");
1201 if build_runner.bcx.gctx.cli_unstable().cargo_lints {
1202 json.push_str(",unused-externs-silent");
1203 }
1204 if let MessageFormat::Short | MessageFormat::Json { short: true, .. } =
1205 build_runner.bcx.build_config.message_format
1206 {
1207 json.push_str(",diagnostic-short");
1208 } else if build_runner.bcx.gctx.shell().err_unicode()
1209 && build_runner.bcx.gctx.cli_unstable().rustc_unicode
1210 {
1211 json.push_str(",diagnostic-unicode");
1212 }
1213 if enable_timings {
1214 json.push_str(",timings");
1215 }
1216 cmd.arg(json);
1217
1218 let gctx = build_runner.bcx.gctx;
1219 if let Some(width) = gctx.shell().err_width().diagnostic_terminal_width() {
1220 cmd.arg(format!("--diagnostic-width={width}"));
1221 }
1222}
1223
1224fn build_base_args(
1226 build_runner: &BuildRunner<'_, '_>,
1227 cmd: &mut ProcessBuilder,
1228 unit: &Unit,
1229) -> CargoResult<()> {
1230 assert!(!unit.mode.is_run_custom_build());
1231
1232 let bcx = build_runner.bcx;
1233 let Profile {
1234 ref opt_level,
1235 codegen_backend,
1236 codegen_units,
1237 debuginfo,
1238 debug_assertions,
1239 split_debuginfo,
1240 overflow_checks,
1241 rpath,
1242 ref panic,
1243 incremental,
1244 strip,
1245 rustflags: profile_rustflags,
1246 trim_paths,
1247 hint_mostly_unused: profile_hint_mostly_unused,
1248 ..
1249 } = unit.profile.clone();
1250 let hints = unit.pkg.hints().cloned().unwrap_or_default();
1251 let test = unit.mode.is_any_test();
1252
1253 let warn = |msg: &str| {
1254 bcx.gctx.shell().warn(format!(
1255 "{}@{}: {msg}",
1256 unit.pkg.package_id().name(),
1257 unit.pkg.package_id().version()
1258 ))
1259 };
1260 let unit_capped_warn = |msg: &str| {
1261 if unit.show_warnings(bcx.gctx) {
1262 warn(msg)
1263 } else {
1264 Ok(())
1265 }
1266 };
1267
1268 cmd.arg("--crate-name").arg(&unit.target.crate_name());
1269
1270 let edition = unit.target.edition();
1271 edition.cmd_edition_arg(cmd);
1272
1273 add_path_args(bcx.ws, unit, cmd);
1274 add_error_format_and_color(build_runner, cmd);
1275 add_allow_features(build_runner, cmd);
1276
1277 let mut contains_dy_lib = false;
1278 if !test {
1279 for crate_type in &unit.target.rustc_crate_types() {
1280 cmd.arg("--crate-type").arg(crate_type.as_str());
1281 contains_dy_lib |= crate_type == &CrateType::Dylib;
1282 }
1283 }
1284
1285 if unit.mode.is_check() {
1286 cmd.arg("--emit=dep-info,metadata");
1287 } else if build_runner.bcx.gctx.cli_unstable().no_embed_metadata {
1288 if unit.benefits_from_no_embed_metadata() {
1298 cmd.arg("--emit=dep-info,metadata,link");
1299 cmd.args(&["-Z", "embed-metadata=no"]);
1300 } else {
1301 cmd.arg("--emit=dep-info,link");
1302 }
1303 } else {
1304 if !unit.requires_upstream_objects() {
1308 cmd.arg("--emit=dep-info,metadata,link");
1309 } else {
1310 cmd.arg("--emit=dep-info,link");
1311 }
1312 }
1313
1314 let prefer_dynamic = (unit.target.for_host() && !unit.target.is_custom_build())
1315 || (contains_dy_lib && !build_runner.is_primary_package(unit));
1316 if prefer_dynamic {
1317 cmd.arg("-C").arg("prefer-dynamic");
1318 }
1319
1320 if opt_level.as_str() != "0" {
1321 cmd.arg("-C").arg(&format!("opt-level={}", opt_level));
1322 }
1323
1324 if *panic != PanicStrategy::Unwind {
1325 cmd.arg("-C").arg(format!("panic={}", panic));
1326 }
1327 if *panic == PanicStrategy::ImmediateAbort {
1328 cmd.arg("-Z").arg("unstable-options");
1329 }
1330
1331 cmd.args(<o_args(build_runner, unit));
1332
1333 if let Some(backend) = codegen_backend {
1334 cmd.arg("-Z").arg(&format!("codegen-backend={}", backend));
1335 }
1336
1337 if let Some(n) = codegen_units {
1338 cmd.arg("-C").arg(&format!("codegen-units={}", n));
1339 }
1340
1341 let debuginfo = debuginfo.into_inner();
1342 if debuginfo != TomlDebugInfo::None {
1344 cmd.arg("-C").arg(format!("debuginfo={debuginfo}"));
1345 if let Some(split) = split_debuginfo {
1352 if build_runner
1353 .bcx
1354 .target_data
1355 .info(unit.kind)
1356 .supports_debuginfo_split(split)
1357 {
1358 cmd.arg("-C").arg(format!("split-debuginfo={split}"));
1359 }
1360 }
1361 }
1362
1363 if let Some(trim_paths) = trim_paths {
1364 trim_paths_args(cmd, build_runner, unit, &trim_paths)?;
1365 }
1366
1367 cmd.args(unit.pkg.manifest().lint_rustflags());
1368 cmd.args(&profile_rustflags);
1369
1370 if opt_level.as_str() != "0" {
1374 if debug_assertions {
1375 cmd.args(&["-C", "debug-assertions=on"]);
1376 if !overflow_checks {
1377 cmd.args(&["-C", "overflow-checks=off"]);
1378 }
1379 } else if overflow_checks {
1380 cmd.args(&["-C", "overflow-checks=on"]);
1381 }
1382 } else if !debug_assertions {
1383 cmd.args(&["-C", "debug-assertions=off"]);
1384 if overflow_checks {
1385 cmd.args(&["-C", "overflow-checks=on"]);
1386 }
1387 } else if !overflow_checks {
1388 cmd.args(&["-C", "overflow-checks=off"]);
1389 }
1390
1391 if test && unit.target.harness() {
1392 cmd.arg("--test");
1393
1394 if *panic == PanicStrategy::Abort || *panic == PanicStrategy::ImmediateAbort {
1402 cmd.arg("-Z").arg("panic-abort-tests");
1403 }
1404 } else if test {
1405 cmd.arg("--cfg").arg("test");
1406 }
1407
1408 cmd.args(&features_args(unit));
1409 cmd.args(&check_cfg_args(unit));
1410
1411 let meta = build_runner.files().metadata(unit);
1412 cmd.arg("-C")
1413 .arg(&format!("metadata={}", meta.c_metadata()));
1414 if let Some(c_extra_filename) = meta.c_extra_filename() {
1415 cmd.arg("-C")
1416 .arg(&format!("extra-filename=-{c_extra_filename}"));
1417 }
1418
1419 if rpath {
1420 cmd.arg("-C").arg("rpath");
1421 }
1422
1423 cmd.arg("--out-dir")
1424 .arg(&build_runner.files().output_dir(unit));
1425
1426 unit.kind.add_target_arg(cmd);
1427
1428 add_codegen_linker(cmd, build_runner, unit, bcx.gctx.target_applies_to_host()?);
1429
1430 if incremental {
1431 add_codegen_incremental(cmd, build_runner, unit)
1432 }
1433
1434 let pkg_hint_mostly_unused = match hints.mostly_unused {
1435 None => None,
1436 Some(toml::Value::Boolean(b)) => Some(b),
1437 Some(v) => {
1438 unit_capped_warn(&format!(
1439 "ignoring unsupported value type ({}) for 'hints.mostly-unused', which expects a boolean",
1440 v.type_str()
1441 ))?;
1442 None
1443 }
1444 };
1445 if profile_hint_mostly_unused
1446 .or(pkg_hint_mostly_unused)
1447 .unwrap_or(false)
1448 {
1449 if bcx.gctx.cli_unstable().profile_hint_mostly_unused {
1450 cmd.arg("-Zhint-mostly-unused");
1451 } else {
1452 if profile_hint_mostly_unused.is_some() {
1453 warn(
1455 "ignoring 'hint-mostly-unused' profile option, pass `-Zprofile-hint-mostly-unused` to enable it",
1456 )?;
1457 } else if pkg_hint_mostly_unused.is_some() {
1458 unit_capped_warn(
1459 "ignoring 'hints.mostly-unused', pass `-Zprofile-hint-mostly-unused` to enable it",
1460 )?;
1461 }
1462 }
1463 }
1464
1465 let strip = strip.into_inner();
1466 if strip != StripInner::None {
1467 cmd.arg("-C").arg(format!("strip={}", strip));
1468 }
1469
1470 if unit.is_std {
1471 cmd.arg("-Z")
1477 .arg("force-unstable-if-unmarked")
1478 .env("RUSTC_BOOTSTRAP", "1");
1479 }
1480
1481 Ok(())
1482}
1483
1484fn features_args(unit: &Unit) -> Vec<OsString> {
1486 let mut args = Vec::with_capacity(unit.features.len() * 2);
1487
1488 for feat in &unit.features {
1489 args.push(OsString::from("--cfg"));
1490 args.push(OsString::from(format!("feature=\"{}\"", feat)));
1491 }
1492
1493 args
1494}
1495
1496fn trim_paths_args_rustdoc(
1498 cmd: &mut ProcessBuilder,
1499 build_runner: &BuildRunner<'_, '_>,
1500 unit: &Unit,
1501 trim_paths: &TomlTrimPaths,
1502) -> CargoResult<()> {
1503 match trim_paths {
1504 TomlTrimPaths::Values(values) if !values.contains(&TomlTrimPathsValue::Diagnostics) => {
1506 return Ok(());
1507 }
1508 _ => {}
1509 }
1510
1511 cmd.arg("-Zunstable-options");
1513
1514 for pair in trim_paths_remap(build_runner, unit) {
1515 let mut arg = OsString::from("--remap-path-prefix=");
1516 arg.push(pair);
1517 cmd.arg(arg);
1518 }
1519
1520 Ok(())
1521}
1522
1523fn trim_paths_args(
1529 cmd: &mut ProcessBuilder,
1530 build_runner: &BuildRunner<'_, '_>,
1531 unit: &Unit,
1532 trim_paths: &TomlTrimPaths,
1533) -> CargoResult<()> {
1534 if trim_paths.is_none() {
1535 return Ok(());
1536 }
1537
1538 cmd.arg(format!("--remap-path-scope={trim_paths}"));
1540
1541 for pair in trim_paths_remap(build_runner, unit) {
1542 let mut arg = OsString::from("--remap-path-prefix=");
1543 arg.push(pair);
1544 cmd.arg(arg);
1545 }
1546
1547 Ok(())
1548}
1549
1550pub(crate) fn trim_paths_remap(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> [OsString; 3] {
1557 [
1558 package_remap(build_runner, unit),
1559 build_dir_remap(build_runner),
1560 sysroot_remap(build_runner, unit),
1561 ]
1562}
1563
1564fn sysroot_remap(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> OsString {
1569 let mut remap = OsString::new();
1570 remap.push({
1571 let mut sysroot = build_runner.bcx.target_data.info(unit.kind).sysroot.clone();
1573 sysroot.push("lib");
1574 sysroot.push("rustlib");
1575 sysroot.push("src");
1576 sysroot.push("rust");
1577 sysroot
1578 });
1579 remap.push("=");
1580 remap.push("/rustc/");
1581 if let Some(commit_hash) = build_runner.bcx.rustc().commit_hash.as_ref() {
1582 remap.push(commit_hash);
1583 } else {
1584 remap.push(build_runner.bcx.rustc().version.to_string());
1585 }
1586 remap
1587}
1588
1589fn package_remap(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> OsString {
1597 let pkg_root = unit.pkg.root();
1598 let ws_root = build_runner.bcx.ws.root();
1599 let mut remap = OsString::new();
1600 let source_id = unit.pkg.package_id().source_id();
1601 if source_id.is_git() {
1602 remap.push(
1603 build_runner
1604 .bcx
1605 .gctx
1606 .git_checkouts_path()
1607 .as_path_unlocked(),
1608 );
1609 remap.push("=");
1610 } else if source_id.is_registry() {
1611 remap.push(
1612 build_runner
1613 .bcx
1614 .gctx
1615 .registry_source_path()
1616 .as_path_unlocked(),
1617 );
1618 remap.push("=");
1619 } else if pkg_root.strip_prefix(ws_root).is_ok() {
1620 remap.push(ws_root);
1621 remap.push("=."); } else {
1623 remap.push(pkg_root);
1624 remap.push("=");
1625 remap.push(unit.pkg.name());
1626 remap.push("-");
1627 remap.push(unit.pkg.version().to_string());
1628 }
1629 remap
1630}
1631
1632fn build_dir_remap(build_runner: &BuildRunner<'_, '_>) -> OsString {
1645 let build_dir = build_runner.bcx.ws.build_dir();
1646 let mut remap = OsString::new();
1647 remap.push(build_dir.as_path_unlocked());
1648 remap.push("=/cargo/build-dir");
1649 remap
1650}
1651
1652fn check_cfg_args(unit: &Unit) -> Vec<OsString> {
1654 let gross_cap_estimation = unit.pkg.summary().features().len() * 7 + 25;
1672 let mut arg_feature = OsString::with_capacity(gross_cap_estimation);
1673
1674 arg_feature.push("cfg(feature, values(");
1675 for (i, feature) in unit.pkg.summary().features().keys().enumerate() {
1676 if i != 0 {
1677 arg_feature.push(", ");
1678 }
1679 arg_feature.push("\"");
1680 arg_feature.push(feature);
1681 arg_feature.push("\"");
1682 }
1683 arg_feature.push("))");
1684
1685 vec![
1694 OsString::from("--check-cfg"),
1695 OsString::from("cfg(docsrs,test)"),
1696 OsString::from("--check-cfg"),
1697 arg_feature,
1698 ]
1699}
1700
1701fn lto_args(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> Vec<OsString> {
1703 let mut result = Vec::new();
1704 let mut push = |arg: &str| {
1705 result.push(OsString::from("-C"));
1706 result.push(OsString::from(arg));
1707 };
1708 match build_runner.lto[unit] {
1709 lto::Lto::Run(None) => push("lto"),
1710 lto::Lto::Run(Some(s)) => push(&format!("lto={}", s)),
1711 lto::Lto::Off => {
1712 push("lto=off");
1713 push("embed-bitcode=no");
1714 }
1715 lto::Lto::ObjectAndBitcode => {} lto::Lto::OnlyBitcode => push("linker-plugin-lto"),
1717 lto::Lto::OnlyObject => push("embed-bitcode=no"),
1718 }
1719 result
1720}
1721
1722fn build_deps_args(
1728 cmd: &mut ProcessBuilder,
1729 build_runner: &BuildRunner<'_, '_>,
1730 unit: &Unit,
1731) -> CargoResult<()> {
1732 let bcx = build_runner.bcx;
1733
1734 for arg in lib_search_paths(build_runner, unit)? {
1735 cmd.arg(arg);
1736 }
1737
1738 let deps = build_runner.unit_deps(unit);
1739
1740 if !deps
1744 .iter()
1745 .any(|dep| !dep.unit.mode.is_doc() && dep.unit.target.is_linkable())
1746 {
1747 if let Some(dep) = deps.iter().find(|dep| {
1748 !dep.unit.mode.is_doc() && dep.unit.target.is_lib() && !dep.unit.artifact.is_true()
1749 }) {
1750 let dep_name = dep.unit.target.crate_name();
1751 let name = unit.target.crate_name();
1752 bcx.gctx.shell().print_report(&[
1753 Level::WARNING.secondary_title(format!("the package `{dep_name}` provides no linkable target"))
1754 .elements([
1755 Level::NOTE.message(format!("this might cause `{name}` to fail compilation")),
1756 Level::NOTE.message("this warning might turn into a hard error in the future"),
1757 Level::HELP.message(format!("consider adding 'dylib' or 'rlib' to key 'crate-type' in `{dep_name}`'s Cargo.toml"))
1758 ])
1759 ], false)?;
1760 }
1761 }
1762
1763 let mut unstable_opts = false;
1764
1765 let first_custom_build_dep = deps.iter().find(|dep| dep.unit.mode.is_run_custom_build());
1767 if let Some(dep) = first_custom_build_dep {
1768 let out_dir = if bcx.gctx.cli_unstable().build_dir_new_layout {
1769 build_runner.files().out_dir_new_layout(&dep.unit)
1770 } else {
1771 build_runner.files().build_script_out_dir(&dep.unit)
1772 };
1773 cmd.env("OUT_DIR", &out_dir);
1774 }
1775
1776 let is_multiple_build_scripts_enabled = unit
1778 .pkg
1779 .manifest()
1780 .unstable_features()
1781 .require(Feature::multiple_build_scripts())
1782 .is_ok();
1783
1784 if is_multiple_build_scripts_enabled {
1785 for dep in deps {
1786 if dep.unit.mode.is_run_custom_build() {
1787 let out_dir = if bcx.gctx.cli_unstable().build_dir_new_layout {
1788 build_runner.files().out_dir_new_layout(&dep.unit)
1789 } else {
1790 build_runner.files().build_script_out_dir(&dep.unit)
1791 };
1792 let target_name = dep.unit.target.name();
1793 let out_dir_prefix = target_name
1794 .strip_prefix("build-script-")
1795 .unwrap_or(target_name);
1796 let out_dir_name = format!("{out_dir_prefix}_OUT_DIR");
1797 cmd.env(&out_dir_name, &out_dir);
1798 }
1799 }
1800 }
1801 for arg in extern_args(build_runner, unit, &mut unstable_opts)? {
1802 cmd.arg(arg);
1803 }
1804
1805 for (var, env) in artifact::get_env(build_runner, unit, deps)? {
1806 cmd.env(&var, env);
1807 }
1808
1809 if unstable_opts {
1812 cmd.arg("-Z").arg("unstable-options");
1813 }
1814
1815 Ok(())
1816}
1817
1818fn add_dep_arg<'a, 'b: 'a>(
1819 map: &mut BTreeMap<&'a Unit, PathBuf>,
1820 build_runner: &'b BuildRunner<'b, '_>,
1821 unit: &'a Unit,
1822) {
1823 if map.contains_key(&unit) {
1824 return;
1825 }
1826 map.insert(&unit, build_runner.files().deps_dir(&unit));
1827
1828 for dep in build_runner.unit_deps(unit) {
1829 add_dep_arg(map, build_runner, &dep.unit);
1830 }
1831}
1832
1833fn add_custom_flags(
1837 cmd: &mut ProcessBuilder,
1838 build_script_outputs: &BuildScriptOutputs,
1839 metadata_vec: Option<Vec<UnitHash>>,
1840) -> CargoResult<()> {
1841 if let Some(metadata_vec) = metadata_vec {
1842 for metadata in metadata_vec {
1843 if let Some(output) = build_script_outputs.get(metadata) {
1844 for cfg in output.cfgs.iter() {
1845 cmd.arg("--cfg").arg(cfg);
1846 }
1847 for check_cfg in &output.check_cfgs {
1848 cmd.arg("--check-cfg").arg(check_cfg);
1849 }
1850 for (name, value) in output.env.iter() {
1851 cmd.env(name, value);
1852 }
1853 }
1854 }
1855 }
1856
1857 Ok(())
1858}
1859
1860pub fn lib_search_paths(
1862 build_runner: &BuildRunner<'_, '_>,
1863 unit: &Unit,
1864) -> CargoResult<Vec<OsString>> {
1865 let mut lib_search_paths = Vec::new();
1866 if build_runner.bcx.gctx.cli_unstable().build_dir_new_layout {
1867 let mut map = BTreeMap::new();
1868
1869 add_dep_arg(&mut map, build_runner, unit);
1871
1872 let paths = map.into_iter().map(|(_, path)| path).sorted_unstable();
1873
1874 for path in paths {
1875 let mut deps = OsString::from("dependency=");
1876 deps.push(path);
1877 lib_search_paths.extend(["-L".into(), deps]);
1878 }
1879 } else {
1880 let mut deps = OsString::from("dependency=");
1881 deps.push(build_runner.files().deps_dir(unit));
1882 lib_search_paths.extend(["-L".into(), deps]);
1883 }
1884
1885 if !unit.kind.is_host() {
1888 let mut deps = OsString::from("dependency=");
1889 deps.push(build_runner.files().host_deps(unit));
1890 lib_search_paths.extend(["-L".into(), deps]);
1891 }
1892
1893 Ok(lib_search_paths)
1894}
1895
1896fn is_public_dependency_enabled(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> bool {
1897 unit.pkg
1898 .manifest()
1899 .unstable_features()
1900 .require(Feature::public_dependency())
1901 .is_ok()
1902 || build_runner.bcx.gctx.cli_unstable().public_dependency
1903}
1904
1905pub fn extern_args(
1907 build_runner: &BuildRunner<'_, '_>,
1908 unit: &Unit,
1909 unstable_opts: &mut bool,
1910) -> CargoResult<Vec<OsString>> {
1911 let mut result = Vec::new();
1912 let deps = build_runner.unit_deps(unit);
1913
1914 let no_embed_metadata = build_runner.bcx.gctx.cli_unstable().no_embed_metadata;
1915 let public_dependency_enabled = is_public_dependency_enabled(build_runner, unit);
1916
1917 let mut link_to = |dep: &UnitDep,
1919 extern_crate_name: InternedString,
1920 noprelude: bool,
1921 nounused: bool|
1922 -> CargoResult<()> {
1923 let mut value = OsString::new();
1924 let mut opts = Vec::new();
1925 if !dep.public && unit.target.is_lib() && public_dependency_enabled {
1926 opts.push("priv");
1927 *unstable_opts = true;
1928 }
1929 if noprelude {
1930 opts.push("noprelude");
1931 *unstable_opts = true;
1932 }
1933 if nounused {
1934 opts.push("nounused");
1935 *unstable_opts = true;
1936 }
1937 if !opts.is_empty() {
1938 value.push(opts.join(","));
1939 value.push(":");
1940 }
1941 value.push(extern_crate_name.as_str());
1942 value.push("=");
1943
1944 let mut pass = |file| {
1945 let mut value = value.clone();
1946 value.push(file);
1947 result.push(OsString::from("--extern"));
1948 result.push(value);
1949 };
1950
1951 let outputs = build_runner.outputs(&dep.unit)?;
1952
1953 if build_runner.only_requires_rmeta(unit, &dep.unit) || dep.unit.mode.is_check() {
1954 let output = outputs
1956 .iter()
1957 .find(|output| output.flavor == FileFlavor::Rmeta)
1958 .expect("failed to find rmeta dep for pipelined dep");
1959 pass(&output.path);
1960 } else {
1961 for output in outputs.iter() {
1963 if output.flavor == FileFlavor::Linkable {
1964 pass(&output.path);
1965 }
1966 else if no_embed_metadata && output.flavor == FileFlavor::Rmeta {
1970 pass(&output.path);
1971 }
1972 }
1973 }
1974 Ok(())
1975 };
1976
1977 for dep in deps {
1978 if dep.unit.target.is_linkable() && !dep.unit.mode.is_doc() {
1979 link_to(dep, dep.extern_crate_name, dep.noprelude, dep.nounused)?;
1980 }
1981 }
1982 if unit.target.proc_macro() {
1983 result.push(OsString::from("--extern"));
1985 result.push(OsString::from("proc_macro"));
1986 }
1987
1988 Ok(result)
1989}
1990
1991fn add_codegen_linker(
1993 cmd: &mut ProcessBuilder,
1994 build_runner: &BuildRunner<'_, '_>,
1995 unit: &Unit,
1996 target_applies_to_host: bool,
1997) {
1998 let linker = if unit.target.for_host() && !target_applies_to_host {
1999 build_runner
2000 .compilation
2001 .host_linker()
2002 .map(|s| s.as_os_str())
2003 } else {
2004 build_runner
2005 .compilation
2006 .target_linker(unit.kind)
2007 .map(|s| s.as_os_str())
2008 };
2009
2010 if let Some(linker) = linker {
2011 let mut arg = OsString::from("linker=");
2012 arg.push(linker);
2013 cmd.arg("-C").arg(arg);
2014 }
2015}
2016
2017fn add_codegen_incremental(
2019 cmd: &mut ProcessBuilder,
2020 build_runner: &BuildRunner<'_, '_>,
2021 unit: &Unit,
2022) {
2023 let dir = build_runner.files().incremental_dir(&unit);
2024 let mut arg = OsString::from("incremental=");
2025 arg.push(dir.as_os_str());
2026 cmd.arg("-C").arg(arg);
2027}
2028
2029fn envify(s: &str) -> String {
2030 s.chars()
2031 .flat_map(|c| c.to_uppercase())
2032 .map(|c| if c == '-' { '_' } else { c })
2033 .collect()
2034}
2035
2036struct OutputOptions {
2039 format: MessageFormat,
2041 cache_cell: Option<(PathBuf, OnceCell<File>)>,
2046 show_diagnostics: bool,
2054 warnings_seen: usize,
2056 errors_seen: usize,
2058}
2059
2060impl OutputOptions {
2061 fn for_dirty(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> OutputOptions {
2062 let path = build_runner.files().message_cache_path(unit);
2063 drop(fs::remove_file(&path));
2065 let cache_cell = Some((path, OnceCell::new()));
2066
2067 let show_diagnostics = true;
2068
2069 let format = build_runner.bcx.build_config.message_format;
2070
2071 OutputOptions {
2072 format,
2073 cache_cell,
2074 show_diagnostics,
2075 warnings_seen: 0,
2076 errors_seen: 0,
2077 }
2078 }
2079
2080 fn for_fresh(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> OutputOptions {
2081 let cache_cell = None;
2082
2083 let show_diagnostics = unit.show_warnings(build_runner.bcx.gctx);
2086
2087 let format = build_runner.bcx.build_config.message_format;
2088
2089 OutputOptions {
2090 format,
2091 cache_cell,
2092 show_diagnostics,
2093 warnings_seen: 0,
2094 errors_seen: 0,
2095 }
2096 }
2097}
2098
2099struct ManifestErrorContext {
2105 path: PathBuf,
2107 spans: Option<toml::Spanned<toml::de::DeTable<'static>>>,
2109 contents: Option<String>,
2111 rename_table: HashMap<InternedString, InternedString>,
2114 requested_kinds: Vec<CompileKind>,
2117 cfgs: Vec<Vec<Cfg>>,
2120 host_name: InternedString,
2121 cwd: PathBuf,
2123 term_width: usize,
2125}
2126
2127fn on_stdout_line(
2128 state: &JobState<'_, '_>,
2129 line: &str,
2130 _package_id: PackageId,
2131 _target: &Target,
2132) -> CargoResult<()> {
2133 state.stdout(line.to_string())?;
2134 Ok(())
2135}
2136
2137fn on_stderr_line(
2138 state: &JobState<'_, '_>,
2139 line: &str,
2140 package_id: PackageId,
2141 manifest: &ManifestErrorContext,
2142 target: &Target,
2143 options: &mut OutputOptions,
2144) -> CargoResult<()> {
2145 if on_stderr_line_inner(state, line, package_id, manifest, target, options)? {
2146 if let Some((path, cell)) = &mut options.cache_cell {
2148 let f = cell.try_borrow_mut_with(|| paths::create(path))?;
2150 debug_assert!(!line.contains('\n'));
2151 f.write_all(line.as_bytes())?;
2152 f.write_all(&[b'\n'])?;
2153 }
2154 }
2155 Ok(())
2156}
2157
2158fn on_stderr_line_inner(
2160 state: &JobState<'_, '_>,
2161 line: &str,
2162 package_id: PackageId,
2163 manifest: &ManifestErrorContext,
2164 target: &Target,
2165 options: &mut OutputOptions,
2166) -> CargoResult<bool> {
2167 if !line.starts_with('{') {
2173 state.stderr(line.to_string())?;
2174 return Ok(true);
2175 }
2176
2177 let mut compiler_message: Box<serde_json::value::RawValue> = match serde_json::from_str(line) {
2178 Ok(msg) => msg,
2179
2180 Err(e) => {
2184 debug!("failed to parse json: {:?}", e);
2185 state.stderr(line.to_string())?;
2186 return Ok(true);
2187 }
2188 };
2189
2190 let count_diagnostic = |level, options: &mut OutputOptions| {
2191 if level == "warning" {
2192 options.warnings_seen += 1;
2193 } else if level == "error" {
2194 options.errors_seen += 1;
2195 }
2196 };
2197
2198 if let Ok(report) = serde_json::from_str::<FutureIncompatReport>(compiler_message.get()) {
2199 for item in &report.future_incompat_report {
2200 count_diagnostic(&*item.diagnostic.level, options);
2201 }
2202 state.future_incompat_report(report.future_incompat_report);
2203 return Ok(true);
2204 }
2205
2206 let res = serde_json::from_str::<SectionTiming>(compiler_message.get());
2207 if let Ok(timing_record) = res {
2208 state.on_section_timing_emitted(timing_record);
2209 return Ok(false);
2210 }
2211
2212 let add_pub_in_priv_diagnostic = |diag: &mut String| -> bool {
2214 static PRIV_DEP_REGEX: LazyLock<Regex> =
2223 LazyLock::new(|| Regex::new("from private dependency '([A-Za-z0-9-_]+)'").unwrap());
2224 if let Some(crate_name) = PRIV_DEP_REGEX.captures(diag).and_then(|m| m.get(1))
2225 && let Some(ref contents) = manifest.contents
2226 && let Some(span) = manifest.find_crate_span(crate_name.as_str())
2227 {
2228 let rel_path = pathdiff::diff_paths(&manifest.path, &manifest.cwd)
2229 .unwrap_or_else(|| manifest.path.clone())
2230 .display()
2231 .to_string();
2232 let report = [Group::with_title(Level::NOTE.secondary_title(format!(
2233 "dependency `{}` declared here",
2234 crate_name.as_str()
2235 )))
2236 .element(
2237 Snippet::source(contents)
2238 .path(rel_path)
2239 .annotation(AnnotationKind::Context.span(span)),
2240 )];
2241
2242 let rendered = Renderer::styled()
2243 .term_width(manifest.term_width)
2244 .render(&report);
2245 diag.push_str(&rendered);
2246 diag.push('\n');
2247 return true;
2248 }
2249 false
2250 };
2251
2252 match options.format {
2255 MessageFormat::Human
2260 | MessageFormat::Short
2261 | MessageFormat::Json {
2262 render_diagnostics: true,
2263 ..
2264 } => {
2265 #[derive(serde::Deserialize)]
2266 struct CompilerMessage<'a> {
2267 rendered: String,
2271 #[serde(borrow)]
2272 message: Cow<'a, str>,
2273 #[serde(borrow)]
2274 level: Cow<'a, str>,
2275 children: Vec<PartialDiagnostic>,
2276 code: Option<DiagnosticCode>,
2277 }
2278
2279 #[derive(serde::Deserialize)]
2288 struct PartialDiagnostic {
2289 spans: Vec<PartialDiagnosticSpan>,
2290 }
2291
2292 #[derive(serde::Deserialize)]
2294 struct PartialDiagnosticSpan {
2295 suggestion_applicability: Option<Applicability>,
2296 }
2297
2298 #[derive(serde::Deserialize)]
2299 struct DiagnosticCode {
2300 code: String,
2301 }
2302
2303 if let Ok(mut msg) = serde_json::from_str::<CompilerMessage<'_>>(compiler_message.get())
2304 {
2305 if msg.message.starts_with("aborting due to")
2306 || msg.message.ends_with("warning emitted")
2307 || msg.message.ends_with("warnings emitted")
2308 {
2309 return Ok(true);
2311 }
2312 if msg.rendered.ends_with('\n') {
2314 msg.rendered.pop();
2315 }
2316 let mut rendered = msg.rendered;
2317 if options.show_diagnostics {
2318 let machine_applicable: bool = msg
2319 .children
2320 .iter()
2321 .map(|child| {
2322 child
2323 .spans
2324 .iter()
2325 .filter_map(|span| span.suggestion_applicability)
2326 .any(|app| app == Applicability::MachineApplicable)
2327 })
2328 .any(|b| b);
2329 count_diagnostic(&msg.level, options);
2330 if msg
2331 .code
2332 .as_ref()
2333 .is_some_and(|c| c.code == "exported_private_dependencies")
2334 && options.format != MessageFormat::Short
2335 {
2336 add_pub_in_priv_diagnostic(&mut rendered);
2337 }
2338 let lint = msg.code.is_some();
2339 state.emit_diag(&msg.level, rendered, lint, machine_applicable)?;
2340 }
2341 return Ok(true);
2342 }
2343 }
2344
2345 MessageFormat::Json { ansi, .. } => {
2346 #[derive(serde::Deserialize, serde::Serialize)]
2347 struct CompilerMessage<'a> {
2348 rendered: String,
2349 #[serde(flatten, borrow)]
2350 other: std::collections::BTreeMap<Cow<'a, str>, serde_json::Value>,
2351 code: Option<DiagnosticCode<'a>>,
2352 }
2353
2354 #[derive(serde::Deserialize, serde::Serialize)]
2355 struct DiagnosticCode<'a> {
2356 code: String,
2357 #[serde(flatten, borrow)]
2358 other: std::collections::BTreeMap<Cow<'a, str>, serde_json::Value>,
2359 }
2360
2361 if let Ok(mut error) =
2362 serde_json::from_str::<CompilerMessage<'_>>(compiler_message.get())
2363 {
2364 let modified_diag = if error
2365 .code
2366 .as_ref()
2367 .is_some_and(|c| c.code == "exported_private_dependencies")
2368 {
2369 add_pub_in_priv_diagnostic(&mut error.rendered)
2370 } else {
2371 false
2372 };
2373
2374 if !ansi {
2378 error.rendered = anstream::adapter::strip_str(&error.rendered).to_string();
2379 }
2380 if !ansi || modified_diag {
2381 let new_line = serde_json::to_string(&error)?;
2382 compiler_message = serde_json::value::RawValue::from_string(new_line)?;
2383 }
2384 }
2385 }
2386 }
2387
2388 #[derive(serde::Deserialize)]
2395 struct ArtifactNotification<'a> {
2396 #[serde(borrow)]
2397 artifact: Cow<'a, str>,
2398 }
2399
2400 if let Ok(artifact) = serde_json::from_str::<ArtifactNotification<'_>>(compiler_message.get()) {
2401 trace!("found directive from rustc: `{}`", artifact.artifact);
2402 if artifact.artifact.ends_with(".rmeta") {
2403 debug!("looks like metadata finished early!");
2404 state.rmeta_produced();
2405 }
2406 return Ok(false);
2407 }
2408
2409 #[derive(serde::Deserialize)]
2410 struct UnusedExterns {
2411 unused_extern_names: std::collections::BTreeSet<InternedString>,
2412 }
2413 if let Ok(uext) = serde_json::from_str::<UnusedExterns>(compiler_message.get()) {
2414 trace!(
2415 "obtained unused externs list from rustc: `{:?}`",
2416 uext.unused_extern_names
2417 );
2418 state.unused_externs(uext.unused_extern_names);
2419 return Ok(true);
2420 }
2421
2422 if !options.show_diagnostics {
2427 return Ok(true);
2428 }
2429
2430 #[derive(serde::Deserialize)]
2431 struct CompilerMessage<'a> {
2432 #[serde(borrow)]
2433 message: Cow<'a, str>,
2434 #[serde(borrow)]
2435 level: Cow<'a, str>,
2436 }
2437
2438 if let Ok(msg) = serde_json::from_str::<CompilerMessage<'_>>(compiler_message.get()) {
2439 if msg.message.starts_with("aborting due to")
2440 || msg.message.ends_with("warning emitted")
2441 || msg.message.ends_with("warnings emitted")
2442 {
2443 return Ok(true);
2445 }
2446 count_diagnostic(&msg.level, options);
2447 }
2448
2449 let msg = machine_message::FromCompiler {
2450 package_id: package_id.to_spec(),
2451 manifest_path: &manifest.path,
2452 target,
2453 message: compiler_message,
2454 }
2455 .to_json_string();
2456
2457 state.stdout(msg)?;
2461 Ok(true)
2462}
2463
2464impl ManifestErrorContext {
2465 fn new(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> ManifestErrorContext {
2466 let mut duplicates = HashSet::new();
2467 let mut rename_table = HashMap::new();
2468
2469 for dep in build_runner.unit_deps(unit) {
2470 let unrenamed_id = dep.unit.pkg.package_id().name();
2471 if duplicates.contains(&unrenamed_id) {
2472 continue;
2473 }
2474 match rename_table.entry(unrenamed_id) {
2475 std::collections::hash_map::Entry::Occupied(occ) => {
2476 occ.remove_entry();
2477 duplicates.insert(unrenamed_id);
2478 }
2479 std::collections::hash_map::Entry::Vacant(vac) => {
2480 vac.insert(dep.extern_crate_name);
2481 }
2482 }
2483 }
2484
2485 let bcx = build_runner.bcx;
2486 ManifestErrorContext {
2487 path: unit.pkg.manifest_path().to_owned(),
2488 spans: unit.pkg.manifest().document().cloned(),
2489 contents: unit.pkg.manifest().contents().map(String::from),
2490 requested_kinds: bcx.target_data.requested_kinds().to_owned(),
2491 host_name: bcx.rustc().host,
2492 rename_table,
2493 cwd: path_args(build_runner.bcx.ws, unit).1,
2494 cfgs: bcx
2495 .target_data
2496 .requested_kinds()
2497 .iter()
2498 .map(|k| bcx.target_data.cfg(*k).to_owned())
2499 .collect(),
2500 term_width: bcx
2501 .gctx
2502 .shell()
2503 .err_width()
2504 .diagnostic_terminal_width()
2505 .unwrap_or(cargo_util_terminal::report::renderer::DEFAULT_TERM_WIDTH),
2506 }
2507 }
2508
2509 fn requested_target_names(&self) -> impl Iterator<Item = &str> {
2510 self.requested_kinds.iter().map(|kind| match kind {
2511 CompileKind::Host => &self.host_name,
2512 CompileKind::Target(target) => target.short_name(),
2513 })
2514 }
2515
2516 fn find_crate_span(&self, unrenamed: &str) -> Option<Range<usize>> {
2530 let Some(ref spans) = self.spans else {
2531 return None;
2532 };
2533
2534 let orig_name = self.rename_table.get(unrenamed)?.as_str();
2535
2536 if let Some((k, v)) = get_key_value(&spans, &["dependencies", orig_name]) {
2537 if let Some(package) = v.get_ref().as_table().and_then(|t| t.get("package")) {
2546 return Some(package.span());
2547 } else {
2548 return Some(k.span());
2549 }
2550 }
2551
2552 if let Some(target) = spans
2557 .as_ref()
2558 .get("target")
2559 .and_then(|t| t.as_ref().as_table())
2560 {
2561 for (platform, platform_table) in target.iter() {
2562 match platform.as_ref().parse::<Platform>() {
2563 Ok(Platform::Name(name)) => {
2564 if !self.requested_target_names().any(|n| n == name) {
2565 continue;
2566 }
2567 }
2568 Ok(Platform::Cfg(cfg_expr)) => {
2569 if !self.cfgs.iter().any(|cfgs| cfg_expr.matches(cfgs)) {
2570 continue;
2571 }
2572 }
2573 Err(_) => continue,
2574 }
2575
2576 let Some(platform_table) = platform_table.as_ref().as_table() else {
2577 continue;
2578 };
2579
2580 if let Some(deps) = platform_table
2581 .get("dependencies")
2582 .and_then(|d| d.as_ref().as_table())
2583 {
2584 if let Some((k, v)) = deps.get_key_value(orig_name) {
2585 if let Some(package) = v.get_ref().as_table().and_then(|t| t.get("package"))
2586 {
2587 return Some(package.span());
2588 } else {
2589 return Some(k.span());
2590 }
2591 }
2592 }
2593 }
2594 }
2595 None
2596 }
2597}
2598
2599fn replay_output_cache(
2603 package_id: PackageId,
2604 manifest: ManifestErrorContext,
2605 target: &Target,
2606 path: PathBuf,
2607 mut output_options: OutputOptions,
2608) -> Work {
2609 let target = target.clone();
2610 Work::new(move |state| {
2611 if !path.exists() {
2612 return Ok(());
2614 }
2615 let file = paths::open(&path)?;
2619 let mut reader = std::io::BufReader::new(file);
2620 let mut line = String::new();
2621 loop {
2622 let length = reader.read_line(&mut line)?;
2623 if length == 0 {
2624 break;
2625 }
2626 let trimmed = line.trim_end_matches(&['\n', '\r'][..]);
2627 on_stderr_line(
2628 state,
2629 trimmed,
2630 package_id,
2631 &manifest,
2632 &target,
2633 &mut output_options,
2634 )?;
2635 line.clear();
2636 }
2637 Ok(())
2638 })
2639}
2640
2641fn descriptive_pkg_name(name: &str, target: &Target, mode: &CompileMode) -> String {
2644 let desc_name = target.description_named();
2645 let mode = if mode.is_rustc_test() && !(target.is_test() || target.is_bench()) {
2646 " test"
2647 } else if mode.is_doc_test() {
2648 " doctest"
2649 } else if mode.is_doc() {
2650 " doc"
2651 } else {
2652 ""
2653 };
2654 format!("`{name}` ({desc_name}{mode})")
2655}
2656
2657pub(crate) fn apply_env_config(
2659 gctx: &crate::GlobalContext,
2660 cmd: &mut ProcessBuilder,
2661) -> CargoResult<()> {
2662 for (key, value) in gctx.env_config()?.iter() {
2663 if cmd.get_envs().contains_key(key) {
2665 continue;
2666 }
2667 cmd.env(key, value);
2668 }
2669 Ok(())
2670}
2671
2672fn should_include_scrape_units(bcx: &BuildContext<'_, '_>, unit: &Unit) -> bool {
2674 unit.mode.is_doc() && bcx.scrape_units.len() > 0 && bcx.ws.unit_needs_doc_scrape(unit)
2675}
2676
2677fn scrape_output_path(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> CargoResult<PathBuf> {
2679 assert!(unit.mode.is_doc() || unit.mode.is_doc_scrape());
2680 build_runner
2681 .outputs(unit)
2682 .map(|outputs| outputs[0].path.clone())
2683}
2684
2685fn rustdoc_dep_info_loc(build_runner: &BuildRunner<'_, '_>, unit: &Unit) -> PathBuf {
2687 let mut loc = build_runner.files().fingerprint_file_path(unit, "");
2688 loc.set_extension("d");
2689 loc
2690}