Skip to content

Commit a20a04e

Browse files
committed
Auto merge of rust-lang#113108 - compiler-errors:normalize-opaques-with-late-bound-vars-again, r=jackh726
Normalize opaques with late-bound vars again We have a hack in the compiler where if an opaque has escaping late-bound vars, we skip revealing it even though we *could* reveal it from a technical perspective. First of all, this is weird, since we really should be revealing all opaques in `Reveal::All` mode. Second of all, it causes subtle bugs (linked below). I attempted to fix this in rust-lang#100980, which was unfortunately reverted due to perf regressions on codebases that used really deeply nested futures in some interesting ways. The worst of which was rust-lang#103423, which caused the project to hang on build. Another one was rust-lang#104842, which was just a slow-down, but not a hang. I took some time afterwards to investigate how to rework `normalize_erasing_regions` to take advantage of better caching, but that effort kinda fizzled out (rust-lang#104133). However, recently, I was made aware of more bugs whose root cause is not revealing opaques during codegen. That made me want to fix this again -- in the process, interestingly, I took the the minimized example from rust-lang#103423 (comment), and it doesn't seem to hang any more... Thinking about this harder, there have been some changes to the way we lower and typecheck async futures that may have reduced the pathologically large number of outlives obligations (see description of rust-lang#103423) that we were encountering when normalizing opaques with bound vars the last time around: * rust-lang#104321 (lower `async { .. }` directly as a generator that implements `Future`, removing the `from_generator` shim) * rust-lang#104833 (removing an `identity_future` fn that was wrapping desugared future generators) ... so given that I can see: * No significant regression on rust perf bot (rust-lang#107620 (comment)) * No timeouts in crater run I did (rust-lang#107620 (comment), rechecked failing crates in rust-lang#107620 (comment)) ... and given that this PR: * Fixes rust-lang#104601 * Fixes rust-lang#107557 * Fixes rust-lang#109464 * Allows us to remove a `DefiningAnchor::Bubble` from codegen (75a8f68) I'm inclined to give this another shot at landing this. Best case, it just works -- worst case, we get more examples to study how we need to improve the compiler to make this work. r? types
2 parents e69c730 + bfc6ca8 commit a20a04e

File tree

5 files changed

+95
-24
lines changed

5 files changed

+95
-24
lines changed

compiler/rustc_trait_selection/src/traits/project.rs

+1-5
Original file line numberDiff line numberDiff line change
@@ -500,10 +500,7 @@ impl<'a, 'b, 'tcx> TypeFolder<TyCtxt<'tcx>> for AssocTypeNormalizer<'a, 'b, 'tcx
500500
// to make sure we don't forget to fold the substs regardless.
501501

502502
match kind {
503-
// This is really important. While we *can* handle this, this has
504-
// severe performance implications for large opaque types with
505-
// late-bound regions. See `issue-88862` benchmark.
506-
ty::Opaque if !data.substs.has_escaping_bound_vars() => {
503+
ty::Opaque => {
507504
// Only normalize `impl Trait` outside of type inference, usually in codegen.
508505
match self.param_env.reveal() {
509506
Reveal::UserFacing => ty.super_fold_with(self),
@@ -529,7 +526,6 @@ impl<'a, 'b, 'tcx> TypeFolder<TyCtxt<'tcx>> for AssocTypeNormalizer<'a, 'b, 'tcx
529526
}
530527
}
531528
}
532-
ty::Opaque => ty.super_fold_with(self),
533529

534530
ty::Projection if !data.has_escaping_bound_vars() => {
535531
// This branch is *mostly* just an optimization: when we don't

compiler/rustc_trait_selection/src/traits/query/normalize.rs

+1-6
Original file line numberDiff line numberDiff line change
@@ -211,10 +211,7 @@ impl<'cx, 'tcx> FallibleTypeFolder<TyCtxt<'tcx>> for QueryNormalizer<'cx, 'tcx>
211211

212212
// Wrap this in a closure so we don't accidentally return from the outer function
213213
let res = match kind {
214-
// This is really important. While we *can* handle this, this has
215-
// severe performance implications for large opaque types with
216-
// late-bound regions. See `issue-88862` benchmark.
217-
ty::Opaque if !data.substs.has_escaping_bound_vars() => {
214+
ty::Opaque => {
218215
// Only normalize `impl Trait` outside of type inference, usually in codegen.
219216
match self.param_env.reveal() {
220217
Reveal::UserFacing => ty.try_super_fold_with(self)?,
@@ -255,8 +252,6 @@ impl<'cx, 'tcx> FallibleTypeFolder<TyCtxt<'tcx>> for QueryNormalizer<'cx, 'tcx>
255252
}
256253
}
257254

258-
ty::Opaque => ty.try_super_fold_with(self)?,
259-
260255
ty::Projection | ty::Inherent | ty::Weak => {
261256
// See note in `rustc_trait_selection::traits::project`
262257

compiler/rustc_traits/src/codegen.rs

+2-13
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,7 @@
55

66
use rustc_infer::infer::TyCtxtInferExt;
77
use rustc_infer::traits::{FulfillmentErrorCode, TraitEngineExt as _};
8-
use rustc_middle::traits::{CodegenObligationError, DefiningAnchor};
8+
use rustc_middle::traits::CodegenObligationError;
99
use rustc_middle::ty::{self, TyCtxt};
1010
use rustc_trait_selection::traits::error_reporting::TypeErrCtxtExt;
1111
use rustc_trait_selection::traits::{
@@ -29,13 +29,7 @@ pub fn codegen_select_candidate<'tcx>(
2929

3030
// Do the initial selection for the obligation. This yields the
3131
// shallow result we are looking for -- that is, what specific impl.
32-
let infcx = tcx
33-
.infer_ctxt()
34-
.ignoring_regions()
35-
.with_opaque_type_inference(DefiningAnchor::Bubble)
36-
.build();
37-
//~^ HACK `Bubble` is required for
38-
// this test to pass: type-alias-impl-trait/assoc-projection-ice.rs
32+
let infcx = tcx.infer_ctxt().ignoring_regions().build();
3933
let mut selcx = SelectionContext::new(&infcx);
4034

4135
let obligation_cause = ObligationCause::dummy();
@@ -79,10 +73,5 @@ pub fn codegen_select_candidate<'tcx>(
7973
let impl_source = infcx.resolve_vars_if_possible(impl_source);
8074
let impl_source = infcx.tcx.erase_regions(impl_source);
8175

82-
// Opaque types may have gotten their hidden types constrained, but we can ignore them safely
83-
// as they will get constrained elsewhere, too.
84-
// (ouz-a) This is required for `type-alias-impl-trait/assoc-projection-ice.rs` to pass
85-
let _ = infcx.take_opaque_types();
86-
8776
Ok(&*tcx.arena.alloc(impl_source))
8877
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
// build-pass
2+
// edition:2021
3+
// compile-flags: -Cdebuginfo=2
4+
5+
// We were not normalizing opaques with escaping bound vars during codegen,
6+
// leading to later errors during debuginfo computation.
7+
8+
#![feature(async_fn_in_trait)]
9+
10+
#[derive(Clone, Copy)]
11+
pub struct SharedState {}
12+
13+
pub trait State {
14+
async fn execute(self, shared_state: &SharedState);
15+
}
16+
17+
pub trait StateComposer {
18+
fn and_then<T, F>(self, map_fn: F) -> AndThen<Self, F>
19+
where
20+
Self: State + Sized,
21+
T: State,
22+
F: FnOnce() -> T,
23+
{
24+
AndThen { previous: self, map_fn }
25+
}
26+
}
27+
28+
impl<T> StateComposer for T where T: State {}
29+
pub struct AndThen<T, F> {
30+
previous: T,
31+
map_fn: F,
32+
}
33+
34+
impl<T, U, F> State for AndThen<T, F>
35+
where
36+
T: State,
37+
U: State,
38+
F: FnOnce() -> U,
39+
{
40+
async fn execute(self, shared_state: &SharedState)
41+
where
42+
Self: Sized,
43+
{
44+
self.previous.execute(shared_state).await;
45+
(self.map_fn)().execute(shared_state).await
46+
}
47+
}
48+
49+
pub struct SomeState {}
50+
51+
impl State for SomeState {
52+
async fn execute(self, shared_state: &SharedState) {}
53+
}
54+
55+
pub fn main() {
56+
let shared_state = SharedState {};
57+
async {
58+
SomeState {}
59+
.and_then(|| SomeState {})
60+
.and_then(|| SomeState {})
61+
.execute(&shared_state)
62+
.await;
63+
};
64+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,27 @@
1+
// build-pass
2+
// edition:2021
3+
// compile-flags: -Cdebuginfo=2
4+
5+
// We were not normalizing opaques with escaping bound vars during codegen,
6+
// leading to later linker errors because of differences in mangled symbol name.
7+
8+
fn func<T>() -> impl Sized {}
9+
10+
trait Trait<'a> {
11+
type Assoc;
12+
13+
fn call() {
14+
let _ = async {
15+
let _value = func::<Self::Assoc>();
16+
std::future::ready(()).await
17+
};
18+
}
19+
}
20+
21+
impl Trait<'static> for () {
22+
type Assoc = ();
23+
}
24+
25+
fn main() {
26+
<()>::call();
27+
}

0 commit comments

Comments
 (0)