-
Notifications
You must be signed in to change notification settings - Fork 1.1k
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
Fix #4031: Check arguments of dependent methods for realizability #4036
Conversation
I now think we should try to avoid this split into isStable (checked in Typer) and checkRealizable in PostTyper, at least as far as lazy vals are concerned. We should try to test for realizibility when checking whether a lazy val is stable. If this runs into cycles, we can always back out by being conservative, i.e. assume non-stable in case of cycles. I think that's at least worth a try. The alternatives look all even less appealing. |
Treating lazy vals as unstable here would already be an improvement. Is “running into cycles” insensitive to compilation order/separate compilation, or does it have a suitable conservative approximation? Beyond “what we can implement” we should probably think of what we can specify and explain to an (advanced) user. |
|
||
def intToString(i: Int): String = xToString(i) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The test did not break. It's masked by the other failures. Taken alone, it would be reported later.
@@ -205,6 +205,11 @@ class PostTyper extends MacroTransform with IdentityDenotTransformer { thisPhase | |||
Checking.checkInstantiable(tree.tpe, nu.pos) | |||
withNoCheckNews(nu :: Nil)(super.transform(tree)) | |||
case _ => | |||
tree.fun.tpe.widen match { | |||
case mt: MethodType if mt.isDependent || mt.isParamDependent => |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Gotta s/isDependent/isResultDependent/
because I merged #4039.
tests/neg/i4031.scala
Outdated
|
||
|
||
def main(args: Array[String]): Unit = { | ||
//println(coerce("Uh oh!")) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Comment back in, that's the actual ClassCastException-triggerer.
tests/neg/i1050c.scala
Outdated
@@ -4,7 +4,7 @@ object Import { | |||
trait B { type L >: Any} | |||
trait U { | |||
lazy val p: B | |||
locally { val x: p.L = ??? } // old-error: nonfinal lazy | |||
locally { val x: p.L = ??? } // error: nonfinal lazy | |||
locally { | |||
import p._ // error: Import.B(U.this.p) is not a legal path | |||
val x: L = ??? // old-error: nonfinal lazy |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Probably this should also be re-enabled, but at least the import p._
fails.
tests/neg/i1050c.scala
Outdated
@@ -4,7 +4,7 @@ object Import { | |||
trait B { type L >: Any} | |||
trait U { | |||
lazy val p: B | |||
locally { val x: p.L = ??? } // old-error: nonfinal lazy | |||
locally { val x: p.L = ??? } // error: nonfinal lazy |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
p.L
has bad bounds, and this test was disabled (by mistake) by unrelated PR #1106 (probably because it had broken), so re-enabling this test.
Since it isn't about a method call it's not addressed by the current fix, but it's still related to realizability checks I guess.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
No, it was because the tests in 1050c are done during typer and the non-final lazy is detected later in PostTyper, so not detected when there are typer errors. The comment was misleading.
There are tests like this in i1050.scala, which get reported correctly.
test performance please |
performance test scheduled: 1 job(s) in queue, 0 running. |
Performance test finished successfully: Visit http://dotty-bench.epfl.ch/4036/ to see the changes. Benchmarks is based on merging with master (eeff384) |
Last commit should fix the broken build. |
Cherry-picked last commit in #4141. |
isType || !is(Erased) && (is(Stable) || !(is(UnstableValue) || info.isInstanceOf[ExprType])) | ||
final def isStable(implicit ctx: Context) = { | ||
def isUnstableValue = is(UnstableValue) || info.isInstanceOf[ExprType] | ||
isType || is(Stable) || !isUnstableValue |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
✅ To be sure: we'd get different results for Stable | Erased
but that combo is currently illegal (and Erased symbols might become Stable in the future anyway).
@@ -155,6 +156,12 @@ object Types { | |||
case _ => false | |||
} | |||
|
|||
/** Does this type denote a realizable stable reference? Much more expensive to checl |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
checl->check
@@ -4136,7 +4145,7 @@ object Types { | |||
forwarded.orElse( | |||
range(super.derivedSelect(tp, preLo), super.derivedSelect(tp, preHi))) | |||
case _ => | |||
super.derivedSelect(tp, pre) | |||
if (pre == defn.AnyType) pre else super.derivedSelect(tp, pre) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
TODO: understand
else { | ||
val widenedArgType = argType.widen | ||
if (realizability(widenedArgType) == Realizable) | ||
tp.substParam(pref, SkolemType(widenedArgType)) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
widenedArgType
can be realizable when argType
isn't if argType is a lazy/erased value.
If argType
is a normal value it's realizable always, even if widenedArgType
isn't.
If neither is realizable, one could try harder, with an even wider type, but we don't have a reliable way to get at it — there are wider types (such as Any
) but it's unclear there's a smallest wider type, since the class hierarchy is a DAG.
tests/pos/z1720.scala
Outdated
@@ -3,7 +3,7 @@ package test | |||
class Thing { | |||
def info: Info[this.type] = InfoRepository.getInfo(this) | |||
def info2: Info[this.type] = { | |||
def self: this.type = this | |||
val self: this.type = this |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
So it seems that this.type
should be as realizable as this
, no?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yes you are right. I added a further tweak to support this in 8389bea
Based on #4108, because that way we could remove a puzzler in |
Should rebase on top of #4108 (now merged). Can try next week. |
if (r == Realizable) sym.setFlag(Stable) | ||
r | ||
} | ||
if (r == Realizable || tp.info.isStableRealizable) Realizable else r |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
r == Realizable ||
can be dropped without changing meaning. In fact, it's unclear why we should check tp.info.isStableRealizable
at this point — if this is necessary, it seems to warrant a comment. Maybe this subtly depends on setFlag
influencing isStableRealizable
.
Should we check tp.info.isStableRealizable
first and skip all the other expensive computations? That doesn't set the sym.is(Stable)
cache, but it can be set on request the next time.*
Otherwise, should tp.info.isStableRealizable
be used when computing r
?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Playing with this myself.
18ab4d0
to
b62f6db
Compare
Status: I modified the checking algorithm (for small optimizations and to cache everything cacheable) but it should check the same "typing rules" — except for any undiscovered bugs due to internal mutations/cycles/... I removed a code which seemed to repeat a call to The changes might make the code faster in some cases — but the reason I made them was to make the code IMHO more sensible: now symbols are marked as TODO: urgently rename |
Martin mentioned he will look at this once I assign him, so I should decide what to do about the small issues I wanted to fix. |
Add the rule T <: Any for any *-Type T. This was not include fully before. We did have the rule that T <: Any, if Any is in the base types of T. However, it could be that the base type wrt Any does not exist. Example: Any#L <: Any yielded false before, now yields true. This error manifested itself in i4031.scala. With the new rule, we can drop again the special case in derivedSelect.
A TermRef is stable if its underlying type is stable. Realizability should behave the same way. This obviates the change in z1720.scala.
I had a TypeError crash in refchecks after screwing up a typeclass encoding in a particularly bad way. This commit reports an error instead.
That is, set is(Stable) for symbols that satisfy isStable.
I wrote this because I feared (incorrectly) exponential slowdowns in realizability checking for this code. But debugging suggests that the complexity of realizability checking is constant in the size of these expressions (even after I disable caching of `Stable`). Beware 1: these expressions almost smash the stack by sheer size. Beware 2: this fails with `-Yno-deep-subtypes`, but simply because the checking heuristics assumes people don't try to do this.
9dd31f8
to
887ff57
Compare
This reduces confusion with `Type.isStable`. I'm tempted to call this `isPure`.
This follows scala#4360. I'd love to add a test but scala#4360 has none, so I'm not 100% sure what it'd look like.
This reverts commit 283f261 because it introduces failures (which I don't get yet), as also shown in http://dotty-ci.epfl.ch/lampepfl/dotty/5085/4.
887ff57
to
ad1167c
Compare
TODO: check this PR fixes https://twitter.com/propensive/status/1031419434320179200?s=21: implicitly[(A with B)#Z =:= Int]
implicitly[(A with B)#Z =:= Char]
implicitly[(A with B)#Z =:= (B with A)#Z] |
After porting the commit from scala#4036, this line newly gives an error. Might be OK, investigate.
After porting the commit from scala#4036, this line newly gives an error. Might be OK, investigate.
After porting the commit from scala#4036, this line newly gives an error. Might be OK, investigate.
Also some additions to realizability tests