Skip to content
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

Closed
wants to merge 4 commits into from

Conversation

odersky
Copy link
Contributor

@odersky odersky commented Jul 7, 2015

Closes #553 and #681. Review by @smarter

odersky added 2 commits July 7, 2015 18:57
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.
@odersky
Copy link
Contributor Author

odersky commented Jul 7, 2015

/rebuild

1 similar comment
@odersky
Copy link
Contributor Author

odersky commented Jul 7, 2015

/rebuild

@odersky
Copy link
Contributor Author

odersky commented Jul 7, 2015

@smarter:

([T] => List[T])[Int] # Apply   <:   [T] => Traversable[T]

or, expanded

Lambda$P { type Apply = List[$HK0] } { $HK0 = Int } # Apply 
<:
Lambda$P { type Apply = Traversable[$HK0] }

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)))
Copy link
Member

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.

Copy link
Member

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.

Copy link
Member

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.

Copy link
Contributor Author

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.

odersky added 2 commits July 8, 2015 17:20
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`.
@smarter
Copy link
Member

smarter commented Jul 8, 2015

Otherwise LGTM, though I wonder if it's possible that we may add something to theMap.discarded in wildApprox which should not be part of the implicit scope (which is determined by ImplicitRunInfo#implicitScope).

@retronym
Copy link
Member

retronym commented Jul 9, 2015

@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:

scala> def foo[M[_], A](m: M[A]): M[A] = ???
warning: there was one feature warning; re-run with -feature for details
foo: [M[_], A](m: M[A])M[A]

scala> def bar = foo(null: T[Int] with U[String])
bar: T[Int]

scala> def bar = foo(null: U[String] with T[Int])
bar: U[String]

scala> def bar = foo(null: U[String] with T[Int] {})
bar: U[String]

scala> type Alias = U[String]
defined type alias Alias

scala> def bar = foo(null: Alias)
bar: U[String]

The logic in play is here: https://github.com/scala/scala/blob/v2.11.6/src/reflect/scala/reflect/internal/Types.scala#L3067-L3087

@odersky
Copy link
Contributor Author

odersky commented Feb 16, 2016

I verified that the "shuffle" test no longer fails. So this seems to be fixed by other means now.

@smarter
Copy link
Member

smarter commented Feb 16, 2016

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.

@odersky
Copy link
Contributor Author

odersky commented Feb 16, 2016

Should we leave the PR open then?

On Tue, Feb 16, 2016 at 4:43 PM, Guillaume Martres <notifications@github.com

wrote:

The shuffle test and many others were fixed by #799
#799 .The bug that dotty-staging@
ec2d7d6
dotty-staging@ec2d7d6
fixes probably still exists but we don't see it anymore because #799
#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 im plicit scope caching that I'd like to fix first
since they might overlap with this.


Reply to this email directly or view it on GitHub
#716 (comment).

Martin Odersky
EPFL

@smarter
Copy link
Member

smarter commented Feb 16, 2016

Sure.

@smarter
Copy link
Member

smarter commented May 31, 2016

The bug that dotty-staging/dotty-1@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.

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.

@odersky odersky closed this Aug 23, 2016
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants