Skip to content

Commit eaf0446

Browse files
committed
Split elided_lifetime_in_paths into tied and untied
Types that contain a reference can be confusing when lifetime elision occurs: ```rust // Confusing fn foo(_: &u8) -> Bar { todo!() } // Less confusing fn foo(_: &u8) -> Bar<'_> { todo!() } ``` However, the previous lint did not distinguish when these types were not "tying" lifetimes across the function inputs / outputs: ```rust // Maybe a little confusing fn foo(_: Bar) {} // More explicit but noisier with less obvious value fn foo(_: Bar<'_>) {} ``` We now report different lints for each case, hopefully paving the way to marking the first case (when lifetimes are tied together) as warn-by-default.
1 parent b3c630b commit eaf0446

10 files changed

+535
-28
lines changed

compiler/rustc_lint/src/lib.rs

+8-1
Original file line numberDiff line numberDiff line change
@@ -303,7 +303,8 @@ fn register_builtins(store: &mut LintStore) {
303303
BARE_TRAIT_OBJECTS,
304304
UNUSED_EXTERN_CRATES,
305305
ELLIPSIS_INCLUSIVE_RANGE_PATTERNS,
306-
ELIDED_LIFETIMES_IN_PATHS,
306+
ELIDED_LIFETIMES_IN_PATHS_TIED,
307+
ELIDED_LIFETIMES_IN_PATHS_UNTIED,
307308
EXPLICIT_OUTLIVES_REQUIREMENTS // FIXME(#52665, #47816) not always applicable and not all
308309
// macros are ready for this yet.
309310
// UNREACHABLE_PUB,
@@ -313,6 +314,12 @@ fn register_builtins(store: &mut LintStore) {
313314
// MACRO_USE_EXTERN_CRATE
314315
);
315316

317+
add_lint_group!(
318+
"elided_lifetimes_in_paths",
319+
ELIDED_LIFETIMES_IN_PATHS_TIED,
320+
ELIDED_LIFETIMES_IN_PATHS_UNTIED,
321+
);
322+
316323
// Register renamed and removed lints.
317324
store.register_renamed("single_use_lifetime", "single_use_lifetimes");
318325
// store.register_renamed("elided_lifetime_in_path", "elided_lifetimes_in_paths");

compiler/rustc_lint_defs/src/builtin.rs

+47-5
Original file line numberDiff line numberDiff line change
@@ -40,7 +40,8 @@ declare_lint_pass! {
4040
DEPRECATED_WHERE_CLAUSE_LOCATION,
4141
DUPLICATE_MACRO_ATTRIBUTES,
4242
ELIDED_LIFETIMES_IN_ASSOCIATED_CONSTANT,
43-
ELIDED_LIFETIMES_IN_PATHS,
43+
ELIDED_LIFETIMES_IN_PATHS_TIED,
44+
ELIDED_LIFETIMES_IN_PATHS_UNTIED,
4445
EXPORTED_PRIVATE_DEPENDENCIES,
4546
FFI_UNWIND_CALLS,
4647
FORBIDDEN_LINT_GROUPS,
@@ -1701,8 +1702,9 @@ declare_lint! {
17011702
}
17021703

17031704
declare_lint! {
1704-
/// The `elided_lifetimes_in_paths` lint detects the use of hidden
1705-
/// lifetime parameters.
1705+
/// The `elided_lifetimes_in_paths_tied` lint detects the use of
1706+
/// hidden lifetime parameters when those lifetime parameters tie
1707+
/// an input lifetime parameter to an output lifetime parameter.
17061708
///
17071709
/// ### Example
17081710
///
@@ -1713,7 +1715,8 @@ declare_lint! {
17131715
/// x: &'a u32
17141716
/// }
17151717
///
1716-
/// fn foo(x: &Foo) {
1718+
/// fn foo(x: &Foo) -> &u32 {
1719+
/// &x.0
17171720
/// }
17181721
/// ```
17191722
///
@@ -1730,12 +1733,51 @@ declare_lint! {
17301733
/// may require a significant transition for old code.
17311734
///
17321735
/// [placeholder lifetime]: https://doc.rust-lang.org/reference/lifetime-elision.html#lifetime-elision-in-functions
1733-
pub ELIDED_LIFETIMES_IN_PATHS,
1736+
pub ELIDED_LIFETIMES_IN_PATHS_TIED,
17341737
Allow,
17351738
"hidden lifetime parameters in types are deprecated",
17361739
crate_level_only
17371740
}
17381741

1742+
declare_lint! {
1743+
/// The `elided_lifetimes_in_paths_untied` lint detects the use of
1744+
/// hidden lifetime parameters when those lifetime parameters do
1745+
/// not tie an input lifetime parameter to an output lifetime
1746+
/// parameter.
1747+
///
1748+
/// ### Example
1749+
///
1750+
/// ```rust,compile_fail
1751+
/// #![deny(elided_lifetimes_in_paths)]
1752+
/// #![deny(warnings)]
1753+
/// struct Foo<'a> {
1754+
/// x: &'a u32
1755+
/// }
1756+
///
1757+
/// fn foo(x: &Foo) -> u32 {
1758+
/// x.0
1759+
/// }
1760+
/// ```
1761+
///
1762+
/// {{produces}}
1763+
///
1764+
/// ### Explanation
1765+
///
1766+
/// Elided lifetime parameters can make it difficult to see at a glance
1767+
/// that borrowing is occurring. This lint ensures that lifetime
1768+
/// parameters are always explicitly stated, even if it is the `'_`
1769+
/// [placeholder lifetime].
1770+
///
1771+
/// This lint is "allow" by default because it has some known issues, and
1772+
/// may require a significant transition for old code.
1773+
///
1774+
/// [placeholder lifetime]: https://doc.rust-lang.org/reference/lifetime-elision.html#lifetime-elision-in-functions
1775+
pub ELIDED_LIFETIMES_IN_PATHS_UNTIED,
1776+
Allow,
1777+
"hidden lifetime parameters in types make it hard to tell when borrowing is happening",
1778+
crate_level_only
1779+
}
1780+
17391781
declare_lint! {
17401782
/// The `bare_trait_objects` lint suggests using `dyn Trait` for trait
17411783
/// objects.

compiler/rustc_resolve/src/late.rs

+96-20
Original file line numberDiff line numberDiff line change
@@ -646,7 +646,22 @@ struct DiagnosticMetadata<'ast> {
646646
}
647647

648648
#[derive(Debug)]
649-
struct ResolvedNestedElisionTarget {
649+
enum ResolvedElisionTarget {
650+
TopLevel(NodeId),
651+
Nested(NestedResolvedElisionTarget),
652+
}
653+
654+
impl ResolvedElisionTarget {
655+
fn node_id(&self) -> NodeId {
656+
match *self {
657+
Self::TopLevel(n) => n,
658+
Self::Nested(NestedResolvedElisionTarget { segment_id, .. }) => segment_id,
659+
}
660+
}
661+
}
662+
663+
#[derive(Debug)]
664+
struct NestedResolvedElisionTarget {
650665
segment_id: NodeId,
651666
elided_lifetime_span: Span,
652667
diagnostic: lint::BuiltinLintDiagnostics,
@@ -694,7 +709,7 @@ struct LateResolutionVisitor<'a, 'b, 'ast, 'tcx> {
694709
lifetime_uses: FxHashMap<LocalDefId, LifetimeUseSet>,
695710

696711
/// Track which types participated in lifetime elision
697-
resolved_lifetime_elisions: Vec<ResolvedNestedElisionTarget>,
712+
resolved_lifetime_elisions: Vec<ResolvedElisionTarget>,
698713
}
699714

700715
/// Walks the whole crate in DFS order, visiting each item, resolving names as it goes.
@@ -1745,6 +1760,8 @@ impl<'a: 'ast, 'b, 'ast, 'tcx> LateResolutionVisitor<'a, 'b, 'ast, 'tcx> {
17451760
LifetimeElisionCandidate::Ignore,
17461761
);
17471762
self.resolve_anonymous_lifetime(&lt, true);
1763+
1764+
self.resolved_lifetime_elisions.push(ResolvedElisionTarget::TopLevel(anchor_id));
17481765
}
17491766

17501767
#[instrument(level = "debug", skip(self))]
@@ -1957,16 +1974,18 @@ impl<'a: 'ast, 'b, 'ast, 'tcx> LateResolutionVisitor<'a, 'b, 'ast, 'tcx> {
19571974
}
19581975

19591976
if should_lint {
1960-
self.resolved_lifetime_elisions.push(ResolvedNestedElisionTarget {
1961-
segment_id,
1962-
elided_lifetime_span,
1963-
diagnostic: lint::BuiltinLintDiagnostics::ElidedLifetimesInPaths(
1964-
expected_lifetimes,
1965-
path_span,
1966-
!segment.has_generic_args,
1977+
self.resolved_lifetime_elisions.push(ResolvedElisionTarget::Nested(
1978+
NestedResolvedElisionTarget {
1979+
segment_id,
19671980
elided_lifetime_span,
1968-
),
1969-
});
1981+
diagnostic: lint::BuiltinLintDiagnostics::ElidedLifetimesInPaths(
1982+
expected_lifetimes,
1983+
path_span,
1984+
!segment.has_generic_args,
1985+
elided_lifetime_span,
1986+
),
1987+
},
1988+
));
19701989
}
19711990
}
19721991
}
@@ -2028,22 +2047,79 @@ impl<'a: 'ast, 'b, 'ast, 'tcx> LateResolutionVisitor<'a, 'b, 'ast, 'tcx> {
20282047

20292048
let elision_failures =
20302049
replace(&mut self.diagnostic_metadata.current_elision_failures, outer_failures);
2031-
if !elision_failures.is_empty() {
2032-
let Err(failure_info) = elision_lifetime else { bug!() };
2033-
self.report_missing_lifetime_specifiers(elision_failures, Some(failure_info));
2034-
}
2050+
2051+
let elision_lifetime = match (elision_failures.is_empty(), elision_lifetime) {
2052+
(true, Ok(lifetime)) => Some(lifetime),
2053+
2054+
(true, Err(e)) => None,
2055+
2056+
(false, Ok(_)) => bug!(),
2057+
2058+
(false, Err(failure_info)) => {
2059+
self.report_missing_lifetime_specifiers(elision_failures, Some(failure_info));
2060+
None
2061+
}
2062+
};
2063+
2064+
// We've recorded all elisions that occurred in the params and
2065+
// outputs, categorized by top-level or nested.
2066+
//
2067+
// Our primary lint case is when an output lifetime is tied to
2068+
// an input lifetime. In that case, we want to warn about any
2069+
// parameters that had a nested elision.
2070+
//
2071+
// The secondary case is for nested elisions that are not part
2072+
// of the tied lifetime relationship.
20352073

20362074
let output_resolved_lifetime_elisions =
20372075
replace(&mut self.resolved_lifetime_elisions, outer_resolved_lifetime_elisions);
20382076

2039-
let resolved_lifetime_elisions =
2040-
param_resolved_lifetime_elisions.into_iter().chain(output_resolved_lifetime_elisions);
2077+
match (output_resolved_lifetime_elisions.is_empty(), elision_lifetime) {
2078+
(true, _) | (_, None) => {
2079+
// Treat all parameters as untied
2080+
self.report_elided_lifetimes_in_paths(
2081+
param_resolved_lifetime_elisions,
2082+
lint::builtin::ELIDED_LIFETIMES_IN_PATHS_UNTIED,
2083+
);
2084+
}
2085+
(false, Some(elision_lifetime)) => {
2086+
let (primary, secondary): (Vec<_>, Vec<_>) =
2087+
param_resolved_lifetime_elisions.into_iter().partition(|re| {
2088+
let lvl1 = &self.r.lifetimes_res_map[&re.node_id()];
2089+
let lvl2 = match lvl1 {
2090+
LifetimeRes::ElidedAnchor { start, .. } => {
2091+
&self.r.lifetimes_res_map[&start]
2092+
}
2093+
o => o,
2094+
};
2095+
2096+
lvl2 == &elision_lifetime
2097+
});
2098+
2099+
self.report_elided_lifetimes_in_paths(
2100+
primary.into_iter().chain(output_resolved_lifetime_elisions),
2101+
lint::builtin::ELIDED_LIFETIMES_IN_PATHS_TIED,
2102+
);
2103+
self.report_elided_lifetimes_in_paths(
2104+
secondary,
2105+
lint::builtin::ELIDED_LIFETIMES_IN_PATHS_UNTIED,
2106+
);
2107+
}
2108+
}
2109+
}
2110+
2111+
fn report_elided_lifetimes_in_paths(
2112+
&mut self,
2113+
resolved_elisions: impl IntoIterator<Item = ResolvedElisionTarget>,
2114+
lint: &'static lint::Lint,
2115+
) {
2116+
for re in resolved_elisions {
2117+
let ResolvedElisionTarget::Nested(d) = re else { continue };
20412118

2042-
for re in resolved_lifetime_elisions {
2043-
let ResolvedNestedElisionTarget { segment_id, elided_lifetime_span, diagnostic } = re;
2119+
let NestedResolvedElisionTarget { segment_id, elided_lifetime_span, diagnostic } = d;
20442120

20452121
self.r.lint_buffer.buffer_lint_with_diagnostic(
2046-
lint::builtin::ELIDED_LIFETIMES_IN_PATHS,
2122+
lint,
20472123
segment_id,
20482124
elided_lifetime_span,
20492125
"hidden lifetime parameters in types are deprecated",

0 commit comments

Comments
 (0)