Skip to content

Commit c751961

Browse files
committed
Extend support of _ in type parameters
- Account for `impl Trait<_>`. - Provide a reasonable `Span` for empty `Generics` in `impl`s. - Account for `fn foo<_>(_: _) {}` to suggest `fn foo<T>(_: T) {}`. - Fix #67995.
1 parent ed6468d commit c751961

File tree

6 files changed

+91
-12
lines changed

6 files changed

+91
-12
lines changed

src/librustc_parse/parser/generics.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -156,7 +156,7 @@ impl<'a> Parser<'a> {
156156
self.expect_gt()?;
157157
(params, span_lo.to(self.prev_span))
158158
} else {
159-
(vec![], self.prev_span.between(self.token.span))
159+
(vec![], self.prev_span.shrink_to_hi())
160160
};
161161
Ok(ast::Generics {
162162
params,

src/librustc_parse/parser/item.rs

+5-1
Original file line numberDiff line numberDiff line change
@@ -555,7 +555,11 @@ impl<'a> Parser<'a> {
555555
let mut generics = if self.choose_generics_over_qpath() {
556556
self.parse_generics()?
557557
} else {
558-
Generics::default()
558+
let mut generics = Generics::default();
559+
// impl A for B {}
560+
// /\ this is where `generics.span` should point when there are no type params.
561+
generics.span = self.prev_span.shrink_to_hi();
562+
generics
559563
};
560564

561565
// Disambiguate `impl !Trait for Type { ... }` and `impl ! { ... }` for the never type.

src/librustc_typeck/astconv.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -2803,7 +2803,7 @@ impl<'o, 'tcx> dyn AstConv<'tcx> + 'o {
28032803
// allowed. `allow_ty_infer` gates this behavior.
28042804
crate::collect::placeholder_type_error(
28052805
tcx,
2806-
ident_span.unwrap_or(DUMMY_SP),
2806+
ident_span.map(|sp| sp.shrink_to_hi()).unwrap_or(DUMMY_SP),
28072807
generic_params,
28082808
visitor.0,
28092809
ident_span.is_some(),

src/librustc_typeck/collect.rs

+28-8
Original file line numberDiff line numberDiff line change
@@ -124,7 +124,7 @@ struct CollectItemTypesVisitor<'tcx> {
124124
/// all already existing generic type parameters to avoid suggesting a name that is already in use.
125125
crate fn placeholder_type_error(
126126
tcx: TyCtxt<'tcx>,
127-
ident_span: Span,
127+
span: Span,
128128
generics: &[hir::GenericParam<'_>],
129129
placeholder_types: Vec<Span>,
130130
suggest: bool,
@@ -150,7 +150,14 @@ crate fn placeholder_type_error(
150150
let mut sugg: Vec<_> =
151151
placeholder_types.iter().map(|sp| (*sp, type_name.to_string())).collect();
152152
if generics.is_empty() {
153-
sugg.push((ident_span.shrink_to_hi(), format!("<{}>", type_name)));
153+
sugg.push((span, format!("<{}>", type_name)));
154+
} else if let Some(arg) = generics.iter().find(|arg| match arg.name {
155+
hir::ParamName::Plain(Ident { name: kw::Underscore, .. }) => true,
156+
_ => false,
157+
}) {
158+
// Account for `_` already present in cases like `struct S<_>(_);` and suggest
159+
// `struct S<T>(T);` instead of `struct S<_, T>(T);`.
160+
sugg.push((arg.span, format!("{}", type_name)));
154161
} else {
155162
sugg.push((
156163
generics.iter().last().unwrap().span.shrink_to_hi(),
@@ -172,16 +179,20 @@ fn reject_placeholder_type_signatures_in_item(tcx: TyCtxt<'tcx>, item: &'tcx hir
172179
let (generics, suggest) = match &item.kind {
173180
hir::ItemKind::Union(_, generics)
174181
| hir::ItemKind::Enum(_, generics)
175-
| hir::ItemKind::Struct(_, generics) => (&generics.params[..], true),
176-
hir::ItemKind::TyAlias(_, generics) => (&generics.params[..], false),
182+
| hir::ItemKind::TraitAlias(generics, _)
183+
| hir::ItemKind::Trait(_, _, generics, ..)
184+
| hir::ItemKind::Impl(_, _, _, generics, ..)
185+
| hir::ItemKind::Struct(_, generics) => (generics, true),
186+
hir::ItemKind::OpaqueTy(hir::OpaqueTy { generics, .. })
187+
| hir::ItemKind::TyAlias(_, generics) => (generics, false),
177188
// `static`, `fn` and `const` are handled elsewhere to suggest appropriate type.
178189
_ => return,
179190
};
180191

181192
let mut visitor = PlaceholderHirTyCollector::default();
182193
visitor.visit_item(item);
183194

184-
placeholder_type_error(tcx, item.ident.span, generics, visitor.0, suggest);
195+
placeholder_type_error(tcx, generics.span, &generics.params[..], visitor.0, suggest);
185196
}
186197

187198
impl Visitor<'tcx> for CollectItemTypesVisitor<'tcx> {
@@ -1789,10 +1800,19 @@ fn find_opaque_ty_constraints(tcx: TyCtxt<'_>, def_id: DefId) -> Ty<'_> {
17891800
/// Whether `ty` is a type with `_` placeholders that can be infered. Used in diagnostics only to
17901801
/// use inference to provide suggestions for the appropriate type if possible.
17911802
fn is_suggestable_infer_ty(ty: &hir::Ty<'_>) -> bool {
1803+
use hir::TyKind::*;
17921804
match &ty.kind {
1793-
hir::TyKind::Infer => true,
1794-
hir::TyKind::Slice(ty) | hir::TyKind::Array(ty, _) => is_suggestable_infer_ty(ty),
1795-
hir::TyKind::Tup(tys) => tys.iter().any(|ty| is_suggestable_infer_ty(ty)),
1805+
Infer => true,
1806+
Slice(ty) | Array(ty, _) => is_suggestable_infer_ty(ty),
1807+
Tup(tys) => tys.iter().any(is_suggestable_infer_ty),
1808+
Ptr(mut_ty) | Rptr(_, mut_ty) => is_suggestable_infer_ty(mut_ty.ty),
1809+
Def(_, generic_args) => generic_args
1810+
.iter()
1811+
.filter_map(|arg| match arg {
1812+
hir::GenericArg::Type(ty) => Some(ty),
1813+
_ => None,
1814+
})
1815+
.any(is_suggestable_infer_ty),
17961816
_ => false,
17971817
}
17981818
}

src/test/ui/typeck/typeck_type_placeholder_item.rs

+13
Original file line numberDiff line numberDiff line change
@@ -131,3 +131,16 @@ trait T {
131131
fn assoc_fn_test3() -> _;
132132
//~^ ERROR the type placeholder `_` is not allowed within types on item signatures
133133
}
134+
135+
struct BadStruct<_>(_);
136+
//~^ ERROR expected identifier, found reserved identifier `_`
137+
//~| ERROR the type placeholder `_` is not allowed within types on item signatures
138+
trait BadTrait<_> {}
139+
//~^ ERROR expected identifier, found reserved identifier `_`
140+
impl BadTrait<_> for BadStruct<_> {}
141+
//~^ ERROR the type placeholder `_` is not allowed within types on item signatures
142+
143+
fn impl_trait() -> impl BadTrait<_> {
144+
//~^ ERROR the type placeholder `_` is not allowed within types on item signatures
145+
unimplemented!()
146+
}

src/test/ui/typeck/typeck_type_placeholder_item.stderr

+43-1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,15 @@
1+
error: expected identifier, found reserved identifier `_`
2+
--> $DIR/typeck_type_placeholder_item.rs:135:18
3+
|
4+
LL | struct BadStruct<_>(_);
5+
| ^ expected identifier, found reserved identifier
6+
7+
error: expected identifier, found reserved identifier `_`
8+
--> $DIR/typeck_type_placeholder_item.rs:138:16
9+
|
10+
LL | trait BadTrait<_> {}
11+
| ^ expected identifier, found reserved identifier
12+
113
error[E0121]: the type placeholder `_` is not allowed within types on item signatures
214
--> $DIR/typeck_type_placeholder_item.rs:4:14
315
|
@@ -255,6 +267,36 @@ LL | fn fn_test13(x: _) -> (i32, _) { (x, x) }
255267
| | not allowed in type signatures
256268
| help: replace with the correct return type: `(i32, i32)`
257269

270+
error[E0121]: the type placeholder `_` is not allowed within types on item signatures
271+
--> $DIR/typeck_type_placeholder_item.rs:135:21
272+
|
273+
LL | struct BadStruct<_>(_);
274+
| ^ not allowed in type signatures
275+
|
276+
help: use type parameters instead
277+
|
278+
LL | struct BadStruct<T>(T);
279+
| ^ ^
280+
281+
error[E0121]: the type placeholder `_` is not allowed within types on item signatures
282+
--> $DIR/typeck_type_placeholder_item.rs:140:15
283+
|
284+
LL | impl BadTrait<_> for BadStruct<_> {}
285+
| ^ ^ not allowed in type signatures
286+
| |
287+
| not allowed in type signatures
288+
|
289+
help: use type parameters instead
290+
|
291+
LL | impl<T> BadTrait<T> for BadStruct<T> {}
292+
| ^^^ ^ ^
293+
294+
error[E0121]: the type placeholder `_` is not allowed within types on item signatures
295+
--> $DIR/typeck_type_placeholder_item.rs:143:34
296+
|
297+
LL | fn impl_trait() -> impl BadTrait<_> {
298+
| ^ not allowed in type signatures
299+
258300
error[E0121]: the type placeholder `_` is not allowed within types on item signatures
259301
--> $DIR/typeck_type_placeholder_item.rs:121:31
260302
|
@@ -405,7 +447,7 @@ help: use type parameters instead
405447
LL | fn clone_from<T>(&mut self, other: T) { *self = FnTest9; }
406448
| ^^^ ^
407449

408-
error: aborting due to 40 previous errors
450+
error: aborting due to 45 previous errors
409451

410452
Some errors have detailed explanations: E0121, E0282.
411453
For more information about an error, try `rustc --explain E0121`.

0 commit comments

Comments
 (0)