Skip to content

Commit

Permalink
Fix #12681: Approximate ThisType in child instantiation
Browse files Browse the repository at this point in the history
  • Loading branch information
liufengyun committed Jun 2, 2021
1 parent 5efc3b5 commit d0097f7
Show file tree
Hide file tree
Showing 2 changed files with 35 additions and 6 deletions.
21 changes: 15 additions & 6 deletions compiler/src/dotty/tools/dotc/core/TypeOps.scala
Original file line number Diff line number Diff line change
Expand Up @@ -692,15 +692,24 @@ object TypeOps:
*/
private def instantiateToSubType(tp1: NamedType, tp2: Type)(using Context): Type = {
// In order for a child type S to qualify as a valid subtype of the parent
// T, we need to test whether it is possible S <: T. Therefore, we replace
// type parameters in T with tvars, and see if the subtyping is true.
val approximateTypeParams = new TypeMap {
// T, we need to test whether it is possible S <: T.
//
// The check is different from subtype checking due to type parameters and
// `this`. We perform the following operations to approximate the parameters:
//
// 1. Replace type parameters in T with tvars
// 2. Replace `A.this.C` with `A#C` (see tests/patmat/i12681.scala)
//
val approximateParent = new TypeMap {
val boundTypeParams = util.HashMap[TypeRef, TypeVar]()

def apply(tp: Type): Type = tp.dealias match {
case _: MatchType =>
tp // break cycles

case ThisType(tref: TypeRef) if !tref.symbol.isStaticOwner =>
tref

case tp: TypeRef if !tp.symbol.isClass =>
def lo = LazyRef.of(apply(tp.underlying.loBound))
def hi = LazyRef.of(apply(tp.underlying.hiBound))
Expand Down Expand Up @@ -787,7 +796,7 @@ object TypeOps:
// we manually patch subtyping check instead of changing TypeComparer.
// See tests/patmat/i3645b.scala
def parentQualify(tp1: Type, tp2: Type) = tp1.classSymbol.info.parents.exists { parent =>
parent.argInfos.nonEmpty && approximateTypeParams(parent) <:< tp2
parent.argInfos.nonEmpty && approximateParent(parent) <:< tp2
}

def instantiate(): Type = {
Expand All @@ -797,8 +806,8 @@ object TypeOps:

if (protoTp1 <:< tp2) instantiate()
else {
val protoTp2 = approximateTypeParams(tp2)
if (protoTp1 <:< protoTp2 || parentQualify(protoTp1, protoTp2)) instantiate()
val approxTp2 = approximateParent(tp2)
if (protoTp1 <:< approxTp2 || parentQualify(protoTp1, approxTp2)) instantiate()
else NoType
}
}
Expand Down
20 changes: 20 additions & 0 deletions tests/patmat/i12681.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,20 @@
object Examples {

case class Leaf1() extends i.Root
case class Leaf2() extends i.Branch

val i = new Inner()

class Inner {

sealed trait Root
sealed trait Branch extends Root

// simulate ordinal method of a Mirror.SumOf generated at this call site
def myOrdinal(r: Root): Int = r match {
case _: Examples.Leaf1 => 0
case _: Inner.this.Branch => 1
}
}

}

0 comments on commit d0097f7

Please sign in to comment.