-
Notifications
You must be signed in to change notification settings - Fork 13.2k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
arbitrary_self_type: insert implied Receiver bound on Deref #138952
base: master
Are you sure you want to change the base?
arbitrary_self_type: insert implied Receiver bound on Deref #138952
Conversation
HIR ty lowering was modified cc @fmease |
@rustbot labels +F-arbitrary_self_types |
The job Click to see the possible cause of the failure (guessed by this bot)
|
@bors try |
…mpl, r=<try> arbitrary_self_type: insert implied Receiver bound on Deref r? `@nikomatsakis` This is an experimental setup which allows us to assess the impact from inserting an implied supertrait bound `Deref: Receiver`. The observations are as follows. - Thanks to the trait solver, not too many hacks are needed to implement the idea behind [this comment](rust-lang#135881 (comment)), where inductive cycles are admissible. - We want to allow users to continue use the `dyn Deref` type as `Deref` has been `dyn`-compatible. Following the current `dyn`-compatibility rule, it would have been impossible to use `dyn Deref<Target = ..>` because one would need to write `dyn Deref<..> + Receiver<..>` while neither `Deref` nor `Receiver` is an `auto` trait. This is the main change that this PR proposes: we fill in the missing projection bound that `Receiver` demands from a `dyn Deref` and this will remain the only exception. - Finally, the type (pretty-)printing is adjusted so as to not present the frivolous `Receiver::Target = ..` bound. It will always be the same term as `Deref::Target`. I am proposing a crater experiment to see the extent of impact on the existing ecosystem and explore corner cases that I have missed.
#[must_use] | ||
#[stable(feature = "rust1", since = "1.0.0")] | ||
#[rustc_diagnostic_item = "deref_method"] | ||
fn deref(&self) -> &<Self as Deref>::Target; |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why did you need to change this? Isn't the point of the hack in hir lowering to avoid breakage around T::Target
shorthand projections?
@@ -6,8 +6,8 @@ use rustc_hir::def::{DefKind, Res}; | |||
use rustc_lint_defs::builtin::UNUSED_ASSOCIATED_TYPE_BOUNDS; | |||
use rustc_middle::ty::elaborate::ClauseWithSupertraitSpan; | |||
use rustc_middle::ty::{ | |||
self, BottomUpFolder, DynKind, ExistentialPredicateStableCmpExt as _, Ty, TyCtxt, TypeFoldable, | |||
TypeVisitableExt, Upcast, | |||
self, BottomUpFolder, DynKind, ExistentialPredicateStableCmpExt as _, Interner as _, Ty, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Why are you importing Interner
? There should be no reason to use that trait.
@@ -112,9 +112,29 @@ impl<'tcx> dyn HirTyLowerer<'tcx> + '_ { | |||
// let _: &dyn Alias<Assoc = u32> = /* ... */; | |||
// ``` | |||
let mut projection_bounds = FxIndexMap::default(); | |||
let mut auto_fill_bounds: SmallVec<[_; 4]> = smallvec![]; | |||
let mut push_auto_fill_receiver_bound = |pred: ty::Binder<'tcx, _>| { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is a bit overkill. Just take the projection and replace its def id with the one for Receiver::Target
.
At that point, this doesn't need to be a callback, just an additional call to insert
into the projection bounds map that you can duplicate twice.
pred.skip_binder().def_id(), | ||
rustc_hir::LangItem::DerefTarget, | ||
) { | ||
// We need to fill in a `Receiver<Target = ?>` bound because |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This comment is missing the more important part that a supertrait like Deref: Receiver<Target = Self::Target>
does not automatically elaborate into a projection bound for Reciever
's Target
type.
.filter(|item| !tcx.generics_require_sized_self(item.def_id)) | ||
.count() | ||
}) | ||
let expected_count = obj.principal_def_id().map_or(0, |principal_def_id| { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
why is this changed?
☀️ Try build successful - checks-actions |
@craterbot check |
👌 Experiment ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more |
🚧 Experiment ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more |
🎉 Experiment
|
r? @nikomatsakis
This is an experimental setup which allows us to assess the impact from inserting an implied supertrait bound
Deref: Receiver
.The observations are as follows.
dyn Deref
type asDeref
has beendyn
-compatible. Following the currentdyn
-compatibility rule, it would have been impossible to usedyn Deref<Target = ..>
because one would need to writedyn Deref<..> + Receiver<..>
while neitherDeref
norReceiver
is anauto
trait. This is the main change that this PR proposes: we fill in the missing projection bound thatReceiver
demands from adyn Deref
and this will remain the only exception.Receiver::Target = ..
bound. It will always be the same term asDeref::Target
.I am proposing a crater experiment to see the extent of impact on the existing ecosystem and explore corner cases that I have missed.