Skip to content
New issue

Have a question about this project? # for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “#”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? # to your account

compilation does not terminate using async closures #135780

Closed
nikomatsakis opened this issue Jan 20, 2025 · 8 comments · Fixed by #135783
Closed

compilation does not terminate using async closures #135780

nikomatsakis opened this issue Jan 20, 2025 · 8 comments · Fixed by #135783
Assignees
Labels
A-async-closures `async || {}` A-closures Area: Closures (`|…| { … }`) A-monomorphization Area: Monomorphization C-bug Category: This is a bug. I-hang Issue: The compiler never terminates, due to infinite loops, deadlock, livelock, etc. T-compiler Relevant to the compiler team, which will review and decide on the PR/issue.

Comments

@nikomatsakis
Copy link
Contributor

nikomatsakis commented Jan 20, 2025

The following code:

https://play.rust-lang.org/?version=nightly&mode=debug&edition=2021&gist=efa07d177105d0c3aecf5bc64a9bf8b3

use std::ops::AsyncFnMut;
use std::pin::Pin;

trait Db {}

impl Db for () { }

struct Env<'db> {
    db: &'db ()
}

#[derive(Debug)]
enum SymPerm<'db> {
    Dummy(&'db ()),
    Apply(Box<SymPerm<'db>>, Box<SymPerm<'db>>)
}

pub struct ToChain<'env, 'db> {
    db: &'db dyn crate::Db,
    env: &'env Env<'db>,
}

impl<'env, 'db> ToChain<'env, 'db> {
    fn perm_pairs<'l>(
        &'l self,
        perm: &'l SymPerm<'db>,
        yield_chain: &'l mut impl AsyncFnMut(&SymPerm<'db>),
    ) -> Pin<Box<dyn std::future::Future<Output = ()> + 'l>> {
        Box::pin(async move {
            match perm {
                SymPerm::Dummy(_) => {
                    yield_chain(perm).await
                }
                SymPerm::Apply(l, r) => {
                    self.perm_pairs(l, &mut async |left_pair| {
                        self.perm_pairs(r, yield_chain)
                            .await
                    })
                    .await
                }
            }
        })
    }
}

#[tokio::main]
async fn main() {
    let pair = SymPerm::Apply(
        Box::new(SymPerm::Dummy(&())),
        Box::new(SymPerm::Dummy(&()))
    );
    ToChain { db: &(), env: &Env { db: &() } }.perm_pairs(
        &pair,
        &mut async |p| {
            eprintln!("{p:?}");
        }
    ).await;
}

causes compilation to enter into an infinite loop. This was reduced from my dada project (run cargo build on the tip of this branch).

Many many thanks to @lqd for helping me minimize this.

cc @compiler-errors @rust-lang/types

@nikomatsakis nikomatsakis added the C-bug Category: This is a bug. label Jan 20, 2025
@rustbot rustbot added the needs-triage This issue may need triage. Remove it if it has been sufficiently triaged. label Jan 20, 2025
@nikomatsakis
Copy link
Contributor Author

Reproducing the hang requires a true cargo build, at least it does on the dada version.

@lqd
Copy link
Member

lqd commented Jan 20, 2025

Random notes along the way to finding the source+offending closure:

  • confirm_async_closure_candidate here had some seemingly exponential behavior here.
FoldEscapingRegions done, in 5.39µs
...
FoldEscapingRegions done, in 4.302674ms
FoldEscapingRegions done, in 4.326781ms
...
FoldEscapingRegions done, in 1.102144628s
FoldEscapingRegions done, in 1.070276436s
FoldEscapingRegions done, in 4.392306293s
FoldEscapingRegions done, in 4.283039861s
FoldEscapingRegions done, in 17.604121163s
FoldEscapingRegions done, in 16.950846174s
  • when minimizing and playing around, there are often reached the recursion limit while instantiating errors for the async closure instead of the hang.

I'll try to minimize this a bit more later today, while errs enjoys his well-deserved holiday ^^.

@compiler-errors
Copy link
Member

This looks like a monomorphization overflow. I expect it to result in a build error, but obviously not a hang.

@lqd
Copy link
Member

lqd commented Jan 20, 2025

use std::ops::AsyncFn;

fn recur(closure: &impl AsyncFn()) {
    let _ = closure();
    if false {
        recur(&async || {
            let _ = closure();
        });
    }
}

fn main() {
    let closure = async || {};
    let _ = recur(&closure);
}

@fmease fmease added T-compiler Relevant to the compiler team, which will review and decide on the PR/issue. I-hang Issue: The compiler never terminates, due to infinite loops, deadlock, livelock, etc. A-monomorphization Area: Monomorphization A-closures Area: Closures (`|…| { … }`) A-async-closures `async || {}` and removed needs-triage This issue may need triage. Remove it if it has been sufficiently triaged. labels Jan 20, 2025
bors added a commit to rust-lang-ci/rust that referenced this issue Jan 21, 2025
…der, r=<try>

Add cache to `FoldEscapingRegions`

Fixes rust-lang#135780

ty `@lqd` for the tests
@Christiantyemele

This comment has been minimized.

@Christiantyemele
Copy link

Hello @compiler-errors this seems interested plz i wish to know who far you are with this issue if i could help

@oli-obk
Copy link
Contributor

oli-obk commented Jan 24, 2025

This issue is already fixed by #135783 which is approved and waiting to be merged

@Christiantyemele
Copy link

This issue is already fixed by #135783 which is approved and waiting to be merged

Okay thx

bors added a commit to rust-lang-ci/rust that referenced this issue Jan 26, 2025
…der, r=lcnr

Add cache to `FoldEscapingRegions`

Fixes rust-lang#135780

ty `@lqd` for the tests
@bors bors closed this as completed in f85c6de Jan 26, 2025
# for free to join this conversation on GitHub. Already have an account? # to comment
Labels
A-async-closures `async || {}` A-closures Area: Closures (`|…| { … }`) A-monomorphization Area: Monomorphization C-bug Category: This is a bug. I-hang Issue: The compiler never terminates, due to infinite loops, deadlock, livelock, etc. T-compiler Relevant to the compiler team, which will review and decide on the PR/issue.
Projects
None yet
Development

Successfully merging a pull request may close this issue.

7 participants