Skip to content

Commit 23040c4

Browse files
committed
Auto merge of #111245 - fee1-dead-contrib:temp-fix-tuple-struct-field, r=lcnr
fix for `Self` not respecting tuple Ctor privacy This PR 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 } } ```
2 parents 19ed0aa + be44860 commit 23040c4

7 files changed

+126
-0
lines changed

compiler/rustc_hir_typeck/messages.ftl

+2
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,8 @@ hir_typeck_const_select_must_be_fn = this argument must be a function item
2525
2626
hir_typeck_convert_to_str = try converting the passed type into a `&str`
2727
28+
hir_typeck_ctor_is_private = tuple struct constructor `{$def}` is private
29+
2830
hir_typeck_expected_default_return_type = expected `()` because of default return type
2931
3032
hir_typeck_expected_return_type = expected `{$expected}` because of return type

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};
@@ -1207,6 +1208,12 @@ impl<'a, 'tcx> FnCtxt<'a, 'tcx> {
12071208
match ty.normalized.ty_adt_def() {
12081209
Some(adt_def) if adt_def.has_ctor() => {
12091210
let (ctor_kind, ctor_def_id) = adt_def.non_enum_variant().ctor.unwrap();
1211+
// Check the visibility of the ctor.
1212+
let vis = tcx.visibility(ctor_def_id);
1213+
if !vis.is_accessible_from(tcx.parent_module(hir_id).to_def_id(), tcx) {
1214+
tcx.sess
1215+
.emit_err(CtorIsPrivate { span, def: tcx.def_path_str(adt_def.did()) });
1216+
}
12101217
let new_res = Res::Def(DefKind::Ctor(CtorOf::Struct, ctor_kind), ctor_def_id);
12111218
let user_substs = Self::user_substs_for_adt(ty);
12121219
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)