1
Fork 0

Disallow impl autotrait for trait object

This commit is contained in:
David Tolnay 2023-01-18 22:43:05 -08:00
parent a94b9fd0ac
commit 9e1c600f74
No known key found for this signature in database
GPG key ID: F9BA143B95FF6D82
6 changed files with 269 additions and 70 deletions

View file

@ -31,8 +31,8 @@ cfg_if! {
pub auto trait Send {} pub auto trait Send {}
pub auto trait Sync {} pub auto trait Sync {}
impl<T: ?Sized> Send for T {} impl<T> Send for T {}
impl<T: ?Sized> Sync for T {} impl<T> Sync for T {}
#[macro_export] #[macro_export]
macro_rules! rustc_erase_owner { macro_rules! rustc_erase_owner {

View file

@ -8,7 +8,7 @@ use rustc_hir as hir;
use rustc_middle::ty::subst::InternalSubsts; use rustc_middle::ty::subst::InternalSubsts;
use rustc_middle::ty::util::IgnoreRegions; use rustc_middle::ty::util::IgnoreRegions;
use rustc_middle::ty::{ use rustc_middle::ty::{
self, ImplPolarity, Ty, TyCtxt, TypeSuperVisitable, TypeVisitable, TypeVisitor, self, AliasKind, ImplPolarity, Ty, TyCtxt, TypeSuperVisitable, TypeVisitable, TypeVisitor,
}; };
use rustc_session::lint; use rustc_session::lint;
use rustc_span::def_id::{DefId, LocalDefId}; use rustc_span::def_id::{DefId, LocalDefId};
@ -86,7 +86,7 @@ fn do_orphan_check_impl<'tcx>(
// struct B { } // struct B { }
// impl Foo for A { } // impl Foo for A { }
// impl Foo for B { } // impl Foo for B { }
// impl !Send for (A, B) { } // impl !Foo for (A, B) { }
// ``` // ```
// //
// This final impl is legal according to the orphan // This final impl is legal according to the orphan
@ -99,50 +99,171 @@ fn do_orphan_check_impl<'tcx>(
tcx.trait_is_auto(trait_def_id) tcx.trait_is_auto(trait_def_id)
); );
if tcx.trait_is_auto(trait_def_id) && !trait_def_id.is_local() { if tcx.trait_is_auto(trait_def_id) {
let self_ty = trait_ref.self_ty(); let self_ty = trait_ref.self_ty();
let opt_self_def_id = match *self_ty.kind() {
ty::Adt(self_def, _) => Some(self_def.did()), // If the impl is in the same crate as the auto-trait, almost anything
ty::Foreign(did) => Some(did), // goes.
_ => None, //
// impl MyAuto for Rc<Something> {} // okay
// impl<T> !MyAuto for *const T {} // okay
// impl<T> MyAuto for T {} // okay
//
// But there is one important exception: implementing for a trait object
// is not allowed.
//
// impl MyAuto for dyn Trait {} // NOT OKAY
// impl<T: ?Sized> MyAuto for T {} // NOT OKAY
//
enum LocalImpl {
Allow,
Disallow { problematic_kind: &'static str },
}
// If the auto-trait is from a dependency, it must only be getting
// implemented for a nominal type, and specifically one local to the
// current crate.
//
// impl<T> Sync for MyStruct<T> {} // okay
//
// impl Sync for Rc<MyStruct> {} // NOT OKAY
enum NonlocalImpl {
Allow,
DisallowBecauseNonlocal,
DisallowOther,
}
// Exhaustive match considering that this logic is essential for
// soundness.
let (local_impl, nonlocal_impl) = match self_ty.kind() {
// struct Struct<T>;
// impl AutoTrait for Struct<Foo> {}
ty::Adt(self_def, _) => (
LocalImpl::Allow,
if self_def.did().is_local() {
NonlocalImpl::Allow
} else {
NonlocalImpl::DisallowBecauseNonlocal
},
),
// extern { type OpaqueType; }
// impl AutoTrait for OpaqueType {}
ty::Foreign(did) => (
LocalImpl::Allow,
if did.is_local() {
NonlocalImpl::Allow
} else {
NonlocalImpl::DisallowBecauseNonlocal
},
),
// impl AutoTrait for dyn Trait {}
ty::Dynamic(..) => (
LocalImpl::Disallow { problematic_kind: "trait object" },
NonlocalImpl::DisallowOther,
),
// impl<T> AutoTrait for T {}
// impl<T: ?Sized> AutoTrait for T {}
ty::Param(..) => (
if self_ty.is_sized(tcx, tcx.param_env(def_id)) {
LocalImpl::Allow
} else {
LocalImpl::Disallow { problematic_kind: "generic type" }
},
NonlocalImpl::DisallowOther,
),
// trait Id { type This: ?Sized; }
// impl<T: ?Sized> Id for T {
// type This = T;
// }
// impl<T: ?Sized> AutoTrait for <T as Id>::This {}
ty::Alias(AliasKind::Projection, _) => (
LocalImpl::Disallow { problematic_kind: "associated type" },
NonlocalImpl::DisallowOther,
),
// type Opaque = impl Trait;
// impl AutoTrait for Opaque {}
ty::Alias(AliasKind::Opaque, _) => (
LocalImpl::Disallow { problematic_kind: "opaque type" },
NonlocalImpl::DisallowOther,
),
ty::Bool
| ty::Char
| ty::Int(..)
| ty::Uint(..)
| ty::Float(..)
| ty::Str
| ty::Array(..)
| ty::Slice(..)
| ty::RawPtr(..)
| ty::Ref(..)
| ty::FnDef(..)
| ty::FnPtr(..)
| ty::Never
| ty::Tuple(..) => (LocalImpl::Allow, NonlocalImpl::DisallowOther),
ty::Closure(..)
| ty::Generator(..)
| ty::GeneratorWitness(..)
| ty::GeneratorWitnessMIR(..)
| ty::Bound(..)
| ty::Placeholder(..)
| ty::Infer(..) => span_bug!(sp, "weird self type for autotrait impl"),
ty::Error(..) => (LocalImpl::Allow, NonlocalImpl::Allow),
}; };
let msg = match opt_self_def_id { if trait_def_id.is_local() {
// We only want to permit nominal types, but not *all* nominal types. match local_impl {
// They must be local to the current crate, so that people LocalImpl::Allow => {}
// can't do `unsafe impl Send for Rc<SomethingLocal>` or LocalImpl::Disallow { problematic_kind } => {
// `impl !Send for Box<SomethingLocalAndSend>`. let msg = format!(
Some(self_def_id) => { "traits with a default impl, like `{trait}`, \
if self_def_id.is_local() { cannot be implemented for {problematic_kind} `{self_ty}`",
None trait = tcx.def_path_str(trait_def_id),
} else { );
Some(( let label = format!(
format!( "a trait object implements `{trait}` if and only if `{trait}` \
"cross-crate traits with a default impl, like `{}`, \ is one of the trait object's trait bounds",
can only be implemented for a struct/enum type \ trait = tcx.def_path_str(trait_def_id),
defined in the current crate", );
tcx.def_path_str(trait_def_id) let reported =
), struct_span_err!(tcx.sess, sp, E0321, "{}", msg).note(label).emit();
"can't implement cross-crate trait for type in another crate", return Err(reported);
))
} }
} }
_ => Some(( } else {
format!( if let Some((msg, label)) = match nonlocal_impl {
"cross-crate traits with a default impl, like `{}`, can \ NonlocalImpl::Allow => None,
NonlocalImpl::DisallowBecauseNonlocal => Some((
format!(
"cross-crate traits with a default impl, like `{}`, \
can only be implemented for a struct/enum type \
defined in the current crate",
tcx.def_path_str(trait_def_id)
),
"can't implement cross-crate trait for type in another crate",
)),
NonlocalImpl::DisallowOther => Some((
format!(
"cross-crate traits with a default impl, like `{}`, can \
only be implemented for a struct/enum type, not `{}`", only be implemented for a struct/enum type, not `{}`",
tcx.def_path_str(trait_def_id), tcx.def_path_str(trait_def_id),
self_ty self_ty
), ),
"can't implement cross-crate trait with a default impl for \ "can't implement cross-crate trait with a default impl for \
non-struct/enum type", non-struct/enum type",
)), )),
}; } {
let reported =
if let Some((msg, label)) = msg { struct_span_err!(tcx.sess, sp, E0321, "{}", msg).span_label(sp, label).emit();
let reported = return Err(reported);
struct_span_err!(tcx.sess, sp, E0321, "{}", msg).span_label(sp, label).emit(); }
return Err(reported);
} }
} }

View file

@ -12,19 +12,26 @@ auto trait Marker2 {}
trait Object: Marker1 {} trait Object: Marker1 {}
// A supertrait marker is illegal... // A supertrait marker is illegal...
impl !Marker1 for dyn Object + Marker2 { } //~ ERROR E0371 impl !Marker1 for dyn Object + Marker2 {} //~ ERROR E0371
//~^ ERROR 0321
// ...and also a direct component. // ...and also a direct component.
impl !Marker2 for dyn Object + Marker2 { } //~ ERROR E0371 impl !Marker2 for dyn Object + Marker2 {} //~ ERROR E0371
//~^ ERROR 0321
// But implementing a marker if it is not present is OK.
impl !Marker2 for dyn Object {} // OK
// A non-principal trait-object type is orphan even in its crate. // A non-principal trait-object type is orphan even in its crate.
impl !Send for dyn Marker2 {} //~ ERROR E0117 impl !Send for dyn Marker2 {} //~ ERROR E0117
// And impl'ing a remote marker for a local trait object is forbidden // Implementing a marker for a local trait object is forbidden by a special
// by one of these special orphan-like rules. // orphan-like rule.
impl !Marker2 for dyn Object {} //~ ERROR E0321
impl !Send for dyn Object {} //~ ERROR E0321 impl !Send for dyn Object {} //~ ERROR E0321
impl !Send for dyn Object + Marker2 {} //~ ERROR E0321 impl !Send for dyn Object + Marker2 {} //~ ERROR E0321
fn main() { } // Blanket impl that applies to dyn Object is equally problematic.
auto trait Marker3 {}
impl<T: ?Sized> !Marker3 for T {} //~ ERROR E0321
auto trait Marker4 {}
impl<T> !Marker4 for T {} // okay
fn main() {}

View file

@ -1,17 +1,41 @@
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker1` error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker1`
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:15:1 --> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:15:1
| |
LL | impl !Marker1 for dyn Object + Marker2 { } LL | impl !Marker1 for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker1` | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker1`
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker2` error[E0321]: traits with a default impl, like `Marker1`, cannot be implemented for trait object `(dyn Object + Marker2 + 'static)`
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:17:1 --> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:15:1
| |
LL | impl !Marker2 for dyn Object + Marker2 { } LL | impl !Marker1 for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: a trait object implements `Marker1` if and only if `Marker1` is one of the trait object's trait bounds
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker2`
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:18:1
|
LL | impl !Marker2 for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker2` | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker2`
error[E0321]: traits with a default impl, like `Marker2`, cannot be implemented for trait object `(dyn Object + Marker2 + 'static)`
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:18:1
|
LL | impl !Marker2 for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: a trait object implements `Marker2` if and only if `Marker2` is one of the trait object's trait bounds
error[E0321]: traits with a default impl, like `Marker2`, cannot be implemented for trait object `(dyn Object + 'static)`
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:26:1
|
LL | impl !Marker2 for dyn Object {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: a trait object implements `Marker2` if and only if `Marker2` is one of the trait object's trait bounds
error[E0117]: only traits defined in the current crate can be implemented for arbitrary types error[E0117]: only traits defined in the current crate can be implemented for arbitrary types
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:23:1 --> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:22:1
| |
LL | impl !Send for dyn Marker2 {} LL | impl !Send for dyn Marker2 {}
| ^^^^^^^^^^^^^^^----------- | ^^^^^^^^^^^^^^^-----------
@ -33,7 +57,15 @@ error[E0321]: cross-crate traits with a default impl, like `Send`, can only be i
LL | impl !Send for dyn Object + Marker2 {} LL | impl !Send for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ can't implement cross-crate trait with a default impl for non-struct/enum type | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ can't implement cross-crate trait with a default impl for non-struct/enum type
error: aborting due to 5 previous errors error[E0321]: traits with a default impl, like `Marker3`, cannot be implemented for generic type `T`
--> $DIR/coherence-impl-trait-for-marker-trait-negative.rs:32:1
|
LL | impl<T: ?Sized> !Marker3 for T {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: a trait object implements `Marker3` if and only if `Marker3` is one of the trait object's trait bounds
error: aborting due to 9 previous errors
Some errors have detailed explanations: E0117, E0321, E0371. Some errors have detailed explanations: E0117, E0321, E0371.
For more information about an error, try `rustc --explain E0117`. For more information about an error, try `rustc --explain E0117`.

View file

@ -12,19 +12,26 @@ auto trait Marker2 {}
trait Object: Marker1 {} trait Object: Marker1 {}
// A supertrait marker is illegal... // A supertrait marker is illegal...
impl Marker1 for dyn Object + Marker2 { } //~ ERROR E0371 impl Marker1 for dyn Object + Marker2 {} //~ ERROR E0371
//~^ ERROR E0321
// ...and also a direct component. // ...and also a direct component.
impl Marker2 for dyn Object + Marker2 { } //~ ERROR E0371 impl Marker2 for dyn Object + Marker2 {} //~ ERROR E0371
//~^ ERROR E0321
// But implementing a marker if it is not present is OK.
impl Marker2 for dyn Object {} // OK
// A non-principal trait-object type is orphan even in its crate. // A non-principal trait-object type is orphan even in its crate.
unsafe impl Send for dyn Marker2 {} //~ ERROR E0117 unsafe impl Send for dyn Marker2 {} //~ ERROR E0117
// And impl'ing a remote marker for a local trait object is forbidden // Implementing a marker for a local trait object is forbidden by a special
// by one of these special orphan-like rules. // orphan-like rule.
impl Marker2 for dyn Object {} //~ ERROR E0321
unsafe impl Send for dyn Object {} //~ ERROR E0321 unsafe impl Send for dyn Object {} //~ ERROR E0321
unsafe impl Send for dyn Object + Marker2 {} //~ ERROR E0321 unsafe impl Send for dyn Object + Marker2 {} //~ ERROR E0321
fn main() { } // Blanket impl that applies to dyn Object is equally problematic.
auto trait Marker3 {}
impl<T: ?Sized> Marker3 for T {} //~ ERROR E0321
auto trait Marker4 {}
impl<T> Marker4 for T {} // okay
fn main() {}

View file

@ -1,17 +1,41 @@
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker1` error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker1`
--> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:15:1 --> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:15:1
| |
LL | impl Marker1 for dyn Object + Marker2 { } LL | impl Marker1 for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker1` | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker1`
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker2` error[E0321]: traits with a default impl, like `Marker1`, cannot be implemented for trait object `(dyn Object + Marker2 + 'static)`
--> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:17:1 --> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:15:1
| |
LL | impl Marker2 for dyn Object + Marker2 { } LL | impl Marker1 for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: a trait object implements `Marker1` if and only if `Marker1` is one of the trait object's trait bounds
error[E0371]: the object type `(dyn Object + Marker2 + 'static)` automatically implements the trait `Marker2`
--> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:18:1
|
LL | impl Marker2 for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker2` | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ `(dyn Object + Marker2 + 'static)` automatically implements trait `Marker2`
error[E0321]: traits with a default impl, like `Marker2`, cannot be implemented for trait object `(dyn Object + Marker2 + 'static)`
--> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:18:1
|
LL | impl Marker2 for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: a trait object implements `Marker2` if and only if `Marker2` is one of the trait object's trait bounds
error[E0321]: traits with a default impl, like `Marker2`, cannot be implemented for trait object `(dyn Object + 'static)`
--> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:26:1
|
LL | impl Marker2 for dyn Object {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: a trait object implements `Marker2` if and only if `Marker2` is one of the trait object's trait bounds
error[E0117]: only traits defined in the current crate can be implemented for arbitrary types error[E0117]: only traits defined in the current crate can be implemented for arbitrary types
--> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:23:1 --> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:22:1
| |
LL | unsafe impl Send for dyn Marker2 {} LL | unsafe impl Send for dyn Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^----------- | ^^^^^^^^^^^^^^^^^^^^^-----------
@ -33,7 +57,15 @@ error[E0321]: cross-crate traits with a default impl, like `Send`, can only be i
LL | unsafe impl Send for dyn Object + Marker2 {} LL | unsafe impl Send for dyn Object + Marker2 {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ can't implement cross-crate trait with a default impl for non-struct/enum type | ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ can't implement cross-crate trait with a default impl for non-struct/enum type
error: aborting due to 5 previous errors error[E0321]: traits with a default impl, like `Marker3`, cannot be implemented for generic type `T`
--> $DIR/coherence-impl-trait-for-marker-trait-positive.rs:32:1
|
LL | impl<T: ?Sized> Marker3 for T {}
| ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
|
= note: a trait object implements `Marker3` if and only if `Marker3` is one of the trait object's trait bounds
error: aborting due to 9 previous errors
Some errors have detailed explanations: E0117, E0321, E0371. Some errors have detailed explanations: E0117, E0321, E0371.
For more information about an error, try `rustc --explain E0117`. For more information about an error, try `rustc --explain E0117`.