Skip to content

Commit 22a13ef

Browse files
committed
resolve: Future-proof against imports referring to local variables and generic parameters
1 parent 696fbc0 commit 22a13ef

File tree

3 files changed

+134
-1
lines changed

3 files changed

+134
-1
lines changed

src/librustc_resolve/lib.rs

+35-1
Original file line numberDiff line numberDiff line change
@@ -2305,6 +2305,36 @@ impl<'a, 'crateloader: 'a> Resolver<'a, 'crateloader> {
23052305
});
23062306
}
23072307

2308+
fn future_proof_import(&mut self, use_tree: &ast::UseTree) {
2309+
if !self.session.rust_2018() {
2310+
return;
2311+
}
2312+
2313+
let segments = &use_tree.prefix.segments;
2314+
if !segments.is_empty() {
2315+
let ident = segments[0].ident;
2316+
if ident.is_path_segment_keyword() {
2317+
return;
2318+
}
2319+
2320+
let nss = match use_tree.kind {
2321+
ast::UseTreeKind::Simple(..) if segments.len() == 1 => &[TypeNS, ValueNS][..],
2322+
_ => &[TypeNS],
2323+
};
2324+
for &ns in nss {
2325+
if let Some(LexicalScopeBinding::Def(..)) =
2326+
self.resolve_ident_in_lexical_scope(ident, ns, None, use_tree.prefix.span) {
2327+
let what = if ns == TypeNS { "type parameters" } else { "local variables" };
2328+
self.session.span_err(ident.span, &format!("imports cannot refer to {}", what));
2329+
}
2330+
}
2331+
} else if let ast::UseTreeKind::Nested(use_trees) = &use_tree.kind {
2332+
for (use_tree, _) in use_trees {
2333+
self.future_proof_import(use_tree);
2334+
}
2335+
}
2336+
}
2337+
23082338
fn resolve_item(&mut self, item: &Item) {
23092339
let name = item.ident.name;
23102340
debug!("(resolving item) resolving {}", name);
@@ -2398,7 +2428,11 @@ impl<'a, 'crateloader: 'a> Resolver<'a, 'crateloader> {
23982428
});
23992429
}
24002430

2401-
ItemKind::Use(..) | ItemKind::ExternCrate(..) |
2431+
ItemKind::Use(ref use_tree) => {
2432+
self.future_proof_import(use_tree);
2433+
}
2434+
2435+
ItemKind::ExternCrate(..) |
24022436
ItemKind::MacroDef(..) | ItemKind::GlobalAsm(..) => {
24032437
// do nothing, these are just around to be encoded
24042438
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,49 @@
1+
// edition:2018
2+
3+
#![feature(uniform_paths, underscore_imports)]
4+
5+
mod T {
6+
pub struct U;
7+
}
8+
mod x {
9+
pub struct y;
10+
}
11+
12+
fn type_param<T>() {
13+
use T as _; //~ ERROR imports cannot refer to type parameters
14+
use T::U; //~ ERROR imports cannot refer to type parameters
15+
use T::*; //~ ERROR imports cannot refer to type parameters
16+
}
17+
18+
fn self_import<T>() {
19+
use T; // FIXME Should be an error, but future-proofing fails due to `T` being "self-shadowed"
20+
}
21+
22+
fn let_binding() {
23+
let x = 10;
24+
25+
use x as _; //~ ERROR imports cannot refer to local variables
26+
use x::y; // OK
27+
use x::*; // OK
28+
}
29+
30+
fn param_binding(x: u8) {
31+
use x; //~ ERROR imports cannot refer to local variables
32+
}
33+
34+
fn match_binding() {
35+
match 0 {
36+
x => {
37+
use x; //~ ERROR imports cannot refer to local variables
38+
}
39+
}
40+
}
41+
42+
fn nested<T>() {
43+
let x = 10;
44+
45+
use {T as _, x}; //~ ERROR imports cannot refer to type parameters
46+
//~| ERROR imports cannot refer to local variables
47+
}
48+
49+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
error: imports cannot refer to type parameters
2+
--> $DIR/future-proofing-locals.rs:13:9
3+
|
4+
LL | use T as _; //~ ERROR imports cannot refer to type parameters
5+
| ^
6+
7+
error: imports cannot refer to type parameters
8+
--> $DIR/future-proofing-locals.rs:14:9
9+
|
10+
LL | use T::U; //~ ERROR imports cannot refer to type parameters
11+
| ^
12+
13+
error: imports cannot refer to type parameters
14+
--> $DIR/future-proofing-locals.rs:15:9
15+
|
16+
LL | use T::*; //~ ERROR imports cannot refer to type parameters
17+
| ^
18+
19+
error: imports cannot refer to local variables
20+
--> $DIR/future-proofing-locals.rs:25:9
21+
|
22+
LL | use x as _; //~ ERROR imports cannot refer to local variables
23+
| ^
24+
25+
error: imports cannot refer to local variables
26+
--> $DIR/future-proofing-locals.rs:31:9
27+
|
28+
LL | use x; //~ ERROR imports cannot refer to local variables
29+
| ^
30+
31+
error: imports cannot refer to local variables
32+
--> $DIR/future-proofing-locals.rs:37:17
33+
|
34+
LL | use x; //~ ERROR imports cannot refer to local variables
35+
| ^
36+
37+
error: imports cannot refer to type parameters
38+
--> $DIR/future-proofing-locals.rs:45:10
39+
|
40+
LL | use {T as _, x}; //~ ERROR imports cannot refer to type parameters
41+
| ^
42+
43+
error: imports cannot refer to local variables
44+
--> $DIR/future-proofing-locals.rs:45:18
45+
|
46+
LL | use {T as _, x}; //~ ERROR imports cannot refer to type parameters
47+
| ^
48+
49+
error: aborting due to 8 previous errors
50+

0 commit comments

Comments
 (0)