Skip to content

Commit 90adafb

Browse files
committed
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.
1 parent cd8377d commit 90adafb

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

@@ -449,16 +459,11 @@ impl Options {
449459
});
450460

451461
let show_coverage = matches.opt_present("show-coverage");
452-
let document_private = matches.opt_present("document-private-items");
453462

454463
let default_passes = if matches.opt_present("no-defaults") {
455464
passes::DefaultPassOption::None
456-
} else if show_coverage && document_private {
457-
passes::DefaultPassOption::PrivateCoverage
458465
} else if show_coverage {
459466
passes::DefaultPassOption::Coverage
460-
} else if document_private {
461-
passes::DefaultPassOption::Private
462467
} else {
463468
passes::DefaultPassOption::Default
464469
};
@@ -497,6 +502,8 @@ impl Options {
497502
let runtool = matches.opt_str("runtool");
498503
let runtool_args = matches.opt_strs("runtool-arg");
499504
let enable_per_target_ignores = matches.opt_present("enable-per-target-ignores");
505+
let document_private = matches.opt_present("document-private-items");
506+
let document_hidden = matches.opt_present("document-hidden-items");
500507

501508
let (lint_opts, describe_lints, lint_cap) = get_cmd_lint_options(matches, error_format);
502509

@@ -523,6 +530,8 @@ impl Options {
523530
should_test,
524531
test_args,
525532
default_passes,
533+
document_private,
534+
document_hidden,
526535
manual_passes,
527536
display_warnings,
528537
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, Input, Options};
3939
pub use rustc::session::search_paths::SearchPath;
@@ -234,6 +234,8 @@ pub fn run_core(options: RustdocOptions) -> (clean::Crate, RenderInfo, RenderOpt
234234
describe_lints,
235235
lint_cap,
236236
mut default_passes,
237+
mut document_private,
238+
document_hidden,
237239
mut manual_passes,
238240
display_warnings,
239241
render_options,
@@ -470,16 +472,14 @@ pub fn run_core(options: RustdocOptions) -> (clean::Crate, RenderInfo, RenderOpt
470472
}
471473

472474
if attr.is_word() && name == sym::document_private_items {
473-
if default_passes == passes::DefaultPassOption::Default {
474-
default_passes = passes::DefaultPassOption::Private;
475-
}
475+
document_private = true;
476476
}
477477
}
478478

479-
let passes = passes::defaults(default_passes).iter().chain(
479+
let passes = passes::defaults(default_passes).iter().copied().chain(
480480
manual_passes.into_iter().flat_map(|name| {
481481
if let Some(pass) = passes::find_pass(&name) {
482-
Some(pass)
482+
Some(ConditionalPass::always(pass))
483483
} else {
484484
error!("unknown pass {}, skipping", name);
485485
None
@@ -489,9 +489,17 @@ pub fn run_core(options: RustdocOptions) -> (clean::Crate, RenderInfo, RenderOpt
489489

490490
info!("Executing passes");
491491

492-
for pass in passes {
493-
debug!("running pass {}", pass.name);
494-
krate = (pass.pass)(krate, &ctxt);
492+
for p in passes {
493+
let run = match p.condition {
494+
Always => true,
495+
WhenDocumentPrivate => document_private,
496+
WhenNotDocumentPrivate => !document_private,
497+
WhenNotDocumentHidden => !document_hidden,
498+
};
499+
if run {
500+
debug!("running pass {}", p.pass.name);
501+
krate = (p.pass.run)(krate, &ctxt);
502+
}
495503
}
496504

497505
ctxt.sess().abort_if_errors();

src/librustdoc/lib.rs

+3
Original file line numberDiff line numberDiff line change
@@ -173,6 +173,9 @@ fn opts() -> Vec<RustcOptGroup> {
173173
stable("document-private-items", |o| {
174174
o.optflag("", "document-private-items", "document private items")
175175
}),
176+
unstable("document-hidden-items", |o| {
177+
o.optflag("", "document-hidden-items", "document items that have doc(hidden)")
178+
}),
176179
stable("test", |o| o.optflag("", "test", "run code examples as tests")),
177180
stable("test-args", |o| {
178181
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
@@ -13,7 +13,7 @@ use crate::passes::Pass;
1313

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

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
@@ -9,6 +9,7 @@ use rustc_span::{InnerSpan, Span, DUMMY_SP};
99
use std::mem;
1010
use std::ops::Range;
1111

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

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

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

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

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

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

108127
/// A shorthand way to refer to which set of passes to use, based on the presence of
109-
/// `--no-defaults` or `--document-private-items`.
128+
/// `--no-defaults` and `--show-coverage`.
110129
#[derive(Copy, Clone, PartialEq, Eq, Debug)]
111130
pub enum DefaultPassOption {
112131
Default,
113-
Private,
114132
Coverage,
115-
PrivateCoverage,
116133
None,
117134
}
118135

119136
/// Returns the given default set of passes.
120-
pub fn defaults(default_set: DefaultPassOption) -> &'static [Pass] {
137+
pub fn defaults(default_set: DefaultPassOption) -> &'static [ConditionalPass] {
121138
match default_set {
122139
DefaultPassOption::Default => DEFAULT_PASSES,
123-
DefaultPassOption::Private => DEFAULT_PRIVATE_PASSES,
124-
DefaultPassOption::Coverage => DEFAULT_COVERAGE_PASSES,
125-
DefaultPassOption::PrivateCoverage => PRIVATE_COVERAGE_PASSES,
140+
DefaultPassOption::Coverage => COVERAGE_PASSES,
126141
DefaultPassOption::None => &[],
127142
}
128143
}
129144

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

135150
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
};

0 commit comments

Comments
 (0)