Skip to content

Commit 2e03130

Browse files
Detect duplicates
1 parent f66558d commit 2e03130

File tree

7 files changed

+78
-8
lines changed

7 files changed

+78
-8
lines changed

compiler/rustc_ast_lowering/src/lib.rs

+7-4
Original file line numberDiff line numberDiff line change
@@ -1531,10 +1531,13 @@ impl<'a, 'hir> LoweringContext<'a, 'hir> {
15311531
let opaque_ty_span = self.mark_span_with_reason(DesugaringKind::OpaqueTy, span, None);
15321532

15331533
let captured_lifetimes_to_duplicate = if let Some(args) =
1534-
bounds.iter().find_map(|bound| match bound {
1535-
ast::GenericBound::Use(a, _) => Some(a),
1536-
_ => None,
1537-
}) {
1534+
// We only look for one `use<...>` syntax since we syntactially reject more than one.
1535+
bounds.iter().find_map(
1536+
|bound| match bound {
1537+
ast::GenericBound::Use(a, _) => Some(a),
1538+
_ => None,
1539+
},
1540+
) {
15381541
// We'll actually validate these later on; all we need is the list of
15391542
// lifetimes to duplicate during this portion of lowering.
15401543
args.iter()

compiler/rustc_ast_passes/messages.ftl

+5-2
Original file line numberDiff line numberDiff line change
@@ -14,8 +14,6 @@ ast_passes_assoc_type_without_body =
1414
associated type in `impl` without body
1515
.suggestion = provide a definition for the type
1616
17-
ast_passes_precise_capturing_not_allowed_here = `use<...>` precise capturing syntax not allowed in {$loc}
18-
1917
ast_passes_at_least_one_trait = at least one trait must be specified
2018
2119
ast_passes_auto_generic = auto traits cannot have generic parameters
@@ -217,6 +215,11 @@ ast_passes_pattern_in_fn_pointer = patterns aren't allowed in function pointer t
217215
ast_passes_pattern_in_foreign = patterns aren't allowed in foreign function declarations
218216
.label = pattern not allowed in foreign function
219217
218+
ast_passes_precise_capturing_duplicated = duplicate `use<...>` precise capturing syntax
219+
.label = second `use<...>` here
220+
221+
ast_passes_precise_capturing_not_allowed_here = `use<...>` precise capturing syntax not allowed in {$loc}
222+
220223
ast_passes_show_span = {$msg}
221224
222225
ast_passes_stability_outside_std = stability attributes may not be used outside of the standard library

compiler/rustc_ast_passes/src/ast_validation.rs

+18-2
Original file line numberDiff line numberDiff line change
@@ -193,8 +193,24 @@ impl<'a> AstValidator<'a> {
193193
// Mirrors `visit::walk_ty`, but tracks relevant state.
194194
fn walk_ty(&mut self, t: &'a Ty) {
195195
match &t.kind {
196-
TyKind::ImplTrait(..) => {
197-
self.with_impl_trait(Some(t.span), |this| visit::walk_ty(this, t))
196+
TyKind::ImplTrait(_, bounds) => {
197+
self.with_impl_trait(Some(t.span), |this| visit::walk_ty(this, t));
198+
199+
// FIXME(precise_capturing): If we were to allow `use` in other positions
200+
// (e.g. GATs), then we must validate those as well. However, we don't have
201+
// a good way of doing this with the current `Visitor` structure.
202+
let mut use_bounds = bounds
203+
.iter()
204+
.filter_map(|bound| match bound {
205+
GenericBound::Use(_, span) => Some(span),
206+
_ => None,
207+
})
208+
.copied();
209+
if let Some(bound1) = use_bounds.next()
210+
&& let Some(bound2) = use_bounds.next()
211+
{
212+
self.dcx().emit_err(errors::DuplicatePreciseCapturing { bound1, bound2 });
213+
}
198214
}
199215
TyKind::TraitObject(..) => self
200216
.with_tilde_const(Some(DisallowTildeConstContext::TraitObject), |this| {

compiler/rustc_ast_passes/src/errors.rs

+9
Original file line numberDiff line numberDiff line change
@@ -852,3 +852,12 @@ pub struct PreciseCapturingNotAllowedHere {
852852
pub span: Span,
853853
pub loc: &'static str,
854854
}
855+
856+
#[derive(Diagnostic)]
857+
#[diag(ast_passes_precise_capturing_duplicated)]
858+
pub struct DuplicatePreciseCapturing {
859+
#[primary_span]
860+
pub bound1: Span,
861+
#[label]
862+
pub bound2: Span,
863+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
warning: the feature `precise_capturing` is incomplete and may not be safe to use and/or cause compiler crashes
2+
--> $DIR/duplicated-use.rs:4:12
3+
|
4+
LL | #![feature(precise_capturing)]
5+
| ^^^^^^^^^^^^^^^^^
6+
|
7+
= note: see issue #123432 <https://github.com/rust-lang/rust/issues/123432> for more information
8+
= note: `#[warn(incomplete_features)]` on by default
9+
10+
warning: 1 warning emitted
11+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
error: duplicate `use<...>` precise capturing syntax
2+
--> $DIR/duplicated-use.rs:8:32
3+
|
4+
LL | fn hello<'a>() -> impl Sized + use<'a> + use<'a> {}
5+
| ^^^^^^^ ------- second `use<...>` here
6+
7+
warning: the feature `precise_capturing` is incomplete and may not be safe to use and/or cause compiler crashes
8+
--> $DIR/duplicated-use.rs:4:12
9+
|
10+
LL | #![feature(precise_capturing)]
11+
| ^^^^^^^^^^^^^^^^^
12+
|
13+
= note: see issue #123432 <https://github.com/rust-lang/rust/issues/123432> for more information
14+
= note: `#[warn(incomplete_features)]` on by default
15+
16+
error: aborting due to 1 previous error; 1 warning emitted
17+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
//@ revisions: real pre_expansion
2+
//@[pre_expansion] check-pass
3+
4+
#![feature(precise_capturing)]
5+
//~^ WARN the feature `precise_capturing` is incomplete
6+
7+
#[cfg(real)]
8+
fn hello<'a>() -> impl Sized + use<'a> + use<'a> {}
9+
//[real]~^ ERROR duplicate `use<...>` precise capturing syntax
10+
11+
fn main() {}

0 commit comments

Comments
 (0)