-
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 problems with higher-kinded implicits #716
Conversation
There ws a missing case, where a lambda projection `T#Apply` is compared to something to a type lambda. In that case we should compare the prefix `T` directly wiht the type lambda. With that modification i553 works if the canBuildFrom implicit is imported. But the implicit is not yet found in the type scope.
lookupRefined used to fall back to unbounded Wildcard, given a Wildcard prefix. But that loses bounds when called from wildApprox, which in turn loses companion objects in the implicit scope of a type. In `i553-shuffle.scala` this happened because we projected with #Apply on a type that approximated to a bounded wildcard. The fix keeps the bounds, projecting in turn on them.
/rebuild |
1 similar comment
/rebuild |
or, expanded
|
pre.optBounds match { | ||
case TypeBounds(lo, hi) if name.isTypeName => | ||
// for term names, forming a Termref with just the name risks losing overloaded instance | ||
WildcardType(TypeBounds(NamedType(lo, name), NamedType(hi, name))) |
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 comment on https://github.com/dotty-staging/dotty/blob/fix/shuffle/src/dotty/tools/dotc/core/Types.scala#L1423-L1424 is now wrong since the wildcard is now bounded, not unbounded. it'd be nice to update it and maybe also document the handling of wildcard in lookupRefined
itself.
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, does this mean that we consider (_ >: Foo <: Bar)#T
to be equivalent to _ >: Foo#T <: Bar#T
? If so, isn't this wrong in general? For example, if T
is contravariant we may have Foo <: Bar
but Foo#T >: Bar#T
so the bounds _ >: Foo#T <: Bar#T
would not be satisfiable.
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.
I think that having a few test cases would be helpful.
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.
@smarter Good point about the change in lookupRefined. I have backed out of it, and replaced it with a different solution.
Test cases: We have Random.shuffle, I think that's already a good one. If you have more, please add them.
This reverts commit ec2d7d6.
lookupRefined on a Wildcard prefix will return an unbounded Wildcard type. This can destroy information that's important for determining the companions making up the implicit scope of a type. We handle this case by maintaining a collection for types that were discarded in that way and taking these types intoa ccount separately when computing companions. Examples where this is necessary include `i553-shuffle.scala` abd `vector1.scala`.
Otherwise LGTM, though I wonder if it's possible that we may add something to |
I also suggest adding the test case https://raw.githubusercontent.com/retronym/scala/e439651befb85e6c3b131db6bf346513ecedc783/test/pending/pos/t6948.scala which fails in Scala 2.11 (see https://issues.scala-lang.org/browse/SI-6948?focusedCommentId=61848&page=com.atlassian.jira.plugin.system.issuetabpanels:comment-tabpanel#comment-61848) but works fine in Dotty (yay!) |
@smarter I would suggest to add make that test case independent of the collections library before using it. It will be less fragile and easier to reason about in isolation. Why does it work in dotty? Here's an approach to quickly and directly see how higher kinded inference works:
The logic in play is here: https://github.com/scala/scala/blob/v2.11.6/src/reflect/scala/reflect/internal/Types.scala#L3067-L3087 |
I verified that the "shuffle" test no longer fails. So this seems to be fixed by other means now. |
The shuffle test and many others were fixed by #799 .The bug that dotty-staging@ec2d7d6 fixes probably still exists but we don't see it anymore because #799 changed inference such that we no longer end up with wildcard prefixes for implicit higher-kinded types. I could try to construct a test case for it, however I'm aware of several issues with implicit scope caching that I'd like to fix first since they might overlap with this. |
Should we leave the PR open then? On Tue, Feb 16, 2016 at 4:43 PM, Guillaume Martres <notifications@github.com
Martin Odersky |
Sure. |
The only test cases I can come up are of this form: class X {
type Elem
}
object X {
implicit def foo: X#Elem = ???
}
object Test {
def get[T <: X](implicit ev: T#Elem) = ev
get
} Which does not compile anyway because of the restrictions on type projections that #1051 added, so we can probably close this for now, but keep this in mind if we ever loosen these restrictions. |
Closes #553 and #681. Review by @smarter