Skip to content

Commit

Permalink
Backport "Fix #21295: Restrict provablyDisjoint with Nothings in …
Browse files Browse the repository at this point in the history
…invariant type params." to 3.6.2 (#21906)

Backports #21891 to the 3.6.2 branch.

PR submitted by the release tooling.
  • Loading branch information
WojciechMazur authored Nov 8, 2024
2 parents bbc2636 + 531f29b commit 8ceb324
Show file tree
Hide file tree
Showing 2 changed files with 27 additions and 2 deletions.
21 changes: 19 additions & 2 deletions compiler/src/dotty/tools/dotc/core/TypeComparer.scala
Original file line number Diff line number Diff line change
Expand Up @@ -3231,6 +3231,12 @@ class TypeComparer(@constructorOnly initctx: Context) extends ConstraintHandling
end provablyDisjointClasses

private def provablyDisjointTypeArgs(cls: ClassSymbol, args1: List[Type], args2: List[Type], pending: util.HashSet[(Type, Type)])(using Context): Boolean =
// sjrd: I will not be surprised when this causes further issues in the future.
// This is a compromise to be able to fix #21295 without breaking the world.
def cannotBeNothing(tp: Type): Boolean = tp match
case tp: TypeParamRef => cannotBeNothing(tp.paramInfo)
case _ => !(tp.loBound.stripTypeVar <:< defn.NothingType)

// It is possible to conclude that two types applied are disjoint by
// looking at covariant type parameters if the said type parameters
// are disjoint and correspond to fields.
Expand All @@ -3239,9 +3245,20 @@ class TypeComparer(@constructorOnly initctx: Context) extends ConstraintHandling
def covariantDisjoint(tp1: Type, tp2: Type, tparam: TypeParamInfo): Boolean =
provablyDisjoint(tp1, tp2, pending) && typeparamCorrespondsToField(cls.appliedRef, tparam)

// In the invariant case, direct type parameter disjointness is enough.
// In the invariant case, we have more ways to prove disjointness:
// - either the type param corresponds to a field, like in the covariant case, or
// - one of the two actual args can never be `Nothing`.
// The latter condition, as tested by `cannotBeNothing`, is ad hoc and was
// not carefully evaluated to be sound. We have it because we had to
// reintroduce the former condition to fix #21295, and alone, that broke a
// lot of existing test cases.
// Having either one of the two conditions be true is better than not requiring
// any, which was the status quo before #21295.
def invariantDisjoint(tp1: Type, tp2: Type, tparam: TypeParamInfo): Boolean =
provablyDisjoint(tp1, tp2, pending)
provablyDisjoint(tp1, tp2, pending) && {
typeparamCorrespondsToField(cls.appliedRef, tparam)
|| (cannotBeNothing(tp1) || cannotBeNothing(tp2))
}

args1.lazyZip(args2).lazyZip(cls.typeParams).exists {
(arg1, arg2, tparam) =>
Expand Down
8 changes: 8 additions & 0 deletions tests/pos/i21295.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
sealed trait Foo[A]
final class Bar extends Foo[Nothing]

object Test:
type Extract[T] = T match
case Foo[_] => Int

val x: Extract[Bar] = 1

0 comments on commit 8ceb324

Please sign in to comment.