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

Cyclic type involving LazyRef is not detected and crash in Pickler. #2941

Closed
gsps opened this issue Aug 2, 2017 · 1 comment · Fixed by #11260
Closed

Cyclic type involving LazyRef is not detected and crash in Pickler. #2941

gsps opened this issue Aug 2, 2017 · 1 comment · Fixed by #11260

Comments

@gsps
Copy link
Contributor

gsps commented Aug 2, 2017

The following example triggers an infinite recursion in TypeAssigner.avoid(). The call to avoid() originates from ensureNoLocalRefs, and, further up the call stack, from the handling of match cases (Typer.scala:951). The actual avoidance is performed on a LazyRef, which is likely at the heart of the problem.

trait FooBase {
  type Bar >: Null <: BarBase { type This <: FooBase.this.Bar }
  type This >: this.type <: FooBase { type This <: FooBase.this.This }

  def derived(bar: Bar): This = ???
}

trait BarBase {
  type This >: Null <: BarBase { type This <: BarBase.this.This }
}

object Test {
  def bad(foo: FooBase): FooBase = foo match {
    case foo: FooBase =>
      foo.derived(???)  // Triggers infinite loop in TypeAssigner.avoid()
  }
}
@smarter
Copy link
Member

smarter commented Jan 11, 2018

The avoidance is no longer infinite, but we generate a really funny type:

         case foo @ _: FooBase => 
            foo.derived(???): 
              
                FooBase{
                  This <: 
                    LazyRef(
                      FooBase{
                        This <: 
                          LazyRef(
                            FooBase{
                              This <: 
                                LazyRef(
                                  FooBase{
                                    This <: 
                                      LazyRef(
                                        FooBase{
                                          This <: 
                                            LazyRef(
                                              FooBase{
                                                This <: 
                                                  LazyRef(
                                                    FooBase{
                                                      This <: 
                                                        LazyRef(
                                                          FooBase{
                                                            This <: 
                                                              LazyRef(
                                                                FooBase{
                                                                  This <: 
                                                                    LazyRef(
                                                                      FooBase{
                                                                        This <: 
                                                                          LazyRef(
                                                                            FooBase
                                                                              {
                                                                            This
                                                                               <: 
                                                                            LazyRef(
                                                                              FooBase
                                                                                {
                                                                              This
                                                                                 <: 
                                                                              LazyRef(
                                                                                FooBase
                                                                                  {
                                                                                This
                                                                                   <: 
                                                                                LazyRef(
                                                                                  FooBase
                                                                                    {
                                                                                  This
                                                                                     <: 
                                                                                  LazyRef(
                                                                                    FooBase
                                                                                      {
                                                                                    This
                                                                                       <: 
                                                                                    LazyRef(
                                                                                      FooBase
                                                                                        {
                                                                                      This
                                                                                         <: 
                                                                                      LazyRef(
                                                                                        FooBase
                                                                                          {
                                                                                        This
                                                                                           <: 
                                                                                        LazyRef(
                                                                                          FooBase
                                                                                            {
                                                                                          This
                                                                                             <: 
                                                                                          LazyRef(
                                                                                            ...
                                                                                              {
                                                                                            This
                                                                                               <: 
                                                                                            ...
                                                                                              }
                                                                                          )
                                                                                            }
                                                                                        )
                                                                                          }
                                                                                      )
                                                                                        }
                                                                                    )
                                                                                      }
                                                                                  )
                                                                                    }
                                                                                )
                                                                                  }
                                                                              )
                                                                              }
                                                                            )
                                                                            }
                                                                          )
                                                                      }
                                                                    )
                                                                }
                                                              )
                                                          }
                                                        )
                                                    }
                                                  )
                                              }
                                            )
                                        }
                                      )
                                  }
                                )
                            }
                          )
                      }
                    )
                }
              
        }
  }
}

And then stackoverflow in pickling.

@smarter smarter changed the title Infinite loop in TypeAssigner.avoid() Cyclic type involving LazyRef is not detected and crash in Pickler. Jan 11, 2018
odersky added a commit to dotty-staging/dotty that referenced this issue Jan 14, 2018
griggt added a commit to griggt/dotty that referenced this issue Jan 31, 2021
liufengyun added a commit that referenced this issue Jan 31, 2021
michelou pushed a commit to michelou/scala3 that referenced this issue Feb 5, 2021
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Projects
None yet
Development

Successfully merging a pull request may close this issue.

3 participants