Skip to content
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

rustdoc: Consider eliminating edition sensitivity around RPIT capturing #135453

Open
fmease opened this issue Jan 13, 2025 · 2 comments
Open

rustdoc: Consider eliminating edition sensitivity around RPIT capturing #135453

fmease opened this issue Jan 13, 2025 · 2 comments
Assignees
Labels
A-impl-trait Area: `impl Trait`. Universally / existentially quantified anonymous types with static dispatch. C-enhancement Category: An issue proposing an enhancement or a PR with one. T-rustdoc Relevant to the rustdoc team, which will review and decide on the PR/issue.

Comments

@fmease
Copy link
Member

fmease commented Jan 13, 2025

Uplifted from #127228 (comment) and #t-lang/blog post about precise capture (comment).


At the time of writing, rustdoc renders RPITs (return-position impl-Trait) basically identical to how they appear in the source code, most important of which is impl-Trait captures (which boils down to the presence, absence and contents of use<…> bounds for all practical purposes). Since RPIT (but not RPITIT) capturing is edition sensitive, it follows that rustdoc's RPIT rendering is edition sensitive as well. This leads us to the point of discussion: Should we (somehow) eliminate this edition sensitivity in rustdoc?

Consider the following signature as it might be shown by rustdoc: pub fn f(x: &()) -> impl Sized. Unless you know the edition of the containing crate (which could be regarded as an implementation detail?), you won't know if the RPIT impl Sized captures the anonymous lifetime of the parameter (i.e., impl Sized + use<'_>) or not (i.e., impl Sized + use<>). I believe this to be less than ideal

Edition insensitivity would solve this:

(A) The solution I initially thought of would comprise rendering synthetic use<…> bounds on use<…>-less1 RPITs whenever we have any2 generic parameters in all editions! Note: If we have anonymous lifetime parameters, we can't always easily synthesize a use<…> bound3.

However, this leads to more verbose output and fixates too much on pre-2024 edition behavior. Assuming the 2021 edition and before lose relevance by the minute, this solution is stuck in the past. Moreover, <2024 RPIT capturing behavior is the odd one out among RPITIT, TAIT, ATPIT (and ≥2024 RPIT).

(B) I guess the more realistic and forward-thinking solution would be to render ≥2024 RPITs as is and to always render a synthetic use<…> bounds on use<…>-less1 <2024 RPITs if there are any2 generic parameters in scope. We don't face the "anon lifetime" issues solution (A) has because the synthetic use<…> bounds should never (need to) contain any lifetime parameters! In fact, if we didn't care about non-lifetime generic parameters, we could get away with just using + use<> for ≤2021 edition crates.


I'm pretty sure this whole discussion is also relevant to rustdoc JSON (does its output even contain the edition?).

Footnotes

  1. Natural use<…> bounds are already edition agnostic/insensitive. 2

  2. I'm aware that opaque types are currently required to capture all in-scope type and const parameters (which is a temporary restriction I assume), so we could relax the rule to only consider lifetime parameters. 2

  3. Consider this Rust 2024(!) snippet: pub fn f(x: &(), y: &()) -> impl Sized. The opaque type impl Sized captures both anonymous/elided lifetime parameters. We can't denote this, use<'_> is not really an option since it's semantically ill-formed. I guess we could still use it as a sort of pseudo-Rust (rustdoc already employs pseudo-Rust syntax in certain cases). Otherwise, we could render it as pub fn f<'a, 'b>(x: &'a (), y: &'b ()) -> impl Sized + use<'a, 'b> (ensuring that 'a and 'b occur free in the signature) or pseudo-Rust pub fn f(x: &'0 (), y: &'1 ()) -> impl Sized + use<'0, '1>.

@fmease fmease added A-impl-trait Area: `impl Trait`. Universally / existentially quantified anonymous types with static dispatch. C-discussion Category: Discussion or questions that doesn't represent real issues. T-rustdoc Relevant to the rustdoc team, which will review and decide on the PR/issue. labels Jan 13, 2025
@rustbot rustbot added the needs-triage This issue may need triage. Remove it if it has been sufficiently triaged. label Jan 13, 2025
@fmease fmease removed the needs-triage This issue may need triage. Remove it if it has been sufficiently triaged. label Jan 13, 2025
@jieyouxu
Copy link
Member

Alternative idea: is it possible to go the opposite direction, and be more "faithful" to the edition differences -- show the edition of the crate on the page?

@fmease fmease self-assigned this Feb 10, 2025
@fmease
Copy link
Member Author

fmease commented Feb 10, 2025

In meeting 2025-02-10 we decided to go with solution (B) for now. I'll spin up a needs-FCP PR and we can go from there.

@fmease fmease added C-enhancement Category: An issue proposing an enhancement or a PR with one. and removed C-discussion Category: Discussion or questions that doesn't represent real issues. labels Mar 26, 2025
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-impl-trait Area: `impl Trait`. Universally / existentially quantified anonymous types with static dispatch. C-enhancement Category: An issue proposing an enhancement or a PR with one. T-rustdoc Relevant to the rustdoc team, which will review and decide on the PR/issue.
Projects
None yet
Development

No branches or pull requests

3 participants