Skip to content

Commit d38a8ad

Browse files
committed
Improve diagnostics for inaccessible constructors
1 parent c9788fd commit d38a8ad

File tree

6 files changed

+103
-12
lines changed

6 files changed

+103
-12
lines changed

src/librustc_resolve/build_reduced_graph.rs

+11-5
Original file line numberDiff line numberDiff line change
@@ -345,9 +345,7 @@ impl<'a> Resolver<'a> {
345345
let ctor_def = Def::StructCtor(self.definitions.local_def_id(struct_def.id()),
346346
CtorKind::from_ast(struct_def));
347347
self.define(parent, ident, ValueNS, (ctor_def, ctor_vis, sp, expansion));
348-
if !ctor_vis.is_at_least(vis, &*self) {
349-
self.legacy_ctor_visibilities.insert(def.def_id(), (ctor_def, ctor_vis));
350-
}
348+
self.struct_constructors.insert(def.def_id(), (ctor_def, ctor_vis));
351349
}
352350
}
353351

@@ -441,9 +439,17 @@ impl<'a> Resolver<'a> {
441439
Def::Variant(..) | Def::TyAlias(..) => {
442440
self.define(parent, ident, TypeNS, (def, vis, DUMMY_SP, Mark::root()));
443441
}
444-
Def::Fn(..) | Def::Static(..) | Def::Const(..) |
445-
Def::VariantCtor(..) | Def::StructCtor(..) => {
442+
Def::Fn(..) | Def::Static(..) | Def::Const(..) | Def::VariantCtor(..) => {
443+
self.define(parent, ident, ValueNS, (def, vis, DUMMY_SP, Mark::root()));
444+
}
445+
Def::StructCtor(..) => {
446446
self.define(parent, ident, ValueNS, (def, vis, DUMMY_SP, Mark::root()));
447+
448+
if let Some(struct_def_id) =
449+
self.session.cstore.def_key(def_id).parent
450+
.map(|index| DefId { krate: def_id.krate, index: index }) {
451+
self.struct_constructors.insert(struct_def_id, (def, vis));
452+
}
447453
}
448454
Def::Trait(..) => {
449455
let module_kind = ModuleKind::Def(def, ident.name);

src/librustc_resolve/lib.rs

+14-4
Original file line numberDiff line numberDiff line change
@@ -1132,8 +1132,9 @@ pub struct Resolver<'a> {
11321132

11331133
potentially_unused_imports: Vec<&'a ImportDirective<'a>>,
11341134

1135-
// Auxiliary map used only for reporting `legacy_constructor_visibility` lint.
1136-
legacy_ctor_visibilities: DefIdMap<(Def, ty::Visibility)>,
1135+
// This table maps struct IDs into struct constructor IDs,
1136+
// it's not used during normal resolution, only for better error reporting.
1137+
struct_constructors: DefIdMap<(Def, ty::Visibility)>,
11371138
}
11381139

11391140
pub struct ResolverArenas<'a> {
@@ -1313,7 +1314,7 @@ impl<'a> Resolver<'a> {
13131314
proc_macro_enabled: features.proc_macro,
13141315
warned_proc_macros: FxHashSet(),
13151316
potentially_unused_imports: Vec::new(),
1316-
legacy_ctor_visibilities: DefIdMap(),
1317+
struct_constructors: DefIdMap(),
13171318
}
13181319
}
13191320

@@ -2209,6 +2210,15 @@ impl<'a> Resolver<'a> {
22092210
_ => {}
22102211
},
22112212
_ if ns == ValueNS && is_struct_like(def) => {
2213+
if let Def::Struct(def_id) = def {
2214+
if let Some((ctor_def, ctor_vis))
2215+
= this.struct_constructors.get(&def_id).cloned() {
2216+
if is_expected(ctor_def) && !this.is_accessible(ctor_vis) {
2217+
err.span_label(span, &format!("constructor is not visible \
2218+
here due to private fields"));
2219+
}
2220+
}
2221+
}
22122222
err.span_label(span, &format!("did you mean `{} {{ /* fields */ }}`?",
22132223
path_str));
22142224
return err;
@@ -2244,7 +2254,7 @@ impl<'a> Resolver<'a> {
22442254
let mut res = None;
22452255
if let Def::Struct(def_id) = resolution.base_def {
22462256
if let Some((ctor_def, ctor_vis))
2247-
= self.legacy_ctor_visibilities.get(&def_id).cloned() {
2257+
= self.struct_constructors.get(&def_id).cloned() {
22482258
if is_expected(ctor_def) && self.is_accessible(ctor_vis) {
22492259
let lint = lint::builtin::LEGACY_CONSTRUCTOR_VISIBILITY;
22502260
self.session.add_lint(lint, id, span,

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

+1
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
fn main() {
1212
let Box(a) = loop { };
1313
//~^ ERROR expected tuple struct/variant, found struct `Box`
14+
//~| ERROR expected tuple struct/variant, found struct `Box`
1415

1516
// (The below is a trick to allow compiler to infer a type for
1617
// variable `a` without attempting to ascribe a type to the

src/test/compile-fail/privacy-struct-ctor.rs src/test/ui/resolve/privacy-struct-ctor.rs

+9-3
Original file line numberDiff line numberDiff line change
@@ -25,18 +25,24 @@ mod m {
2525

2626
fn f() {
2727
n::Z; //~ ERROR tuple struct `Z` is private
28-
Z; //~ ERROR expected value, found struct `Z`
28+
Z;
29+
//~^ ERROR expected value, found struct `Z`
30+
//~| NOTE tuple struct constructors with private fields are invisible outside of their mod
2931
}
3032
}
3133

3234
use m::S; // OK, only the type is imported
3335

3436
fn main() {
3537
m::S; //~ ERROR tuple struct `S` is private
36-
S; //~ ERROR expected value, found struct `S`
38+
S;
39+
//~^ ERROR expected value, found struct `S`
40+
//~| NOTE constructor is not visible here due to private fields
3741
m::n::Z; //~ ERROR tuple struct `Z` is private
3842

3943
xcrate::m::S; //~ ERROR tuple struct `S` is private
40-
xcrate::S; //~ ERROR expected value, found struct `xcrate::S`
44+
xcrate::S;
45+
//~^ ERROR expected value, found struct `xcrate::S`
46+
//~| NOTE constructor is not visible here due to private fields
4147
xcrate::m::n::Z; //~ ERROR tuple struct `Z` is private
4248
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
error[E0423]: expected value, found struct `Z`
2+
--> $DIR/privacy-struct-ctor.rs:28:9
3+
|
4+
28 | Z;
5+
| ^
6+
| |
7+
| did you mean `Z { /* fields */ }`?
8+
| constructor is not visible here due to private fields
9+
|
10+
= help: possible better candidate is found in another module, you can import it into scope:
11+
`use m::n::Z;`
12+
13+
error[E0423]: expected value, found struct `S`
14+
--> $DIR/privacy-struct-ctor.rs:38:5
15+
|
16+
38 | S;
17+
| ^
18+
| |
19+
| did you mean `S { /* fields */ }`?
20+
| constructor is not visible here due to private fields
21+
|
22+
= help: possible better candidate is found in another module, you can import it into scope:
23+
`use m::S;`
24+
25+
error[E0423]: expected value, found struct `xcrate::S`
26+
--> $DIR/privacy-struct-ctor.rs:44:5
27+
|
28+
44 | xcrate::S;
29+
| ^^^^^^^^^
30+
| |
31+
| did you mean `xcrate::S { /* fields */ }`?
32+
| constructor is not visible here due to private fields
33+
|
34+
= help: possible better candidate is found in another module, you can import it into scope:
35+
`use m::S;`
36+
37+
error: tuple struct `Z` is private
38+
--> $DIR/privacy-struct-ctor.rs:27:9
39+
|
40+
27 | n::Z; //~ ERROR tuple struct `Z` is private
41+
| ^^^^
42+
43+
error: tuple struct `S` is private
44+
--> $DIR/privacy-struct-ctor.rs:37:5
45+
|
46+
37 | m::S; //~ ERROR tuple struct `S` is private
47+
| ^^^^
48+
49+
error: tuple struct `Z` is private
50+
--> $DIR/privacy-struct-ctor.rs:41:5
51+
|
52+
41 | m::n::Z; //~ ERROR tuple struct `Z` is private
53+
| ^^^^^^^
54+
55+
error: tuple struct `S` is private
56+
--> $DIR/privacy-struct-ctor.rs:43:5
57+
|
58+
43 | xcrate::m::S; //~ ERROR tuple struct `S` is private
59+
| ^^^^^^^^^^^^
60+
61+
error: tuple struct `Z` is private
62+
--> $DIR/privacy-struct-ctor.rs:47:5
63+
|
64+
47 | xcrate::m::n::Z; //~ ERROR tuple struct `Z` is private
65+
| ^^^^^^^^^^^^^^^
66+
67+
error: aborting due to 8 previous errors
68+

0 commit comments

Comments
 (0)