Skip to content

Commit dee65a4

Browse files
committed
fix for Self not respecting tuple Ctor privacy
This fixes #111220 by checking the privacy of tuple constructors using `Self`, so the following code now errors ```rust mod my { pub struct Foo(&'static str); } impl AsRef<str> for my::Foo { fn as_ref(&self) -> &str { let Self(s) = self; // previously compiled, now errors correctly s } } ```
1 parent 70db836 commit dee65a4

7 files changed

+126
-0
lines changed

compiler/rustc_hir_typeck/messages.ftl

+2
Original file line numberDiff line numberDiff line change
@@ -90,3 +90,5 @@ hir_typeck_candidate_trait_note = `{$trait_name}` defines an item `{$item_name}`
9090
[implement] , perhaps you need to implement it
9191
*[other] , perhaps you need to restrict type parameter `{$action_or_ty}` with it
9292
}
93+
94+
hir_typeck_ctor_is_private = tuple struct constructor `{$def}` is private

compiler/rustc_hir_typeck/src/errors.rs

+8
Original file line numberDiff line numberDiff line change
@@ -319,3 +319,11 @@ pub struct CandidateTraitNote {
319319
pub item_name: Ident,
320320
pub action_or_ty: String,
321321
}
322+
323+
#[derive(Diagnostic)]
324+
#[diag(hir_typeck_ctor_is_private, code = "E0603")]
325+
pub struct CtorIsPrivate {
326+
#[primary_span]
327+
pub span: Span,
328+
pub def: String,
329+
}

compiler/rustc_hir_typeck/src/fn_ctxt/_impl.rs

+7
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,5 @@
11
use crate::callee::{self, DeferredCallResolution};
2+
use crate::errors::CtorIsPrivate;
23
use crate::method::{self, MethodCallee, SelfSource};
34
use crate::rvalue_scopes;
45
use crate::{BreakableCtxt, Diverges, Expectation, FnCtxt, LocalTy, RawTy};
@@ -1210,6 +1211,12 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
12101211
match ty.normalized.ty_adt_def() {
12111212
Some(adt_def) if adt_def.has_ctor() => {
12121213
let (ctor_kind, ctor_def_id) = adt_def.non_enum_variant().ctor.unwrap();
1214+
// Check the visibility of the ctor.
1215+
let vis = tcx.visibility(ctor_def_id);
1216+
if !vis.is_accessible_from(tcx.parent_module(hir_id).to_def_id(), tcx) {
1217+
tcx.sess
1218+
.emit_err(CtorIsPrivate { span, def: tcx.def_path_str(adt_def.did()) });
1219+
}
12131220
let new_res = Res::Def(DefKind::Ctor(CtorOf::Struct, ctor_kind), ctor_def_id);
12141221
let user_substs = Self::user_substs_for_adt(ty);
12151222
user_self_ty = user_substs.user_self_ty;
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
mod b {
2+
pub struct A(u32);
3+
}
4+
5+
trait Id {
6+
type Assoc;
7+
}
8+
impl Id for b::A {
9+
type Assoc = b::A;
10+
}
11+
impl Id for u32 {
12+
type Assoc = u32;
13+
}
14+
15+
16+
trait Trait<T> {
17+
fn method(&self)
18+
where
19+
T: Id<Assoc = b::A>;
20+
}
21+
22+
impl<T: Id> Trait<T> for <T as Id>::Assoc {
23+
fn method(&self)
24+
where
25+
T: Id<Assoc = b::A>,
26+
{
27+
let Self(a) = self;
28+
//~^ ERROR: tuple struct constructor `A` is private
29+
println!("{a}");
30+
}
31+
}
32+
33+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
error[E0603]: tuple struct constructor `A` is private
2+
--> $DIR/issue-111220-2-tuple-struct-fields-projection.rs:27:13
3+
|
4+
LL | let Self(a) = self;
5+
| ^^^^^^^
6+
7+
error: aborting due to previous error
8+
9+
For more information about this error, try `rustc --explain E0603`.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,46 @@
1+
mod b {
2+
#[derive(Default)]
3+
pub struct A(u32);
4+
}
5+
6+
impl b::A {
7+
fn inherent_bypass(&self) {
8+
let Self(x) = self;
9+
//~^ ERROR: tuple struct constructor `A` is private
10+
println!("{x}");
11+
}
12+
}
13+
14+
pub trait B {
15+
fn f(&self);
16+
}
17+
18+
impl B for b::A {
19+
fn f(&self) {
20+
let Self(a) = self;
21+
//~^ ERROR: tuple struct constructor `A` is private
22+
println!("{}", a);
23+
}
24+
}
25+
26+
pub trait Projector {
27+
type P;
28+
}
29+
30+
impl Projector for () {
31+
type P = b::A;
32+
}
33+
34+
pub trait Bypass2 {
35+
fn f2(&self);
36+
}
37+
38+
impl Bypass2 for <() as Projector>::P {
39+
fn f2(&self) {
40+
let Self(a) = self;
41+
//~^ ERROR: tuple struct constructor `A` is private
42+
println!("{}", a);
43+
}
44+
}
45+
46+
fn main() {}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
error[E0603]: tuple struct constructor `A` is private
2+
--> $DIR/issue-111220-tuple-struct-fields.rs:8:13
3+
|
4+
LL | let Self(x) = self;
5+
| ^^^^^^^
6+
7+
error[E0603]: tuple struct constructor `A` is private
8+
--> $DIR/issue-111220-tuple-struct-fields.rs:20:13
9+
|
10+
LL | let Self(a) = self;
11+
| ^^^^^^^
12+
13+
error[E0603]: tuple struct constructor `A` is private
14+
--> $DIR/issue-111220-tuple-struct-fields.rs:40:13
15+
|
16+
LL | let Self(a) = self;
17+
| ^^^^^^^
18+
19+
error: aborting due to 3 previous errors
20+
21+
For more information about this error, try `rustc --explain E0603`.

0 commit comments

Comments
 (0)