blob: d33a127580ba3de8c83c1090154e551983cb97a7 [file] [log] [blame]
Mads Ager418d1ca2017-05-22 09:35:49 +02001// Copyright (c) 2016, the R8 project authors. Please see the AUTHORS file
2// for details. All rights reserved. Use of this source code is governed by a
3// BSD-style license that can be found in the LICENSE file.
4import org.gradle.internal.os.OperatingSystem;
5
6apply plugin: 'java'
7apply plugin: 'idea'
8apply plugin: 'jacoco'
9
10apply from: 'copyAdditionalJctfCommonFiles.gradle'
11
12repositories {
13 mavenCentral()
14}
15
16// Custom source set for example tests and generated tests.
17sourceSets {
18 test {
19 java {
20 srcDirs = [
21 'src/test/java',
22 'build/generated/test/java',
23 ]
24 }
25 }
26 debugTestResources {
27 java {
28 srcDirs = ['src/test/debugTestResources']
29 }
30 output.resourcesDir = 'build/classes/debugTestResources'
31 }
32 examples {
33 java {
34 srcDirs = ['src/test/examples']
35 }
36 output.resourcesDir = 'build/classes/examples'
37 }
38 examplesAndroidN {
39 java {
40 srcDirs = ['src/test/examplesAndroidN']
41 }
42 output.resourcesDir = 'build/classes/examplesAndroidN'
43 }
44 examplesAndroidO {
45 java {
46 srcDirs = ['src/test/examplesAndroidO']
47 }
48 output.resourcesDir = 'build/classes/examplesAndroidO'
49 }
50 jctfCommon {
51 java {
52 srcDirs = [
53 'third_party/jctf/Harness/src',
54 'third_party/jctf/LibTests/src/com/google/jctf/test/categories',
55 'third_party/jctf/LibTests/src/com/google/jctf/test/helper',
56 'third_party/jctf/LibTests/src/com/google/jctf/testHelpers',
57 'third_party/jctf/LibTests/src/org',
58 'build/additionalJctfCommonFiles'
59 ]
60 }
61 resources {
62 srcDirs = ['third_party/jctf/LibTests/resources']
63 }
64 }
65 jctfTests {
66 java {
67 srcDirs = [
68 'third_party/jctf/LibTests/src/com/google/jctf/test/lib',
69 // 'third_party/jctf/VMTests/src',
70 ]
71 }
72 }
73}
74
75dependencies {
76 compile 'net.sf.jopt-simple:jopt-simple:4.6'
77 compile group: 'com.google.guava', name: 'guava', version: '19.0'
Stephan Herhutb17bb8d2017-05-23 12:34:55 +020078 compile group: 'it.unimi.dsi', name: 'fastutil', version: '7.2.0'
Mads Ager418d1ca2017-05-22 09:35:49 +020079 compile group: 'org.apache.commons', name: 'commons-compress', version: '1.12'
80 compile group: 'org.ow2.asm', name: 'asm', version: '5.1'
81 compile group: 'org.ow2.asm', name: 'asm-commons', version: '5.1'
82 compile group: 'org.ow2.asm', name: 'asm-tree', version: '5.1'
83 compile group: 'org.ow2.asm', name: 'asm-util', version: '5.1'
84 testCompile sourceSets.examples.output
85 testCompile 'junit:junit:4.12'
86 testCompile group: 'org.smali', name: 'smali', version: '2.2b4'
87 testCompile files('third_party/jasmin/jasmin-2.4.jar')
88 testCompile files('third_party/jdwp-tests/apache-harmony-jdwp-tests-host.jar')
89 jctfCommonCompile 'junit:junit:4.12'
90 jctfTestsCompile 'junit:junit:4.12'
91 jctfTestsCompile sourceSets.jctfCommon.output
92 examplesAndroidOCompile group: 'org.ow2.asm', name: 'asm', version: '5.1'
93}
94
95def osString = OperatingSystem.current().isLinux() ? "linux" : "mac"
96
97def cloudDependencies = [
98 "tests" : [
99 "art.tar.gz"
100 ],
101 "third_party": [
102 "android_jar/lib-v14.tar.gz",
103 "android_jar/lib-v19.tar.gz",
104 "android_jar/lib-v24.tar.gz",
105 "android_jar/lib-v25.tar.gz",
106 "android_jar/lib-v26.tar.gz",
107 "proguard/proguard5.2.1.tar.gz",
108 "gradle/gradle.tar.gz",
109 "jdwp-tests.tar.gz",
110 "jasmin.tar.gz",
111 "jctf.tar.gz",
112 "android_cts_baseline.tar.gz",
113 ],
114 // All dex-vms have a fixed OS of Linux, as they are only supported on Linux, and will be run in a Docker
115 // container on other platforms where supported.
116 "tools" : [
117 "linux/art.tar.gz",
118 "linux/art-5.1.1.tar.gz",
119 "linux/art-6.0.1.tar.gz",
120 "linux/art-7.0.0.tar.gz",
121 "linux/dalvik.tar.gz",
122 "${osString}/dx.tar.gz",
123 ]
124]
125
126cloudDependencies.each { entry ->
127 entry.value.each { entryFile ->
128 task "download_deps_${entry.key}/${entryFile}"(type: Exec) {
129 def gzFile = "${entry.key}/${entryFile}"
130 def sha1File = "${gzFile}.sha1"
131 inputs.file sha1File
132 outputs.file gzFile
133 executable "bash"
134 args "-c", "download_from_google_storage -n -b r8-deps -u -s ${sha1File}"
135 }
136 }
137}
138
139def x20Dependencies = [
140 "third_party": [
141 "gmscore/v4.tar.gz",
142 "gmscore/v5.tar.gz",
143 "gmscore/v6.tar.gz",
144 "gmscore/v7.tar.gz",
145 "gmscore/v8.tar.gz",
146 "gmscore/gmscore_v9.tar.gz",
147 "gmscore/gmscore_v10.tar.gz",
148 "youtube/youtube.android_11.47.tar.gz",
149 "youtube/youtube.android_12.10.tar.gz",
Søren Gjessefeac2ef2017-05-22 17:09:29 +0200150 "youtube/youtube.android_12.17.tar.gz",
Mads Ager418d1ca2017-05-22 09:35:49 +0200151 "proguardsettings.tar.gz",
152 ],
153]
154
155x20Dependencies.each { entry ->
156 entry.value.each { entryFile ->
157 task "download_deps_${entry.key}/${entryFile}"(type: Exec) {
158 def gzFile = "${entry.key}/${entryFile}"
159 def sha1File = "${gzFile}.sha1"
160 inputs.file sha1File
161 outputs.file gzFile
162 executable "bash"
163 args "-c", "tools/download_from_x20.py ${sha1File}"
164 }
165 }
166}
167
Rico Wind897bb712017-05-23 10:44:29 +0200168task downloadProguard {
169 cloudDependencies.each { entry ->
170 entry.value.each { entryFile ->
171 if (entryFile.contains("proguard")) {
172 dependsOn "download_deps_${entry.key}/${entryFile}"
173 }
174 }
175 }
176}
177
Mads Ager418d1ca2017-05-22 09:35:49 +0200178task downloadDeps {
179 cloudDependencies.each { entry ->
180 entry.value.each { entryFile ->
181 dependsOn "download_deps_${entry.key}/${entryFile}"
182 }
183 }
184 if (!project.hasProperty('no_internal')) {
185 x20Dependencies.each { entry ->
186 entry.value.each { entryFile ->
187 dependsOn "download_deps_${entry.key}/${entryFile}"
188 }
189 }
190 }
191}
192
193allprojects {
194 sourceCompatibility = JavaVersion.VERSION_1_8
195 targetCompatibility = JavaVersion.VERSION_1_8
196}
197
198tasks.withType(JavaCompile) {
199 options.compilerArgs << '-Xlint:unchecked'
200}
201
202compileJctfCommonJava {
203 dependsOn 'copyAdditionalJctfCommonFiles'
204 options.compilerArgs = ['-Xlint:none']
205}
206
207compileJctfTestsJava {
208 dependsOn 'jctfCommonClasses'
209 options.compilerArgs = ['-Xlint:none']
210}
211
212task R8(type: Jar) {
213 from sourceSets.main.output
214 baseName 'r8'
215 manifest {
216 attributes 'Main-Class': 'com.android.tools.r8.R8'
217 }
218 // In order to build without dependencies, pass the exclude_deps property using:
219 // gradle -Pexclude_deps R8
220 if (!project.hasProperty('exclude_deps')) {
221 // Also include dependencies
222 from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
223 }
224}
225
226task D8(type: Jar) {
227 from sourceSets.main.output
228 baseName 'd8'
229 manifest {
230 attributes 'Main-Class': 'com.android.tools.r8.D8'
231 }
232 // In order to build without dependencies, pass the exclude_deps property using:
233 // gradle -Pexclude_deps D8
234 if (!project.hasProperty('exclude_deps')) {
235 // Also include dependencies
236 from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
237 }
238}
239
240task CompatDx(type: Jar) {
241 from sourceSets.main.output
242 baseName 'compatdx'
243 manifest {
244 attributes 'Main-Class': 'com.android.tools.r8.compatdx.CompatDx'
245 }
246 // In order to build without dependencies, pass the exclude_deps property using:
247 // gradle -Pexclude_deps CompatDx
248 if (!project.hasProperty('exclude_deps')) {
249 // Also include dependencies
250 from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
251 }
252}
253
254task disasm(type: Jar) {
255 from sourceSets.main.output
256 baseName 'disasm'
257 manifest {
258 attributes 'Main-Class': 'com.android.tools.r8.Disassemble'
259 }
260 // In order to build without dependencies, pass the exclude_deps property using:
261 // gradle -Pexclude_deps D8
262 if (!project.hasProperty('exclude_deps')) {
263 // Also include dependencies
264 from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
265 }
266}
267
268task bisect(type: Jar) {
269 from sourceSets.main.output
270 baseName 'bisect'
271 manifest {
272 attributes 'Main-Class': 'com.android.tools.r8.bisect.Bisect'
273 }
274 // In order to build without dependencies, pass the exclude_deps property using:
275 // gradle -Pexclude_deps R8
276 if (!project.hasProperty('exclude_deps')) {
277 // Also include dependencies
278 from { configurations.compile.collect { it.isDirectory() ? it : zipTree(it) } }
279 }
280}
281
282task sourceJar(type: Jar, dependsOn: classes) {
283 classifier = 'src'
284 from sourceSets.main.allSource
285}
286
287task jctfCommonJar(type: Jar) {
288 from sourceSets.jctfCommon.output
289 baseName 'jctfCommon'
290}
291
292artifacts {
293 archives sourceJar
294}
295
296task createArtTests(type: Exec) {
297 def outputDir = "build/generated/test/java/com/android/tools/r8/art"
Mads Ager7e5bd722017-05-24 07:17:27 +0200298 def createArtTestsScript = "tools/create_art_tests.py"
Mads Ager418d1ca2017-05-22 09:35:49 +0200299 inputs.file "tests/art.tar.gz"
300 inputs.file createArtTestsScript
301 outputs.dir outputDir
302 dependsOn downloadDeps
Mads Ager677e3002017-05-24 07:54:51 +0200303 commandLine "python", createArtTestsScript
Mads Ager418d1ca2017-05-22 09:35:49 +0200304 workingDir = projectDir
305}
306
307task createJctfTests(type: Exec) {
Stephan Herhutea6ee582017-05-23 13:14:34 +0200308 def outputDir = "build/generated/test/java/com/android/tools/r8/jctf"
Mads Ager418d1ca2017-05-22 09:35:49 +0200309 def script = "scripts/create-jctf-tests.sh"
310 inputs.file script
311 outputs.dir outputDir
312 dependsOn downloadDeps
313 executable "bash"
314 args "${projectDir}/${script}"
315 workingDir = projectDir
316}
317
318compileTestJava {
319 dependsOn createArtTests
320 dependsOn createJctfTests
321}
322
323task buildDebugInfoExamplesDex {
324 def examplesDir = file("src/test/java")
325 def hostJar = "debuginfo_examples.jar"
326 def hostDexJar = "debuginfo_examples_dex.jar"
327 task "compile_debuginfo_examples"(type: JavaCompile) {
328 source = fileTree(dir: examplesDir, include: "com/android/tools/r8/debuginfo/*Test.java")
329 destinationDir = file("build/test/debuginfo_examples/classes")
330 classpath = sourceSets.main.compileClasspath
331 sourceCompatibility = JavaVersion.VERSION_1_7
332 targetCompatibility = JavaVersion.VERSION_1_7
333 options.compilerArgs += ["-Xlint:-options"]
334 }
335 task "jar_debuginfo_examples"(type: Jar, dependsOn: "compile_debuginfo_examples") {
336 archiveName = hostJar
337 destinationDir = file("build/test/")
338 from "build/test/debuginfo_examples/classes"
339 include "**/*.class"
340 }
341 task "dex_debuginfo_examples"(type: Exec,
342 dependsOn: ["jar_debuginfo_examples", "downloadDeps"]) {
343 executable file("tools/linux/dx/bin/dx");
344 args "--dex"
345 args "--output=build/test/${hostDexJar}"
346 args "build/test/${hostJar}"
347 inputs.file file("build/test/${hostJar}")
348 outputs.file file("build/test/${hostDexJar}")
349 }
350 dependsOn dex_debuginfo_examples
351}
352
353task buildDebugTestResourcesJars {
354 dependsOn downloadDeps
355 def resourcesDir = file("src/test/debugTestResources")
356 def hostJar = "debug_test_resources.jar"
357 task "compile_debugTestResources"(type: JavaCompile) {
358 source = fileTree(dir: resourcesDir, include: '**/*.java')
359 destinationDir = file("build/test/debugTestResources/classes")
360 classpath = sourceSets.main.compileClasspath
361 sourceCompatibility = JavaVersion.VERSION_1_7
362 targetCompatibility = JavaVersion.VERSION_1_7
363 options.compilerArgs += ["-g", "-Xlint:-options"]
364 }
365 task "jar_debugTestResources"(type: Jar, dependsOn: "compile_debugTestResources") {
366 archiveName = hostJar
367 destinationDir = file("build/test/")
368 from "build/test/debugTestResources/classes"
369 include "**/*.class"
370 }
371 dependsOn jar_debugTestResources
372}
373
374task buildExampleJars {
Rico Wind897bb712017-05-23 10:44:29 +0200375 dependsOn downloadProguard
Mads Ager418d1ca2017-05-22 09:35:49 +0200376 def examplesDir = file("src/test/examples")
377 def proguardScript = "third_party/proguard/proguard5.2.1/bin/proguard.sh"
378 task "compile_examples"(type: JavaCompile) {
379 source = fileTree(dir: examplesDir, include: '**/*.java')
380 destinationDir = file("build/test/examples/classes")
381 classpath = sourceSets.main.compileClasspath
382 sourceCompatibility = JavaVersion.VERSION_1_7
383 targetCompatibility = JavaVersion.VERSION_1_7
384 options.compilerArgs += ["-Xlint:-options"]
385 }
386 examplesDir.eachDir { dir ->
387 def name = dir.getName();
388 def exampleOutputDir = file("build/test/examples");
389 def jarName = "${name}.jar"
390 dependsOn "jar_example_${name}"
391 // The "throwing" test verifies debugging/stack info on the post-proguarded output.
392 def proguardConfigPath = "${dir}/proguard.cfg"
393 if (new File(proguardConfigPath).exists()) {
394 task "pre_proguard_example_${name}"(type: Jar, dependsOn: "compile_examples") {
395 archiveName = "${name}_pre_proguard.jar"
396 destinationDir = exampleOutputDir
397 from "build/test/examples/classes"
398 include "**/" + name + "/**/*.class"
399 }
400 def jarPath = files(tasks.getByPath("pre_proguard_example_${name}")).files.first();
401 def proguardJarPath = "${exampleOutputDir}/${jarName}"
402 def proguardMapPath = "${exampleOutputDir}/${name}/${name}.map"
403 task "jar_example_${name}"(type: Exec, dependsOn: "pre_proguard_example_${name}") {
404 inputs.files tasks.getByPath("pre_proguard_example_${name}")
405 inputs.file proguardConfigPath
406 // Enable these to get stdout and stderr redirected to files...
407 // standardOutput = new FileOutputStream('proguard.stdout')
408 // errorOutput = new FileOutputStream('proguard.stderr')
409 executable "bash"
410 args "-c", "${proguardScript} '-verbose -dontwarn java.** -injars ${jarPath}" +
411 " -outjars ${proguardJarPath}" +
412 " -include ${proguardConfigPath}" +
413 " -printmapping ${proguardMapPath}'"
414 outputs.file proguardJarPath
415 }
416 } else {
417 task "jar_example_${name}"(type: Jar, dependsOn: "compile_examples") {
418 archiveName = jarName
419 destinationDir = exampleOutputDir
420 from "build/test/examples/classes"
421 include "**/" + name + "/**/*.class"
422 }
423 }
424 }
425}
426
427task buildExampleAndroidNJars {
428 dependsOn downloadDeps
429 def examplesDir = file("src/test/examplesAndroidN")
430 task "compile_examplesAndroidN"(type: JavaCompile) {
431 source = fileTree(dir: examplesDir, include: '**/*.java')
432 destinationDir = file("build/test/examplesAndroidN/classes")
433 classpath = sourceSets.main.compileClasspath
434 sourceCompatibility = JavaVersion.VERSION_1_8
435 targetCompatibility = JavaVersion.VERSION_1_8
436 options.compilerArgs += ["-Xlint:-options"]
437 }
438 examplesDir.eachDir { dir ->
439 def name = dir.getName();
440 def exampleOutputDir = file("build/test/examplesAndroidN");
441 def jarName = "${name}.jar"
442 dependsOn "jar_examplesAndroidN_${name}"
443 task "jar_examplesAndroidN_${name}"(type: Jar, dependsOn: "compile_examplesAndroidN") {
444 archiveName = jarName
445 destinationDir = exampleOutputDir
446 from "build/test/examplesAndroidN/classes"
447 include "**/" + name + "/**/*.class"
448 }
449 }
450}
451
452
453task buildExampleAndroidOJars {
454 dependsOn downloadDeps
455 def examplesDir = file("src/test/examplesAndroidO")
456 // NOTE: we want to enable a scenario when test needs to reference some
457 // classes generated by legacy (1.6) Java compiler to test some specific
458 // behaviour. To do so we compile all the java files located in sub-directory
459 // called 'legacy' with Java 1.6, then compile the rest of the files with
460 // Java 1.8 and a reference to previously generated 1.6 classes.
461
462 // Compiling all classes in dirs 'legacy' with old Java version.
463 task "compile_examplesAndroidO_Legacy"(type: JavaCompile) {
464 source = fileTree(dir: examplesDir, include: '**/legacy/**/*.java')
465 destinationDir = file("build/test/examplesAndroidOLegacy/classes")
466 classpath = sourceSets.main.compileClasspath
467 sourceCompatibility = JavaVersion.VERSION_1_6
468 targetCompatibility = JavaVersion.VERSION_1_6
469 options.compilerArgs += ["-Xlint:-options", "-parameters"]
470 }
471 // Compiling the rest of the files as Java 1.8 code.
472 task "compile_examplesAndroidO"(type: JavaCompile) {
473 dependsOn "compile_examplesAndroidO_Legacy"
474 source = fileTree(dir: examplesDir, include: '**/*.java', exclude: '**/legacy/**/*.java')
475 destinationDir = file("build/test/examplesAndroidO/classes")
476 classpath = sourceSets.main.compileClasspath
477 classpath += files("build/test/examplesAndroidOLegacy/classes")
478 sourceCompatibility = JavaVersion.VERSION_1_8
479 targetCompatibility = JavaVersion.VERSION_1_8
480 options.compilerArgs += ["-Xlint:-options", "-parameters"]
481 }
482 examplesDir.eachDir { dir ->
483 def name = dir.getName();
484 def destinationDir = file("build/test/examplesAndroidO/classes");
485 if (file("src/test/examplesAndroidO/" + name + "/TestGenerator.java").isFile()) {
486 task "generate_examplesAndroidO_${name}"(type: JavaExec,
487 dependsOn: "compile_examplesAndroidO") {
488 main = name + ".TestGenerator"
489 classpath = files(destinationDir, sourceSets.main.compileClasspath)
490 args destinationDir
491 }
492 } else {
493 task "generate_examplesAndroidO_${name}" () {}
494 }
495 }
496 examplesDir.eachDir { dir ->
497 def name = dir.getName();
498 def exampleOutputDir = file("build/test/examplesAndroidO");
499 def jarName = "${name}.jar"
500 dependsOn "jar_examplesAndroidO_${name}"
501 task "jar_examplesAndroidO_${name}"(type: Jar, dependsOn: ["compile_examplesAndroidO",
502 "generate_examplesAndroidO_${name}"]) {
503 archiveName = jarName
504 destinationDir = exampleOutputDir
505 from "build/test/examplesAndroidO/classes" // Java 1.8 classes
506 from "build/test/examplesAndroidOLegacy/classes" // Java 1.6 classes
507 include "**/" + name + "/**/*.class"
508 // Do not include generator into the test runtime jar, it is not useful.
509 // Otherwise, shrinking will need ASM jars.
510 exclude "**/TestGenerator*"
511 }
512 }
513}
514
515task buildExamples {
516 if (OperatingSystem.current().isMacOsX()) {
517 logger.lifecycle("WARNING: Testing (including building examples) is only partially supported on Mac OS.")
518 } else if (!OperatingSystem.current().isLinux()) {
519 logger.lifecycle("WARNING: Testing (including building examples) is not supported on your platform. " +
520 "It is fully supported on Linux and partially supported on Mac OS")
521 return;
522 }
523 dependsOn buildDebugTestResourcesJars
524 dependsOn buildExampleJars
525 dependsOn buildExampleAndroidNJars
526 dependsOn buildExampleAndroidOJars
527 def examplesDir = file("src/test/examples")
528 examplesDir.eachDir { dir ->
529 def name = dir.getName();
530 dependsOn "dex_example_${name}"
531 def exampleOutputDir = file("build/test/examples/" + name);
532 def dexPath = file("${exampleOutputDir}")
533 def debug = (name == "throwing")
534 if (!dexPath.exists()) {
535 dexPath.mkdirs()
536 }
537 task "dex_example_${name}"(type: dx.Dx, dependsOn: "jar_example_${name}") {
538 source = files(tasks.getByPath("jar_example_${name}")).asFileTree
539 destination = dexPath
540 debug = debug
541 }
542 }
543}
544
545task buildSmali {
546 def smaliDir = file("src/test/smali")
547 smaliDir.eachDirRecurse() { dir ->
548 def name = dir.getName();
549 def relativeDir = smaliDir.toPath().relativize(dir.toPath());
550 def smaliOutputDir = file("build/test/smali/" + relativeDir);
551 smaliOutputDir.mkdirs()
552 outputs.dir smaliOutputDir
553 def taskName = "smali_build_${relativeDir.toString().replace('/', '_')}"
554 def smaliFiles = fileTree(dir: dir, include: '*.smali')
555 def javaFiles = fileTree(dir: dir, include: '*.java')
556 def destDir = smaliOutputDir;
557 def destFile = destDir.toPath().resolve("${name}.dex").toFile()
558 def intermediateFileName = "${name}-intermediate.dex";
559 def intermediateFile = destDir.toPath().resolve(intermediateFileName).toFile()
560 if (javaFiles.empty) {
561 if (!smaliFiles.empty) {
562 dependsOn "${taskName}_smali"
563 task "${taskName}_smali"(type: smali.Smali) {
564 source = smaliFiles
565 destination = destFile
566 }
567 }
568 } else {
569 dependsOn "${taskName}_dexmerger"
570 task "${taskName}_smali"(type: smali.Smali) {
571 source = smaliFiles
572 destination = intermediateFile
573 }
574 task "${taskName}_java"(type: JavaCompile) {
575 source = javaFiles
576 destinationDir destDir
577 classpath = sourceSets.main.compileClasspath
578 sourceCompatibility = JavaVersion.VERSION_1_7
579 targetCompatibility = JavaVersion.VERSION_1_7
580 options.compilerArgs += ["-Xlint:-options"]
581 }
582 task "${taskName}_jar"(type: Jar, dependsOn: "${taskName}_java") {
583 archiveName = "Test.jar"
584 destinationDir = destDir
585 from fileTree(dir: destDir, include: 'Test.class')
586 }
587 task "${taskName}_dx"(type: dx.Dx, dependsOn: "${taskName}_jar") {
588 source = fileTree(dir: destDir, include: 'Test.jar')
589 destination = destDir
590 }
591 task "${taskName}_dexmerger"(
592 type: dx.DexMerger, dependsOn: ["${taskName}_dx", "${taskName}_smali"]) {
593 source = fileTree(dir: destDir, include: ["classes.dex", intermediateFileName])
594 destination = destFile
595 }
596 }
597 }
598}
599
600tasks.withType(Test) {
601 def userDefinedCoresPerFork = System.getenv('R8_GRADLE_CORES_PER_FORK')
602 def coresPerFork = userDefinedCoresPerFork ? userDefinedCoresPerFork.toInteger() : 3
603 // See https://docs.gradle.org/current/dsl/org.gradle.api.tasks.testing.Test.html.
604 maxParallelForks = Runtime.runtime.availableProcessors().intdiv(coresPerFork) ?: 1
605 forkEvery = 0
606 // Use the Concurrent Mark Sweep GC (CMS) to keep memory usage at a resonable level.
607 jvmArgs = ["-XX:+UseConcMarkSweepGC"]
608}
609
610task buildPreNJdwpTestsJar(type: Jar) {
611 baseName = 'jdwp-tests-preN'
612 from zipTree('third_party/jdwp-tests/apache-harmony-jdwp-tests-host.jar')
613 // Exclude the classes containing java8
614 exclude 'org/apache/harmony/jpda/tests/jdwp/InterfaceType/*.class'
615 exclude 'org/apache/harmony/jpda/tests/jdwp/ObjectReference/InvokeMethodDefault*.class'
616 includeEmptyDirs = false
617}
618
619test {
620 testLogging.exceptionFormat = 'full'
621 if (project.hasProperty('print_test_stdout')) {
622 testLogging.showStandardStreams = true
623 }
624 if (project.hasProperty('dex_vm')) {
625 println "Running with non default vm: " + project.property('dex_vm')
626 systemProperty 'dex_vm', project.property('dex_vm')
627 if (project.property('dex_vm') == '5.1.1' || project.property('dex_vm') == '6.0.1') {
628 // R8 and D8 compute the dex file version number based on the input.
629 // Jack generates dex files with version 37 which art 5.1.1 and 6.0.1 will not run.
630 // Therefore we skip the jack generated art tests with those art versions.
631 exclude "com/android/tools/r8/art/jack/**"
632 }
633 }
634 if (project.hasProperty('one_line_per_test')) {
635 beforeTest { desc ->
636 println "Start executing test ${desc.name} [${desc.className}]"
637 }
638 afterTest { desc, result ->
639 println "Done executing test ${desc.name} [${desc.className}] with result: ${result.resultType}"
640 }
641 }
642 if (project.hasProperty('no_internal')) {
643 exclude "com/android/tools/r8/internal/**"
644 }
645 if (project.hasProperty('only_internal')) {
646 include "com/android/tools/r8/internal/**"
647 }
648 if (project.hasProperty('tool')) {
649 if (project.property('tool') == 'r8') {
650 exclude "com/android/tools/r8/art/*/d8/**"
651 } else {
652 assert(project.property('tool') == 'd8')
653 exclude "com/android/tools/r8/art/*/r8/**"
654 }
655 }
656 if (!project.hasProperty('all_tests')) {
657 exclude "com/android/tools/r8/art/dx/**"
658 exclude "com/android/tools/r8/art/jack/**"
659 }
660 // TODO(tamaskenez) enable jctf on all_tests when consolidated
661 if (!project.hasProperty('jctf') && !project.hasProperty('only_jctf')) {
Stephan Herhutea6ee582017-05-23 13:14:34 +0200662 exclude "com/android/tools/r8/jctf/**"
Mads Ager418d1ca2017-05-22 09:35:49 +0200663 }
664 if (project.hasProperty('only_jctf')) {
Stephan Herhutea6ee582017-05-23 13:14:34 +0200665 include "com/android/tools/r8/jctf/**"
Mads Ager418d1ca2017-05-22 09:35:49 +0200666 }
667 if (project.hasProperty('jctf_compile_only')) {
668 println "JCTF: compiling only"
669 systemProperty 'jctf_compile_only', '1'
670 }
671
672 if (OperatingSystem.current().isLinux() || OperatingSystem.current().isMacOsX()) {
673 if (OperatingSystem.current().isMacOsX()) {
674 logger.lifecycle("WARNING: Testing in only partially supported on Mac OS. " +
675 "Art only runs on Linux and tests requiring Art runs in a Docker container, which must be present. " +
676 "See tools/docker/README.md for details.")
677 }
678 dependsOn downloadDeps
679 dependsOn buildExamples
680 dependsOn buildSmali
681 dependsOn jctfCommonJar
682 dependsOn jctfTestsClasses
683 dependsOn buildDebugInfoExamplesDex
684 dependsOn buildPreNJdwpTestsJar
685 } else {
686 logger.lifecycle("WARNING: Testing in not supported on your platform. Testing is only fully supported on " +
687 "Linux and partially supported on Mac OS. Art does not run on other platforms.")
688 }
689}
690
691// The Art tests we use for R8 are pre-build and downloaded from Google Cloud Storage.
692//
693// To build and upload a new set of the Art tests for use with R8 follow these steps:
694//
695// First of all an Android checkout is required. Currently it must be located
696// in $HOME/android/master.
697//
698// TODO(ricow): simplify this
699//
700// Before: update the checked in art, see scripts/update-host-art.sh
701//
702// 1. Get an android checkout in $HOME/android/master and apply the patch from
703// https://android-review.googlesource.com/#/c/294187/
704//
705// 2. run the following commands in the Android checkout directory:
706//
707// source build/envsetup.sh
708// lunch aosp_angler-userdebug
709// make -j30 test-art-host
710//
711// 3. In the R8 project root directory, make sure we have a clean state before starting:
712// tools/gradle.py downloadDeps
713// tools/gradle.py clean
714// rm -rf tests/art
715//
716// 4. Now build in the R8 checkout (-P hack to not generate dirs when not running this target)
717// Make sure you have smali on your path, there is a build binary in the out directory of
718// the android checkout:
719//
720// tools/gradle.py -Pandroid_source buildArtTests
721//
722// 4a. If any failures are produced in step 4, figure out what went wrong and add an entry in
723// skippedTests with an explanation. Rerun from step 3.
724//
725// 5. Run the tests:
726// tools/gradle.py clean
727// tools/test.py
728//
729// 5a. If any more tests fail, either fix the issue or add them to the toBeTriaged list (note that
730// you need to change "_" to "-" from stdout). Rerun from step 3 if anything was added to
731// toBeTriaged.
732//
733// 6. To upload a new version to Google Cloud Storage
734// cd tests
735// upload_to_google_storage.py -a --bucket r8-deps art
736
737enum DexTool {
738 JACK,
739 DX
740}
741
742def androidCheckoutDir = file("${System.env.HOME}/android/master")
743def androidCheckoutJack = file("${androidCheckoutDir}/out/host/linux-x86/bin/jack");
744def androidCheckoutJackServer = file("${androidCheckoutDir}/out/host/linux-x86/bin/jack-admin");
745
746def artTestDir = file("${androidCheckoutDir}/art/test")
747
748if (project.hasProperty('android_source')) {
749 task buildArtTests {
750 outputs.upToDateWhen { false }
751 def toBeTriaged = [
752 "903-hello-tagging",
753 "904-object-allocation",
754 "905-object-free",
755 "906-iterate-heap",
756 "907-get-loaded-classes",
757 "908-gc-start-finish",
758 "954-invoke-polymorphic-verifier",
759 "955-methodhandles-smali",
760 "596-monitor-inflation",
761 ]
762 def skippedTests = toBeTriaged + [
763 // This test produces no jar.
764 "000-nop",
765 // This does not build, as it tests the error when the application exceeds more
766 // than 65536 methods
767 "089-many-methods",
768 // Requires some jack beta jar
769 "956-methodhandles",
770 ]
771
772 def skippedTestsDx = [
773 // Tests with custom build scripts, where javac is not passed the options
774 // -source 1.7 -target 1.7.
775 "462-checker-inlining-across-dex-files",
776 "556-invoke-super",
777 "569-checker-pattern-replacement",
778 // These tests use jack even when --build-with-javac-dx is specified.
779 "004-JniTest",
780 "048-reflect-v8",
781 "146-bad-interface",
782 "563-checker-invoke-super",
783 "580-checker-string-fact-intrinsics", // java.lang.StringFactory
784 "604-hot-static-interface",
785 "957-methodhandle-transforms",
786 "958-methodhandle-emulated-stackframe",
787 "959-invoke-polymorphic-accessors",
788 "961-default-iface-resolution-gen",
789 "962-iface-static",
790 "963-default-range-smali",
791 "964-default-iface-init-gen",
792 "965-default-verify",
793 "966-default-conflict",
794 "967-default-ame",
795 "968-default-partial-compile-gen",
796 "969-iface-super",
797 "970-iface-super-resolution-gen",
798 "971-iface-super",
799 // These tests does not build with --build-with-javac-dx
800 "004-NativeAllocations", // Javac error
801 "031-class-attributes",
802 "138-duplicate-classes-check",
803 "157-void-class", // Javac error
804 "580-checker-string-factory-intrinsics",
805 "612-jit-dex-cache",
806 "613-inlining-dex-cache",
807 "900-hello-plugin", // --experimental agents
808 "901-hello-ti-agent", // --experimental agents
809 "902-hello-transformation", // --experimental agents
810 "909-attach-agent", // --experimental agents
811 "946-obsolete-throw", // -source 1.7 -target 1.7, but use lambda
812 "950-redefine-intrinsic", // -source 1.7 -target 1.7, but use method references
813 "951-threaded-obsolete", // -source 1.7 -target 1.7, but use lambda
814 "960-default-smali", // --experimental default-methods
815 // These tests force the build to use jack
816 "953-invoke-polymorphic-compiler",
817 "958-methodhandle-stackframe",
818 ]
819
820 def artTestBuildDir = file("${projectDir}/tests/art")
821
822 if (androidCheckoutDir.exists()) {
823 dependsOn downloadDeps
824 artTestBuildDir.mkdirs()
825 // Ensure Jack server is running.
826 "${androidCheckoutJackServer} start-server".execute()
827 artTestDir.eachDir { dir ->
828 def name = dir.getName();
829 def markerFile = dir.toPath().resolve("info.txt").toFile();
830 if (markerFile.exists() && !(name in skippedTests)) {
831 if (!(name in skippedTestsDx)) {
832 dependsOn buildArtTest(androidCheckoutDir, artTestBuildDir, dir, DexTool.DX);
833 }
834 dependsOn buildArtTest(androidCheckoutDir, artTestBuildDir, dir, DexTool.JACK);
835 }
836 }
837 }
838 doFirst {
839 if (!androidCheckoutDir.exists()) {
840 throw new InvalidUserDataException(
841 "This task requires an Android checkout in ${androidCheckoutDir}");
842 } else if (!androidCheckoutJack.exists() ||
843 !androidCheckoutJackServer.exists()) {
844 throw new InvalidUserDataException(
845 "This task requires that tools for host testing have been build in the " +
846 "Android checkout in ${androidCheckoutDir}");
847 }
848 }
849 doLast {
850 copy {
851 from file("${androidCheckoutDir}/out/host/linux-x86/nativetest64")
852 into file("${artTestBuildDir}/lib64")
853 include 'lib*.so'
854 }
855 copy {
856 from file("${androidCheckoutDir}/out/host/linux-x86/lib64")
857 into file("${artTestBuildDir}/lib64")
858 include 'libart.so'
859 include 'libbacktrace.so'
860 include 'libbase.so'
861 include 'libc++.so'
862 include 'libcutils.so'
863 include 'liblz4.so'
864 include 'liblzma.so'
865 include 'libnativebridge.so'
866 include 'libnativeloader.so'
867 include 'libsigchain.so'
868 include 'libunwind.so'
869 include 'libziparchive.so'
870 }
871 copy {
872 from file("${androidCheckoutDir}/out/host/linux-x86/nativetest")
873 into file("${artTestBuildDir}/lib")
874 include 'lib*.so'
875 }
876 copy {
877 from file("${androidCheckoutDir}/out/host/linux-x86/lib")
878 into file("${artTestBuildDir}/lib")
879 include 'libart.so'
880 include 'libbacktrace.so'
881 include 'libbase.so'
882 include 'libc++.so'
883 include 'libcutils.so'
884 include 'liblz4.so'
885 include 'liblzma.so'
886 include 'libnativebridge.so'
887 include 'libnativeloader.so'
888 include 'libsigchain.so'
889 include 'libunwind.so'
890 include 'libziparchive.so'
891 }
892 }
893 }
894}
895
896def buildArtTest(androidCheckoutDir, artTestBuildDir, dir, dexTool) {
897 def artTestDir = file("${androidCheckoutDir}/art/test")
898 def artRunTestScript = file("${artTestDir}/run-test")
899 def dxExecutable = new File("tools/linux/dx/bin/dx");
900 def dexMergerExecutable = new File("tools/linux/dx/bin/dexmerger");
901 def dexToolName = dexTool == DexTool.DX ? "dx" : "jack"
902
903 def name = dir.getName();
904 def buildTask = "build_art_test_${dexToolName}_${name}"
905 def sanitizeTask = "sanitize_art_test_${dexToolName}_${name}"
906 def copyCheckTask = "copy_check_art_test_${dexToolName}_${name}"
907 def smaliToDexTask = "smali_to_dex_${dexToolName}_${name}"
908
909 def buildInputs = fileTree(dir: dir, include: '**/*')
910 def testDir = file("${artTestBuildDir}/${dexToolName}/${name}")
911 def outputJar = testDir.toPath().resolve("${name}.jar").toFile()
912 testDir.mkdirs()
913 if (dexTool == DexTool.DX) {
914 task "$buildTask"(type: Exec) {
915 outputs.upToDateWhen { false }
916 inputs.file buildInputs
917 executable "${artRunTestScript}"
918 args "--host"
919 args "--build-only"
920 args "--build-with-javac-dx"
921 args "--output-path", "${testDir}"
922 args "${name}"
923 environment DX: "${dxExecutable.absolutePath}"
924 environment DXMERGER: "${dexMergerExecutable.absolutePath}"
925 outputs.file outputJar
926 }
927 } else {
928 assert dexTool == DexTool.JACK
929 def javaLibs = "${androidCheckoutDir}/out/host/common/obj/JAVA_LIBRARIES"
930 def jackClasspath = "${javaLibs}/core-libart-hostdex_intermediates/classes.jack:" +
931 "${javaLibs}/core-oj-hostdex_intermediates/classes.jack"
932 task "$buildTask"(type: Exec) {
933 outputs.upToDateWhen { false }
934 inputs.file buildInputs
935 executable "${artRunTestScript}"
936 args "--host"
937 args "--build-only"
938 args "--output-path", "${testDir}"
939 args "${name}"
940 environment JACK: "${androidCheckoutDir}/out/host/linux-x86/bin/jack"
941 environment JACK_CLASSPATH: jackClasspath
942 environment DXMERGER: "${dexMergerExecutable.absolutePath}"
943 environment ANDROID_BUILD_TOP: "${androidCheckoutDir}"
944 outputs.file outputJar
945 }
946 }
947 task "${sanitizeTask}"(type: Exec, dependsOn: buildTask) {
948 outputs.upToDateWhen { false }
949 executable "/bin/bash"
950 args "-c"
951 args "rm -rf ${testDir}/smali_*.dex ${testDir}/*-ex.dex ${testDir}/*-ex.jar" +
952 " ${testDir}/classes-ex ${testDir}/check"
953 }
954
955 task "${smaliToDexTask}"(type: Exec) {
956 workingDir "${testDir}/smali"
957 executable "/bin/bash"
958 args "-c", "smali -o out.dex *.smali"
959 }
960
961 task "${copyCheckTask}"(type: Copy, dependsOn: sanitizeTask) {
962 def smali_dir = file("${dir}/smali")
963 outputs.upToDateWhen { false }
964 if (smali_dir.exists() && dexTool == DexTool.DX) {
965 dependsOn smaliToDexTask
966 }
967 from("${artTestDir}/${name}") {
968 include 'check'
969 }
970 into testDir
971 }
972
973 return copyCheckTask
974}
975
976task javadocD8(type: Javadoc) {
977 classpath = sourceSets.main.compileClasspath
978 source = sourceSets.main.allJava
979 include '**/com/android/tools/r8/BaseCommand.java'
980 include '**/com/android/tools/r8/BaseOutput.java'
981 include '**/com/android/tools/r8/CompilationException.java'
982 include '**/com/android/tools/r8/CompilationMode.java'
983 include '**/com/android/tools/r8/D8.java'
984 include '**/com/android/tools/r8/D8Command.java'
985 include '**/com/android/tools/r8/D8Output.java'
986 include '**/com/android/tools/r8/Resource.java'
987}