-
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
Initial support for auto traits with default bounds #120706
Conversation
@rustbot author |
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
So, what are the goals here:
|
The issue right now is that there are regressions, some previously passing code now fails due to cycles in trait solver or something similar, @Bryanskiy has been trying to investigate it, but without success. @lcnr, this is the work I've been talking about today. |
This comment has been minimized.
This comment has been minimized.
I want to reproduce regressions in CI @rustbot author |
it would be good to get an MVCE for the new cycles, otherwise debugging this without fully going through the PR is hard |
|
||
#![feature(auto_traits, lang_items, no_core, rustc_attrs, trait_alias)] | ||
#![no_std] | ||
#![no_core] |
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.
do these tests have to be no_core
?
There's some mini_core or whatever somewhere. It's better to reuse that one rather than manually adding required lang items. Otherwise these tests have to all be separately fixed if it starts to rely on additional core lang itms
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.
I would prefer to keep these tests isolated. Default auto traits can poison lang item trait implementations like in #120706 (comment).
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.
There's //@ add-core-stubs
which means you still don't depend on all of core but avoid breaking the tests in case we end up with another required lang item.
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.
I guess you want to avoid DispatchFromDyn
in your test... however, if that already doesn't work, is experimenting with this feature at all useful 😅 I feel like you won't learn anything too insightful about it if something this core already does not work
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.
What do you mean by "it does not work"? A little more context on the issue with lang items:
For !Leak
types to be able to call methods, Leak
bound should be relaxed for Receiver
implementations:
#[lang = "legacy_receiver"]
trait LegacyReceiver {}
impl<T: ?Sized + ?Leak> LegacyReceiver for &T {}
impl<T: ?Sized + ?Leak> LegacyReceiver for &mut T {}
...
For !Leak
trait objects to be able to call methods, Leak
bound should be relaxed in DispatchFromDyn
:
#[lang = "dispatch_from_dyn"]
pub trait DispatchFromDyn<T: ?Leak> {}
(according to receiver_is_dispatchable
compiler will try to prove Receiver: DispatchFromDyn<Receiver>
).
I have 2 tests to check this behaviour:
- maybe-bounds-in-dyn-traits.rs (for
DispatchFromDyn
) - maybe-bounds-in-traits.rs (for
Receiver
)
I can't reuse mini_core in this case because of lang items redefinition. Adding experimental lang items to mini_core, instead, and relaxing additional bounds seems like a bad idea.
Maybe I'm missing something, but what is wrong here?
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.
That makes sense, you actually need to modify mini_core
to relax some of the new default bounds
okay, this seems fine then. In case we end up with a lot of tests for default auto traits we should get a new mini_core.rs which is modified for them (or add cfg
to the existing one), but for now that seems fine 🤔
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.
some final comments, after that cleanup, r=me
where | ||
'tcx: 'a, |
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 here instead of in the method, why is it even required instead of just using hir_bounds: &[hir::GenericBound<'tcx>]
?
let sized_def_id = tcx.require_lang_item(LangItem::Sized, Some(span)); | ||
let trait_ref = ty::TraitRef::new(tcx, sized_def_id, [ty]); | ||
// Preferable to put this obligation first, since we report better errors for sized ambiguity. | ||
bounds.insert(0, (trait_ref.upcast(tcx), span)); |
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.
can you instead only do the if lang_:item == LangItem::Sized
for the final push/insert instead of duplicating the trait_ref creation?
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.
It may be worth deleting this branch completely and leaving only "insert", since this only affects performance and diagnostics with experimental traits enabled.
@bors r+ |
☀️ Test successful - checks-actions |
What is this?This is an experimental post-merge analysis report that shows differences in test outcomes between the merged PR and its parent PR.Comparing 4fd8c04 (parent) -> 9e14530 (this PR) Test differencesShow 16 test diffsStage 1
Stage 2
Additionally, 4 doctest diffs were found. These are ignored, as they are noisy. Job group index
Job duration changes
How to interpret the job duration changes?Job durations can vary a lot, based on the actual runner instance |
Finished benchmarking commit (9e14530): comparison URL. Overall result: ❌ regressions - please read the text belowOur benchmarks found a performance regression caused by this PR. Next Steps:
@rustbot label: +perf-regression Instruction countThis is the most reliable metric that we have; it was used to determine the overall result at the top of this comment. However, even this metric can sometimes exhibit noise.
Max RSS (memory usage)Results (secondary -0.9%)This is a less reliable metric that may be of interest but was not used to determine the overall result at the top of this comment.
CyclesThis benchmark run did not return any relevant results for this metric. Binary sizeThis benchmark run did not return any relevant results for this metric. Bootstrap: 776.101s -> 773.774s (-0.30%) |
Does this include some extra work on a "good path", when no additional auto traits are defined? Perhaps it can be avoided? |
Yeah, we can skip computing |
Default auto traits: fix perf Skip computing `requires_default_supertraits` if `experimental-default-bounds` option is not enabled. Possible perf fix for rust-lang#120706 r? lcnr
This PR is part of "MCP: Low level components for async drop"
Tracking issue: #138781
Summary: #120706 (comment)
Intro
Sometimes we want to use type system to express specific behavior and provide safety guarantees. This behavior can be specified by various "marker" traits. For example, we use
Send
andSync
to keep track of which types are thread safe. As the language develops, there are more problems that could be solved by adding new marker traits:SyncDrop
could be used Async destructors, async genericity and completion futures.Leak
(orForget
) trait.Move
trait instead of aPin
.All the traits proposed above are supposed to be auto traits implemented for most types, and usually implemented automatically by compiler.
For backward compatibility these traits have to be added implicitly to all bound lists in old code (see below). Adding new default bounds involves many difficulties: many standard library interfaces may need to opt out of those default bounds, and therefore be infected with confusing
?Trait
syntax, migration to a new edition may contain backward compatibility holes, supporting new traits in the compiler can be quite difficult and so forth. Anyway, it's hard to evaluate the complexity until we try the system on a practice.In this PR we introduce new optional lang items for traits that are added to all bound lists by default, similarly to existing
Sized
. The examples of such traits could beLeak
,Move
,SyncDrop
or something else, it doesn't matter much right now (further I will call themDefaultAutoTrait
's). We want to land this change into rustc under an option, so it becomes available in bootstrap compiler. Then we'll be able to do standard library experiments with the aforementioned traits without adding hundreds of#[cfg(not(bootstrap))]
s. Based on the experiments, we can come up with some scheme for the next edition, in which such bounds are added in a more targeted way, and not just everywhere.Most of the implementation is basically a refactoring that replaces hardcoded uses of
Sized
with iterating over a list of traits including bothSized
and the new traits when-Zexperimental-default-bounds
is enabled (or justSized
as before, if the option is not enabled).Default bounds for old editions
All existing types, including generic parameters, are considered
Leak
/Move
/SyncDrop
and can be forgotten, moved or destroyed in generic contexts without specifying any bounds. New types that cannot be, for example, forgotten and do not implementLeak
can be added at some point, and they should not be usable in such generic contexts in existing code.To both maintain this property and keep backward compatibility with existing code, the new traits should be added as default bounds everywhere in previous editions. Besides the implicit
Sized
bound contexts that includes supertrait lists and trait lists in trait objects (dyn Trait1 + ... + TraitN
). Compiler should also generate implicitDefaultAutoTrait
implementations for foreign types (extern { type Foo; }
) because they are also currently usable in generic contexts without any bounds.Supertraits
Adding the new traits as supertraits to all existing traits is potentially necessary, because, for example, using a
Self
param in a trait's associated item may be a breaking change otherwise:However, default supertraits can significantly affect compiler performance. For example, if we know that
T: Trait
, the compiler would deduce thatT: DefaultAutoTrait
. It also implies provingF: DefaultAutoTrait
for each fieldF
of typeT
until an explicit impl is be provided.If the standard library is not modified, then even traits like
Copy
orSend
would get these supertraits.In this PR for optimization purposes instead of adding default supertraits, bounds are added to the associated items:
It is not always possible to do this optimization because of backward compatibility:
or
Therefore,
DefaultAutoTrait
's are still being added to supertraits if theSelf
params or type bindings were found in the trait header.Trait objects
Trait objects requires explicit
+ Trait
bound to implement corresponding trait which is not backward compatible:So, for a trait object
dyn Trait
we should add an implicit bounddyn Trait + DefaultAutoTrait
to make it usable, and allow relaxing it with a question mark syntaxdyn Trait + ?DefaultAutoTrait
when it's not necessary.Foreign types
If compiler doesn't generate auto trait implementations for a foreign type, then it's a breaking change if the default bounds are added everywhere else:
We'll have to enable implicit
DefaultAutoTrait
implementations for foreign types at least for previous editions:Unresolved questions
New default bounds affect all existing Rust code complicating an already complex type system.
Also migration to a new edition could be quite ugly and enormous, but that's actually what we want to solve. For other issues there's a chance that they could be solved by a new solver.