Skip to content

Commit

Permalink
Refine AsSeenFrom approximation scheme
Browse files Browse the repository at this point in the history
The previous scheme did not set the approximated flag in some cases, which led
to the appearance of unhelpful Nothing types in error messages.

We now always mark the AsSeenFrom map as approximated when encountering an illegal prefix at variance <= 0.
But we reset it again when a range in a prefix is dropped because we can follow a type alias or use a
singleton type info.

Furthermore, we use an `Int`, `approxCount` instead of `approximated` to keep track of
multiple approximation points.
  • Loading branch information
odersky committed Sep 3, 2022
1 parent bf03086 commit 46e3d77
Show file tree
Hide file tree
Showing 4 changed files with 119 additions and 22 deletions.
49 changes: 33 additions & 16 deletions compiler/src/dotty/tools/dotc/core/TypeOps.scala
Original file line number Diff line number Diff line change
Expand Up @@ -45,7 +45,7 @@ object TypeOps:
val widenedAsf = new AsSeenFromMap(pre.info, cls)
val ret = widenedAsf.apply(tp)

if (!widenedAsf.approximated)
if widenedAsf.approxCount == 0 then
return ret

Stats.record("asSeenFrom skolem prefix required")
Expand All @@ -58,7 +58,14 @@ object TypeOps:
/** The TypeMap handling the asSeenFrom */
class AsSeenFromMap(pre: Type, cls: Symbol)(using Context) extends ApproximatingTypeMap, IdempotentCaptRefMap {
/** Set to true when the result of `apply` was approximated to avoid an unstable prefix. */
var approximated: Boolean = false
private var approximated: Boolean = false

/** The number of range approximations in invariant or contravariant positions
* - Incremented each time we produce a range.
* - Decremented each time we drop a prefix range by forwarding to a type alias
* or singleton type.
*/
private[TypeOps] var approxCount: Int = 0

def apply(tp: Type): Type = {

Expand All @@ -76,17 +83,22 @@ object TypeOps:
case _ =>
if (thiscls.derivesFrom(cls) && pre.baseType(thiscls).exists)
if (variance <= 0 && !isLegalPrefix(pre))
if (variance < 0) {
approximated = true
defn.NothingType
}
else
// Don't set the `approximated` flag yet: if this is a prefix
// of a path, we might be able to dealias the path instead
// (this is handled in `ApproximatingTypeMap`). If dealiasing
// is not possible, then `expandBounds` will end up being
// called which we override to set the `approximated` flag.
if true then
approxCount += 1
range(defn.NothingType, pre)
else
if (variance < 0) {
approximated = true
defn.NothingType
}
else
// Don't set the `approximated` flag yet: if this is a prefix
// of a path, we might be able to dealias the path instead
// (this is handled in `ApproximatingTypeMap`). If dealiasing
// is not possible, then `expandBounds` will end up being
// called which we override to set the `approximated` flag.
range(defn.NothingType, pre)

else pre
else if (pre.termSymbol.is(Package) && !thiscls.is(Package))
toPrefix(pre.select(nme.PACKAGE), cls, thiscls)
Expand Down Expand Up @@ -119,10 +131,15 @@ object TypeOps:
// derived infos have already been subjected to asSeenFrom, hence to need to apply the map again.
tp

override protected def expandBounds(tp: TypeBounds): Type = {
approximated = true
super.expandBounds(tp)
}
//override protected def expandBounds(tp: TypeBounds): Type = {
// approximated = true
// super.expandBounds(tp)
//}

override protected def useAlternate(tp: Type): Type =
assert(approxCount > 0)
approxCount -= 1
tp
}

def isLegalPrefix(pre: Type)(using Context): Boolean =
Expand Down
23 changes: 17 additions & 6 deletions compiler/src/dotty/tools/dotc/core/Types.scala
Original file line number Diff line number Diff line change
Expand Up @@ -5766,6 +5766,13 @@ object Types {

private var expandingBounds: Boolean = false

/** Use an alterate type `tp` that replaces a range. This can happen if the
* prefix of a Select is a range and the selected symbol is an alias type
* or a value with a singleton type. In both cases we can forget the prefix
* and use the symbol's type.
*/
protected def useAlternate(tp: Type): Type = reapply(tp)

/** Whether it is currently expanding bounds
*
* It is used to avoid following LazyRef in F-Bounds
Expand All @@ -5789,15 +5796,15 @@ object Types {
case TypeAlias(alias) =>
// if H#T = U, then for any x in L..H, x.T =:= U,
// hence we can replace with U under all variances
reapply(alias.rewrapAnnots(tp1))
useAlternate(alias.rewrapAnnots(tp1))
case bounds: TypeBounds =>
// If H#T = ? >: S <: U, then for any x in L..H, S <: x.T <: U,
// hence we can replace with S..U under all variances
expandBounds(bounds)
case info: SingletonType =>
// if H#x: y.type, then for any x in L..H, x.type =:= y.type,
// hence we can replace with y.type under all variances
reapply(info)
useAlternate(info)
case _ =>
NoType
}
Expand All @@ -5812,11 +5819,15 @@ object Types {
tp.argForParam(pre) match {
case arg @ TypeRef(pre, _) if pre.isArgPrefixOf(arg.symbol) =>
arg.info match {
case argInfo: TypeBounds => expandBounds(argInfo)
case argInfo => reapply(arg)
case argInfo: TypeBounds =>
expandBounds(argInfo)
case argInfo =>
useAlternate(arg)
}
case arg: TypeBounds => expandBounds(arg)
case arg => reapply(arg)
case arg: TypeBounds =>
expandBounds(arg)
case arg =>
useAlternate(arg)
}

/** Derived selection.
Expand Down
45 changes: 45 additions & 0 deletions tests/neg/i15939.check
Original file line number Diff line number Diff line change
@@ -0,0 +1,45 @@
-- [E007] Type Mismatch Error: tests/neg/i15939.scala:19:16 ------------------------------------------------------------
19 | mkFoo.andThen(mkBarString) // error
| ^^^^^^^^^^^
| Found: String
| Required: ?1.Bar
|
| where: ?1 is an unknown value of type Test.Foo
|
| longer explanation available when compiling with `-explain`
-- [E007] Type Mismatch Error: tests/neg/i15939.scala:20:2 -------------------------------------------------------------
20 | mkBarString andThen_: mkFoo // error
| ^^^^^^^^^^^
| Found: String
| Required: ?2.Bar
|
| where: ?2 is an unknown value of type Test.Foo
|
| longer explanation available when compiling with `-explain`
-- [E007] Type Mismatch Error: tests/neg/i15939.scala:21:18 ------------------------------------------------------------
21 | mkFoo.andThen_:(mkBarString) // error
| ^^^^^^^^^^^
| Found: String
| Required: ?3.Bar
|
| where: ?3 is an unknown value of type Test.Foo
|
| longer explanation available when compiling with `-explain`
-- [E007] Type Mismatch Error: tests/neg/i15939.scala:22:2 -------------------------------------------------------------
22 | mkBarString andThenByName_: mkFoo // error
| ^^^^^^^^^^^
| Found: String
| Required: ?4.Bar
|
| where: ?4 is an unknown value of type Test.Foo
|
| longer explanation available when compiling with `-explain`
-- [E007] Type Mismatch Error: tests/neg/i15939.scala:23:24 ------------------------------------------------------------
23 | mkFoo.andThenByName_:(mkBarString) // error
| ^^^^^^^^^^^
| Found: String
| Required: ?5.Bar
|
| where: ?5 is an unknown value of type Test.Foo
|
| longer explanation available when compiling with `-explain`
24 changes: 24 additions & 0 deletions tests/neg/i15939.scala
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import scala.language.implicitConversions

object Test {
class Foo {
class Bar {
override def toString() = "bar"
}
object Bar {
implicit def fromString(a: String): Bar = { println("convert bar") ; new Bar }
}

def andThen(b: Bar): Unit = { println("pre") ; println(s"use $b") ; println("post") }
def andThen_:(b: Bar) = { println("pre") ; println(s"use $b") ; println("post") }
def andThenByName_:(b: => Bar) = { println("pre") ; println(s"use $b") ; println(s"use $b") ; println("post") }
}

def mkFoo: Foo = ???
def mkBarString: String = ???
mkFoo.andThen(mkBarString) // error
mkBarString andThen_: mkFoo // error
mkFoo.andThen_:(mkBarString) // error
mkBarString andThenByName_: mkFoo // error
mkFoo.andThenByName_:(mkBarString) // error
}

0 comments on commit 46e3d77

Please sign in to comment.