Skip to content

Commit cf636c2

Browse files
committed
Auto merge of #22093 - petrochenkov:builtin, r=pnkfelix
Names of structs, enums, traits, type aliases and type parameters (i.e. all identifiers that can be used as full paths in type position) are not allowed to match the names of primitive types. See #20427 for more information. This is a minor [breaking-change]
2 parents ba2efe9 + 8ed58d8 commit cf636c2

File tree

4 files changed

+110
-2
lines changed

4 files changed

+110
-2
lines changed

src/librustc_resolve/diagnostics.rs

+2-1
Original file line numberDiff line numberDiff line change
@@ -23,7 +23,8 @@ register_diagnostics! {
2323
E0257, // inherent implementations are only allowed on types defined in the current module
2424
E0258, // import conflicts with existing submodule
2525
E0259, // an extern crate has already been imported into this module
26-
E0260 // name conflicts with an external crate that has been imported into this module
26+
E0260, // name conflicts with an external crate that has been imported into this module
27+
E0317 // user-defined types or type parameters cannot shadow the primitive types
2728
}
2829

2930
__build_diagnostic_array! { DIAGNOSTICS }

src/librustc_resolve/lib.rs

+29-1
Original file line numberDiff line numberDiff line change
@@ -2788,6 +2788,13 @@ impl<'a, 'tcx> Resolver<'a, 'tcx> {
27882788
visit::walk_crate(self, krate);
27892789
}
27902790

2791+
fn check_if_primitive_type_name(&self, name: Name, span: Span) {
2792+
if let Some(_) = self.primitive_type_table.primitive_types.get(&name) {
2793+
span_err!(self.session, span, E0317,
2794+
"user-defined types or type parameters cannot shadow the primitive types");
2795+
}
2796+
}
2797+
27912798
fn resolve_item(&mut self, item: &Item) {
27922799
let name = item.ident.name;
27932800

@@ -2799,6 +2806,8 @@ impl<'a, 'tcx> Resolver<'a, 'tcx> {
27992806
// enum item: resolve all the variants' discrs,
28002807
// then resolve the ty params
28012808
ItemEnum(ref enum_def, ref generics) => {
2809+
self.check_if_primitive_type_name(name, item.span);
2810+
28022811
for variant in &(*enum_def).variants {
28032812
if let Some(ref dis_expr) = variant.node.disr_expr {
28042813
// resolve the discriminator expr
@@ -2824,6 +2833,8 @@ impl<'a, 'tcx> Resolver<'a, 'tcx> {
28242833
}
28252834

28262835
ItemTy(_, ref generics) => {
2836+
self.check_if_primitive_type_name(name, item.span);
2837+
28272838
self.with_type_parameter_rib(HasTypeParameters(generics,
28282839
TypeSpace,
28292840
item.id,
@@ -2847,6 +2858,8 @@ impl<'a, 'tcx> Resolver<'a, 'tcx> {
28472858
}
28482859

28492860
ItemTrait(_, ref generics, ref bounds, ref trait_items) => {
2861+
self.check_if_primitive_type_name(name, item.span);
2862+
28502863
// Create a new rib for the self type.
28512864
let mut self_type_rib = Rib::new(ItemRibKind);
28522865

@@ -2919,6 +2932,8 @@ impl<'a, 'tcx> Resolver<'a, 'tcx> {
29192932
}
29202933

29212934
ItemStruct(ref struct_def, ref generics) => {
2935+
self.check_if_primitive_type_name(name, item.span);
2936+
29222937
self.resolve_struct(item.id,
29232938
generics,
29242939
&struct_def.fields[]);
@@ -2972,7 +2987,19 @@ impl<'a, 'tcx> Resolver<'a, 'tcx> {
29722987
});
29732988
}
29742989

2975-
ItemExternCrate(_) | ItemUse(_) | ItemMac(..) => {
2990+
ItemUse(ref view_path) => {
2991+
// check for imports shadowing primitive types
2992+
if let ast::ViewPathSimple(ident, _) = view_path.node {
2993+
match self.def_map.borrow().get(&item.id) {
2994+
Some(&DefTy(..)) | Some(&DefStruct(..)) | Some(&DefTrait(..)) | None => {
2995+
self.check_if_primitive_type_name(ident.name, item.span);
2996+
}
2997+
_ => {}
2998+
}
2999+
}
3000+
}
3001+
3002+
ItemExternCrate(_) | ItemMac(..) => {
29763003
// do nothing, these are just around to be encoded
29773004
}
29783005
}
@@ -3114,6 +3141,7 @@ impl<'a, 'tcx> Resolver<'a, 'tcx> {
31143141

31153142
fn resolve_type_parameter(&mut self,
31163143
type_parameter: &TyParam) {
3144+
self.check_if_primitive_type_name(type_parameter.ident.name, type_parameter.span);
31173145
for bound in &*type_parameter.bounds {
31183146
self.resolve_type_parameter_bound(type_parameter.id, bound,
31193147
TraitBoundingTypeParameter);

src/test/auxiliary/i8.rs

+13
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,13 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
// A crate named after a built-in type.
12+
13+
pub struct Test;

src/test/compile-fail/issue-20427.rs

+66
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,66 @@
1+
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
2+
// file at the top-level directory of this distribution and at
3+
// http://rust-lang.org/COPYRIGHT.
4+
//
5+
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
6+
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
7+
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
8+
// option. This file may not be copied, modified, or distributed
9+
// except according to those terms.
10+
11+
// aux-build:i8.rs
12+
extern crate i8;
13+
use std::string as i16;
14+
static i32: i32 = 0;
15+
const i64: i64 = 0;
16+
fn u8(f32: f32) {}
17+
fn f<f64>(f64: f64) {}
18+
//~^ ERROR user-defined types or type parameters cannot shadow the primitive types
19+
type u16 = u16; //~ ERROR user-defined types or type parameters cannot shadow the primitive types
20+
enum u32 {} //~ ERROR user-defined types or type parameters cannot shadow the primitive types
21+
struct u64; //~ ERROR user-defined types or type parameters cannot shadow the primitive types
22+
trait bool {} //~ ERROR user-defined types or type parameters cannot shadow the primitive types
23+
24+
mod char {
25+
extern crate i8;
26+
static i32_: i32 = 0;
27+
const i64_: i64 = 0;
28+
fn u8_(f32: f32) {}
29+
fn f_<f64_>(f64: f64_) {}
30+
type u16_ = u16;
31+
enum u32_ {}
32+
struct u64_;
33+
trait bool_ {}
34+
mod char_ {}
35+
36+
mod str {
37+
use super::i8 as i8;
38+
use super::i32_ as i32;
39+
use super::i64_ as i64;
40+
use super::u8_ as u8;
41+
use super::f_ as f64;
42+
use super::u16_ as u16;
43+
//~^ ERROR user-defined types or type parameters cannot shadow the primitive types
44+
use super::u32_ as u32;
45+
//~^ ERROR user-defined types or type parameters cannot shadow the primitive types
46+
use super::u64_ as u64;
47+
//~^ ERROR user-defined types or type parameters cannot shadow the primitive types
48+
use super::bool_ as bool;
49+
//~^ ERROR user-defined types or type parameters cannot shadow the primitive types
50+
use super::char_ as char;
51+
}
52+
}
53+
54+
trait isize_ {
55+
type isize; //~ ERROR user-defined types or type parameters cannot shadow the primitive types
56+
}
57+
58+
fn usize<'usize>(usize: &'usize usize) -> &'usize usize { usize }
59+
60+
fn main() {
61+
let bool = true;
62+
match bool {
63+
str @ true => if str { i32 as i64 } else { 0 },
64+
false => i64,
65+
}
66+
}

0 commit comments

Comments
 (0)