-
Notifications
You must be signed in to change notification settings - Fork 21
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
substitution broken for dependent types #7753
Comments
Imported From: https://issues.scala-lang.org/browse/SI-7753?orig=1 |
@paulp said: trait CC[A]
trait HasElem[Repr] { type A }
object ccInt extends HasElem[CC[Int]] { type A = Int }
object Test {
final class View[Repr, AIn](repr: Repr, val tc: HasElem[Repr] { type A = AIn }) {
// For a time it's completely convinced AIn and tc.A are the same type.
def equivalence1[B](implicit ev: B =:= AIn) = true
def equivalence2[B](implicit ev: B =:= tc.A) = true
def evidences = List(equivalence1[tc.A], equivalence1[AIn], equivalence2[tc.A], equivalence2[AIn]) // 4 out of 4
// Foreshadow.
def f1(p: AIn): AIn = p
def f2(p: tc.A): tc.A = p
}
def xs: CC[Int] = ???
val view = new View[CC[Int], Int](xs, ccInt)
// No longer equivalent, transitivity lost.
def g0 = List(view.equivalence1[Int], view.equivalence2[Int])
// b.scala:33: error: Cannot prove that Int =:= Test.view.tc.A.
// def g0 = List(view.equivalence1[Int], view.equivalence2[Int])
// ^
def g1 = view f1 5 // compiles
def g2 = view f2 5 // fails
// b.scala:31: error: type mismatch;
// found : Int(5)
// required: Test.view.tc.A
// (which expands to) AIn
// def g2 = view f2 5 // fails
// ^
} |
@retronym said: import scala.language.{ higherKinds, implicitConversions }
trait Foo { type Out }
trait SI {
val instance: Foo
type Out
}
object Test {
def test {
def indirect(si: SI)(v: si.instance.Out) = v
val foo: Foo { type Out = Int } = ???
def conv(i: Foo): SI { type Out = i.Out; val instance: i.type } = ???
val converted = conv(foo)
val v1: Int = indirect(converted)(23) // Okay (after refining the return type `instance` in the return type of `conv`)
/*
indirect(converted){(v: converted.instance.Out)converted.instance.Out}(
23{Int(23)}
){converted.instance.Out};
*/
val v2: Int = indirect(conv(foo))(23) // Fails.
/*
indirect(
conv(foo){si.SI{type Out = foo.Out; val instance: si.Test.<refinement>.type}}
){(v: si.instance.Out)si.instance.Out}(
23{<error>}
){<error>};
*/
}
} First of all, I needed to refine However, after doing that, we can still see a difference between the case when the first argument to
The unstable type of the argument is intentionally not substituted for My conclusion is that this isn't a bug. I'm going to mark it as such, but feel free to reopen if you think I've got it wrong. I'll move Paul's test case over to #8177. It is a bug in as-seen-from. |
@retronym said: |
@adriaanm said: |
@clhodapp said (edited on Feb 6, 2014 10:48:48 AM UTC): class A {
class B
def b: B = new B
}
class B {
def f(a: A)(b: a.B): a.B = b
val a = new A
f(a)(a.b)
f({ val fresh = new A; fresh: fresh.type })(a.b) // doesn't compile
} You would find that the code still wouldn't compile, since import scala.language.{ higherKinds, implicitConversions }
trait Foo { type Out }
trait SI {
val instance: Foo
type Out
}
object Test {
def test {
def indirect(si: SI)(v: si.instance.Out) = v
val foo: Foo { type Out = Int } = ???
def conv(i: Foo): SI { type Out = i.Out; val instance: i.type } = ???
val converted = conv(foo)
val v1: Int = indirect(converted)(23)
val v2: Int = indirect({ val fresh = conv(foo); fresh: fresh.type })(23) // Works now
}
} You will find that the code compiles now. But then, of course it does: this basically comes to the same thing as what you already had above with While I'm at it, though, I'm extremely unconfident that the "stability lambdas" above ( |
@retronym said: I tend to agree that the restrictions placed by #3873 didn't get to the essence of the unsoundness. I also agree that the "stability lambda" (nicely coined!) is pretty dubious.
As you point out, this has an ill-scoped reference to Here's an analagous situation where block typing widens to a structural type to avoid leaking a locally scoped type: scala> {class A { def foo = 0 }; new A }
res1: AnyRef{def foo: Int} = A$1@23905e3 With regards to eta-expansion, that's no surprise as you currently can't express dependently typed functions in scala. |
@paulp said: Why is that a problem? The "lexical" in "lexical scope" tells you what scopes are about: names. When properties of the value which can be expressed in the type without reference to its name are visible outside of the defining scope, that is the type system doing its job. That's what a type system is for. |
@paulp said: I say that because the types look right when I do good old -Xprint:typer for both these cases. Looking at the positions where Singleton is shown for f, I see no way x could possibly be escaping. And in g it is given the correct existential type. object Test {
def f = { val x = ""; x: x.type }: Singleton
def g = { val x = "" ; identity[x.type] _ }
}
// def f: Singleton = ({
// val x: String = "";
// (x: x.type)
// }: Singleton);
// def g: x.type => x.type forSome { val x: String } = {
// val x: String = "";
// {
// ((x: x.type) => scala.this.Predef.identity[x.type](x))
// }
// } |
@paulp said: |
@paulp said: |
@clhodapp said: |
@adriaanm said (edited on Feb 13, 2014 1:51:23 AM UTC): |
@adriaanm said: |
@adriaanm said: |
@milessabin said: |
In the following example (I'm afraid I haven't been able to significantly reduce it further), the equivalence of
si.Out
andsi.instance.Out
has been lost at the call sites ofdirect
andindirect
.Notice that in
direct
the typesi.Out
is known in the second parameter block at the call site, and so the expression typechecks. However inindirect
the typesi.instance.Out
is unknown or, at least, is sufficiently opaque to prevent the second expression from typechecking.si.Out
is defined in terms of the implicitly summonedi
, so ifsi.Out =:= i.Out
is available at the call sites ofdirect
andindirect
then intuitively it seems thatsi.instance.Out =:= i.Out
(wheresi.instance: i.type
) should be available too.The text was updated successfully, but these errors were encountered: