Skip to content

Commit aa6e46e

Browse files
committed
Auto merge of #13012 - Urgau:check-cfg-fingerprint, r=epage
Include declared list of features in fingerprint for `-Zcheck-cfg` This PR include the declared list of features in fingerprint for `-Zcheck-cfg` (#10554) `--check-cfg` verifies that `#[cfg()]` attributes are valid in Rust code, which includes `--cfg features foo` definitions that came from `[features]` tables in `Cargo.toml`. For us to correctly re-check cfgs on feature changes,we need to include them in the fingerprint. For example, if a user deletes an entry from `[features]`, they should then get warnings about any `#[cfg()]` attributes left in the code. Without this change, that won't happen. With this change, it now does.
2 parents bbd2dcd + f32f43d commit aa6e46e

File tree

3 files changed

+101
-0
lines changed

3 files changed

+101
-0
lines changed

src/cargo/core/compiler/fingerprint/dirty_reason.rs

+7
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,10 @@ pub enum DirtyReason {
1515
old: String,
1616
new: String,
1717
},
18+
DeclaredFeaturesChanged {
19+
old: String,
20+
new: String,
21+
},
1822
TargetConfigurationChanged,
1923
PathToSourceChanged,
2024
ProfileConfigurationChanged,
@@ -141,6 +145,9 @@ impl DirtyReason {
141145
DirtyReason::FeaturesChanged { .. } => {
142146
s.dirty_because(unit, "the list of features changed")
143147
}
148+
DirtyReason::DeclaredFeaturesChanged { .. } => {
149+
s.dirty_because(unit, "the list of declared features changed")
150+
}
144151
DirtyReason::TargetConfigurationChanged => {
145152
s.dirty_because(unit, "the target configuration changed")
146153
}

src/cargo/core/compiler/fingerprint/mod.rs

+23
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,7 @@
6565
//! Target Name | ✓ | ✓
6666
//! TargetKind (bin/lib/etc.) | ✓ | ✓
6767
//! Enabled Features | ✓ | ✓
68+
//! Declared Features | ✓ |
6869
//! Immediate dependency’s hashes | ✓[^1] | ✓
6970
//! [`CompileKind`] (host/target) | ✓ | ✓
7071
//! __CARGO_DEFAULT_LIB_METADATA[^4] | | ✓
@@ -572,6 +573,8 @@ pub struct Fingerprint {
572573
rustc: u64,
573574
/// Sorted list of cfg features enabled.
574575
features: String,
576+
/// Sorted list of all the declared cfg features.
577+
declared_features: String,
575578
/// Hash of the `Target` struct, including the target name,
576579
/// package-relative source path, edition, etc.
577580
target: u64,
@@ -876,6 +879,7 @@ impl Fingerprint {
876879
profile: 0,
877880
path: 0,
878881
features: String::new(),
882+
declared_features: String::new(),
879883
deps: Vec::new(),
880884
local: Mutex::new(Vec::new()),
881885
memoized_hash: Mutex::new(None),
@@ -922,6 +926,12 @@ impl Fingerprint {
922926
new: self.features.clone(),
923927
};
924928
}
929+
if self.declared_features != old.declared_features {
930+
return DirtyReason::DeclaredFeaturesChanged {
931+
old: old.declared_features.clone(),
932+
new: self.declared_features.clone(),
933+
};
934+
}
925935
if self.target != old.target {
926936
return DirtyReason::TargetConfigurationChanged;
927937
}
@@ -1200,6 +1210,7 @@ impl hash::Hash for Fingerprint {
12001210
let Fingerprint {
12011211
rustc,
12021212
ref features,
1213+
ref declared_features,
12031214
target,
12041215
path,
12051216
profile,
@@ -1215,6 +1226,7 @@ impl hash::Hash for Fingerprint {
12151226
(
12161227
rustc,
12171228
features,
1229+
declared_features,
12181230
target,
12191231
path,
12201232
profile,
@@ -1431,6 +1443,9 @@ fn calculate_normal(cx: &mut Context<'_, '_>, unit: &Unit) -> CargoResult<Finger
14311443
allow_features.hash(&mut config);
14321444
}
14331445
let compile_kind = unit.kind.fingerprint_hash();
1446+
let mut declared_features = unit.pkg.summary().features().keys().collect::<Vec<_>>();
1447+
declared_features.sort(); // to avoid useless rebuild if the user orders it's features
1448+
// differently
14341449
Ok(Fingerprint {
14351450
rustc: util::hash_u64(&cx.bcx.rustc().verbose_version),
14361451
target: util::hash_u64(&unit.target),
@@ -1439,6 +1454,14 @@ fn calculate_normal(cx: &mut Context<'_, '_>, unit: &Unit) -> CargoResult<Finger
14391454
// actually affect the output artifact so there's no need to hash it.
14401455
path: util::hash_u64(path_args(cx.bcx.ws, unit).0),
14411456
features: format!("{:?}", unit.features),
1457+
// Note we curently only populate `declared_features` when `-Zcheck-cfg`
1458+
// is passed since it's the only user-facing toggle that will make this
1459+
// fingerprint relevant.
1460+
declared_features: if cx.bcx.config.cli_unstable().check_cfg {
1461+
format!("{declared_features:?}")
1462+
} else {
1463+
"".to_string()
1464+
},
14421465
deps,
14431466
local: Mutex::new(local),
14441467
memoized_hash: Mutex::new(None),

tests/testsuite/check_cfg.rs

+71
Original file line numberDiff line numberDiff line change
@@ -141,6 +141,77 @@ fn features_with_namespaced_features() {
141141
.run();
142142
}
143143

144+
#[cargo_test(nightly, reason = "--check-cfg is unstable")]
145+
fn features_fingerprint() {
146+
let p = project()
147+
.file(
148+
"Cargo.toml",
149+
r#"
150+
[package]
151+
name = "foo"
152+
version = "0.1.0"
153+
154+
[features]
155+
f_a = []
156+
f_b = []
157+
"#,
158+
)
159+
.file("src/lib.rs", "#[cfg(feature = \"f_b\")] fn entry() {}")
160+
.build();
161+
162+
p.cargo("check -v -Zcheck-cfg")
163+
.masquerade_as_nightly_cargo(&["check-cfg"])
164+
.with_stderr_contains(x!("rustc" => "cfg" of "feature" with "f_a" "f_b"))
165+
.with_stderr_does_not_contain("[..]unexpected_cfgs[..]")
166+
.run();
167+
168+
p.cargo("check -v -Zcheck-cfg")
169+
.masquerade_as_nightly_cargo(&["check-cfg"])
170+
.with_stderr_does_not_contain("[..]rustc[..]")
171+
.run();
172+
173+
// checking that re-ordering the features does not invalid the fingerprint
174+
p.change_file(
175+
"Cargo.toml",
176+
r#"
177+
[package]
178+
name = "foo"
179+
version = "0.1.0"
180+
181+
[features]
182+
f_b = []
183+
f_a = []
184+
"#,
185+
);
186+
187+
p.cargo("check -v -Zcheck-cfg")
188+
.masquerade_as_nightly_cargo(&["check-cfg"])
189+
.with_stderr_does_not_contain("[..]rustc[..]")
190+
.run();
191+
192+
p.change_file(
193+
"Cargo.toml",
194+
r#"
195+
[package]
196+
name = "foo"
197+
version = "0.1.0"
198+
199+
[features]
200+
f_a = []
201+
"#,
202+
);
203+
204+
p.cargo("check -v -Zcheck-cfg")
205+
.masquerade_as_nightly_cargo(&["check-cfg"])
206+
// we check that the fingerprint is indeed dirty
207+
.with_stderr_contains("[..]Dirty[..]the list of declared features changed")
208+
// that is cause rustc to be called again with the new check-cfg args
209+
.with_stderr_contains(x!("rustc" => "cfg" of "feature" with "f_a"))
210+
// and that we indeed found a new warning from the unexpected_cfgs lint
211+
.with_stderr_contains("[..]unexpected_cfgs[..]")
212+
.run();
213+
}
214+
144215
#[cargo_test(nightly, reason = "--check-cfg is unstable")]
145216
fn well_known_names_values() {
146217
let p = project()

0 commit comments

Comments
 (0)