Skip to content

Commit 03fe834

Browse files
authoredJan 8, 2020
Rollup merge of rust-lang#67875 - dtolnay:hidden, r=GuillaumeGomez
Distinguish between private items and hidden items in rustdoc I believe rustdoc should not be conflating private items (visibility lower than `pub`) and hidden items (attribute `doc(hidden)`). This matters now that Cargo is passing --document-private-items by default for bin crates. In bin crates that rely on macros, intentionally hidden implementation details of the macros can overwhelm the actual useful internal API that one would want to document. This PR restores the strip-hidden pass when documenting private items, and introduces a separate unstable --document-hidden-items option to skip the strip-hidden pass. The two options are orthogonal to one another. Fixes rust-lang#67851. Closes rust-lang#60884.
2 parents 429a7e7 + 90adafb commit 03fe834

20 files changed

+138
-78
lines changed
 

‎src/librustdoc/config.rs

+26-17
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ use crate::html;
2424
use crate::html::markdown::IdMap;
2525
use crate::html::static_files;
2626
use crate::opts;
27-
use crate::passes::{self, DefaultPassOption};
27+
use crate::passes::{self, Condition, DefaultPassOption};
2828
use crate::theme;
2929

3030
/// Configuration options for rustdoc.
@@ -98,6 +98,10 @@ pub struct Options {
9898
///
9999
/// Be aware: This option can come both from the CLI and from crate attributes!
100100
pub default_passes: DefaultPassOption,
101+
/// Document items that have lower than `pub` visibility.
102+
pub document_private: bool,
103+
/// Document items that have `doc(hidden)`.
104+
pub document_hidden: bool,
101105
/// Any passes manually selected by the user.
102106
///
103107
/// Be aware: This option can come both from the CLI and from crate attributes!
@@ -146,6 +150,8 @@ impl fmt::Debug for Options {
146150
.field("test_args", &self.test_args)
147151
.field("persist_doctests", &self.persist_doctests)
148152
.field("default_passes", &self.default_passes)
153+
.field("document_private", &self.document_private)
154+
.field("document_hidden", &self.document_hidden)
149155
.field("manual_passes", &self.manual_passes)
150156
.field("display_warnings", &self.display_warnings)
151157
.field("show_coverage", &self.show_coverage)
@@ -240,22 +246,26 @@ impl Options {
240246
println!("{:>20} - {}", pass.name, pass.description);
241247
}
242248
println!("\nDefault passes for rustdoc:");
243-
for pass in passes::DEFAULT_PASSES {
244-
println!("{:>20}", pass.name);
245-
}
246-
println!("\nPasses run with `--document-private-items`:");
247-
for pass in passes::DEFAULT_PRIVATE_PASSES {
248-
println!("{:>20}", pass.name);
249+
for p in passes::DEFAULT_PASSES {
250+
print!("{:>20}", p.pass.name);
251+
println_condition(p.condition);
249252
}
250253

251254
if nightly_options::is_nightly_build() {
252255
println!("\nPasses run with `--show-coverage`:");
253-
for pass in passes::DEFAULT_COVERAGE_PASSES {
254-
println!("{:>20}", pass.name);
256+
for p in passes::COVERAGE_PASSES {
257+
print!("{:>20}", p.pass.name);
258+
println_condition(p.condition);
255259
}
256-
println!("\nPasses run with `--show-coverage --document-private-items`:");
257-
for pass in passes::PRIVATE_COVERAGE_PASSES {
258-
println!("{:>20}", pass.name);
260+
}
261+
262+
fn println_condition(condition: Condition) {
263+
use Condition::*;
264+
match condition {
265+
Always => println!(),
266+
WhenDocumentPrivate => println!(" (when --document-private-items)"),
267+
WhenNotDocumentPrivate => println!(" (when not --document-private-items)"),
268+
WhenNotDocumentHidden => println!(" (when not --document-hidden-items)"),
259269
}
260270
}
261271

@@ -444,16 +454,11 @@ impl Options {
444454
});
445455

446456
let show_coverage = matches.opt_present("show-coverage");
447-
let document_private = matches.opt_present("document-private-items");
448457

449458
let default_passes = if matches.opt_present("no-defaults") {
450459
passes::DefaultPassOption::None
451-
} else if show_coverage && document_private {
452-
passes::DefaultPassOption::PrivateCoverage
453460
} else if show_coverage {
454461
passes::DefaultPassOption::Coverage
455-
} else if document_private {
456-
passes::DefaultPassOption::Private
457462
} else {
458463
passes::DefaultPassOption::Default
459464
};
@@ -492,6 +497,8 @@ impl Options {
492497
let runtool = matches.opt_str("runtool");
493498
let runtool_args = matches.opt_strs("runtool-arg");
494499
let enable_per_target_ignores = matches.opt_present("enable-per-target-ignores");
500+
let document_private = matches.opt_present("document-private-items");
501+
let document_hidden = matches.opt_present("document-hidden-items");
495502

496503
let (lint_opts, describe_lints, lint_cap) = get_cmd_lint_options(matches, error_format);
497504

@@ -518,6 +525,8 @@ impl Options {
518525
should_test,
519526
test_args,
520527
default_passes,
528+
document_private,
529+
document_hidden,
521530
manual_passes,
522531
display_warnings,
523532
show_coverage,

‎src/librustdoc/core.rs

+17-9
Original file line numberDiff line numberDiff line change
@@ -33,7 +33,7 @@ use crate::clean::{AttributesExt, MAX_DEF_ID};
3333
use crate::config::{Options as RustdocOptions, RenderOptions};
3434
use crate::html::render::RenderInfo;
3535

36-
use crate::passes;
36+
use crate::passes::{self, Condition::*, ConditionalPass};
3737

3838
pub use rustc::session::config::{CodegenOptions, DebuggingOptions, Input, Options};
3939
pub use rustc::session::search_paths::SearchPath;
@@ -221,6 +221,8 @@ pub fn run_core(options: RustdocOptions) -> (clean::Crate, RenderInfo, RenderOpt
221221
describe_lints,
222222
lint_cap,
223223
mut default_passes,
224+
mut document_private,
225+
document_hidden,
224226
mut manual_passes,
225227
display_warnings,
226228
render_options,
@@ -457,16 +459,14 @@ pub fn run_core(options: RustdocOptions) -> (clean::Crate, RenderInfo, RenderOpt
457459
}
458460

459461
if attr.is_word() && name == sym::document_private_items {
460-
if default_passes == passes::DefaultPassOption::Default {
461-
default_passes = passes::DefaultPassOption::Private;
462-
}
462+
document_private = true;
463463
}
464464
}
465465

466-
let passes = passes::defaults(default_passes).iter().chain(
466+
let passes = passes::defaults(default_passes).iter().copied().chain(
467467
manual_passes.into_iter().flat_map(|name| {
468468
if let Some(pass) = passes::find_pass(&name) {
469-
Some(pass)
469+
Some(ConditionalPass::always(pass))
470470
} else {
471471
error!("unknown pass {}, skipping", name);
472472
None
@@ -476,9 +476,17 @@ pub fn run_core(options: RustdocOptions) -> (clean::Crate, RenderInfo, RenderOpt
476476

477477
info!("Executing passes");
478478

479-
for pass in passes {
480-
debug!("running pass {}", pass.name);
481-
krate = (pass.pass)(krate, &ctxt);
479+
for p in passes {
480+
let run = match p.condition {
481+
Always => true,
482+
WhenDocumentPrivate => document_private,
483+
WhenNotDocumentPrivate => !document_private,
484+
WhenNotDocumentHidden => !document_hidden,
485+
};
486+
if run {
487+
debug!("running pass {}", p.pass.name);
488+
krate = (p.pass.run)(krate, &ctxt);
489+
}
482490
}
483491

484492
ctxt.sess().abort_if_errors();

‎src/librustdoc/lib.rs

+3
Original file line numberDiff line numberDiff line change
@@ -174,6 +174,9 @@ fn opts() -> Vec<RustcOptGroup> {
174174
stable("document-private-items", |o| {
175175
o.optflag("", "document-private-items", "document private items")
176176
}),
177+
unstable("document-hidden-items", |o| {
178+
o.optflag("", "document-hidden-items", "document items that have doc(hidden)")
179+
}),
177180
stable("test", |o| o.optflag("", "test", "run code examples as tests")),
178181
stable("test-args", |o| {
179182
o.optmulti("", "test-args", "arguments to pass to the test runner", "ARGS")

‎src/librustdoc/passes/calculate_doc_coverage.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ use std::ops;
1212

1313
pub const CALCULATE_DOC_COVERAGE: Pass = Pass {
1414
name: "calculate-doc-coverage",
15-
pass: calculate_doc_coverage,
15+
run: calculate_doc_coverage,
1616
description: "counts the number of items with and without documentation",
1717
};
1818

‎src/librustdoc/passes/check_code_block_syntax.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ use crate::passes::Pass;
1414

1515
pub const CHECK_CODE_BLOCK_SYNTAX: Pass = Pass {
1616
name: "check-code-block-syntax",
17-
pass: check_code_block_syntax,
17+
run: check_code_block_syntax,
1818
description: "validates syntax inside Rust code blocks",
1919
};
2020

‎src/librustdoc/passes/collapse_docs.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use std::mem::take;
88

99
pub const COLLAPSE_DOCS: Pass = Pass {
1010
name: "collapse-docs",
11-
pass: collapse_docs,
11+
run: collapse_docs,
1212
description: "concatenates all document attributes into one document attribute",
1313
};
1414

‎src/librustdoc/passes/collect_intra_doc_links.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -28,7 +28,7 @@ use super::span_of_attrs;
2828

2929
pub const COLLECT_INTRA_DOC_LINKS: Pass = Pass {
3030
name: "collect-intra-doc-links",
31-
pass: collect_intra_doc_links,
31+
run: collect_intra_doc_links,
3232
description: "reads a crate's documentation to resolve intra-doc-links",
3333
};
3434

‎src/librustdoc/passes/collect_trait_impls.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,7 @@ use rustc_span::symbol::sym;
99

1010
pub const COLLECT_TRAIT_IMPLS: Pass = Pass {
1111
name: "collect-trait-impls",
12-
pass: collect_trait_impls,
12+
run: collect_trait_impls,
1313
description: "retrieves trait impls for items in the crate",
1414
};
1515

‎src/librustdoc/passes/mod.rs

+51-36
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ use rustc_span::{InnerSpan, Span, DUMMY_SP};
88
use std::mem;
99
use std::ops::Range;
1010

11+
use self::Condition::*;
1112
use crate::clean::{self, GetDefId, Item};
1213
use crate::core::DocContext;
1314
use crate::fold::{DocFolder, StripItem};
@@ -52,10 +53,29 @@ pub use self::calculate_doc_coverage::CALCULATE_DOC_COVERAGE;
5253
#[derive(Copy, Clone)]
5354
pub struct Pass {
5455
pub name: &'static str,
55-
pub pass: fn(clean::Crate, &DocContext<'_>) -> clean::Crate,
56+
pub run: fn(clean::Crate, &DocContext<'_>) -> clean::Crate,
5657
pub description: &'static str,
5758
}
5859

60+
/// In a list of passes, a pass that may or may not need to be run depending on options.
61+
#[derive(Copy, Clone)]
62+
pub struct ConditionalPass {
63+
pub pass: Pass,
64+
pub condition: Condition,
65+
}
66+
67+
/// How to decide whether to run a conditional pass.
68+
#[derive(Copy, Clone)]
69+
pub enum Condition {
70+
Always,
71+
/// When `--document-private-items` is passed.
72+
WhenDocumentPrivate,
73+
/// When `--document-private-items` is not passed.
74+
WhenNotDocumentPrivate,
75+
/// When `--document-hidden-items` is not passed.
76+
WhenNotDocumentHidden,
77+
}
78+
5979
/// The full list of passes.
6080
pub const PASSES: &[Pass] = &[
6181
CHECK_PRIVATE_ITEMS_DOC_TESTS,
@@ -72,63 +92,58 @@ pub const PASSES: &[Pass] = &[
7292
];
7393

7494
/// The list of passes run by default.
75-
pub const DEFAULT_PASSES: &[Pass] = &[
76-
COLLECT_TRAIT_IMPLS,
77-
COLLAPSE_DOCS,
78-
UNINDENT_COMMENTS,
79-
CHECK_PRIVATE_ITEMS_DOC_TESTS,
80-
STRIP_HIDDEN,
81-
STRIP_PRIVATE,
82-
COLLECT_INTRA_DOC_LINKS,
83-
CHECK_CODE_BLOCK_SYNTAX,
84-
PROPAGATE_DOC_CFG,
95+
pub const DEFAULT_PASSES: &[ConditionalPass] = &[
96+
ConditionalPass::always(COLLECT_TRAIT_IMPLS),
97+
ConditionalPass::always(COLLAPSE_DOCS),
98+
ConditionalPass::always(UNINDENT_COMMENTS),
99+
ConditionalPass::always(CHECK_PRIVATE_ITEMS_DOC_TESTS),
100+
ConditionalPass::new(STRIP_HIDDEN, WhenNotDocumentHidden),
101+
ConditionalPass::new(STRIP_PRIVATE, WhenNotDocumentPrivate),
102+
ConditionalPass::new(STRIP_PRIV_IMPORTS, WhenDocumentPrivate),
103+
ConditionalPass::always(COLLECT_INTRA_DOC_LINKS),
104+
ConditionalPass::always(CHECK_CODE_BLOCK_SYNTAX),
105+
ConditionalPass::always(PROPAGATE_DOC_CFG),
85106
];
86107

87-
/// The list of default passes run with `--document-private-items` is passed to rustdoc.
88-
pub const DEFAULT_PRIVATE_PASSES: &[Pass] = &[
89-
COLLECT_TRAIT_IMPLS,
90-
COLLAPSE_DOCS,
91-
UNINDENT_COMMENTS,
92-
CHECK_PRIVATE_ITEMS_DOC_TESTS,
93-
STRIP_PRIV_IMPORTS,
94-
COLLECT_INTRA_DOC_LINKS,
95-
CHECK_CODE_BLOCK_SYNTAX,
96-
PROPAGATE_DOC_CFG,
108+
/// The list of default passes run when `--doc-coverage` is passed to rustdoc.
109+
pub const COVERAGE_PASSES: &[ConditionalPass] = &[
110+
ConditionalPass::always(COLLECT_TRAIT_IMPLS),
111+
ConditionalPass::new(STRIP_HIDDEN, WhenNotDocumentHidden),
112+
ConditionalPass::new(STRIP_PRIVATE, WhenNotDocumentPrivate),
113+
ConditionalPass::always(CALCULATE_DOC_COVERAGE),
97114
];
98115

99-
/// The list of default passes run when `--doc-coverage` is passed to rustdoc.
100-
pub const DEFAULT_COVERAGE_PASSES: &[Pass] =
101-
&[COLLECT_TRAIT_IMPLS, STRIP_HIDDEN, STRIP_PRIVATE, CALCULATE_DOC_COVERAGE];
116+
impl ConditionalPass {
117+
pub const fn always(pass: Pass) -> Self {
118+
Self::new(pass, Always)
119+
}
102120

103-
/// The list of default passes run when `--doc-coverage --document-private-items` is passed to
104-
/// rustdoc.
105-
pub const PRIVATE_COVERAGE_PASSES: &[Pass] = &[COLLECT_TRAIT_IMPLS, CALCULATE_DOC_COVERAGE];
121+
pub const fn new(pass: Pass, condition: Condition) -> Self {
122+
ConditionalPass { pass, condition }
123+
}
124+
}
106125

107126
/// A shorthand way to refer to which set of passes to use, based on the presence of
108-
/// `--no-defaults` or `--document-private-items`.
127+
/// `--no-defaults` and `--show-coverage`.
109128
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
110129
pub enum DefaultPassOption {
111130
Default,
112-
Private,
113131
Coverage,
114-
PrivateCoverage,
115132
None,
116133
}
117134

118135
/// Returns the given default set of passes.
119-
pub fn defaults(default_set: DefaultPassOption) -> &'static [Pass] {
136+
pub fn defaults(default_set: DefaultPassOption) -> &'static [ConditionalPass] {
120137
match default_set {
121138
DefaultPassOption::Default => DEFAULT_PASSES,
122-
DefaultPassOption::Private => DEFAULT_PRIVATE_PASSES,
123-
DefaultPassOption::Coverage => DEFAULT_COVERAGE_PASSES,
124-
DefaultPassOption::PrivateCoverage => PRIVATE_COVERAGE_PASSES,
139+
DefaultPassOption::Coverage => COVERAGE_PASSES,
125140
DefaultPassOption::None => &[],
126141
}
127142
}
128143

129144
/// If the given name matches a known pass, returns its information.
130-
pub fn find_pass(pass_name: &str) -> Option<&'static Pass> {
131-
PASSES.iter().find(|p| p.name == pass_name)
145+
pub fn find_pass(pass_name: &str) -> Option<Pass> {
146+
PASSES.iter().find(|p| p.name == pass_name).copied()
132147
}
133148

134149
struct Stripper<'a> {

‎src/librustdoc/passes/private_items_doc_tests.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ use crate::passes::{look_for_tests, Pass};
55

66
pub const CHECK_PRIVATE_ITEMS_DOC_TESTS: Pass = Pass {
77
name: "check-private-items-doc-tests",
8-
pass: check_private_items_doc_tests,
8+
run: check_private_items_doc_tests,
99
description: "check private items doc tests",
1010
};
1111

‎src/librustdoc/passes/propagate_doc_cfg.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ use crate::passes::Pass;
88

99
pub const PROPAGATE_DOC_CFG: Pass = Pass {
1010
name: "propagate-doc-cfg",
11-
pass: propagate_doc_cfg,
11+
run: propagate_doc_cfg,
1212
description: "propagates `#[doc(cfg(...))]` to child items",
1313
};
1414

‎src/librustdoc/passes/strip_hidden.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@ use crate::passes::{ImplStripper, Pass};
1010

1111
pub const STRIP_HIDDEN: Pass = Pass {
1212
name: "strip-hidden",
13-
pass: strip_hidden,
13+
run: strip_hidden,
1414
description: "strips all doc(hidden) items from the output",
1515
};
1616

‎src/librustdoc/passes/strip_priv_imports.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@ use crate::passes::{ImportStripper, Pass};
55

66
pub const STRIP_PRIV_IMPORTS: Pass = Pass {
77
name: "strip-priv-imports",
8-
pass: strip_priv_imports,
8+
run: strip_priv_imports,
99
description: "strips all private import statements (`use`, `extern crate`) from a crate",
1010
};
1111

‎src/librustdoc/passes/strip_private.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ use crate::passes::{ImplStripper, ImportStripper, Pass, Stripper};
77

88
pub const STRIP_PRIVATE: Pass = Pass {
99
name: "strip-private",
10-
pass: strip_private,
10+
run: strip_private,
1111
description: "strips all private items from a crate which cannot be seen externally, \
1212
implies strip-priv-imports",
1313
};

‎src/librustdoc/passes/unindent_comments.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ mod tests;
1212

1313
pub const UNINDENT_COMMENTS: Pass = Pass {
1414
name: "unindent-comments",
15-
pass: unindent_comments,
15+
run: unindent_comments,
1616
description: "removes excess indentation on comments in order for markdown to like it",
1717
};
1818

‎src/test/rustdoc/issue-46380.rs

-5
This file was deleted.

‎src/test/rustdoc/issue-67851-both.rs

+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
// compile-flags: -Zunstable-options --document-private-items --document-hidden-items
2+
3+
// @has issue_67851_both/struct.Hidden.html
4+
#[doc(hidden)]
5+
pub struct Hidden;
6+
7+
// @has issue_67851_both/struct.Private.html
8+
struct Private;
+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
// compile-flags: -Zunstable-options --document-hidden-items
2+
3+
// @has issue_67851_hidden/struct.Hidden.html
4+
#[doc(hidden)]
5+
pub struct Hidden;
6+
7+
// @!has issue_67851_hidden/struct.Private.html
8+
struct Private;
+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
// @!has issue_67851_neither/struct.Hidden.html
2+
#[doc(hidden)]
3+
pub struct Hidden;
4+
5+
// @!has issue_67851_neither/struct.Private.html
6+
struct Private;
+8
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,8 @@
1+
// compile-flags: --document-private-items
2+
3+
// @!has issue_67851_private/struct.Hidden.html
4+
#[doc(hidden)]
5+
pub struct Hidden;
6+
7+
// @has issue_67851_private/struct.Private.html
8+
struct Private;

0 commit comments

Comments
 (0)
Please sign in to comment.