Skip to content

Commit 24efe7d

Browse files
authored
Correct Java signature for value classes appearing in type arguments (#20463)
As suggested in #10846 the fix to this issue should be to port scala/scala#8127 to scala3 I started by adding the same tests as in the scala2 PR and then tried to find the place where to do the fix by adding some log traces. Unfortunately I am still pretty lost because this is my first time looking at the compiler code. Any tips where this needs to be fixed are very welcome. Meanwhile a few questions: * The scala2 fix was done in `src/compiler/scala/tools/nsc/transform/Erasure.scala`, should I do the fix for scala3 in `compiler/src/dotty/tools/dotc/transform/Erasure.scala` as well? * I think I need to do the fix around `ErasedValueType`, either when it is created (in `TypeErasure#eraseDerivedValueClass`) or when it is used (in `Erasure#unbox`). Please also send me any pointers besides https://dotty.epfl.ch/docs/contributing/index.html regarding compiler contributions
2 parents 1f00e4a + 2217634 commit 24efe7d

File tree

4 files changed

+45
-15
lines changed

4 files changed

+45
-15
lines changed

compiler/src/dotty/tools/dotc/transform/GenericSignatures.scala

+13-14
Original file line numberDiff line numberDiff line change
@@ -64,7 +64,7 @@ object GenericSignatures {
6464
ps.foreach(boxedSig)
6565
}
6666

67-
def boxedSig(tp: Type): Unit = jsig(tp.widenDealias, primitiveOK = false)
67+
def boxedSig(tp: Type): Unit = jsig(tp.widenDealias, unboxedVCs = false)
6868

6969
/** The signature of the upper-bound of a type parameter.
7070
*
@@ -232,7 +232,7 @@ object GenericSignatures {
232232
}
233233

234234
@noinline
235-
def jsig(tp0: Type, toplevel: Boolean = false, primitiveOK: Boolean = true): Unit = {
235+
def jsig(tp0: Type, toplevel: Boolean = false, unboxedVCs: Boolean = true): Unit = {
236236

237237
val tp = tp0.dealias
238238
tp match {
@@ -241,7 +241,7 @@ object GenericSignatures {
241241
val erasedUnderlying = fullErasure(ref.underlying.bounds.hi)
242242
// don't emit type param name if the param is upper-bounded by a primitive type (including via a value class)
243243
if erasedUnderlying.isPrimitiveValueType then
244-
jsig(erasedUnderlying, toplevel, primitiveOK)
244+
jsig(erasedUnderlying, toplevel, unboxedVCs)
245245
else typeParamSig(ref.paramName.lastPart)
246246

247247
case defn.ArrayOf(elemtp) =>
@@ -269,15 +269,14 @@ object GenericSignatures {
269269
else if (sym == defn.NullClass)
270270
builder.append("Lscala/runtime/Null$;")
271271
else if (sym.isPrimitiveValueClass)
272-
if (!primitiveOK) jsig(defn.ObjectType)
272+
if (!unboxedVCs) jsig(defn.ObjectType)
273273
else if (sym == defn.UnitClass) jsig(defn.BoxedUnitClass.typeRef)
274274
else builder.append(defn.typeTag(sym.info))
275275
else if (sym.isDerivedValueClass) {
276-
val erasedUnderlying = fullErasure(tp)
277-
if (erasedUnderlying.isPrimitiveValueType && !primitiveOK)
278-
classSig(sym, pre, args)
279-
else
280-
jsig(erasedUnderlying, toplevel, primitiveOK)
276+
if (unboxedVCs) {
277+
val erasedUnderlying = fullErasure(tp)
278+
jsig(erasedUnderlying, toplevel)
279+
} else classSig(sym, pre, args)
281280
}
282281
else if (defn.isSyntheticFunctionClass(sym)) {
283282
val erasedSym = defn.functionTypeErasure(sym).typeSymbol
@@ -286,7 +285,7 @@ object GenericSignatures {
286285
else if sym.isClass then
287286
classSig(sym, pre, args)
288287
else
289-
jsig(erasure(tp), toplevel, primitiveOK)
288+
jsig(erasure(tp), toplevel, unboxedVCs)
290289

291290
case ExprType(restpe) if toplevel =>
292291
builder.append("()")
@@ -339,23 +338,23 @@ object GenericSignatures {
339338
val (reprParents, _) = splitIntersection(parents)
340339
val repr =
341340
reprParents.find(_.typeSymbol.is(TypeParam)).getOrElse(reprParents.head)
342-
jsig(repr, primitiveOK = primitiveOK)
341+
jsig(repr, unboxedVCs = unboxedVCs)
343342

344343
case ci: ClassInfo =>
345344
val tParams = tp.typeParams
346345
if (toplevel) polyParamSig(tParams)
347346
superSig(ci.typeSymbol, ci.parents)
348347

349348
case AnnotatedType(atp, _) =>
350-
jsig(atp, toplevel, primitiveOK)
349+
jsig(atp, toplevel, unboxedVCs)
351350

352351
case hktl: HKTypeLambda =>
353-
jsig(hktl.finalResultType, toplevel, primitiveOK)
352+
jsig(hktl.finalResultType, toplevel, unboxedVCs)
354353

355354
case _ =>
356355
val etp = erasure(tp)
357356
if (etp eq tp) throw new UnknownSig
358-
else jsig(etp, toplevel, primitiveOK)
357+
else jsig(etp, toplevel, unboxedVCs)
359358
}
360359
}
361360
val throwsArgs = sym0.annotations flatMap ThrownException.unapply

tests/pos/i10347/C_2.java

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
public class C_2 {
22
String hi = A.foo().head();
3-
String hy = A.bar().head();
3+
String hy = A.bar().head().s();
44
String hj = A.baz("").head();
55
}

tests/run/i10846.check

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
i10846.V: scala.Option<i10846.V>
2+
i10846.U: scala.Option<i10846.U>
3+
i10846.W: scala.Option<i10846.W<scala.Function1<java.lang.Object, java.lang.String>>>

tests/run/i10846/i10846.scala

+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
// scalajs: --skip
2+
3+
package i10846 {
4+
final class V(val x: Int) extends AnyVal
5+
object V { def get: Option[V] = null }
6+
7+
final class U(val y: String) extends AnyVal
8+
object U { def get: Option[U] = null }
9+
10+
final class W[T](val z: T) extends AnyVal
11+
object W { def get: Option[W[Int => String]] = null }
12+
}
13+
14+
15+
object Test extends scala.App {
16+
def check[T](implicit tt: reflect.ClassTag[T]): Unit = {
17+
val companion = tt.runtimeClass.getClassLoader.loadClass(tt.runtimeClass.getName + '$')
18+
val get = companion.getMethod("get")
19+
assert(get.getReturnType == classOf[Option[_]])
20+
println(s"${tt.runtimeClass.getName}: ${get.getGenericReturnType}")
21+
}
22+
23+
import i10846._
24+
25+
check[V]
26+
check[U]
27+
check[W[_]]
28+
}

0 commit comments

Comments
 (0)