Skip to content

Commit

Permalink
add more vararg tests
Browse files Browse the repository at this point in the history
  • Loading branch information
G1ng3r committed Apr 1, 2024
1 parent c680f7d commit 1f3f7e7
Show file tree
Hide file tree
Showing 6 changed files with 248 additions and 118 deletions.
31 changes: 14 additions & 17 deletions compiler/src/dotty/tools/dotc/typer/Dynamic.scala
Original file line number Diff line number Diff line change
Expand Up @@ -185,23 +185,20 @@ trait Dynamic {
val vargss = termArgss(tree)

def handleRepeated(base: untpd.Tree, possiblyCurried: List[List[Tree]], isReflectSelectable: Boolean) = {
val handled = possiblyCurried.map { args =>
val isRepeated = args.exists(_.tpe.widen.isRepeatedParam)
if isRepeated && isReflectSelectable then
List(untpd.TypedSplice(tpd.repeatedSeq(args, TypeTree(defn.AnyType))))
else args.map { t =>
val clzSym = t.tpe.resultType.classSymbol.asClass
if ValueClasses.isDerivedValueClass(clzSym) && isReflectSelectable then
val underlying = ValueClasses.valueClassUnbox(clzSym).asTerm
tpd.Select(t, underlying.name)
else
t
}.map(untpd.TypedSplice(_))
val args = possiblyCurried.flatten.map { t =>
val clzSym = t.tpe.resultType.classSymbol.asClass
if clzSym.isDerivedValueClass && isReflectSelectable then
val underlying = ValueClasses.valueClassUnbox(clzSym).asTerm
tpd.Select(t, underlying.name)
else
t
}
val handledFlat =
if isReflectSelectable then List(untpd.TypedSplice(tpd.repeatedSeq(args, TypeTree(defn.AnyType))))
else if args.isEmpty then List()
else List(untpd.TypedSplice(tpd.repeatedSeq(args, TypeTree(args.map(_.tpe).reduce(_ | _)))))

if isReflectSelectable
then untpd.Apply(base, handled.flatten)
else handled.foldLeft(base)((base, args) => untpd.Apply(base, args))
untpd.Apply(base, handledFlat)
}

def structuralCall(selectorName: TermName, classOfs: => List[Tree]) = {
Expand Down Expand Up @@ -265,7 +262,7 @@ trait Dynamic {
*/
def maybeBoxingCast(tpe: Type) =
val maybeBoxed =
if tpe.classSymbol.isDerivedValueClass && qual.tpe <:< defn.ReflectSelectableTypeRef then
if tpe.classSymbol.isDerivedValueClass && qual.tpe.isReflectSelectableTypeRef then
val genericUnderlying = ValueClasses.valueClassUnbox(tpe.classSymbol.asClass)
val underlying = tpe.select(genericUnderlying).widen.resultType
New(tpe.widen, tree.cast(underlying) :: Nil)
Expand Down Expand Up @@ -295,7 +292,7 @@ trait Dynamic {
fail(i"has a parameter type with an unstable erasure") :: Nil
else
TypeErasure.erasure(tpe).asInstanceOf[MethodType].paramInfos.map { tpe =>
if ValueClasses.isDerivedValueClass(tpe.widen.classSymbol) then
if tpe.widen.classSymbol.isDerivedValueClass then
clsOf(ValueClasses.valueClassUnbox(tpe.classSymbol.asClass).info)
else
clsOf(tpe)
Expand Down
2 changes: 1 addition & 1 deletion tests/pos/i17100.scala
Original file line number Diff line number Diff line change
Expand Up @@ -2,7 +2,7 @@ trait Sel extends Selectable

extension (s: Sel)
def selectDynamic(name: String) = ???
def applyDynamic(name: String)(x: Int) = ???
def applyDynamic(name: String)(x: Int*) = ???
def applyDynamic(name: String)() = ???

val sel = (new Sel {}).asInstanceOf[Sel{ def foo: String; def bar(x: Int): Int; def baz(): Int }]
Expand Down
10 changes: 5 additions & 5 deletions tests/run/i14340.scala
Original file line number Diff line number Diff line change
@@ -1,8 +1,8 @@
class Container1 extends reflect.Selectable

class Container2(values: Map[String, Any], methods: Map[String, Int => Any]) extends Selectable:
class Container2(values: Map[String, Any], methods: Map[String, Seq[Int] => Any]) extends Selectable:
def selectDynamic(name: String) = values(name)
def applyDynamic(name: String)(arg: Int) = methods(name)(arg)
def applyDynamic(name: String)(arg: Int*) = methods(name)(arg)

class Foo(val value: Int) extends AnyVal
class Bar[A](val value: A) extends AnyVal
Expand All @@ -29,7 +29,7 @@ object Helpers:
)

val cont2methods = Map(
"fooFromInt" -> { (i: Int) => Foo(i) }
"fooFromInt" -> { (i: Seq[Int]) => Foo(i.head) }
)

val cont2 = Container2(cont2values, cont2methods).asInstanceOf[Container2 {
Expand All @@ -39,7 +39,7 @@ object Helpers:
def qux2: Bar[Container2 { def foo: Foo }]
def fooFromInt(i: Int): Foo
}]


println(cont1.foo.value)
println(cont2.foo.value)
Expand All @@ -49,7 +49,7 @@ object Helpers:

println(cont1.qux1.value.foo.value)
println(cont2.qux1.value.foo.value)

println(cont1.qux2.value.foo.value)
println(cont2.qux2.value.foo.value)

Expand Down
72 changes: 62 additions & 10 deletions tests/run/i16995.check
Original file line number Diff line number Diff line change
@@ -1,10 +1,62 @@
1
check
6
7
5
3
3
3
7
3
foo
foo0
fooI 1
fooN 1
fooII 1 2
fooNN 1 2
fooIvarargs 1 2
fooNvarargs 1 2
fooIseq 1 2
fooNseq 1 2
fooIIvarargs 1 2
fooNNvarargs 1 2
fooI_I 1 2
fooN_N 1 2
foo0_II 1 2
foo0_NN 1 2
foo0_Ivarargs 1 2
foo0_Nvarargs 1 2
foo0_I_I 1 2
foo0_N_N 1 2

foo
foo0
fooI 1
fooN 1
fooII 1 2
fooNN 1 2
fooIvarargs 1 2
fooNvarargs 1 2
fooIseq 1 2
fooNseq 1 2
fooIIvarargs 1 2 3
fooNNvarargs 1 2
fooI_I 1 2
fooN_N 1 2
foo0_II 1 2
foo0_NN 1 2
foo0_Ivarargs 1 2
foo0_Nvarargs 1 2
foo0_I_I 1 2
foo0_N_N 1 2

foo
foo0
fooI 1
fooN 1
fooII 1 2
fooNN 1 2
fooIvarargs 1 2
fooNvarargs 1 2
fooIseq 1 2
fooNseq 1 2
fooIIvarargs 1 2
fooNNvarargs 1 2
fooI_I 1 2
fooN_N 1 2
foo0_II 1 2
foo0_NN 1 2
foo0_Ivarargs 1 2
foo0_Nvarargs 1 2
foo0_I_I 1 2
foo0_N_N 1 2
Loading

0 comments on commit 1f3f7e7

Please sign in to comment.