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

cargo doc stack overflow (1.31) #56701

Closed
rusty-objects opened this issue Dec 11, 2018 · 4 comments · Fixed by #56722
Closed

cargo doc stack overflow (1.31) #56701

rusty-objects opened this issue Dec 11, 2018 · 4 comments · Fixed by #56722
Labels
A-synthetic-impls Area: Synthetic impls, used by rustdoc to document auto traits and traits with blanket impls I-crash Issue: The compiler crashes (SIGSEGV, SIGABRT, etc). Use I-ICE instead when the compiler panics. T-rustdoc Relevant to the rustdoc team, which will review and decide on the PR/issue.

Comments

@rusty-objects
Copy link

The following code causes a stack overflow:

https://play.rust-lang.org/?version=stable&mode=debug&edition=2018&gist=df017106fc6a032b1b0959b1701b9781

Also here:

// compiling all this presents a stack overflow.  But removing any trait or impl fixes it

pub trait SimpleTrait {
    type SimpleT;
}

impl<Inner: SimpleTrait, Outer: Deref<Target = Inner>> SimpleTrait for Outer {
    type SimpleT = Inner::SimpleT;
}

pub trait AnotherTrait {
    type AnotherT;
}

impl<T, Simple: SimpleTrait<SimpleT = Vec<T>>> AnotherTrait for Simple {
    type AnotherT = T;
}

// this code block seems completely unrelated to the above, however Unrelated compiles alone, and SimpleTrait/AnotherTrait compile alone.
// But the combination won't compile.

pub struct Unrelated<Inner, UnrelatedT: DerefMut<Target = Vec<Inner>>>(UnrelatedT);

impl<Inner, UnrelatedT: DerefMut<Target = Vec<Inner>>> Deref for Unrelated<Inner, UnrelatedT> {
    type Target = Vec<Inner>;

    fn deref(&self) -> &Self::Target {
        &self.0
    }
}


pub fn main() { }

The output:

$ cargo doc
Documenting overflow-example v0.1.0 (/private/tmp/overflow_example)                                                                                      
                                                                                                                                                         
thread '<unknown>' has overflowed its stack                                                                                                               
fatal runtime error: stack overflow                                                                                                                       
error: Could not document `overflow-example`.                                                                                                             
 
Caused by:
 process didn't exit successfully: `rustdoc --edition=2018 --crate-name overflow_example src/main.rs --color always -o /private/tmp/overflow_example/target/doc -L dependency=/private/tmp/overflow_example/target/debug/deps` (signal: 6, SIGABRT: process abort signal)

$ cargo --version
cargo 1.31.0 (339d9f9c8 2018-11-16)

$ rustc --version
rustc 1.31.0 (abe02cefd 2018-12-04)
@rusty-objects
Copy link
Author

Note that this will compile just fine. It's the doc that overflows. So in the playground you'll be able to get it to compile, and cargo build works just fine.

@rusty-objects
Copy link
Author

$ rustdoc --version
rustdoc 1.31.0 (abe02cefd 2018-12-04)

@estebank estebank added I-crash Issue: The compiler crashes (SIGSEGV, SIGABRT, etc). Use I-ICE instead when the compiler panics. T-rustdoc Relevant to the rustdoc team, which will review and decide on the PR/issue. labels Dec 11, 2018
@Aaron1011
Copy link
Member

Aaron1011 commented Dec 11, 2018

The bottom part of the stack trace: https://gist.github.com/Aaron1011/6ee3bab4a34d6ef895cfb19d25bcd36c

This appears to be caused by BlanketImplFinder, which is somehow causing SelectionContext::evaluate_predicate_recursively to infinitely recurse.

@Aaron1011
Copy link
Member

The issue appears to be caused by the fact that recursion_depth is only increased during confirmation of candidates. Since BlanketImplFinder calls evaluate_obligation_recursively, which skips confirmation, the recursion depth is never incremented. This will lead to a stack overflow if the recursion goes deep enough, since we'll never end up bailing out with an overflow error.

Aaron1011 added a commit to Aaron1011/rust that referenced this issue Jan 4, 2019
Currently, SelectionContext tries to prevent stack overflow by keeping
track of the current recursion depth. However, this depth tracking is
only used when performing normal section (which includes confirmation).
No such tracking is performed for evaluate_obligation_recursively, which
can allow a stack overflow to occur.

To fix this, this commit tracks the current predicate evaluation depth.
This is done separately from the existing obligation depth tracking:
an obligation overflow can occur across multiple calls to 'select' (e.g.
when fulfilling a trait), while a predicate evaluation overflow can only
happen as a result of a deep recursive call stack.

Fixes rust-lang#56701
@QuietMisdreavus QuietMisdreavus added the A-synthetic-impls Area: Synthetic impls, used by rustdoc to document auto traits and traits with blanket impls label Jan 16, 2019
bors added a commit that referenced this issue Jan 19, 2019
…sakis

Fix stack overflow when finding blanket impls

Currently, SelectionContext tries to prevent stack overflow by keeping
track of the current recursion depth. However, this depth tracking is
only used when performing normal section (which includes confirmation).
No such tracking is performed for evaluate_obligation_recursively, which
can allow a stack overflow to occur.

To fix this, this commit tracks the current predicate evaluation depth.
This is done separately from the existing obligation depth tracking:
an obligation overflow can occur across multiple calls to 'select' (e.g.
when fulfilling a trait), while a predicate evaluation overflow can only
happen as a result of a deep recursive call stack.

Fixes #56701

I've re-used `tcx.sess.recursion_limit` when checking for predication evaluation overflows. This is such a weird corner case that I don't believe it's necessary to have a separate setting controlling the maximum depth.
VardhanThigle pushed a commit to jethrogb/rust that referenced this issue Jan 31, 2019
Currently, SelectionContext tries to prevent stack overflow by keeping
track of the current recursion depth. However, this depth tracking is
only used when performing normal section (which includes confirmation).
No such tracking is performed for evaluate_obligation_recursively, which
can allow a stack overflow to occur.

To fix this, this commit tracks the current predicate evaluation depth.
This is done separately from the existing obligation depth tracking:
an obligation overflow can occur across multiple calls to 'select' (e.g.
when fulfilling a trait), while a predicate evaluation overflow can only
happen as a result of a deep recursive call stack.

Fixes rust-lang#56701
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
A-synthetic-impls Area: Synthetic impls, used by rustdoc to document auto traits and traits with blanket impls I-crash Issue: The compiler crashes (SIGSEGV, SIGABRT, etc). Use I-ICE instead when the compiler panics. T-rustdoc Relevant to the rustdoc team, which will review and decide on the PR/issue.
Projects
None yet
Development

Successfully merging a pull request may close this issue.

4 participants