Skip to content

Conversation

@dianne
Copy link
Contributor

@dianne dianne commented Oct 17, 2024

Motivation

Many unstable library items require the stabilization of multiple features before they can be stabilized. By allowing them to be annotated with multiple #[unstable] attributes, this prevents their accidental stabilization when some of those features are stabilized, and helps mitigate upgrade pains for unstable toolchain users.

New stability attribute semantics

  • If any #[unstable] attribute is present on an item, it is unstable. Likewise, any #[rustc_const_unstable] marks an item as const-unstable, and any #[rustc_default_body_unstable] marks it as default body-unstable.
  • In order to use an unstable item, all feature gates specified by its #[unstable] attributes must be enabled. Conceptually, this is because it depends on multiple unstable features. Practically, this also means nightly toolchain users are less likely to need to replace one unstable feature gate with another in their crate-level attributes.
  • Multiple #[stable] attributes may be present. The stabilization version of an item is the most recent stabilization version in present #[stable] attributes. Likewise, #[stable] attributes may be present on an unstable item. This simplifies macros that can apply stability attributes to items.

This is documented in rust-lang/rustc-dev-guide#2128.

New syntactic constraints

  • As a sanity check, an item is not allowed to have multiple stability attributes for the same feature.
  • At most one #[unstable] attribute on an item may have a reason provided (and likewise for #[rustc_const_unstable] and #[rustc_default_body_unstable]). At a glance, the reason meta-item was used to describe the reason for an item being unstable, rather than an individual feature it depends on. Enforcing this makes the diagnostic formatting much cleaner.
  • If any #[unstable] attribute on an item has a soft marker, it must be present on all #[unstable] attributes on that item. Items can't be partially soft-unstable, and this helps prevent accidentally making a soft-unstable item unusable on stable when stabilizing one of the features it requires.

Fixes #94770

Based on #94988

PR for a diagnostic formatting change this uses: #132544

Some notes on assumptions I've made and how I've handled them:

  • Stability structs are specialized to how they're used for stability checking and rustdoc. rustc_passes::lib_features also reads stability attributes, but it parses them itself. I considered it out of the scope of this PR refactor that too, but once attribute handling is reworked to be more unified in the future, this may need further adjustment (but hopefully not too much).
  • Under the assumption that most library items will still have at most one #[unstable] attribute, this uses SmallVec<[_; 1]> for storage. Is this assumption reasonable? I haven't done any performance testing.
  • Since StabilityLevel no longer has Copy, stability structs are now arena-allocated; that was the simplest fix. I'm not sure what the performance implications are. If it's an issue, I can try try making it only allocate for multiple unstable features (which would also let StabilityLevel regain Copy).
  • As a stopgap until Proper support for cross-crate recursive const stability checks #132541 lands, this is using a ConstStabilityLevel enum for const-stability levels. Once that's merged, I should be able to move const-stability levels back to using StabilityLevel.

This doesn't update any libraries to add additional unstable attributes to items that should have them.

@rustbot
Copy link
Collaborator

rustbot commented Oct 17, 2024

r? @wesleywiser

rustbot has assigned @wesleywiser.
They will have a look at your PR within the next two weeks and either review your PR or reassign to another reviewer.

Use r? to explicitly pick a reviewer

@rustbot rustbot added S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. T-rustdoc Relevant to the rustdoc team, which will review and decide on the PR/issue. labels Oct 17, 2024
@rustbot
Copy link
Collaborator

rustbot commented Oct 17, 2024

Some changes occurred in src/librustdoc/clean/types.rs

cc @camelid

These commits modify the Cargo.lock file. Unintentional changes to Cargo.lock can be introduced when switching branches and rebasing PRs.

If this was unintentional then you should revert the changes before this PR is merged.
Otherwise, you can ignore this comment.

@rust-log-analyzer

This comment has been minimized.

@dianne dianne force-pushed the multiple-unstables branch from ebf7852 to 14b8556 Compare October 17, 2024 10:03
@bors
Copy link
Collaborator

bors commented Oct 22, 2024

☔ The latest upstream changes (presumably #132020) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the multiple-unstables branch from 14b8556 to 6561424 Compare October 23, 2024 02:37
@bors
Copy link
Collaborator

bors commented Oct 23, 2024

☔ The latest upstream changes (presumably #132027) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the multiple-unstables branch from 6561424 to 0a730d9 Compare October 23, 2024 19:52
@bors
Copy link
Collaborator

bors commented Oct 24, 2024

☔ The latest upstream changes (presumably #131985) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the multiple-unstables branch from 0a730d9 to 86fac38 Compare October 25, 2024 01:50
@bors
Copy link
Collaborator

bors commented Oct 26, 2024

☔ The latest upstream changes (presumably #131349) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the multiple-unstables branch from 86fac38 to 88e33bf Compare October 26, 2024 23:16
@rustbot
Copy link
Collaborator

rustbot commented Oct 26, 2024

Some changes occurred in src/tools/clippy

cc @rust-lang/clippy

@dianne
Copy link
Contributor Author

dianne commented Oct 26, 2024

cc @RalfJung @compiler-errors This rebase was pretty substantial and required some minor refactors. Would either of you mind giving the const-stability parts a look to make sure they make sense and don't go against planned future changes?

cc @rust-lang/libs-api I realize I should probably ping here as well. Does the design supporting a mix of unstable/stable attributes sound good? I'll try documenting it in the dev guide and std dev guide once that's finalized.

@bors
Copy link
Collaborator

bors commented Oct 27, 2024

☔ The latest upstream changes (presumably #131284) made this pull request unmergeable. Please resolve the merge conflicts.

@RalfJung
Copy link
Member

The PR should explain what the semantics are of having multiple unstable attributes. Do all gates need to be active to use the feature, or only some?

This supports multiple #[stable] attributes, and a mix of #[stable] and #[unstable], in case that's helpful for tracking which features library items depended on as they stabilize (and because it's less difficult to reword E0544 this way).

That sounds really strange. What is the semantics of this? And what is the point?

@dianne
Copy link
Contributor Author

dianne commented Oct 27, 2024

The PR should explain what the semantics are of having multiple unstable attributes. Do all gates need to be active to use the feature, or only some?

This supports multiple #[stable] attributes, and a mix of #[stable] and #[unstable], in case that's helpful for tracking which features library items depended on as they stabilize (and because it's less difficult to reword E0544 this way).

That sounds really strange. What is the semantics of this? And what is the point?

I've updated the PR description to address this. To your second point, I'm working off of this specification. Treating it as a matter of preference, I decided to implement it both with and without support for multiple #[stable] attributes in separate commits so that I'd have an implementation to point to before asking which approach is nicer. I'll work on addressing your review comments shortly. Thank you for giving this a look!

@dianne dianne force-pushed the multiple-unstables branch from 88e33bf to bc6c3b0 Compare October 28, 2024 06:44
@dianne dianne requested a review from RalfJung October 28, 2024 06:56
@RalfJung
Copy link
Member

The updated PR description is great, thanks. :)

Personally I don't think multiple stable attributes, or both stable and unstable, are worth it. But let's see what t-libs-api thinks.

@RalfJung RalfJung added I-libs-api-nominated Nominated for discussion during a libs-api team meeting. T-libs-api Relevant to the library API team, which will review and decide on the PR/issue. labels Oct 28, 2024
@bors
Copy link
Collaborator

bors commented Oct 28, 2024

☔ The latest upstream changes (presumably #132145) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the multiple-unstables branch from bc6c3b0 to 5a1fd55 Compare October 28, 2024 23:40
@bors
Copy link
Collaborator

bors commented Nov 1, 2024

☔ The latest upstream changes (presumably #132435) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne
Copy link
Contributor Author

dianne commented Nov 11, 2024

I did some cleanup to the diagnostic formatting and commit history. Now only the tests for stability attribute sanity (and the tests I added) have diffs. Hopefully it will be more digestible now, and easier to split apart if needed. Once changes to attribute handling and/or const-stability checks land it should hopefully be able to be cleaned up some more.

@bors
Copy link
Collaborator

bors commented Nov 12, 2024

☔ The latest upstream changes (presumably #132954) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the multiple-unstables branch from 4bbcf10 to 814ab38 Compare November 13, 2024 04:59
@petrochenkov
Copy link
Contributor

The updated PR description is great, thanks. :)

Personally I don't think multiple stable attributes, or both stable and unstable, are worth it. But let's see what t-libs-api thinks.

I agree that the cost/benefit ratio here is too high, from the compiler maintenance point of view.
It's not even a user facing feature, just a minor convenience for the library people.

@bors
Copy link
Collaborator

bors commented Nov 15, 2024

☔ The latest upstream changes (presumably #132910) made this pull request unmergeable. Please resolve the merge conflicts.

@dianne dianne force-pushed the multiple-unstables branch 2 times, most recently from 6f28bda to 9138a57 Compare November 16, 2024 04:54
@bors
Copy link
Collaborator

bors commented Nov 18, 2024

☔ The latest upstream changes (presumably #133179) made this pull request unmergeable. Please resolve the merge conflicts.

This moves stability structs' `feature` fields into
`StabilityLevel::Unstable` and `ConstStabilityLevel::Unstable`, in
preparation to support multiple unstable attributes on items.

Seemingly, the `feature` field isn't used with the
`StabilityLevel::Stable` variant, so I haven't included it.
`rustc_passes::lib_features` uses the 'feature' meta-item for 'stable'
attributes, but it extracts them itself, rather than relying on
`rustc_attr`.
…ions

the logic for adding unstable attrs gets a bit messier when supporting multiple
instances thereof. this keeps that from being duplicated in 3 places.
Translation is in an awkward position, but this provides a similar
interface for both errors and soft-unstable lints, and enables the use
of `DiagSymbolList` for simplifying error message construction.
This changes the text for E0544. Unfortunately, the example doesn't make
much sense anymore.

The way this merges stability levels together is made to work for
stability-checking and rustdoc; as far as I can tell, only
`rustc_passes::lib_features` needs them separate, and it extracts them
itself.

This includes the clarified/fixed const-unstable semantics that were
previously in a different commit.
`reason` must appear at most once: it's the reason for the item being unstable, rather than a
particular feature. This simplifies diagnostic formatting.

`soft` must either be on all or no unstable attributes: it doesn't make sense for something to be
partially soft, and allowing inconsistent softness markers would risk an item accidentally becoming
properly unstable as its features stabilize.
@dianne dianne force-pushed the multiple-unstables branch from 9138a57 to b23f28d Compare November 19, 2024 06:28
@Dylan-DPC
Copy link
Member

@dianne i know this is waiting for review, but this has gathered conflicts so it would be better to rebase it and we can get it reviewed after that. Thanks

@Dylan-DPC Dylan-DPC added S-waiting-on-author Status: This is awaiting some action (such as code changes or more information) from the author. and removed S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. labels Mar 1, 2025
@jdonszelmann
Copy link
Contributor

I'm terribly sorry but I'm afraid the conflicts are rather complicated because the way attributes work itself was partially changed. Let me know if I need to help out with info about the new system.

@dianne
Copy link
Contributor Author

dianne commented Mar 1, 2025

This was really conflict-prone anyway, so I've enjoyed the break from maintaining it. ^^ When I rebase/rewrite it, I'll see how easy it'd be to split the const-stability changes into a separate PR; the const-eval parts were the most conflict-heavy, and that that should hopefully make this a bit more digestible too. I'll also see about once again starting off with only supporting multiple unstable attributes.

@jdonszelmann ty! I'll let you know if I have any specific questions. Do you expect any more reworks on your end to how stability attributes are handled? No worries either way! I'm not in a rush at least, so I'm happy to wait for the dust to settle if that means fewer future conflicts.

@alex-semenyuk
Copy link
Member

@dianne
Thanks for your contribution
From wg-triage. Any updates on this PR?

@dianne
Copy link
Contributor Author

dianne commented May 15, 2025

None yet. I'm planning on getting back to this in a week or two and paring it down as much as I can.

@dianne
Copy link
Contributor Author

dianne commented Jun 24, 2025

I've been looking at reviving this. As part of cleaning it up and lessening its perf/stdlib-binary-size impact, I think it'd help to arena-allocate the lists of unstable features that definitions are gated by. StabilityLevel::Unstable would be able to contain a slice reference1, rather than a heap-allocated collection, so Stability and ConstStability still be Copy. Giving the attribute parser access to the dropless arena seems easy enough, but giving hir::Attribute a lifetime parameter is a pretty invasive change, so I figure I should ask: @jdonszelmann, does this sound reasonable? I haven't considered too much whether it would be useful for other attributes, but I've noticed stability attributes are the largest AttributeKind variants, so there's some potential for reducing the size of Attribute (and thus rlib size) by putting the whole Stability/etc. in the arena. Potentially they could be de-duplicated at some point too to further reduce the size of the standard library2 a bit, since most(?) stability attributes are shared between multiple definitions, but I'm not sure whether that's worth the effort.

Relatedly, I've been wondering: is there a reason AttributeKind::Stability and friends are EncodeCrossCrate::Yes? I think this would only affect the size of the standard library2, but those are encoded in their own tables too, corresponding to the lookup_stability family of queries. I might have missed something, but I couldn't find any use of them via get_attr, attrs_for_def, or anything else defined in terms of those.

Footnotes

  1. I checked, and StabilityLevel::Stable would be big enough that using a slice reference in StabilityLevel::Unstable (i.e. storing the length in the variant) doesn't make StabilityLevel any larger. It may not be the ideal representation, but it seems like hopefully a decent compromise between efficiency/simplicity/convenience?

  2. And anything built with -Zforce-unstable-if-unmarked while building the compiler. 2

@alex-semenyuk
Copy link
Member

@dianne
Are you waiting answer from @jdonszelmann?
Meantime, please rebase

@dianne
Copy link
Contributor Author

dianne commented Oct 3, 2025

The rebase requires a substantial amount of rewriting, so since making Stability non-Copy is a fairly messy change, I was hoping to figure out the path forward for this PR before then (ideally with Stability still being Copy), hence the ask. I'll convert the PR to a draft for now, since I'd prefer not to maintain it without a clearer idea of how it should look in the end. Besides the rebase requiring a lot of rewriting (that would again need to be redone if making Stability Copy), it's a very bitrotty PR so I'd ideally like to minimize the window in which I'd need to keep it up-to-date. If it would be preferable (or if someone else would like to tackle this), I can close this PR instead and release my assignment.

@dianne dianne marked this pull request as draft October 3, 2025 02:01
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

S-waiting-on-author Status: This is awaiting some action (such as code changes or more information) from the author. T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. T-libs-api Relevant to the library API team, which will review and decide on the PR/issue. T-rustdoc Relevant to the rustdoc team, which will review and decide on the PR/issue.

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Allow multiple #[unstable] attributes on one item