any.type does not convert to `_ <: any.type` when `-Yliteral-types` is on
Atry opened this issue · 6 comments
Atry commented
scalaOrganization in ThisBuild := "org.typelevel"
scalacOptions in ThisBuild += "-Yliteral-types"
import scala.collection.mutable
// Compiles, as expected
def pass(a: AnyRef): mutable.Buffer[_ <: a.type] = {
mutable.Buffer.empty[a.type]
}
// Expect compile, got error
def error(a: Any): mutable.Buffer[_ <: a.type] = {
mutable.Buffer.empty[a.type]
}
CoAny.scala:15: type mismatch;
found : scala.collection.mutable.Buffer[a.type]
required: scala.collection.mutable.Buffer[_]
mutable.Buffer.empty[a.type]
^
The Scala version is TLS 2.12.1
Atry commented
A workaround
def workaround(a: Any): mutable.Buffer[_ <: a.type] = {
def aux[A](buffer: mutable.Buffer[A]): mutable.Buffer[_ <: A] = buffer
aux[a.type](mutable.Buffer.empty[a.type])
}
milessabin commented
That does look like a bug.
But isn't the existential redundant here? Given that a.type
is a singleton type, surely _ <: a.type =:= a.type
? In which case a simpler equivalent signature would be,
def workaround(a: Any): mutable.Buffer[a.type] = ...
Atry commented
That's the point. (_ <: a.type) =:= a.type
is correct. However, scalac does not realise that. At least scalac did not know mutable.Buffer[a.type] <:< mutable.Buffer[_ <: a.type]
if a
is Any
.
milessabin commented
Confirming that this compiles as expected in Dotty.
milessabin commented
Fixed in: milessabin@e4f55f6.
milessabin commented
PR against Lightbend Scala is here: scala#6155.