diff --git a/src/cargo/ops/cargo_compile/compile_filter.rs b/src/cargo/ops/cargo_compile/compile_filter.rs index 2cbb149aba2..3615529382f 100644 --- a/src/cargo/ops/cargo_compile/compile_filter.rs +++ b/src/cargo/ops/cargo_compile/compile_filter.rs @@ -28,11 +28,11 @@ pub enum FilterRule { /// Filter to apply to the root package to select which Cargo targets will be built. /// (examples, bins, benches, tests, ...) /// -/// The actual filter process happens inside [`generate_targets`]. +/// The actual filter process happens inside [`generate_units`]. /// /// Not to be confused with [`Packages`], which opts in packages to be built. /// -/// [`generate_targets`]: super::generate_targets +/// [`generate_units`]: super::UnitGenerator::generate_units /// [`Packages`]: crate::ops::Packages #[derive(Debug)] pub enum CompileFilter { @@ -176,7 +176,7 @@ impl CompileFilter { /// may include additional example targets to ensure they can be compiled. /// /// Note that the actual behavior is subject to `filter_default_targets` - /// and `generate_targets` though. + /// and `generate_units` though. pub fn all_test_targets() -> Self { Self::Only { all_targets: false, @@ -234,7 +234,7 @@ impl CompileFilter { } /// Selects targets for "cargo run". for logic to select targets for other - /// subcommands, see `generate_targets` and `filter_default_targets`. + /// subcommands, see `generate_units` and `filter_default_targets`. pub fn target_run(&self, target: &Target) -> bool { match *self { CompileFilter::Default { .. } => true, diff --git a/src/cargo/ops/cargo_compile/mod.rs b/src/cargo/ops/cargo_compile/mod.rs index b451b3b734d..7c2ebb080d6 100644 --- a/src/cargo/ops/cargo_compile/mod.rs +++ b/src/cargo/ops/cargo_compile/mod.rs @@ -7,10 +7,10 @@ //! rough outline is: //! //! - Resolve the dependency graph (see [`ops::resolve`]). -//! - Download any packages needed (see [`PackageSet`]). +//! - Download any packages needed (see [`PackageSet`](crate::core::PackageSet)). //! - Generate a list of top-level "units" of work for the targets the user //! requested on the command-line. Each [`Unit`] corresponds to a compiler -//! invocation. This is done in this module ([`generate_targets`]). +//! invocation. This is done in this module ([`UnitGenerator::generate_units`]). //! - Build the graph of `Unit` dependencies (see [`unit_dependencies`]). //! - Create a [`Context`] which will perform the following steps: //! - Prepare the `target` directory (see [`Layout`]). @@ -30,36 +30,35 @@ //! ["Cargo Target"]: https://doc.rust-lang.org/nightly/cargo/reference/cargo-targets.html use std::collections::{HashMap, HashSet}; -use std::fmt::Write; use std::hash::{Hash, Hasher}; use std::sync::Arc; use crate::core::compiler::rustdoc::RustdocScrapeExamples; -use crate::core::compiler::unit_dependencies::{build_unit_dependencies, IsArtifact}; +use crate::core::compiler::unit_dependencies::build_unit_dependencies; use crate::core::compiler::unit_graph::{self, UnitDep, UnitGraph}; use crate::core::compiler::{standard_lib, CrateType, TargetInfo}; use crate::core::compiler::{BuildConfig, BuildContext, Compilation, Context}; use crate::core::compiler::{CompileKind, CompileMode, CompileTarget, RustcTargetData, Unit}; use crate::core::compiler::{DefaultExecutor, Executor, UnitInterner}; -use crate::core::dependency::DepKind; -use crate::core::profiles::{Profiles, UnitFor}; +use crate::core::profiles::Profiles; use crate::core::resolver::features::{self, CliFeatures, FeaturesFor}; use crate::core::resolver::{HasDevUnits, Resolve}; -use crate::core::{FeatureValue, Package, PackageSet, Shell, Summary, Target}; -use crate::core::{PackageId, SourceId, TargetKind, Workspace}; +use crate::core::{PackageId, PackageSet, SourceId, TargetKind, Workspace}; use crate::drop_println; use crate::ops; use crate::ops::resolve::WorkspaceResolve; use crate::util::config::Config; use crate::util::interning::InternedString; -use crate::util::restricted_names::is_glob_pattern; -use crate::util::{closest_msg, profile, CargoResult, StableHasher}; +use crate::util::{profile, CargoResult, StableHasher}; mod compile_filter; pub use compile_filter::{CompileFilter, FilterRule, LibRule}; +mod unit_generator; +use unit_generator::UnitGenerator; + mod packages; -use packages::build_glob; + pub use packages::Packages; /// Contains information about how a package should be compiled. @@ -345,25 +344,26 @@ pub fn create_bcx<'a, 'cfg>( .collect(); // Passing `build_config.requested_kinds` instead of - // `explicit_host_kinds` here so that `generate_targets` can do + // `explicit_host_kinds` here so that `generate_units` can do // its own special handling of `CompileKind::Host`. It will // internally replace the host kind by the `explicit_host_kind` // before setting as a unit. - let mut units = generate_targets( + let generator = UnitGenerator { ws, - &to_builds, + packages: &to_builds, filter, - &build_config.requested_kinds, + requested_kinds: &build_config.requested_kinds, explicit_host_kind, - build_config.mode, - &resolve, - &workspace_resolve, - &resolved_features, - &pkg_set, - &profiles, + mode: build_config.mode, + resolve: &resolve, + workspace_resolve: &workspace_resolve, + resolved_features: &resolved_features, + package_set: &pkg_set, + profiles: &profiles, interner, has_dev_units, - )?; + }; + let mut units = generator.generate_units()?; if let Some(args) = target_rustc_crate_types { override_rustc_crate_types(&mut units, args, interner)?; @@ -371,21 +371,11 @@ pub fn create_bcx<'a, 'cfg>( let should_scrape = build_config.mode.is_doc() && config.cli_unstable().rustdoc_scrape_examples; let mut scrape_units = if should_scrape { - let all_units = generate_targets( - ws, - &to_builds, - &filter, - &build_config.requested_kinds, - explicit_host_kind, - CompileMode::Docscrape, - &resolve, - &workspace_resolve, - &resolved_features, - &pkg_set, - &profiles, - interner, - has_dev_units, - )?; + let scrape_generator = UnitGenerator { + mode: CompileMode::Docscrape, + ..generator + }; + let all_units = scrape_generator.generate_units()?; let valid_units = all_units .into_iter() @@ -560,683 +550,6 @@ pub fn create_bcx<'a, 'cfg>( Ok(bcx) } -/// A proposed target. -/// -/// Proposed targets are later filtered into actual `Unit`s based on whether or -/// not the target requires its features to be present. -#[derive(Debug)] -struct Proposal<'a> { - pkg: &'a Package, - target: &'a Target, - /// Indicates whether or not all required features *must* be present. If - /// false, and the features are not available, then it will be silently - /// skipped. Generally, targets specified by name (`--bin foo`) are - /// required, all others can be silently skipped if features are missing. - requires_features: bool, - mode: CompileMode, -} - -/// Generates all the base targets for the packages the user has requested to -/// compile. Dependencies for these targets are computed later in `unit_dependencies`. -fn generate_targets( - ws: &Workspace<'_>, - packages: &[&Package], - filter: &CompileFilter, - requested_kinds: &[CompileKind], - explicit_host_kind: CompileKind, - mode: CompileMode, - resolve: &Resolve, - workspace_resolve: &Option, - resolved_features: &features::ResolvedFeatures, - package_set: &PackageSet<'_>, - profiles: &Profiles, - interner: &UnitInterner, - has_dev_units: HasDevUnits, -) -> CargoResult> { - let config = ws.config(); - // Helper for creating a list of `Unit` structures - let new_unit = |units: &mut HashSet, - pkg: &Package, - target: &Target, - initial_target_mode: CompileMode| { - // Custom build units are added in `build_unit_dependencies`. - assert!(!target.is_custom_build()); - let target_mode = match initial_target_mode { - CompileMode::Test => { - if target.is_example() && !filter.is_specific() && !target.tested() { - // Examples are included as regular binaries to verify - // that they compile. - CompileMode::Build - } else { - CompileMode::Test - } - } - CompileMode::Build => match *target.kind() { - TargetKind::Test => CompileMode::Test, - TargetKind::Bench => CompileMode::Bench, - _ => CompileMode::Build, - }, - // `CompileMode::Bench` is only used to inform `filter_default_targets` - // which command is being used (`cargo bench`). Afterwards, tests - // and benches are treated identically. Switching the mode allows - // de-duplication of units that are essentially identical. For - // example, `cargo build --all-targets --release` creates the units - // (lib profile:bench, mode:test) and (lib profile:bench, mode:bench) - // and since these are the same, we want them to be de-duplicated in - // `unit_dependencies`. - CompileMode::Bench => CompileMode::Test, - _ => initial_target_mode, - }; - - let is_local = pkg.package_id().source_id().is_path(); - - // No need to worry about build-dependencies, roots are never build dependencies. - let features_for = FeaturesFor::from_for_host(target.proc_macro()); - let features = resolved_features.activated_features(pkg.package_id(), features_for); - - // If `--target` has not been specified, then the unit - // graph is built almost like if `--target $HOST` was - // specified. See `rebuild_unit_graph_shared` for more on - // why this is done. However, if the package has its own - // `package.target` key, then this gets used instead of - // `$HOST` - let explicit_kinds = if let Some(k) = pkg.manifest().forced_kind() { - vec![k] - } else { - requested_kinds - .iter() - .map(|kind| match kind { - CompileKind::Host => { - pkg.manifest().default_kind().unwrap_or(explicit_host_kind) - } - CompileKind::Target(t) => CompileKind::Target(*t), - }) - .collect() - }; - - for kind in explicit_kinds.iter() { - let unit_for = if initial_target_mode.is_any_test() { - // NOTE: the `UnitFor` here is subtle. If you have a profile - // with `panic` set, the `panic` flag is cleared for - // tests/benchmarks and their dependencies. If this - // was `normal`, then the lib would get compiled three - // times (once with panic, once without, and once with - // `--test`). - // - // This would cause a problem for doc tests, which would fail - // because `rustdoc` would attempt to link with both libraries - // at the same time. Also, it's probably not important (or - // even desirable?) for rustdoc to link with a lib with - // `panic` set. - // - // As a consequence, Examples and Binaries get compiled - // without `panic` set. This probably isn't a bad deal. - // - // Forcing the lib to be compiled three times during `cargo - // test` is probably also not desirable. - UnitFor::new_test(config, *kind) - } else if target.for_host() { - // Proc macro / plugin should not have `panic` set. - UnitFor::new_compiler(*kind) - } else { - UnitFor::new_normal(*kind) - }; - let profile = profiles.get_profile( - pkg.package_id(), - ws.is_member(pkg), - is_local, - unit_for, - *kind, - ); - let unit = interner.intern( - pkg, - target, - profile, - kind.for_target(target), - target_mode, - features.clone(), - /*is_std*/ false, - /*dep_hash*/ 0, - IsArtifact::No, - ); - units.insert(unit); - } - }; - - // Create a list of proposed targets. - let mut proposals: Vec> = Vec::new(); - - match *filter { - CompileFilter::Default { - required_features_filterable, - } => { - for pkg in packages { - let default = filter_default_targets(pkg.targets(), mode); - proposals.extend(default.into_iter().map(|target| Proposal { - pkg, - target, - requires_features: !required_features_filterable, - mode, - })); - if mode == CompileMode::Test { - if let Some(t) = pkg - .targets() - .iter() - .find(|t| t.is_lib() && t.doctested() && t.doctestable()) - { - proposals.push(Proposal { - pkg, - target: t, - requires_features: false, - mode: CompileMode::Doctest, - }); - } - } - } - } - CompileFilter::Only { - all_targets, - ref lib, - ref bins, - ref examples, - ref tests, - ref benches, - } => { - if *lib != LibRule::False { - let mut libs = Vec::new(); - for proposal in filter_targets(packages, Target::is_lib, false, mode) { - let Proposal { target, pkg, .. } = proposal; - if mode.is_doc_test() && !target.doctestable() { - let types = target.rustc_crate_types(); - let types_str: Vec<&str> = types.iter().map(|t| t.as_str()).collect(); - ws.config().shell().warn(format!( - "doc tests are not supported for crate type(s) `{}` in package `{}`", - types_str.join(", "), - pkg.name() - ))?; - } else { - libs.push(proposal) - } - } - if !all_targets && libs.is_empty() && *lib == LibRule::True { - let names = packages.iter().map(|pkg| pkg.name()).collect::>(); - if names.len() == 1 { - anyhow::bail!("no library targets found in package `{}`", names[0]); - } else { - anyhow::bail!("no library targets found in packages: {}", names.join(", ")); - } - } - proposals.extend(libs); - } - - // If `--tests` was specified, add all targets that would be - // generated by `cargo test`. - let test_filter = match tests { - FilterRule::All => Target::tested, - FilterRule::Just(_) => Target::is_test, - }; - let test_mode = match mode { - CompileMode::Build => CompileMode::Test, - CompileMode::Check { .. } => CompileMode::Check { test: true }, - _ => mode, - }; - // If `--benches` was specified, add all targets that would be - // generated by `cargo bench`. - let bench_filter = match benches { - FilterRule::All => Target::benched, - FilterRule::Just(_) => Target::is_bench, - }; - let bench_mode = match mode { - CompileMode::Build => CompileMode::Bench, - CompileMode::Check { .. } => CompileMode::Check { test: true }, - _ => mode, - }; - - proposals.extend(list_rule_targets( - packages, - bins, - "bin", - Target::is_bin, - mode, - )?); - proposals.extend(list_rule_targets( - packages, - examples, - "example", - Target::is_example, - mode, - )?); - proposals.extend(list_rule_targets( - packages, - tests, - "test", - test_filter, - test_mode, - )?); - proposals.extend(list_rule_targets( - packages, - benches, - "bench", - bench_filter, - bench_mode, - )?); - } - } - - if mode.is_doc_scrape() { - // In general, the goal is to scrape examples from (a) whatever targets - // the user is documenting, and (b) Example targets. However, if the user - // is documenting a library with dev-dependencies, those dev-deps are not - // needed for the library, while dev-deps are needed for the examples. - // - // If scrape-examples caused `cargo doc` to start requiring dev-deps, this - // would be a breaking change to crates whose dev-deps don't compile. - // Therefore we ONLY want to scrape Example targets if either: - // (1) No package has dev-dependencies, so this is a moot issue, OR - // (2) The provided CompileFilter requires dev-dependencies anyway. - // - // The next two variables represent these two conditions. - let no_pkg_has_dev_deps = packages.iter().all(|pkg| { - pkg.summary() - .dependencies() - .iter() - .all(|dep| !matches!(dep.kind(), DepKind::Development)) - }); - let reqs_dev_deps = matches!(has_dev_units, HasDevUnits::Yes); - let safe_to_scrape_example_targets = no_pkg_has_dev_deps || reqs_dev_deps; - - let proposed_targets: HashSet<&Target> = proposals.iter().map(|p| p.target).collect(); - let can_scrape = |target: &Target| { - let not_redundant = !proposed_targets.contains(target); - not_redundant - && match (target.doc_scrape_examples(), target.is_example()) { - // Targets configured by the user to not be scraped should never be scraped - (RustdocScrapeExamples::Disabled, _) => false, - // Targets configured by the user to be scraped should always be scraped - (RustdocScrapeExamples::Enabled, _) => true, - // Example targets with no configuration should be conditionally scraped if - // it's guaranteed not to break the build - (RustdocScrapeExamples::Unset, true) => safe_to_scrape_example_targets, - // All other targets are ignored for now. This may change in the future! - (RustdocScrapeExamples::Unset, false) => false, - } - }; - proposals.extend(filter_targets( - packages, - can_scrape, - false, - CompileMode::Docscrape, - )); - } - - // Only include targets that are libraries or have all required - // features available. - // - // `features_map` is a map of &Package -> enabled_features - // It is computed by the set of enabled features for the package plus - // every enabled feature of every enabled dependency. - let mut features_map = HashMap::new(); - // This needs to be a set to de-duplicate units. Due to the way the - // targets are filtered, it is possible to have duplicate proposals for - // the same thing. - let mut units = HashSet::new(); - for Proposal { - pkg, - target, - requires_features, - mode, - } in proposals - { - let unavailable_features = match target.required_features() { - Some(rf) => { - validate_required_features( - workspace_resolve, - target.name(), - rf, - pkg.summary(), - &mut config.shell(), - )?; - - let features = features_map.entry(pkg).or_insert_with(|| { - resolve_all_features(resolve, resolved_features, package_set, pkg.package_id()) - }); - rf.iter().filter(|f| !features.contains(*f)).collect() - } - None => Vec::new(), - }; - if target.is_lib() || unavailable_features.is_empty() { - new_unit(&mut units, pkg, target, mode); - } else if requires_features { - let required_features = target.required_features().unwrap(); - let quoted_required_features: Vec = required_features - .iter() - .map(|s| format!("`{}`", s)) - .collect(); - anyhow::bail!( - "target `{}` in package `{}` requires the features: {}\n\ - Consider enabling them by passing, e.g., `--features=\"{}\"`", - target.name(), - pkg.name(), - quoted_required_features.join(", "), - required_features.join(" ") - ); - } - // else, silently skip target. - } - let mut units: Vec<_> = units.into_iter().collect(); - unmatched_target_filters(&units, filter, &mut ws.config().shell())?; - - // Keep the roots in a consistent order, which helps with checking test output. - units.sort_unstable(); - Ok(units) -} - -/// Checks if the unit list is empty and the user has passed any combination of -/// --tests, --examples, --benches or --bins, and we didn't match on any targets. -/// We want to emit a warning to make sure the user knows that this run is a no-op, -/// and their code remains unchecked despite cargo not returning any errors -fn unmatched_target_filters( - units: &[Unit], - filter: &CompileFilter, - shell: &mut Shell, -) -> CargoResult<()> { - if let CompileFilter::Only { - all_targets, - lib: _, - ref bins, - ref examples, - ref tests, - ref benches, - } = *filter - { - if units.is_empty() { - let mut filters = String::new(); - let mut miss_count = 0; - - let mut append = |t: &FilterRule, s| { - if let FilterRule::All = *t { - miss_count += 1; - filters.push_str(s); - } - }; - - if all_targets { - filters.push_str(" `all-targets`"); - } else { - append(bins, " `bins`,"); - append(tests, " `tests`,"); - append(examples, " `examples`,"); - append(benches, " `benches`,"); - filters.pop(); - } - - return shell.warn(format!( - "Target {}{} specified, but no targets matched. This is a no-op", - if miss_count > 1 { "filters" } else { "filter" }, - filters, - )); - } - } - - Ok(()) -} - -/// Warns if a target's required-features references a feature that doesn't exist. -/// -/// This is a warning because historically this was not validated, and it -/// would cause too much breakage to make it an error. -fn validate_required_features( - resolve: &Option, - target_name: &str, - required_features: &[String], - summary: &Summary, - shell: &mut Shell, -) -> CargoResult<()> { - let resolve = match resolve { - None => return Ok(()), - Some(resolve) => resolve, - }; - - for feature in required_features { - let fv = FeatureValue::new(feature.into()); - match &fv { - FeatureValue::Feature(f) => { - if !summary.features().contains_key(f) { - shell.warn(format!( - "invalid feature `{}` in required-features of target `{}`: \ - `{}` is not present in [features] section", - fv, target_name, fv - ))?; - } - } - FeatureValue::Dep { .. } => { - anyhow::bail!( - "invalid feature `{}` in required-features of target `{}`: \ - `dep:` prefixed feature values are not allowed in required-features", - fv, - target_name - ); - } - FeatureValue::DepFeature { weak: true, .. } => { - anyhow::bail!( - "invalid feature `{}` in required-features of target `{}`: \ - optional dependency with `?` is not allowed in required-features", - fv, - target_name - ); - } - // Handling of dependent_crate/dependent_crate_feature syntax - FeatureValue::DepFeature { - dep_name, - dep_feature, - weak: false, - } => { - match resolve - .deps(summary.package_id()) - .find(|(_dep_id, deps)| deps.iter().any(|dep| dep.name_in_toml() == *dep_name)) - { - Some((dep_id, _deps)) => { - let dep_summary = resolve.summary(dep_id); - if !dep_summary.features().contains_key(dep_feature) - && !dep_summary - .dependencies() - .iter() - .any(|dep| dep.name_in_toml() == *dep_feature && dep.is_optional()) - { - shell.warn(format!( - "invalid feature `{}` in required-features of target `{}`: \ - feature `{}` does not exist in package `{}`", - fv, target_name, dep_feature, dep_id - ))?; - } - } - None => { - shell.warn(format!( - "invalid feature `{}` in required-features of target `{}`: \ - dependency `{}` does not exist", - fv, target_name, dep_name - ))?; - } - } - } - } - } - Ok(()) -} - -/// Gets all of the features enabled for a package, plus its dependencies' -/// features. -/// -/// Dependencies are added as `dep_name/feat_name` because `required-features` -/// wants to support that syntax. -pub fn resolve_all_features( - resolve_with_overrides: &Resolve, - resolved_features: &features::ResolvedFeatures, - package_set: &PackageSet<'_>, - package_id: PackageId, -) -> HashSet { - let mut features: HashSet = resolved_features - .activated_features(package_id, FeaturesFor::NormalOrDev) - .iter() - .map(|s| s.to_string()) - .collect(); - - // Include features enabled for use by dependencies so targets can also use them with the - // required-features field when deciding whether to be built or skipped. - for (dep_id, deps) in resolve_with_overrides.deps(package_id) { - let is_proc_macro = package_set - .get_one(dep_id) - .expect("packages downloaded") - .proc_macro(); - for dep in deps { - let features_for = FeaturesFor::from_for_host(is_proc_macro || dep.is_build()); - for feature in resolved_features - .activated_features_unverified(dep_id, features_for) - .unwrap_or_default() - { - features.insert(format!("{}/{}", dep.name_in_toml(), feature)); - } - } - } - - features -} - -/// Given a list of all targets for a package, filters out only the targets -/// that are automatically included when the user doesn't specify any targets. -fn filter_default_targets(targets: &[Target], mode: CompileMode) -> Vec<&Target> { - match mode { - CompileMode::Bench => targets.iter().filter(|t| t.benched()).collect(), - CompileMode::Test => targets - .iter() - .filter(|t| t.tested() || t.is_example()) - .collect(), - CompileMode::Build | CompileMode::Check { .. } => targets - .iter() - .filter(|t| t.is_bin() || t.is_lib()) - .collect(), - CompileMode::Doc { .. } | CompileMode::Docscrape => { - // `doc` does lib and bins (bin with same name as lib is skipped). - targets - .iter() - .filter(|t| { - t.documented() - && (!t.is_bin() - || !targets.iter().any(|l| l.is_lib() && l.name() == t.name())) - }) - .collect() - } - CompileMode::Doctest | CompileMode::RunCustomBuild => { - panic!("Invalid mode {:?}", mode) - } - } -} - -/// Returns a list of proposed targets based on command-line target selection flags. -fn list_rule_targets<'a>( - packages: &[&'a Package], - rule: &FilterRule, - target_desc: &'static str, - is_expected_kind: fn(&Target) -> bool, - mode: CompileMode, -) -> CargoResult>> { - let mut proposals = Vec::new(); - match rule { - FilterRule::All => { - proposals.extend(filter_targets(packages, is_expected_kind, false, mode)) - } - FilterRule::Just(names) => { - for name in names { - proposals.extend(find_named_targets( - packages, - name, - target_desc, - is_expected_kind, - mode, - )?); - } - } - } - Ok(proposals) -} - -/// Finds the targets for a specifically named target. -fn find_named_targets<'a>( - packages: &[&'a Package], - target_name: &str, - target_desc: &'static str, - is_expected_kind: fn(&Target) -> bool, - mode: CompileMode, -) -> CargoResult>> { - let is_glob = is_glob_pattern(target_name); - let proposals = if is_glob { - let pattern = build_glob(target_name)?; - let filter = |t: &Target| is_expected_kind(t) && pattern.matches(t.name()); - filter_targets(packages, filter, true, mode) - } else { - let filter = |t: &Target| t.name() == target_name && is_expected_kind(t); - filter_targets(packages, filter, true, mode) - }; - - if proposals.is_empty() { - let targets = packages - .iter() - .flat_map(|pkg| { - pkg.targets() - .iter() - .filter(|target| is_expected_kind(target)) - }) - .collect::>(); - let suggestion = closest_msg(target_name, targets.iter(), |t| t.name()); - if !suggestion.is_empty() { - anyhow::bail!( - "no {} target {} `{}`{}", - target_desc, - if is_glob { "matches pattern" } else { "named" }, - target_name, - suggestion - ); - } else { - let mut msg = String::new(); - writeln!( - msg, - "no {} target {} `{}`.", - target_desc, - if is_glob { "matches pattern" } else { "named" }, - target_name, - )?; - if !targets.is_empty() { - writeln!(msg, "Available {} targets:", target_desc)?; - for target in targets { - writeln!(msg, " {}", target.name())?; - } - } - anyhow::bail!(msg); - } - } - Ok(proposals) -} - -fn filter_targets<'a>( - packages: &[&'a Package], - predicate: impl Fn(&Target) -> bool, - requires_features: bool, - mode: CompileMode, -) -> Vec> { - let mut proposals = Vec::new(); - for pkg in packages { - for target in pkg.targets().iter().filter(|t| predicate(t)) { - proposals.push(Proposal { - pkg, - target, - requires_features, - mode, - }); - } - } - proposals -} - /// This is used to rebuild the unit graph, sharing host dependencies if possible. /// /// This will translate any unit's `CompileKind::Target(host)` to @@ -1503,3 +816,41 @@ fn override_rustc_crate_types( Ok(()) } + +/// Gets all of the features enabled for a package, plus its dependencies' +/// features. +/// +/// Dependencies are added as `dep_name/feat_name` because `required-features` +/// wants to support that syntax. +pub fn resolve_all_features( + resolve_with_overrides: &Resolve, + resolved_features: &features::ResolvedFeatures, + package_set: &PackageSet<'_>, + package_id: PackageId, +) -> HashSet { + let mut features: HashSet = resolved_features + .activated_features(package_id, FeaturesFor::NormalOrDev) + .iter() + .map(|s| s.to_string()) + .collect(); + + // Include features enabled for use by dependencies so targets can also use them with the + // required-features field when deciding whether to be built or skipped. + for (dep_id, deps) in resolve_with_overrides.deps(package_id) { + let is_proc_macro = package_set + .get_one(dep_id) + .expect("packages downloaded") + .proc_macro(); + for dep in deps { + let features_for = FeaturesFor::from_for_host(is_proc_macro || dep.is_build()); + for feature in resolved_features + .activated_features_unverified(dep_id, features_for) + .unwrap_or_default() + { + features.insert(format!("{}/{}", dep.name_in_toml(), feature)); + } + } + } + + features +} diff --git a/src/cargo/ops/cargo_compile/unit_generator.rs b/src/cargo/ops/cargo_compile/unit_generator.rs new file mode 100644 index 00000000000..24016dd0dc4 --- /dev/null +++ b/src/cargo/ops/cargo_compile/unit_generator.rs @@ -0,0 +1,666 @@ +use std::collections::{HashMap, HashSet}; +use std::fmt::Write; + +use crate::core::compiler::rustdoc::RustdocScrapeExamples; +use crate::core::compiler::unit_dependencies::IsArtifact; +use crate::core::compiler::UnitInterner; +use crate::core::compiler::{CompileKind, CompileMode, Unit}; +use crate::core::dependency::DepKind; +use crate::core::profiles::{Profiles, UnitFor}; +use crate::core::resolver::features::{self, FeaturesFor}; +use crate::core::resolver::{HasDevUnits, Resolve}; +use crate::core::{FeatureValue, Package, PackageSet, Summary, Target}; +use crate::core::{TargetKind, Workspace}; +use crate::util::restricted_names::is_glob_pattern; +use crate::util::{closest_msg, CargoResult}; + +use super::compile_filter::{CompileFilter, FilterRule, LibRule}; +use super::packages::build_glob; + +/// A proposed target. +/// +/// Proposed targets are later filtered into actual `Unit`s based on whether or +/// not the target requires its features to be present. +#[derive(Debug)] +struct Proposal<'a> { + pkg: &'a Package, + target: &'a Target, + /// Indicates whether or not all required features *must* be present. If + /// false, and the features are not available, then it will be silently + /// skipped. Generally, targets specified by name (`--bin foo`) are + /// required, all others can be silently skipped if features are missing. + requires_features: bool, + mode: CompileMode, +} + +/// The context needed for generating units. +pub(super) struct UnitGenerator<'a, 'cfg> { + pub ws: &'a Workspace<'cfg>, + pub packages: &'a [&'a Package], + pub filter: &'a CompileFilter, + pub requested_kinds: &'a [CompileKind], + pub explicit_host_kind: CompileKind, + pub mode: CompileMode, + pub resolve: &'a Resolve, + pub workspace_resolve: &'a Option, + pub resolved_features: &'a features::ResolvedFeatures, + pub package_set: &'a PackageSet<'cfg>, + pub profiles: &'a Profiles, + pub interner: &'a UnitInterner, + pub has_dev_units: HasDevUnits, +} + +impl<'a> UnitGenerator<'a, '_> { + /// Helper for creating a list of `Unit` structures + fn new_units( + &self, + pkg: &Package, + target: &Target, + initial_target_mode: CompileMode, + ) -> Vec { + // Custom build units are added in `build_unit_dependencies`. + assert!(!target.is_custom_build()); + let target_mode = match initial_target_mode { + CompileMode::Test => { + if target.is_example() && !self.filter.is_specific() && !target.tested() { + // Examples are included as regular binaries to verify + // that they compile. + CompileMode::Build + } else { + CompileMode::Test + } + } + CompileMode::Build => match *target.kind() { + TargetKind::Test => CompileMode::Test, + TargetKind::Bench => CompileMode::Bench, + _ => CompileMode::Build, + }, + // `CompileMode::Bench` is only used to inform `filter_default_targets` + // which command is being used (`cargo bench`). Afterwards, tests + // and benches are treated identically. Switching the mode allows + // de-duplication of units that are essentially identical. For + // example, `cargo build --all-targets --release` creates the units + // (lib profile:bench, mode:test) and (lib profile:bench, mode:bench) + // and since these are the same, we want them to be de-duplicated in + // `unit_dependencies`. + CompileMode::Bench => CompileMode::Test, + _ => initial_target_mode, + }; + + let is_local = pkg.package_id().source_id().is_path(); + + // No need to worry about build-dependencies, roots are never build dependencies. + let features_for = FeaturesFor::from_for_host(target.proc_macro()); + let features = self + .resolved_features + .activated_features(pkg.package_id(), features_for); + + // If `--target` has not been specified, then the unit + // graph is built almost like if `--target $HOST` was + // specified. See `rebuild_unit_graph_shared` for more on + // why this is done. However, if the package has its own + // `package.target` key, then this gets used instead of + // `$HOST` + let explicit_kinds = if let Some(k) = pkg.manifest().forced_kind() { + vec![k] + } else { + self.requested_kinds + .iter() + .map(|kind| match kind { + CompileKind::Host => pkg + .manifest() + .default_kind() + .unwrap_or(self.explicit_host_kind), + CompileKind::Target(t) => CompileKind::Target(*t), + }) + .collect() + }; + + explicit_kinds + .into_iter() + .map(move |kind| { + let unit_for = if initial_target_mode.is_any_test() { + // NOTE: the `UnitFor` here is subtle. If you have a profile + // with `panic` set, the `panic` flag is cleared for + // tests/benchmarks and their dependencies. If this + // was `normal`, then the lib would get compiled three + // times (once with panic, once without, and once with + // `--test`). + // + // This would cause a problem for doc tests, which would fail + // because `rustdoc` would attempt to link with both libraries + // at the same time. Also, it's probably not important (or + // even desirable?) for rustdoc to link with a lib with + // `panic` set. + // + // As a consequence, Examples and Binaries get compiled + // without `panic` set. This probably isn't a bad deal. + // + // Forcing the lib to be compiled three times during `cargo + // test` is probably also not desirable. + UnitFor::new_test(self.ws.config(), kind) + } else if target.for_host() { + // Proc macro / plugin should not have `panic` set. + UnitFor::new_compiler(kind) + } else { + UnitFor::new_normal(kind) + }; + let profile = self.profiles.get_profile( + pkg.package_id(), + self.ws.is_member(pkg), + is_local, + unit_for, + kind, + ); + self.interner.intern( + pkg, + target, + profile, + kind.for_target(target), + target_mode, + features.clone(), + /*is_std*/ false, + /*dep_hash*/ 0, + IsArtifact::No, + ) + }) + .collect() + } + + /// Given a list of all targets for a package, filters out only the targets + /// that are automatically included when the user doesn't specify any targets. + fn filter_default_targets<'b>(&self, targets: &'b [Target]) -> Vec<&'b Target> { + match self.mode { + CompileMode::Bench => targets.iter().filter(|t| t.benched()).collect(), + CompileMode::Test => targets + .iter() + .filter(|t| t.tested() || t.is_example()) + .collect(), + CompileMode::Build | CompileMode::Check { .. } => targets + .iter() + .filter(|t| t.is_bin() || t.is_lib()) + .collect(), + CompileMode::Doc { .. } | CompileMode::Docscrape => { + // `doc` does lib and bins (bin with same name as lib is skipped). + targets + .iter() + .filter(|t| { + t.documented() + && (!t.is_bin() + || !targets.iter().any(|l| l.is_lib() && l.name() == t.name())) + }) + .collect() + } + CompileMode::Doctest | CompileMode::RunCustomBuild => { + panic!("Invalid mode {:?}", self.mode) + } + } + } + + /// Filters the set of all possible targets based on the provided predicate. + fn filter_targets( + &self, + predicate: impl Fn(&Target) -> bool, + requires_features: bool, + mode: CompileMode, + ) -> Vec> { + self.packages + .iter() + .flat_map(|pkg| { + pkg.targets() + .iter() + .filter(|t| predicate(t)) + .map(|target| Proposal { + pkg, + target, + requires_features, + mode, + }) + }) + .collect() + } + + /// Finds the targets for a specifically named target. + fn find_named_targets( + &self, + target_name: &str, + target_desc: &'static str, + is_expected_kind: fn(&Target) -> bool, + mode: CompileMode, + ) -> CargoResult>> { + let is_glob = is_glob_pattern(target_name); + let proposals = if is_glob { + let pattern = build_glob(target_name)?; + let filter = |t: &Target| is_expected_kind(t) && pattern.matches(t.name()); + self.filter_targets(filter, true, mode) + } else { + let filter = |t: &Target| t.name() == target_name && is_expected_kind(t); + self.filter_targets(filter, true, mode) + }; + + if proposals.is_empty() { + let targets = self + .packages + .iter() + .flat_map(|pkg| { + pkg.targets() + .iter() + .filter(|target| is_expected_kind(target)) + }) + .collect::>(); + let suggestion = closest_msg(target_name, targets.iter(), |t| t.name()); + if !suggestion.is_empty() { + anyhow::bail!( + "no {} target {} `{}`{}", + target_desc, + if is_glob { "matches pattern" } else { "named" }, + target_name, + suggestion + ); + } else { + let mut msg = String::new(); + writeln!( + msg, + "no {} target {} `{}`.", + target_desc, + if is_glob { "matches pattern" } else { "named" }, + target_name, + )?; + if !targets.is_empty() { + writeln!(msg, "Available {} targets:", target_desc)?; + for target in targets { + writeln!(msg, " {}", target.name())?; + } + } + anyhow::bail!(msg); + } + } + Ok(proposals) + } + + /// Returns a list of proposed targets based on command-line target selection flags. + fn list_rule_targets( + &self, + rule: &FilterRule, + target_desc: &'static str, + is_expected_kind: fn(&Target) -> bool, + mode: CompileMode, + ) -> CargoResult>> { + let mut proposals = Vec::new(); + match rule { + FilterRule::All => proposals.extend(self.filter_targets(is_expected_kind, false, mode)), + FilterRule::Just(names) => { + for name in names { + proposals.extend(self.find_named_targets( + name, + target_desc, + is_expected_kind, + mode, + )?); + } + } + } + Ok(proposals) + } + + /// Create a list of proposed targets given the context in `TargetGenerator` + fn create_proposals(&self) -> CargoResult>> { + let mut proposals: Vec> = Vec::new(); + + match *self.filter { + CompileFilter::Default { + required_features_filterable, + } => { + for pkg in self.packages { + let default = self.filter_default_targets(pkg.targets()); + proposals.extend(default.into_iter().map(|target| Proposal { + pkg, + target, + requires_features: !required_features_filterable, + mode: self.mode, + })); + if self.mode == CompileMode::Test { + if let Some(t) = pkg + .targets() + .iter() + .find(|t| t.is_lib() && t.doctested() && t.doctestable()) + { + proposals.push(Proposal { + pkg, + target: t, + requires_features: false, + mode: CompileMode::Doctest, + }); + } + } + } + } + CompileFilter::Only { + all_targets, + ref lib, + ref bins, + ref examples, + ref tests, + ref benches, + } => { + if *lib != LibRule::False { + let mut libs = Vec::new(); + for proposal in self.filter_targets(Target::is_lib, false, self.mode) { + let Proposal { target, pkg, .. } = proposal; + if self.mode.is_doc_test() && !target.doctestable() { + let types = target.rustc_crate_types(); + let types_str: Vec<&str> = types.iter().map(|t| t.as_str()).collect(); + self.ws.config().shell().warn(format!( + "doc tests are not supported for crate type(s) `{}` in package `{}`", + types_str.join(", "), + pkg.name() + ))?; + } else { + libs.push(proposal) + } + } + if !all_targets && libs.is_empty() && *lib == LibRule::True { + let names = self + .packages + .iter() + .map(|pkg| pkg.name()) + .collect::>(); + if names.len() == 1 { + anyhow::bail!("no library targets found in package `{}`", names[0]); + } else { + anyhow::bail!( + "no library targets found in packages: {}", + names.join(", ") + ); + } + } + proposals.extend(libs); + } + + // If `--tests` was specified, add all targets that would be + // generated by `cargo test`. + let test_filter = match tests { + FilterRule::All => Target::tested, + FilterRule::Just(_) => Target::is_test, + }; + let test_mode = match self.mode { + CompileMode::Build => CompileMode::Test, + CompileMode::Check { .. } => CompileMode::Check { test: true }, + _ => self.mode, + }; + // If `--benches` was specified, add all targets that would be + // generated by `cargo bench`. + let bench_filter = match benches { + FilterRule::All => Target::benched, + FilterRule::Just(_) => Target::is_bench, + }; + let bench_mode = match self.mode { + CompileMode::Build => CompileMode::Bench, + CompileMode::Check { .. } => CompileMode::Check { test: true }, + _ => self.mode, + }; + + proposals.extend(self.list_rule_targets(bins, "bin", Target::is_bin, self.mode)?); + proposals.extend(self.list_rule_targets( + examples, + "example", + Target::is_example, + self.mode, + )?); + proposals.extend(self.list_rule_targets(tests, "test", test_filter, test_mode)?); + proposals.extend(self.list_rule_targets( + benches, + "bench", + bench_filter, + bench_mode, + )?); + } + } + + if self.mode.is_doc_scrape() { + self.add_docscrape_proposals(&mut proposals); + } + + Ok(proposals) + } + + /// Add additional targets from which to scrape examples for documentation + fn add_docscrape_proposals(&self, proposals: &mut Vec>) { + // In general, the goal is to scrape examples from (a) whatever targets + // the user is documenting, and (b) Example targets. However, if the user + // is documenting a library with dev-dependencies, those dev-deps are not + // needed for the library, while dev-deps are needed for the examples. + // + // If scrape-examples caused `cargo doc` to start requiring dev-deps, this + // would be a breaking change to crates whose dev-deps don't compile. + // Therefore we ONLY want to scrape Example targets if either: + // (1) No package has dev-dependencies, so this is a moot issue, OR + // (2) The provided CompileFilter requires dev-dependencies anyway. + // + // The next two variables represent these two conditions. + let no_pkg_has_dev_deps = self.packages.iter().all(|pkg| { + pkg.summary() + .dependencies() + .iter() + .all(|dep| !matches!(dep.kind(), DepKind::Development)) + }); + let reqs_dev_deps = matches!(self.has_dev_units, HasDevUnits::Yes); + let safe_to_scrape_example_targets = no_pkg_has_dev_deps || reqs_dev_deps; + + let proposed_targets: HashSet<&Target> = proposals.iter().map(|p| p.target).collect(); + let can_scrape = |target: &Target| { + let not_redundant = !proposed_targets.contains(target); + not_redundant + && match (target.doc_scrape_examples(), target.is_example()) { + // Targets configured by the user to not be scraped should never be scraped + (RustdocScrapeExamples::Disabled, _) => false, + // Targets configured by the user to be scraped should always be scraped + (RustdocScrapeExamples::Enabled, _) => true, + // Example targets with no configuration should be conditionally scraped if + // it's guaranteed not to break the build + (RustdocScrapeExamples::Unset, true) => safe_to_scrape_example_targets, + // All other targets are ignored for now. This may change in the future! + (RustdocScrapeExamples::Unset, false) => false, + } + }; + proposals.extend(self.filter_targets(can_scrape, false, CompileMode::Docscrape)); + } + + /// Checks if the unit list is empty and the user has passed any combination of + /// --tests, --examples, --benches or --bins, and we didn't match on any targets. + /// We want to emit a warning to make sure the user knows that this run is a no-op, + /// and their code remains unchecked despite cargo not returning any errors + fn unmatched_target_filters(&self, units: &[Unit]) -> CargoResult<()> { + let mut shell = self.ws.config().shell(); + if let CompileFilter::Only { + all_targets, + lib: _, + ref bins, + ref examples, + ref tests, + ref benches, + } = *self.filter + { + if units.is_empty() { + let mut filters = String::new(); + let mut miss_count = 0; + + let mut append = |t: &FilterRule, s| { + if let FilterRule::All = *t { + miss_count += 1; + filters.push_str(s); + } + }; + + if all_targets { + filters.push_str(" `all-targets`"); + } else { + append(bins, " `bins`,"); + append(tests, " `tests`,"); + append(examples, " `examples`,"); + append(benches, " `benches`,"); + filters.pop(); + } + + return shell.warn(format!( + "Target {}{} specified, but no targets matched. This is a no-op", + if miss_count > 1 { "filters" } else { "filter" }, + filters, + )); + } + } + + Ok(()) + } + + /// Warns if a target's required-features references a feature that doesn't exist. + /// + /// This is a warning because historically this was not validated, and it + /// would cause too much breakage to make it an error. + fn validate_required_features( + &self, + target_name: &str, + required_features: &[String], + summary: &Summary, + ) -> CargoResult<()> { + let resolve = match self.workspace_resolve { + None => return Ok(()), + Some(resolve) => resolve, + }; + + let mut shell = self.ws.config().shell(); + for feature in required_features { + let fv = FeatureValue::new(feature.into()); + match &fv { + FeatureValue::Feature(f) => { + if !summary.features().contains_key(f) { + shell.warn(format!( + "invalid feature `{}` in required-features of target `{}`: \ + `{}` is not present in [features] section", + fv, target_name, fv + ))?; + } + } + FeatureValue::Dep { .. } => { + anyhow::bail!( + "invalid feature `{}` in required-features of target `{}`: \ + `dep:` prefixed feature values are not allowed in required-features", + fv, + target_name + ); + } + FeatureValue::DepFeature { weak: true, .. } => { + anyhow::bail!( + "invalid feature `{}` in required-features of target `{}`: \ + optional dependency with `?` is not allowed in required-features", + fv, + target_name + ); + } + // Handling of dependent_crate/dependent_crate_feature syntax + FeatureValue::DepFeature { + dep_name, + dep_feature, + weak: false, + } => { + match resolve.deps(summary.package_id()).find(|(_dep_id, deps)| { + deps.iter().any(|dep| dep.name_in_toml() == *dep_name) + }) { + Some((dep_id, _deps)) => { + let dep_summary = resolve.summary(dep_id); + if !dep_summary.features().contains_key(dep_feature) + && !dep_summary.dependencies().iter().any(|dep| { + dep.name_in_toml() == *dep_feature && dep.is_optional() + }) + { + shell.warn(format!( + "invalid feature `{}` in required-features of target `{}`: \ + feature `{}` does not exist in package `{}`", + fv, target_name, dep_feature, dep_id + ))?; + } + } + None => { + shell.warn(format!( + "invalid feature `{}` in required-features of target `{}`: \ + dependency `{}` does not exist", + fv, target_name, dep_name + ))?; + } + } + } + } + } + Ok(()) + } + + /// Converts proposals to units based on each target's required features. + fn proposals_to_units(&self, proposals: Vec>) -> CargoResult> { + // Only include targets that are libraries or have all required + // features available. + // + // `features_map` is a map of &Package -> enabled_features + // It is computed by the set of enabled features for the package plus + // every enabled feature of every enabled dependency. + let mut features_map = HashMap::new(); + // This needs to be a set to de-duplicate units. Due to the way the + // targets are filtered, it is possible to have duplicate proposals for + // the same thing. + let mut units = HashSet::new(); + for Proposal { + pkg, + target, + requires_features, + mode, + } in proposals + { + let unavailable_features = match target.required_features() { + Some(rf) => { + self.validate_required_features(target.name(), rf, pkg.summary())?; + + let features = features_map.entry(pkg).or_insert_with(|| { + super::resolve_all_features( + self.resolve, + self.resolved_features, + self.package_set, + pkg.package_id(), + ) + }); + rf.iter().filter(|f| !features.contains(*f)).collect() + } + None => Vec::new(), + }; + if target.is_lib() || unavailable_features.is_empty() { + units.extend(self.new_units(pkg, target, mode)); + } else if requires_features { + let required_features = target.required_features().unwrap(); + let quoted_required_features: Vec = required_features + .iter() + .map(|s| format!("`{}`", s)) + .collect(); + anyhow::bail!( + "target `{}` in package `{}` requires the features: {}\n\ + Consider enabling them by passing, e.g., `--features=\"{}\"`", + target.name(), + pkg.name(), + quoted_required_features.join(", "), + required_features.join(" ") + ); + } + // else, silently skip target. + } + let mut units: Vec<_> = units.into_iter().collect(); + self.unmatched_target_filters(&units)?; + + // Keep the roots in a consistent order, which helps with checking test output. + units.sort_unstable(); + Ok(units) + } + + /// Generates all the base units for the packages the user has requested to + /// compile. Dependencies for these units are computed later in `unit_dependencies`. + pub fn generate_units(&self) -> CargoResult> { + let proposals = self.create_proposals()?; + self.proposals_to_units(proposals) + } +} diff --git a/src/doc/contrib/src/architecture/compilation.md b/src/doc/contrib/src/architecture/compilation.md index c88d6567a29..ecccee07152 100644 --- a/src/doc/contrib/src/architecture/compilation.md +++ b/src/doc/contrib/src/architecture/compilation.md @@ -9,7 +9,7 @@ module. The compilation can be conceptually broken into these steps: 1. Perform dependency resolution (see [the resolution chapter]). 2. Generate the root `Unit`s, the things the user requested to compile on the - command-line. This is done in [`generate_targets`]. + command-line. This is done in the [`unit_generator`] module. 3. Starting from the root `Unit`s, generate the [`UnitGraph`] by walking the dependency graph from the resolver. The `UnitGraph` contains all of the `Unit` structs, and information about the dependency relationships between @@ -26,8 +26,8 @@ module. The compilation can be conceptually broken into these steps: can be used for various things, such as running tests after the compilation has finished. -[`cargo_compile`]: https://github.com/rust-lang/cargo/blob/master/src/cargo/ops/cargo_compile.rs -[`generate_targets`]: https://github.com/rust-lang/cargo/blob/e4b65bdc80f2a293447f2f6a808fa7c84bf9a357/src/cargo/ops/cargo_compile.rs#L725-L739 +[`cargo_compile`]: https://github.com/rust-lang/cargo/blob/master/src/cargo/ops/cargo_compile/mod.rs +[`unit_generator`]: https://github.com/rust-lang/cargo/blob/master/src/cargo/ops/cargo_compile/unit_generator.rs [`UnitGraph`]: https://github.com/rust-lang/cargo/blob/master/src/cargo/core/compiler/unit_graph.rs [the resolution chapter]: packages.md [`Unit`]: https://github.com/rust-lang/cargo/blob/master/src/cargo/core/compiler/unit.rs diff --git a/src/doc/contrib/src/tests/writing.md b/src/doc/contrib/src/tests/writing.md index bdd105d4582..05a97681d83 100644 --- a/src/doc/contrib/src/tests/writing.md +++ b/src/doc/contrib/src/tests/writing.md @@ -269,7 +269,7 @@ environment. The general process is: * `/path/to/my/cargo/target/debug/cargo check` * Using a debugger like `lldb` or `gdb`: 1. `lldb /path/to/my/cargo/target/debug/cargo` - 2. Set a breakpoint, for example: `b generate_targets` + 2. Set a breakpoint, for example: `b generate_units` 3. Run with arguments: `r check` [`testsuite`]: https://github.com/rust-lang/cargo/tree/master/tests/testsuite/