Skip to content

Commit 297c88c

Browse files
committed
Only skip reflection code on JS
- add basic support for `//> using target.platform jvm|scala-js` in Vulpix. - if the directive is present, filter out files in compilation that dont match the test platform, also add a suffix to the expected check-file name. - duplicate UnrollTestPlatformSpecific files for jvm and scala-js platforms, deleting the reflection code in scala-js version.
1 parent 5e285cd commit 297c88c

File tree

61 files changed

+244
-261
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

61 files changed

+244
-261
lines changed

compiler/test/dotty/tools/utils.scala

+37-4
Original file line numberDiff line numberDiff line change
@@ -57,25 +57,54 @@ def assertThrows[T <: Throwable: ClassTag](p: T => Boolean)(body: => Any): Unit
5757
case NonFatal(other) => throw AssertionError(s"Wrong exception: expected ${implicitly[ClassTag[T]]} but was ${other.getClass.getName}").tap(_.addSuppressed(other))
5858
end assertThrows
5959

60+
enum TestPlatform:
61+
case JVM, ScalaJS
62+
override def toString: String = this match
63+
case JVM => "jvm"
64+
case ScalaJS => "scala-js"
65+
66+
object TestPlatform:
67+
def named(s: String): TestPlatform = s match
68+
case "jvm" => TestPlatform.JVM
69+
case "scala-js" => TestPlatform.ScalaJS
70+
case _ => throw IllegalArgumentException(s)
71+
6072
/** Famous tool names in the ecosystem. Used for tool args in test files. */
6173
enum ToolName:
62-
case Scala, Scalac, Java, Javac, ScalaJS, Test
74+
case Scala, Scalac, Java, Javac, ScalaJS, Test, Target
6375
object ToolName:
6476
def named(s: String): ToolName = values.find(_.toString.equalsIgnoreCase(s)).getOrElse(throw IllegalArgumentException(s))
6577

6678
type ToolArgs = Map[ToolName, List[String]]
79+
type PlatformFiles = Map[TestPlatform, List[String]]
6780

6881
/** Take a prefix of each file, extract tool args, parse, and combine.
6982
* Arg parsing respects quotation marks. Result is a map from ToolName to the combined tokens.
7083
*/
7184
def toolArgsFor(files: List[JPath], charset: Charset = UTF_8): ToolArgs =
72-
files.foldLeft(Map.empty[ToolName, List[String]]) { (res, path) =>
85+
val (_, toolArgs) = platformAndToolArgsFor(files, charset)
86+
toolArgs
87+
88+
/** Take a prefix of each file, extract tool args, parse, and combine.
89+
* Arg parsing respects quotation marks. Result is a map from ToolName to the combined tokens.
90+
* If the ToolName is Target, then also accumulate the file name associated with the given platform.
91+
*/
92+
def platformAndToolArgsFor(files: List[JPath], charset: Charset = UTF_8): (PlatformFiles, ToolArgs) =
93+
files.foldLeft(Map.empty[TestPlatform, List[String]] -> Map.empty[ToolName, List[String]]) { (res, path) =>
7394
val toolargs = toolArgsParse(resource(Files.lines(path, charset))(_.limit(10).toScala(List)), Some(path.toString))
7495
toolargs.foldLeft(res) {
75-
case (acc, (tool, args)) =>
96+
case ((plat, acc), (tool, args)) =>
7697
val name = ToolName.named(tool)
7798
val tokens = CommandLineParser.tokenize(args)
78-
acc.updatedWith(name)(v0 => v0.map(_ ++ tokens).orElse(Some(tokens)))
99+
100+
val plat1 = if name eq ToolName.Target then
101+
val testPlatform = TestPlatform.named(tokens.head)
102+
val fileName = path.toString
103+
plat.updatedWith(testPlatform)(_.map(fileName :: _).orElse(Some(fileName :: Nil)))
104+
else
105+
plat
106+
107+
plat1 -> acc.updatedWith(name)(v0 => v0.map(_ ++ tokens).orElse(Some(tokens)))
79108
}
80109
}
81110

@@ -94,6 +123,8 @@ private val toolArg = raw"(?://|/\*| \*) ?(?i:(${ToolName.values.mkString("|")})
94123
/** Directive to specify to vulpix the options to pass to Dotty */
95124
private val directiveOptionsArg = raw"//> using options (.*)".r.unanchored
96125
private val directiveJavacOptions = raw"//> using javacOpt (.*)".r.unanchored
126+
private val directiveTargetOptions = raw"//> using target.platform (jvm|scala-js)".r.unanchored
127+
private val directiveUnknown = raw"//> using (.*)".r.unanchored
97128

98129
// Inspect the lines for compiler options of the form
99130
// `//> using options args`, `// scalajs: args`, `/* scalajs: args`, ` * scalajs: args` etc.
@@ -109,6 +140,8 @@ def toolArgsParse(lines: List[String], filename: Option[String]): List[(String,S
109140
lines.flatMap {
110141
case directiveOptionsArg(args) => List(("scalac", args))
111142
case directiveJavacOptions(args) => List(("javac", args))
143+
case directiveTargetOptions(platform) => List(("target", platform))
144+
case directiveUnknown(rest) => sys.error(s"Unknown directive: `//> using ${CommandLineParser.tokenize(rest).headOption.getOrElse("''")}`${filename.fold("")(f => s" in file $f")}")
112145
case _ => Nil
113146
}
114147

compiler/test/dotty/tools/vulpix/ParallelTesting.scala

+20-3
Original file line numberDiff line numberDiff line change
@@ -60,6 +60,8 @@ trait ParallelTesting extends RunnerOrchestration { self =>
6060
/** Contains a list of failed tests to run, if list is empty no tests will run */
6161
def failedTests: Option[List[String]]
6262

63+
protected def testPlatform: TestPlatform = TestPlatform.JVM
64+
6365
/** A test source whose files or directory of files is to be compiled
6466
* in a specific way defined by the `Test`
6567
*/
@@ -343,7 +345,11 @@ trait ParallelTesting extends RunnerOrchestration { self =>
343345
new JFile(f.getPath.replaceFirst("\\.(scala|java)$", ".check"))
344346
}
345347
case ts: SeparateCompilationSource =>
346-
Option(new JFile(ts.dir.getPath + ".check"))
348+
val platform =
349+
if testSource.allToolArgs.getOrElse(ToolName.Target, Nil).nonEmpty then
350+
s".$testPlatform"
351+
else ""
352+
Option(new JFile(ts.dir.getPath + platform + ".check"))
347353
}
348354
}
349355

@@ -499,7 +505,8 @@ trait ParallelTesting extends RunnerOrchestration { self =>
499505

500506
val files: Array[JFile] = files0.flatMap(flattenFiles)
501507

502-
val toolArgs = toolArgsFor(files.toList.map(_.toPath), getCharsetFromEncodingOpt(flags0))
508+
val (platformFiles, toolArgs) =
509+
platformAndToolArgsFor(files.toList.map(_.toPath), getCharsetFromEncodingOpt(flags0))
503510

504511
val spec = raw"(\d+)(\+)?".r
505512
val testIsFiltered = toolArgs.get(ToolName.Test) match
@@ -557,7 +564,17 @@ trait ParallelTesting extends RunnerOrchestration { self =>
557564
// If a test contains a Java file that cannot be parsed by Dotty's Java source parser, its
558565
// name must contain the string "JAVA_ONLY".
559566
val dottyFiles = files.filterNot(_.getName.contains("JAVA_ONLY")).map(_.getPath)
560-
driver.process(allArgs ++ dottyFiles, reporter = reporter)
567+
568+
val dottyFiles0 =
569+
if platformFiles.isEmpty then dottyFiles
570+
else
571+
val excludedFiles = platformFiles
572+
.collect { case (plat, files) if plat != testPlatform => files }
573+
.flatten
574+
.toSet
575+
dottyFiles.filterNot(excludedFiles)
576+
577+
driver.process(allArgs ++ dottyFiles0, reporter = reporter)
561578

562579
// todo a better mechanism than ONLY. test: -scala-only?
563580
val javaFiles = files.filter(_.getName.endsWith(".java")).filterNot(_.getName.contains("SCALA_ONLY")).map(_.getPath)

sjs-compiler-tests/test/scala/dotty/tools/dotc/ScalaJSCompilationTests.scala

+2
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,8 @@ class ScalaJSCompilationTests extends ParallelTesting {
4040
override protected def shouldSkipTestSource(testSource: TestSource): Boolean =
4141
testSource.allToolArgs.get(ToolName.ScalaJS).exists(_.contains("--skip"))
4242

43+
override protected def testPlatform: TestPlatform = TestPlatform.ScalaJS
44+
4345
override def runMain(classPath: String, toolArgs: ToolArgs)(implicit summaryReport: SummaryReporting): Status =
4446
import scala.concurrent.ExecutionContext.Implicits.global
4547

tests/run/unroll-caseclass.check tests/run/unroll-caseclass-integration.scala-js.check

-7
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,10 @@
1-
=== Unrolled Test V1 ===
21
Assertion passed: found "cow1" + "true0"
32
Assertion passed: found "cow2" + "true0"
43
Assertion passed: found "cow1" + "true0"
54
Assertion passed: found "cow2" + "true0"
65
Assertion passed: found "cow1" + "true0"
76
Assertion passed: found "cow2" + "true0"
87
Assertion passed: found "hello31337" + "true0"
9-
=== Unrolled Test V2 ===
108
Assertion passed: found "cow1true" + "0"
119
Assertion passed: found "cow2true" + "0"
1210
Assertion passed: found "cow2false" + "0"
@@ -17,12 +15,7 @@ Assertion passed: found "cow1true" + "0"
1715
Assertion passed: found "cow2true" + "0"
1816
Assertion passed: found "cow2false" + "0"
1917
Assertion passed: found "hello31337false" + "0"
20-
=== Unrolled Test V3 ===
2118
Assertion passed: found "hello31337false12345"
22-
as expected, no constructor for Unrolled(s: String)
23-
public example.Unrolled(java.lang.String,int)
24-
public example.Unrolled(java.lang.String,int,boolean)
25-
public example.Unrolled(java.lang.String,int,boolean,long)
2619
Assertion passed: found "cow1true0"
2720
Assertion passed: found "cow2true0"
2821
Assertion passed: found "cow2false0"

tests/run/unroll-caseclass-integration/Test_4.scala

-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,4 @@
11
//> using options -experimental
2-
// scalajs: --skip
32
import unroll.*
43

54
@main def Test: Unit =
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
//> using options -experimental
2+
//> using target.platform scala-js
3+
package unroll
4+
5+
object UnrollTestPlatformSpecificV3{
6+
def apply() = {
7+
// do nothing for scala-js
8+
}
9+
}

tests/run/unroll-caseclass-integration/UnrollTestPlatformSpecific_3.scala

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
//> using options -experimental
2-
// scalajs: --skip
2+
//> using target.platform jvm
33
package unroll
44

55
object UnrollTestPlatformSpecificV3{

tests/run/unroll-caseclass/Test_4.scala

-13
This file was deleted.

tests/run/unroll-caseclass/unrolledV1_1.scala

-62
This file was deleted.

tests/run/unroll-caseclass/unrolledV2_2.scala

-56
This file was deleted.

0 commit comments

Comments
 (0)