1
Fork 0

Move type parameter shadowing errors to resolve

For some reason type checking did this. Further it didn't consider
hygiene.
This commit is contained in:
Matthew Jasper 2019-08-11 18:09:50 +01:00
parent 3296d0ed6d
commit f70c90c677
14 changed files with 116 additions and 117 deletions

View file

@ -169,12 +169,14 @@ impl<'a> Resolver<'a> {
err err
} }
ResolutionError::NameAlreadyUsedInParameterList(name, first_use_span) => { ResolutionError::NameAlreadyUsedInParameterList(name, first_use_span) => {
let mut err = struct_span_err!(self.session, let mut err = struct_span_err!(
span, self.session,
E0403, span,
"the name `{}` is already used for a generic \ E0403,
parameter in this list of generic parameters", "the name `{}` is already used for a generic \
name); parameter in this item's generic parameters",
name,
);
err.span_label(span, "already used"); err.span_label(span, "already used");
err.span_label(first_use_span, format!("first use of `{}`", name)); err.span_label(first_use_span, format!("first use of `{}`", name));
err err

View file

@ -526,15 +526,25 @@ Some type parameters have the same name.
Erroneous code example: Erroneous code example:
```compile_fail,E0403 ```compile_fail,E0403
fn foo<T, T>(s: T, u: T) {} // error: the name `T` is already used for a type fn f<T, T>(s: T, u: T) {} // error: the name `T` is already used for a generic
// parameter in this type parameter list // parameter in this item's generic parameters
``` ```
Please verify that none of the type parameters are misspelled, and rename any Please verify that none of the type parameters are misspelled, and rename any
clashing parameters. Example: clashing parameters. Example:
``` ```
fn foo<T, Y>(s: T, u: Y) {} // ok! fn f<T, Y>(s: T, u: Y) {} // ok!
```
Type parameters in an associated item also cannot shadow parameters from the
containing item:
```compile_fail,E0403
trait Foo<T> {
fn do_something(&self) -> T;
fn do_something_else<T: Clone>(&self, bar: T);
}
``` ```
"##, "##,

View file

@ -111,6 +111,24 @@ crate enum RibKind<'a> {
TyParamAsConstParamTy, TyParamAsConstParamTy,
} }
impl RibKind<'_> {
// Whether this rib kind contains generic parameters, as opposed to local
// variables.
crate fn contains_params(&self) -> bool {
match self {
NormalRibKind
| FnItemRibKind
| ConstantItemRibKind
| ModuleRibKind(_)
| MacroDefinition(_) => false,
AssocItemRibKind
| ItemRibKind
| ForwardTyParamBanRibKind
| TyParamAsConstParamTy => true,
}
}
}
/// A single local scope. /// A single local scope.
/// ///
/// A rib represents a scope names can live in. Note that these appear in many places, not just /// A rib represents a scope names can live in. Note that these appear in many places, not just
@ -798,6 +816,19 @@ impl<'a, 'b> LateResolutionVisitor<'a, '_> {
let mut function_type_rib = Rib::new(rib_kind); let mut function_type_rib = Rib::new(rib_kind);
let mut function_value_rib = Rib::new(rib_kind); let mut function_value_rib = Rib::new(rib_kind);
let mut seen_bindings = FxHashMap::default(); let mut seen_bindings = FxHashMap::default();
// We also can't shadow bindings from the parent item
if let AssocItemRibKind = rib_kind {
let mut add_bindings_for_ns = |ns| {
let parent_rib = self.ribs[ns].iter()
.rfind(|rib| if let ItemRibKind = rib.kind { true } else { false })
.expect("associated item outside of an item");
seen_bindings.extend(
parent_rib.bindings.iter().map(|(ident, _)| (*ident, ident.span)),
);
};
add_bindings_for_ns(ValueNS);
add_bindings_for_ns(TypeNS);
}
for param in &generics.params { for param in &generics.params {
match param.kind { match param.kind {
GenericParamKind::Lifetime { .. } => {} GenericParamKind::Lifetime { .. } => {}

View file

@ -1499,7 +1499,7 @@ impl<'a> Resolver<'a> {
debug!("walk rib\n{:?}", ribs[i].bindings); debug!("walk rib\n{:?}", ribs[i].bindings);
// Use the rib kind to determine whether we are resolving parameters // Use the rib kind to determine whether we are resolving parameters
// (modern hygiene) or local variables (legacy hygiene). // (modern hygiene) or local variables (legacy hygiene).
let rib_ident = if let AssocItemRibKind | ItemRibKind = ribs[i].kind { let rib_ident = if ribs[i].kind.contains_params() {
modern_ident modern_ident
} else { } else {
ident ident

View file

@ -203,7 +203,6 @@ fn check_associated_item(
fcx.register_wf_obligation(ty, span, code.clone()); fcx.register_wf_obligation(ty, span, code.clone());
} }
ty::AssocKind::Method => { ty::AssocKind::Method => {
reject_shadowing_parameters(fcx.tcx, item.def_id);
let sig = fcx.tcx.fn_sig(item.def_id); let sig = fcx.tcx.fn_sig(item.def_id);
let sig = fcx.normalize_associated_types_in(span, &sig); let sig = fcx.normalize_associated_types_in(span, &sig);
check_fn_or_method(tcx, fcx, span, sig, check_fn_or_method(tcx, fcx, span, sig,
@ -998,34 +997,6 @@ fn report_bivariance(tcx: TyCtxt<'_>, span: Span, param_name: ast::Name) {
err.emit(); err.emit();
} }
fn reject_shadowing_parameters(tcx: TyCtxt<'_>, def_id: DefId) {
let generics = tcx.generics_of(def_id);
let parent = tcx.generics_of(generics.parent.unwrap());
let impl_params: FxHashMap<_, _> = parent.params.iter().flat_map(|param| match param.kind {
GenericParamDefKind::Lifetime => None,
GenericParamDefKind::Type { .. } | GenericParamDefKind::Const => {
Some((param.name, param.def_id))
}
}).collect();
for method_param in &generics.params {
// Shadowing is checked in `resolve_lifetime`.
if let GenericParamDefKind::Lifetime = method_param.kind {
continue
}
if impl_params.contains_key(&method_param.name) {
// Tighten up the span to focus on only the shadowing type.
let type_span = tcx.def_span(method_param.def_id);
// The expectation here is that the original trait declaration is
// local so it should be okay to just unwrap everything.
let trait_def_id = impl_params[&method_param.name];
let trait_decl_span = tcx.def_span(trait_def_id);
error_194(tcx, type_span, trait_decl_span, &method_param.name.as_str()[..]);
}
}
}
/// Feature gates RFC 2056 -- trivial bounds, checking for global bounds that /// Feature gates RFC 2056 -- trivial bounds, checking for global bounds that
/// aren't true. /// aren't true.
fn check_false_global_bounds(fcx: &FnCtxt<'_, '_>, span: Span, id: hir::HirId) { fn check_false_global_bounds(fcx: &FnCtxt<'_, '_>, span: Span, id: hir::HirId) {
@ -1152,12 +1123,3 @@ fn error_392(
err.span_label(span, "unused parameter"); err.span_label(span, "unused parameter");
err err
} }
fn error_194(tcx: TyCtxt<'_>, span: Span, trait_decl_span: Span, name: &str) {
struct_span_err!(tcx.sess, span, E0194,
"type parameter `{}` shadows another type parameter of the same name",
name)
.span_label(span, "shadows another type parameter")
.span_label(trait_decl_span, format!("first `{}` declared here", name))
.emit();
}

View file

@ -1718,22 +1718,6 @@ Since we know for certain that `Wrapper<u32>` implements `Clone`, there's no
reason to also specify it in a `where` clause. reason to also specify it in a `where` clause.
"##, "##,
E0194: r##"
A type parameter was declared which shadows an existing one. An example of this
error:
```compile_fail,E0194
trait Foo<T> {
fn do_something(&self) -> T;
fn do_something_else<T: Clone>(&self, bar: T);
}
```
In this example, the trait `Foo` and the trait method `do_something_else` both
define a type parameter `T`. This is not allowed: if the method wishes to
define a type parameter, it must use a different name for it.
"##,
E0195: r##" E0195: r##"
Your method's lifetime parameters do not match the trait declaration. Your method's lifetime parameters do not match the trait declaration.
Erroneous code example: Erroneous code example:
@ -4837,6 +4821,7 @@ register_diagnostics! {
// E0188, // can not cast an immutable reference to a mutable pointer // E0188, // can not cast an immutable reference to a mutable pointer
// E0189, // deprecated: can only cast a boxed pointer to a boxed object // E0189, // deprecated: can only cast a boxed pointer to a boxed object
// E0190, // deprecated: can only cast a &-pointer to an &-object // E0190, // deprecated: can only cast a &-pointer to an &-object
// E0194, // merged into E0403
// E0196, // cannot determine a type for this closure // E0196, // cannot determine a type for this closure
E0203, // type parameter has more than one relaxed default bound, E0203, // type parameter has more than one relaxed default bound,
// and only one is supported // and only one is supported

View file

@ -1,4 +1,4 @@
error[E0403]: the name `T` is already used for a generic parameter in this list of generic parameters error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/duplicate-type-parameter.rs:1:12 --> $DIR/duplicate-type-parameter.rs:1:12
| |
LL | type Foo<T,T> = Option<T>; LL | type Foo<T,T> = Option<T>;
@ -6,7 +6,7 @@ LL | type Foo<T,T> = Option<T>;
| | | |
| first use of `T` | first use of `T`
error[E0403]: the name `T` is already used for a generic parameter in this list of generic parameters error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/duplicate-type-parameter.rs:4:14 --> $DIR/duplicate-type-parameter.rs:4:14
| |
LL | struct Bar<T,T>(T); LL | struct Bar<T,T>(T);
@ -14,7 +14,7 @@ LL | struct Bar<T,T>(T);
| | | |
| first use of `T` | first use of `T`
error[E0403]: the name `T` is already used for a generic parameter in this list of generic parameters error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/duplicate-type-parameter.rs:7:14 --> $DIR/duplicate-type-parameter.rs:7:14
| |
LL | struct Baz<T,T> { LL | struct Baz<T,T> {
@ -22,7 +22,7 @@ LL | struct Baz<T,T> {
| | | |
| first use of `T` | first use of `T`
error[E0403]: the name `T` is already used for a generic parameter in this list of generic parameters error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/duplicate-type-parameter.rs:12:12 --> $DIR/duplicate-type-parameter.rs:12:12
| |
LL | enum Boo<T,T> { LL | enum Boo<T,T> {
@ -30,7 +30,7 @@ LL | enum Boo<T,T> {
| | | |
| first use of `T` | first use of `T`
error[E0403]: the name `T` is already used for a generic parameter in this list of generic parameters error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/duplicate-type-parameter.rs:18:11 --> $DIR/duplicate-type-parameter.rs:18:11
| |
LL | fn quux<T,T>(x: T) {} LL | fn quux<T,T>(x: T) {}
@ -38,7 +38,7 @@ LL | fn quux<T,T>(x: T) {}
| | | |
| first use of `T` | first use of `T`
error[E0403]: the name `T` is already used for a generic parameter in this list of generic parameters error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/duplicate-type-parameter.rs:21:13 --> $DIR/duplicate-type-parameter.rs:21:13
| |
LL | trait Qux<T,T> {} LL | trait Qux<T,T> {}
@ -46,7 +46,7 @@ LL | trait Qux<T,T> {}
| | | |
| first use of `T` | first use of `T`
error[E0403]: the name `T` is already used for a generic parameter in this list of generic parameters error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/duplicate-type-parameter.rs:24:8 --> $DIR/duplicate-type-parameter.rs:24:8
| |
LL | impl<T,T> Qux<T,T> for Option<T> {} LL | impl<T,T> Qux<T,T> for Option<T> {}

View file

@ -1,7 +1,7 @@
trait Foo<T> { trait Foo<T> {
fn do_something(&self) -> T; fn do_something(&self) -> T;
fn do_something_else<T: Clone>(&self, bar: T); fn do_something_else<T: Clone>(&self, bar: T);
//~^ ERROR E0194 //~^ ERROR E0403
} }
fn main() { fn main() {

View file

@ -1,12 +1,12 @@
error[E0194]: type parameter `T` shadows another type parameter of the same name error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/E0194.rs:3:26 --> $DIR/E0194.rs:3:26
| |
LL | trait Foo<T> { LL | trait Foo<T> {
| - first `T` declared here | - first use of `T`
LL | fn do_something(&self) -> T; LL | fn do_something(&self) -> T;
LL | fn do_something_else<T: Clone>(&self, bar: T); LL | fn do_something_else<T: Clone>(&self, bar: T);
| ^ shadows another type parameter | ^ already used
error: aborting due to previous error error: aborting due to previous error
For more information about this error, try `rustc --explain E0194`. For more information about this error, try `rustc --explain E0403`.

View file

@ -1,4 +1,4 @@
error[E0403]: the name `T` is already used for a generic parameter in this list of generic parameters error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/E0403.rs:1:11 --> $DIR/E0403.rs:1:11
| |
LL | fn foo<T, T>(s: T, u: T) {} LL | fn foo<T, T>(s: T, u: T) {}

View file

@ -1,12 +1,9 @@
#![allow(incomplete_features)]
#![feature(generic_associated_types)] #![feature(generic_associated_types)]
//FIXME(#44265): The lifetime shadowing and type parameter shadowing
// should cause an error. Now it compiles (erroneously) and this will be addressed
// by a future PR. Then remove the following:
// build-pass (FIXME(62277): could be check-pass?)
trait Shadow<'a> { trait Shadow<'a> {
type Bar<'a>; // Error: shadowed lifetime //FIXME(#44265): The lifetime parameter shadowing should cause an error.
type Bar<'a>;
} }
trait NoShadow<'a> { trait NoShadow<'a> {
@ -14,11 +11,12 @@ trait NoShadow<'a> {
} }
impl<'a> NoShadow<'a> for &'a u32 { impl<'a> NoShadow<'a> for &'a u32 {
type Bar<'a> = i32; // Error: shadowed lifetime //FIXME(#44265): The lifetime parameter shadowing should cause an error.
type Bar<'a> = i32;
} }
trait ShadowT<T> { trait ShadowT<T> {
type Bar<T>; // Error: shadowed type parameter type Bar<T>; //~ ERROR the name `T` is already used
} }
trait NoShadowT<T> { trait NoShadowT<T> {
@ -26,7 +24,7 @@ trait NoShadowT<T> {
} }
impl<T> NoShadowT<T> for Option<T> { impl<T> NoShadowT<T> for Option<T> {
type Bar<T> = i32; // Error: shadowed type parameter type Bar<T> = i32; //~ ERROR the name `T` is already used
} }
fn main() {} fn main() {}

View file

@ -1,8 +1,19 @@
warning: the feature `generic_associated_types` is incomplete and may cause the compiler to crash error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/shadowing.rs:1:12 --> $DIR/shadowing.rs:19:14
| |
LL | #![feature(generic_associated_types)] LL | trait ShadowT<T> {
| ^^^^^^^^^^^^^^^^^^^^^^^^ | - first use of `T`
| LL | type Bar<T>;
= note: `#[warn(incomplete_features)]` on by default | ^ already used
error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/shadowing.rs:27:14
|
LL | impl<T> NoShadowT<T> for Option<T> {
| - first use of `T`
LL | type Bar<T> = i32;
| ^ already used
error: aborting due to 2 previous errors
For more information about this error, try `rustc --explain E0403`.

View file

@ -6,7 +6,7 @@ struct Foo<T>(T);
impl<T> Foo<T> { impl<T> Foo<T> {
fn shadow_in_method<T>(&self) {} fn shadow_in_method<T>(&self) {}
//~^ ERROR type parameter `T` shadows another type parameter //~^ ERROR the name `T` is already used
fn not_shadow_in_item<U>(&self) { fn not_shadow_in_item<U>(&self) {
struct Bar<T, U>(T,U); // not a shadow, separate item struct Bar<T, U>(T,U); // not a shadow, separate item
@ -18,10 +18,10 @@ trait Bar<T> {
fn dummy(&self) -> T; fn dummy(&self) -> T;
fn shadow_in_required<T>(&self); fn shadow_in_required<T>(&self);
//~^ ERROR type parameter `T` shadows another type parameter //~^ ERROR the name `T` is already used
fn shadow_in_provided<T>(&self) {} fn shadow_in_provided<T>(&self) {}
//~^ ERROR type parameter `T` shadows another type parameter //~^ ERROR the name `T` is already used
fn not_shadow_in_required<U>(&self); fn not_shadow_in_required<U>(&self);
fn not_shadow_in_provided<U>(&self) {} fn not_shadow_in_provided<U>(&self) {}

View file

@ -1,29 +1,29 @@
error[E0194]: type parameter `T` shadows another type parameter of the same name error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/shadowed-type-parameter.rs:20:27
|
LL | trait Bar<T> {
| - first `T` declared here
...
LL | fn shadow_in_required<T>(&self);
| ^ shadows another type parameter
error[E0194]: type parameter `T` shadows another type parameter of the same name
--> $DIR/shadowed-type-parameter.rs:23:27
|
LL | trait Bar<T> {
| - first `T` declared here
...
LL | fn shadow_in_provided<T>(&self) {}
| ^ shadows another type parameter
error[E0194]: type parameter `T` shadows another type parameter of the same name
--> $DIR/shadowed-type-parameter.rs:8:25 --> $DIR/shadowed-type-parameter.rs:8:25
| |
LL | impl<T> Foo<T> { LL | impl<T> Foo<T> {
| - first `T` declared here | - first use of `T`
LL | fn shadow_in_method<T>(&self) {} LL | fn shadow_in_method<T>(&self) {}
| ^ shadows another type parameter | ^ already used
error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/shadowed-type-parameter.rs:20:27
|
LL | trait Bar<T> {
| - first use of `T`
...
LL | fn shadow_in_required<T>(&self);
| ^ already used
error[E0403]: the name `T` is already used for a generic parameter in this item's generic parameters
--> $DIR/shadowed-type-parameter.rs:23:27
|
LL | trait Bar<T> {
| - first use of `T`
...
LL | fn shadow_in_provided<T>(&self) {}
| ^ already used
error: aborting due to 3 previous errors error: aborting due to 3 previous errors
For more information about this error, try `rustc --explain E0194`. For more information about this error, try `rustc --explain E0403`.