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

borrowck typeck children together with their root #138499

Open
wants to merge 3 commits into
base: master
Choose a base branch
from

Conversation

lcnr
Copy link
Contributor

@lcnr lcnr commented Mar 14, 2025

This introduces new cycle errors, even with feature(inline_const_pat) removed, see the non-structural-match-types-cycle-err.rs test.

The new cycle error happens as the layout of async-blocks relies on their optimized_mir. As that now depends on mir_borrowck of its typeck parent, computing the layout of an async-block during MIR building, e.g. when evaluating a named const pattern. I think there's currently no way to have a named const pattern whose type references an async block while being allowed? cc @oli-obk @RalfJung

I cannot think of other cases where we currently rely on the MIR of a typeck children while borrowchecking their parent. The crater run came back without any breakage. My work here will prevent any future features which rely on this as we'll get locked into borrowchecking them together as I continue to work on rust-lang/types-team#129, cc @rust-lang/types.

r? compiler-errors

@rustbot
Copy link
Collaborator

rustbot commented Mar 14, 2025

r? @oli-obk

rustbot has assigned @oli-obk.
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 A-tidy Area: The tidy tool S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. T-bootstrap Relevant to the bootstrap subteam: Rust's build system (x.py and src/bootstrap) T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. labels Mar 14, 2025
@rustbot rustbot assigned compiler-errors and unassigned oli-obk Mar 14, 2025
@lcnr
Copy link
Contributor Author

lcnr commented Mar 14, 2025

@bors try

bors added a commit to rust-lang-ci/rust that referenced this pull request Mar 14, 2025
borrowck typeck children together with their root

This is a theoretical breaking change, see the `non-structural-match-types-cycle-err.rs` test. Opening for a crater run.

r? compiler-errors
@bors
Copy link
Collaborator

bors commented Mar 14, 2025

⌛ Trying commit f319a98 with merge d534db0...

@rust-log-analyzer

This comment has been minimized.

@bors
Copy link
Collaborator

bors commented Mar 14, 2025

☀️ Try build successful - checks-actions
Build commit: d534db0 (d534db0721b23cf977073bf4cd306dbf501e6b4d)

@compiler-errors
Copy link
Member

@craterbot check

@craterbot
Copy link
Collaborator

👌 Experiment pr-138499 created and queued.
🤖 Automatically detected try build d534db0
🔍 You can check out the queue and this experiment's details.

ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more

@craterbot craterbot added S-waiting-on-crater Status: Waiting on a crater run to be completed. and removed S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. labels Mar 14, 2025
@craterbot
Copy link
Collaborator

🚧 Experiment pr-138499 is now running

ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more

@craterbot
Copy link
Collaborator

🎉 Experiment pr-138499 is completed!
📊 9 regressed and 3 fixed (597764 total)
📰 Open the full report.

⚠️ If you notice any spurious failure please add them to the denylist!
ℹ️ Crater is a tool to run experiments across parts of the Rust ecosystem. Learn more

@craterbot craterbot added S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. and removed S-waiting-on-crater Status: Waiting on a crater run to be completed. labels Mar 17, 2025
@lcnr
Copy link
Contributor Author

lcnr commented Mar 17, 2025

all regressions look either spurious or due to the removal of inline_const_pat

@lcnr lcnr force-pushed the borrowck-typeck_root branch from f319a98 to 35f93c5 Compare March 17, 2025 12:02
@rust-log-analyzer

This comment has been minimized.

@lcnr lcnr force-pushed the borrowck-typeck_root branch from 35f93c5 to 519ed02 Compare March 20, 2025 12:51
@rust-log-analyzer

This comment has been minimized.

@bors
Copy link
Collaborator

bors commented Mar 22, 2025

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

@lcnr

This comment was marked as outdated.

@lcnr lcnr closed this Apr 1, 2025
@lcnr lcnr reopened this Apr 1, 2025
@lcnr lcnr force-pushed the borrowck-typeck_root branch from 519ed02 to 48a3c5e Compare April 2, 2025 11:41
@lcnr lcnr marked this pull request as ready for review April 2, 2025 11:53
@rustbot
Copy link
Collaborator

rustbot commented Apr 2, 2025

Some changes occurred to MIR optimizations

cc @rust-lang/wg-mir-opt

@lcnr
Copy link
Contributor Author

lcnr commented Apr 2, 2025

@bors try @rust-timer queue

@rust-timer

This comment has been minimized.

@rustbot rustbot added the S-waiting-on-perf Status: Waiting on a perf run to be completed. label Apr 2, 2025
bors added a commit to rust-lang-ci/rust that referenced this pull request Apr 2, 2025
borrowck typeck children together with their root

This introduces new cycle errors, even with `feature(inline_const_pat)` removed, see the `non-structural-match-types-cycle-err.rs` test.

The new cycle error happens as the layout of `async`-blocks relies on their `optimized_mir`. As that now depends on `mir_borrowck` of its typeck parent, computing the layout of an `async`-block during MIR building, e.g. when evaluating a named `const` pattern. I think there's currently no way to have a named const pattern whose type references an async block while being allowed? cc `@oli-obk` `@RalfJung`

I cannot think of other cases where we currently rely on the MIR of a typeck children while borrowchecking their parent. The crater run came back without any breakage. My work here will prevent any future features which rely on this as we'll get locked into borrowchecking them together as I continue to work on rust-lang/types-team#129, cc `@rust-lang/types.`

r? compiler-errors
@bors
Copy link
Collaborator

bors commented Apr 2, 2025

⌛ Trying commit 48a3c5e with merge 8565895...

@lcnr lcnr force-pushed the borrowck-typeck_root branch from 48a3c5e to 1bcbc5c Compare April 2, 2025 12:06
we already collect opaque types from nested items
during `mir_borrowck` of the root, checking that they
are consistent this way.
@lcnr lcnr force-pushed the borrowck-typeck_root branch from 1bcbc5c to e60daca Compare April 2, 2025 12:24
@compiler-errors
Copy link
Member

@bors try @rust-timer queue

@rust-timer

This comment has been minimized.

@bors
Copy link
Collaborator

bors commented Apr 3, 2025

⌛ Trying commit e60daca with merge 60ec9868717c8bc856a18fb96d637c2f5286a1a6...

bors added a commit to rust-lang-ci/rust that referenced this pull request Apr 3, 2025
borrowck typeck children together with their root

This introduces new cycle errors, even with `feature(inline_const_pat)` removed, see the `non-structural-match-types-cycle-err.rs` test.

The new cycle error happens as the layout of `async`-blocks relies on their `optimized_mir`. As that now depends on `mir_borrowck` of its typeck parent, computing the layout of an `async`-block during MIR building, e.g. when evaluating a named `const` pattern. I think there's currently no way to have a named const pattern whose type references an async block while being allowed? cc `@oli-obk` `@RalfJung`

I cannot think of other cases where we currently rely on the MIR of a typeck children while borrowchecking their parent. The crater run came back without any breakage. My work here will prevent any future features which rely on this as we'll get locked into borrowchecking them together as I continue to work on rust-lang/types-team#129, cc `@rust-lang/types.`

r? compiler-errors
@bors
Copy link
Collaborator

bors commented Apr 3, 2025

☀️ Try build successful - checks-actions
Build commit: 60ec986 (60ec9868717c8bc856a18fb96d637c2f5286a1a6)

@rust-timer

This comment has been minimized.

@rust-timer
Copy link
Collaborator

Finished benchmarking commit (60ec986): comparison URL.

Overall result: ✅ improvements - no action needed

Benchmarking this pull request likely means that it is perf-sensitive, so we're automatically marking it as not fit for rolling up. While you can manually mark this PR as fit for rollup, we strongly recommend not doing so since this PR may lead to changes in compiler perf.

@bors rollup=never
@rustbot label: -S-waiting-on-perf -perf-regression

Instruction count

This 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.

mean range count
Regressions ❌
(primary)
- - 0
Regressions ❌
(secondary)
- - 0
Improvements ✅
(primary)
- - 0
Improvements ✅
(secondary)
-0.8% [-1.4%, -0.5%] 9
All ❌✅ (primary) - - 0

Max RSS (memory usage)

This benchmark run did not return any relevant results for this metric.

Cycles

Results (secondary 1.1%)

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.

mean range count
Regressions ❌
(primary)
- - 0
Regressions ❌
(secondary)
1.7% [1.4%, 2.1%] 5
Improvements ✅
(primary)
- - 0
Improvements ✅
(secondary)
-1.9% [-1.9%, -1.9%] 1
All ❌✅ (primary) - - 0

Binary size

This benchmark run did not return any relevant results for this metric.

Bootstrap: 777.539s -> 775.513s (-0.26%)
Artifact size: 365.91 MiB -> 366.02 MiB (0.03%)

@rustbot rustbot removed the S-waiting-on-perf Status: Waiting on a perf run to be completed. label Apr 3, 2025
@RalfJung
Copy link
Member

RalfJung commented Apr 3, 2025

closing in favor of #138785

So is this outdated information now? The state of this PR is quite confusing.

I think there's currently no way to have a named const pattern whose type references an async block while being allowed?

Sorry, I don't understand this question. What do you mean?

@lcnr
Copy link
Contributor Author

lcnr commented Apr 3, 2025

So is this outdated information now? The state of this PR is quite confusing.

yeah 😅 originally contained part of this PR in #138785 but ended up cleanly(kinda) separating afterwards

Sorry, I don't understand this question. What do you mean?

The following test went from a "constant of non-structural type" error to a cycle error:

struct AnyOption<T>(T);
impl<T> AnyOption<T> {
    const NONE: Option<T> = None;
}

fn uwu() {}
fn defines() {
    match Some(async {}) {
        AnyOption::<_>::NONE => {}
        //[nightly]~^ ERROR constant of non-structural type
        //[thisPR]~^ ERROR cycle detected
        _ => {}
    }
}
fn main() {}

The change is that computing the layout of async blocks inside of their parent function now cycles. From my knowledge the only case where we actually do it is when evaluating constants in match patterns. As the type of the async block is unnameable, it has to be part of the constants type signature.

Does having an async block/generator in your type always result in structural match errors or are there ways to use a constant whose type references an async block without such an error?

I think it doesn't matter too much, as I don't expect people to write this code (as can be seem by the clean crater run), but it would be the difference between a purely internal refactoring and a theoretical breaking change.

@lcnr lcnr force-pushed the borrowck-typeck_root branch from e60daca to f94b96a Compare April 3, 2025 08:25
@RalfJung
Copy link
Member

RalfJung commented Apr 3, 2025

Does having an async block/generator in your type always result in structural match errors or are there ways to use a constant whose type references an async block without such an error?

I assume async blocks / generators never implement PartialEq?

We require the type of the const to implement PartialEq. So if async blocks / generators never do that, the only way they could appear in a legal const pattern is by having a type where PartialEq was hand-implemented to skip certain type parameters, which means StructuralPartialEq must also have been hand-implemented, which stable code cannot currently do.

So, I think the answer to your question is "no". But I am also rather clueless about everything related to async blocks / generators / opaque types so I am not 100% confident about this.

In the future we might want to let people implement StructuralPartialEq themselves and then there could definitely be examples hitting that. So if we are fully committing to having borrowck work like that, we should be okay with getting weird cycle errors in corner cases like this.

@lcnr lcnr force-pushed the borrowck-typeck_root branch from f94b96a to 95de83f Compare April 3, 2025 08:41
@lcnr lcnr mentioned this pull request Apr 3, 2025
bors added a commit to rust-lang-ci/rust that referenced this pull request Apr 4, 2025
…r-errors,oli-obk

add `TypingMode::Borrowck`

Shares the first commit with rust-lang#138499, doesn't really matter which PR to land first 😊 😁

Introduces `TypingMode::Borrowck` which unlike `TypingMode::Analysis`, uses the hidden type computed by HIR typeck as the initial value of opaques instead of an unconstrained infer var. This is a part of rust-lang/types-team#129.

Using this new `TypingMode` is unfortunately a breaking change for now, see tests/ui/impl-trait/non-defining-uses/as-projection-term.rs. Using an inference variable as the initial value results in non-defining uses in the defining scope. We therefore only enable it if with `-Znext-solver=globally` or `-Ztyping-mode-borrowck`

To do that the PR contains the following changes:
- `TypeckResults::concrete_opaque_type` are already mapped to the definition of the opaque type
  - writeback now checks that the non-lifetime parameters of the opaque are universal
  - for this, `fn check_opaque_type_parameter_valid` is moved from `rustc_borrowck` to `rustc_trait_selection`
- we add a new `query type_of_opaque_hir_typeck` which, using the same visitors as MIR typeck, attempts to merge the hidden types from HIR typeck from all defining scopes
  - done by adding a `DefiningScopeKind` flag to toggle between using borrowck and HIR typeck
  - the visitors stop checking that the MIR type matches the HIR type. This is trivial as the HIR type are now used as the initial hidden types of the opaque. This check is useful as a safeguard when not using `TypingMode::Borrowck`, but adding it to the new structure is annoying and it's not soundness critical, so I intend to not add it back.
- add a `TypingMode::Borrowck`  which behaves just like `TypingMode::Analysis` except when normalizing opaque types
   - it uses `type_of_opaque_hir_typeck(opaque)` as the initial value after replacing its regions with new inference vars
   - it uses structural lookup in the new solver

fixes rust-lang#112201, fixes rust-lang#132335, fixes rust-lang#137751

r? `@compiler-errors` `@oli-obk`
@bors
Copy link
Collaborator

bors commented Apr 5, 2025

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

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-tidy Area: The tidy tool S-waiting-on-review Status: Awaiting review from the assignee but also interested parties. T-bootstrap Relevant to the bootstrap subteam: Rust's build system (x.py and src/bootstrap) T-compiler Relevant to the compiler team, which will review and decide on the PR/issue.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

9 participants