Skip to content

Commit e828e85

Browse files
committed
Add -Z check-cfg-features support for rustdoc
1 parent c7dbc42 commit e828e85

File tree

5 files changed

+111
-34
lines changed

5 files changed

+111
-34
lines changed

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

+1-4
Original file line numberDiff line numberDiff line change
@@ -224,11 +224,8 @@ impl<'a, 'cfg> Context<'a, 'cfg> {
224224
let mut unstable_opts = false;
225225
let mut args = compiler::extern_args(&self, unit, &mut unstable_opts)?;
226226
args.extend(compiler::lto_args(&self, unit));
227+
args.extend(compiler::features_args(&self, unit));
227228

228-
for feature in &unit.features {
229-
args.push("--cfg".into());
230-
args.push(format!("feature=\"{}\"", feature).into());
231-
}
232229
let script_meta = self.find_build_script_metadata(unit);
233230
if let Some(meta) = script_meta {
234231
if let Some(output) = self.build_script_outputs.lock().unwrap().get(meta) {

src/cargo/core/compiler/mod.rs

+31-27
Original file line numberDiff line numberDiff line change
@@ -645,10 +645,7 @@ fn rustdoc(cx: &mut Context<'_, '_>, unit: &Unit) -> CargoResult<Work> {
645645
paths::create_dir_all(&doc_dir)?;
646646

647647
rustdoc.arg("-o").arg(&doc_dir);
648-
649-
for feat in &unit.features {
650-
rustdoc.arg("--cfg").arg(&format!("feature=\"{}\"", feat));
651-
}
648+
rustdoc.args(&features_args(cx, unit));
652649

653650
add_error_format_and_color(cx, &mut rustdoc, unit);
654651
add_allow_features(cx, &mut rustdoc);
@@ -791,28 +788,6 @@ fn add_allow_features(cx: &Context<'_, '_>, cmd: &mut ProcessBuilder) {
791788
}
792789
}
793790

794-
/// Add all features as cfg
795-
fn add_features(cx: &Context<'_, '_>, cmd: &mut ProcessBuilder, unit: &Unit) {
796-
for feat in &unit.features {
797-
cmd.arg("--cfg").arg(&format!("feature=\"{}\"", feat));
798-
}
799-
800-
if cx.bcx.config.cli_unstable().check_cfg_features {
801-
// This generate something like this:
802-
// - values(feature)
803-
// - values(feature, "foo", "bar")
804-
let mut arg = String::from("values(feature");
805-
for (&feat, _) in unit.pkg.summary().features() {
806-
arg.push_str(", \"");
807-
arg.push_str(&feat);
808-
arg.push_str("\"");
809-
}
810-
arg.push(')');
811-
812-
cmd.arg("-Zunstable-options").arg("--check-cfg").arg(&arg);
813-
}
814-
}
815-
816791
/// Add error-format flags to the command.
817792
///
818793
/// Cargo always uses JSON output. This has several benefits, such as being
@@ -1009,7 +984,7 @@ fn build_base_args(
1009984
cmd.arg("--cfg").arg("test");
1010985
}
1011986

1012-
add_features(cx, cmd, unit);
987+
cmd.args(&features_args(cx, unit));
1013988

1014989
let meta = cx.files().metadata(unit);
1015990
cmd.arg("-C").arg(&format!("metadata={}", meta));
@@ -1083,6 +1058,35 @@ fn build_base_args(
10831058
Ok(())
10841059
}
10851060

1061+
/// Features with --cfg and all features with --check-cfg
1062+
fn features_args(cx: &Context<'_, '_>, unit: &Unit) -> Vec<OsString> {
1063+
let mut args = Vec::with_capacity(unit.features.len() + 2);
1064+
1065+
for feat in &unit.features {
1066+
args.push(OsString::from("--cfg"));
1067+
args.push(OsString::from(format!("feature=\"{}\"", feat)));
1068+
}
1069+
1070+
if cx.bcx.config.cli_unstable().check_cfg_features {
1071+
// This generate something like this:
1072+
// - values(feature)
1073+
// - values(feature, "foo", "bar")
1074+
let mut arg = OsString::from("values(feature");
1075+
for (&feat, _) in unit.pkg.summary().features() {
1076+
arg.push(", \"");
1077+
arg.push(&feat);
1078+
arg.push("\"");
1079+
}
1080+
arg.push(")");
1081+
1082+
args.push(OsString::from("-Zunstable-options"));
1083+
args.push(OsString::from("--check-cfg"));
1084+
args.push(arg);
1085+
}
1086+
1087+
args
1088+
}
1089+
10861090
fn lto_args(cx: &Context<'_, '_>, unit: &Unit) -> Vec<OsString> {
10871091
let mut result = Vec::new();
10881092
let mut push = |arg: &str| {

src/doc/src/reference/unstable.md

+3-3
Original file line numberDiff line numberDiff line change
@@ -1132,9 +1132,9 @@ cargo doc -Z unstable-options -Z rustdoc-scrape-examples=examples
11321132
* RFC: [#3013](https://github.com/rust-lang/rfcs/pull/3013)
11331133

11341134
The `-Z check-cfg-features` argument tells Cargo to pass all possible features of a package to
1135-
`rustc` unstable `--check-cfg` command line as `--check-cfg=values(feature, ...)`. This enables
1136-
compile time checking of feature values in `#[cfg]`, `cfg!` and `#[cfg_attr]`. Note than this
1137-
command line options will probably become the default when stabilizing.
1135+
`rustc` and `rustdoc` unstable `--check-cfg` command line as `--check-cfg=values(feature, ...)`.
1136+
This enables compile time checking of feature values in `#[cfg]`, `cfg!` and `#[cfg_attr]`.
1137+
Note than this command line options will probably become the default when stabilizing.
11381138
For instance:
11391139

11401140
```

tests/testsuite/doc.rs

+37
Original file line numberDiff line numberDiff line change
@@ -2697,3 +2697,40 @@ fn doc_lib_false_dep() {
26972697
assert!(p.build_dir().join("doc/foo").exists());
26982698
assert!(!p.build_dir().join("doc/bar").exists());
26992699
}
2700+
2701+
#[cfg_attr(windows, ignore)] // weird normalization issue with windows and cargo-test-support
2702+
#[cargo_test]
2703+
fn doc_check_cfg_features() {
2704+
if !is_nightly() {
2705+
// --check-cfg is a nightly only rustdoc command line
2706+
return;
2707+
}
2708+
2709+
let p = project()
2710+
.file(
2711+
"Cargo.toml",
2712+
r#"
2713+
[project]
2714+
name = "foo"
2715+
version = "0.1.0"
2716+
2717+
[features]
2718+
default = ["f_a"]
2719+
f_a = []
2720+
f_b = []
2721+
"#,
2722+
)
2723+
.file("src/lib.rs", "#[allow(dead_code)] fn foo() {}")
2724+
.build();
2725+
2726+
p.cargo("doc -v -Z check-cfg-features")
2727+
.masquerade_as_nightly_cargo()
2728+
.with_stderr(
2729+
"\
2730+
[DOCUMENTING] foo v0.1.0 [..]
2731+
[RUNNING] `rustdoc [..] --check-cfg 'values(feature, \"default\", \"f_a\", \"f_b\")' [..]
2732+
[FINISHED] [..]
2733+
",
2734+
)
2735+
.run();
2736+
}

tests/testsuite/test.rs

+39
Original file line numberDiff line numberDiff line change
@@ -4536,3 +4536,42 @@ fn check_cfg_features() {
45364536
)
45374537
.run();
45384538
}
4539+
4540+
#[cfg_attr(windows, ignore)] // weird normalization issue with windows and cargo-test-support
4541+
#[cargo_test]
4542+
fn check_cfg_features_doc() {
4543+
if !is_nightly() {
4544+
// --check-cfg is a nightly only rustc and rustdoc command line
4545+
return;
4546+
}
4547+
4548+
let p = project()
4549+
.file(
4550+
"Cargo.toml",
4551+
r#"
4552+
[project]
4553+
name = "foo"
4554+
version = "0.1.0"
4555+
4556+
[features]
4557+
default = ["f_a"]
4558+
f_a = []
4559+
f_b = []
4560+
"#,
4561+
)
4562+
.file("src/lib.rs", "#[allow(dead_code)] fn foo() {}")
4563+
.build();
4564+
4565+
p.cargo("test -v --doc -Z check-cfg-features")
4566+
.masquerade_as_nightly_cargo()
4567+
.with_stderr(
4568+
"\
4569+
[COMPILING] foo v0.1.0 [..]
4570+
[RUNNING] `rustc [..] --check-cfg 'values(feature, \"default\", \"f_a\", \"f_b\")' [..]
4571+
[FINISHED] test [unoptimized + debuginfo] target(s) in [..]
4572+
[DOCTEST] foo
4573+
[RUNNING] `rustdoc [..] --check-cfg 'values(feature, \"default\", \"f_a\", \"f_b\")' [..]
4574+
",
4575+
)
4576+
.run();
4577+
}

0 commit comments

Comments
 (0)