Skip to content

Commit

Permalink
rollup merge of rust-lang#20728: huonw/type-param-shadowing
Browse files Browse the repository at this point in the history
Conflicts:
	src/librustc_typeck/check/wf.rs
  • Loading branch information
alexcrichton committed Jan 8, 2015
2 parents 773fdb3 + 92cd8ea commit 6afda64
Show file tree
Hide file tree
Showing 3 changed files with 78 additions and 2 deletions.
42 changes: 41 additions & 1 deletion src/librustc_typeck/check/wf.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@ use syntax::ast;
use syntax::ast_util::{local_def};
use syntax::attr;
use syntax::codemap::Span;
use syntax::parse::token;
use syntax::visit;
use syntax::visit::Visitor;

Expand Down Expand Up @@ -281,12 +282,45 @@ fn reject_non_type_param_bounds<'tcx>(tcx: &ty::ctxt<'tcx>,
}
}

fn reject_shadowing_type_parameters<'tcx>(tcx: &ty::ctxt<'tcx>,
span: Span,
generics: &ty::Generics<'tcx>) {
let impl_params = generics.types.get_slice(subst::TypeSpace).iter()
.map(|tp| tp.name).collect::<HashSet<_>>();

for method_param in generics.types.get_slice(subst::FnSpace).iter() {
if impl_params.contains(&method_param.name) {
tcx.sess.span_err(
span,
&*format!("type parameter `{}` shadows another type parameter of the same name",
token::get_name(method_param.name)));
}
}
}

impl<'ccx, 'tcx, 'v> Visitor<'v> for CheckTypeWellFormedVisitor<'ccx, 'tcx> {
fn visit_item(&mut self, i: &ast::Item) {
self.check_item_well_formed(i);
visit::walk_item(self, i);
}

fn visit_fn(&mut self,
fk: visit::FnKind<'v>, fd: &'v ast::FnDecl,
b: &'v ast::Block, span: Span, id: ast::NodeId) {
match fk {
visit::FkFnBlock | visit::FkItemFn(..) => {}
visit::FkMethod(..) => {
match ty::impl_or_trait_item(self.ccx.tcx, local_def(id)) {
ty::ImplOrTraitItem::MethodTraitItem(ty_method) => {
reject_shadowing_type_parameters(self.ccx.tcx, span, &ty_method.generics)
}
_ => {}
}
}
}
visit::walk_fn(self, fk, fd, b, span)
}

fn visit_trait_item(&mut self, t: &'v ast::TraitItem) {
match t {
&ast::TraitItem::ProvidedMethod(_) |
Expand All @@ -297,12 +331,18 @@ impl<'ccx, 'tcx, 'v> Visitor<'v> for CheckTypeWellFormedVisitor<'ccx, 'tcx> {
reject_non_type_param_bounds(
self.ccx.tcx,
method.span,
&ty_method.generics)
&ty_method.generics);
reject_shadowing_type_parameters(
self.ccx.tcx,
method.span,
&ty_method.generics);
}
_ => {}
}
}
}

visit::walk_trait_item(self, t)
}
}

Expand Down
2 changes: 1 addition & 1 deletion src/libserialize/json.rs
Original file line number Diff line number Diff line change
Expand Up @@ -1396,7 +1396,7 @@ impl<T: Iterator<Item=char>> Parser<T> {
self.ch == Some(c)
}

fn error<T>(&self, reason: ErrorCode) -> Result<T, ParserError> {
fn error<U>(&self, reason: ErrorCode) -> Result<U, ParserError> {
Err(SyntaxError(reason, self.line, self.col))
}

Expand Down
36 changes: 36 additions & 0 deletions src/test/compile-fail/shadowed-type-parameter.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
// Copyright 2013 The Rust Project Developers. See the COPYRIGHT
// file at the top-level directory of this distribution and at
// http://rust-lang.org/COPYRIGHT.
//
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
// option. This file may not be copied, modified, or distributed
// except according to those terms.

// Test that shadowed lifetimes generate an error.

struct Foo<T>;

impl<T> Foo<T> {
fn shadow_in_method<T>(&self) {}
//~^ ERROR type parameter `T` shadows another type parameter

fn not_shadow_in_item<U>(&self) {
struct Bar<T, U>; // not a shadow, separate item
fn foo<T, U>() {} // same
}
}

trait<T> Bar<T> {
fn shadow_in_required<T>(&self);
//~^ ERROR type parameter `T` shadows another type parameter

fn shadow_in_provided<T>(&self) {}
//~^ ERROR type parameter `T` shadows another type parameter

fn not_shadow_in_required<U>(&self);
fn not_shadow_in_provided<U>(&self) {}
}

fn main() {}

0 comments on commit 6afda64

Please sign in to comment.