Skip to content

Commit 8a4f8e6

Browse files
authored
Rollup merge of rust-lang#58381 - davidtwco:issue-42944, r=estebank
Only suggest imports if not imported. Fixes rust-lang#42944 and fixes rust-lang#53430. This commit modifies name resolution error reporting so that if a name is in scope and has been imported then we do not suggest importing it. This can occur when we add a label about constructors not being visible due to private fields. In these cases, we know that the struct/variant has been imported and we should silence any suggestions to import the struct/variant. r? @estebank
2 parents b39b625 + 48b0c9d commit 8a4f8e6

File tree

5 files changed

+63
-17
lines changed

5 files changed

+63
-17
lines changed

src/librustc_resolve/error_reporting.rs

+9-1
Original file line numberDiff line numberDiff line change
@@ -106,7 +106,15 @@ impl<'a> Resolver<'a> {
106106

107107
// Try to lookup name in more relaxed fashion for better error reporting.
108108
let ident = path.last().unwrap().ident;
109-
let candidates = self.lookup_import_candidates(ident, ns, is_expected);
109+
let candidates = self.lookup_import_candidates(ident, ns, is_expected)
110+
.drain(..)
111+
.filter(|ImportSuggestion { did, .. }| {
112+
match (did, def.and_then(|def| def.opt_def_id())) {
113+
(Some(suggestion_did), Some(actual_did)) => *suggestion_did != actual_did,
114+
_ => true,
115+
}
116+
})
117+
.collect::<Vec<_>>();
110118
if candidates.is_empty() && is_expected(Def::Enum(DefId::local(CRATE_DEF_INDEX))) {
111119
let enum_candidates =
112120
self.lookup_import_candidates(ident, ns, is_enum_variant);

src/librustc_resolve/lib.rs

+12-4
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ use rustc::hir::def::*;
2424
use rustc::hir::def::Namespace::*;
2525
use rustc::hir::def_id::{CRATE_DEF_INDEX, LOCAL_CRATE, DefId};
2626
use rustc::hir::{Freevar, FreevarMap, TraitCandidate, TraitMap, GlobMap};
27-
use rustc::ty;
27+
use rustc::ty::{self, DefIdTree};
2828
use rustc::util::nodemap::{NodeMap, NodeSet, FxHashMap, FxHashSet, DefIdMap};
2929
use rustc::{bug, span_bug};
3030

@@ -92,6 +92,7 @@ enum ScopeSet {
9292

9393
/// A free importable items suggested in case of resolution failure.
9494
struct ImportSuggestion {
95+
did: Option<DefId>,
9596
path: Path,
9697
}
9798

@@ -4391,7 +4392,8 @@ impl<'a> Resolver<'a> {
43914392

43924393
// collect results based on the filter function
43934394
if ident.name == lookup_ident.name && ns == namespace {
4394-
if filter_fn(name_binding.def()) {
4395+
let def = name_binding.def();
4396+
if filter_fn(def) {
43954397
// create the path
43964398
let mut segms = path_segments.clone();
43974399
if lookup_ident.span.rust_2018() {
@@ -4415,7 +4417,12 @@ impl<'a> Resolver<'a> {
44154417
// declared as public (due to pruning, we don't explore
44164418
// outside crate private modules => no need to check this)
44174419
if !in_module_is_extern || name_binding.vis == ty::Visibility::Public {
4418-
candidates.push(ImportSuggestion { path });
4420+
let did = match def {
4421+
Def::StructCtor(did, _) | Def::VariantCtor(did, _) =>
4422+
self.parent(did),
4423+
_ => def.opt_def_id(),
4424+
};
4425+
candidates.push(ImportSuggestion { did, path });
44194426
}
44204427
}
44214428
}
@@ -4512,7 +4519,8 @@ impl<'a> Resolver<'a> {
45124519
span: name_binding.span,
45134520
segments: path_segments,
45144521
};
4515-
result = Some((module, ImportSuggestion { path }));
4522+
let did = module.def().and_then(|def| def.opt_def_id());
4523+
result = Some((module, ImportSuggestion { did, path }));
45164524
} else {
45174525
// add the module to the lookup
45184526
if seen_modules.insert(module.def_id().unwrap()) {

src/test/ui/issue-42944.rs

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
mod foo {
2+
pub struct B(());
3+
}
4+
5+
mod bar {
6+
use foo::B;
7+
8+
fn foo() {
9+
B(()); //~ ERROR expected function, found struct `B` [E0423]
10+
}
11+
}
12+
13+
mod baz {
14+
fn foo() {
15+
B(()); //~ ERROR cannot find function `B` in this scope [E0425]
16+
}
17+
}
18+
19+
fn main() {}

src/test/ui/issue-42944.stderr

+20
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
error[E0423]: expected function, found struct `B`
2+
--> $DIR/issue-42944.rs:9:9
3+
|
4+
LL | B(()); //~ ERROR expected function, found struct `B` [E0423]
5+
| ^ constructor is not visible here due to private fields
6+
7+
error[E0425]: cannot find function `B` in this scope
8+
--> $DIR/issue-42944.rs:15:9
9+
|
10+
LL | B(()); //~ ERROR cannot find function `B` in this scope [E0425]
11+
| ^ not found in this scope
12+
help: possible candidate is found in another module, you can import it into scope
13+
|
14+
LL | use foo::B;
15+
|
16+
17+
error: aborting due to 2 previous errors
18+
19+
Some errors occurred: E0423, E0425.
20+
For more information about an error, try `rustc --explain E0423`.

src/test/ui/resolve/privacy-struct-ctor.stderr

+3-12
Original file line numberDiff line numberDiff line change
@@ -2,25 +2,16 @@ error[E0423]: expected value, found struct `Z`
22
--> $DIR/privacy-struct-ctor.rs:20:9
33
|
44
LL | Z;
5-
| ^ constructor is not visible here due to private fields
6-
help: a tuple struct with a similar name exists
7-
|
8-
LL | S;
95
| ^
10-
help: possible better candidate is found in another module, you can import it into scope
11-
|
12-
LL | use m::n::Z;
13-
|
6+
| |
7+
| constructor is not visible here due to private fields
8+
| help: a tuple struct with a similar name exists: `S`
149

1510
error[E0423]: expected value, found struct `S`
1611
--> $DIR/privacy-struct-ctor.rs:33:5
1712
|
1813
LL | S;
1914
| ^ constructor is not visible here due to private fields
20-
help: possible better candidate is found in another module, you can import it into scope
21-
|
22-
LL | use m::S;
23-
|
2415

2516
error[E0423]: expected value, found struct `S2`
2617
--> $DIR/privacy-struct-ctor.rs:38:5

0 commit comments

Comments
 (0)