Infrastructure for test based benchmarks.
Bug: 210397080
Change-Id: I6938fc23a7578f100de0003656df3104646e66d9
diff --git a/build.gradle b/build.gradle
index 46548e4..eac0f5c 100644
--- a/build.gradle
+++ b/build.gradle
@@ -264,14 +264,23 @@
main11Implementation group: 'org.ow2.asm', name: 'asm-util', version: asmVersion
examplesTestNGRunnerCompile group: 'org.testng', name: 'testng', version: testngVersion
+
testCompile sourceSets.examples.output
testCompile "junit:junit:$junitVersion"
+ testCompile "com.google.guava:guava:$guavaVersion"
testCompile "org.jetbrains.kotlin:kotlin-stdlib:$kotlinVersion"
testCompile "org.jetbrains.kotlin:kotlin-reflect:$kotlinVersion"
testCompile group: 'org.smali', name: 'smali', version: smaliVersion
testCompile files('third_party/jasmin/jasmin-2.4.jar')
testCompile files('third_party/jdwp-tests/apache-harmony-jdwp-tests-host.jar')
testCompile files('third_party/ddmlib/ddmlib.jar')
+ testCompile group: 'org.ow2.asm', name: 'asm', version: asmVersion
+ testCompile group: 'org.ow2.asm', name: 'asm-commons', version: asmVersion
+ testCompile group: 'org.ow2.asm', name: 'asm-tree', version: asmVersion
+ testCompile group: 'org.ow2.asm', name: 'asm-analysis', version: asmVersion
+ testCompile group: 'org.ow2.asm', name: 'asm-util', version: asmVersion
+ testCompile group: 'it.unimi.dsi', name: 'fastutil', version: fastutilVersion
+
jctfCommonCompile "junit:junit:$junitVersion"
jctfTestsCompile "junit:junit:$junitVersion"
jctfTestsCompile sourceSets.jctfCommon.output
diff --git a/src/test/java/com/android/tools/r8/D8TestBuilder.java b/src/test/java/com/android/tools/r8/D8TestBuilder.java
index 1c74df5..cf27d33 100644
--- a/src/test/java/com/android/tools/r8/D8TestBuilder.java
+++ b/src/test/java/com/android/tools/r8/D8TestBuilder.java
@@ -5,6 +5,7 @@
import com.android.tools.r8.D8Command.Builder;
import com.android.tools.r8.TestBase.Backend;
+import com.android.tools.r8.benchmarks.BenchmarkResults;
import com.android.tools.r8.desugar.desugaredlibrary.DesugaredLibraryTestBase.KeepRuleConsumer;
import com.android.tools.r8.origin.Origin;
import com.android.tools.r8.utils.AndroidApiLevel;
@@ -64,10 +65,13 @@
@Override
D8TestCompileResult internalCompile(
- Builder builder, Consumer<InternalOptions> optionsConsumer, Supplier<AndroidApp> app)
+ Builder builder,
+ Consumer<InternalOptions> optionsConsumer,
+ Supplier<AndroidApp> app,
+ BenchmarkResults benchmarkResults)
throws CompilationFailedException {
libraryDesugaringTestConfiguration.configure(builder);
- ToolHelper.runD8(builder, optionsConsumer);
+ ToolHelper.runAndBenchmarkD8(builder, optionsConsumer, benchmarkResults);
return new D8TestCompileResult(
getState(),
app.get(),
diff --git a/src/test/java/com/android/tools/r8/DXTestBuilder.java b/src/test/java/com/android/tools/r8/DXTestBuilder.java
index 436f1e7..fcef696 100644
--- a/src/test/java/com/android/tools/r8/DXTestBuilder.java
+++ b/src/test/java/com/android/tools/r8/DXTestBuilder.java
@@ -7,6 +7,7 @@
import com.android.tools.r8.D8Command.Builder;
import com.android.tools.r8.TestBase.Backend;
import com.android.tools.r8.ToolHelper.ProcessResult;
+import com.android.tools.r8.benchmarks.BenchmarkResults;
import com.android.tools.r8.errors.Unimplemented;
import com.android.tools.r8.utils.AndroidApp;
import com.android.tools.r8.utils.InternalOptions;
@@ -44,8 +45,12 @@
@Override
DXTestCompileResult internalCompile(
- Builder builder, Consumer<InternalOptions> optionsConsumer, Supplier<AndroidApp> app)
+ Builder builder,
+ Consumer<InternalOptions> optionsConsumer,
+ Supplier<AndroidApp> app,
+ BenchmarkResults benchmarkResults)
throws CompilationFailedException {
+ assert benchmarkResults == null;
assert !libraryDesugaringTestConfiguration.isEnabled();
try {
Path dxOutputFolder = getState().getNewTempFolder();
diff --git a/src/test/java/com/android/tools/r8/ExternalR8TestBuilder.java b/src/test/java/com/android/tools/r8/ExternalR8TestBuilder.java
index 6d34346..2808f05 100644
--- a/src/test/java/com/android/tools/r8/ExternalR8TestBuilder.java
+++ b/src/test/java/com/android/tools/r8/ExternalR8TestBuilder.java
@@ -10,6 +10,7 @@
import com.android.tools.r8.R8Command.Builder;
import com.android.tools.r8.TestBase.Backend;
import com.android.tools.r8.ToolHelper.ProcessResult;
+import com.android.tools.r8.benchmarks.BenchmarkResults;
import com.android.tools.r8.errors.Unimplemented;
import com.android.tools.r8.utils.AndroidApp;
import com.android.tools.r8.utils.DescriptorUtils;
@@ -110,8 +111,12 @@
@Override
ExternalR8TestCompileResult internalCompile(
- Builder builder, Consumer<InternalOptions> optionsConsumer, Supplier<AndroidApp> app)
+ Builder builder,
+ Consumer<InternalOptions> optionsConsumer,
+ Supplier<AndroidApp> app,
+ BenchmarkResults benchmarkResults)
throws CompilationFailedException {
+ assert benchmarkResults == null;
assert !libraryDesugaringTestConfiguration.isEnabled();
try {
Path outputFolder = getState().getNewTempFolder();
diff --git a/src/test/java/com/android/tools/r8/ProguardTestBuilder.java b/src/test/java/com/android/tools/r8/ProguardTestBuilder.java
index 8d5bdea..66e5fc8 100644
--- a/src/test/java/com/android/tools/r8/ProguardTestBuilder.java
+++ b/src/test/java/com/android/tools/r8/ProguardTestBuilder.java
@@ -6,6 +6,7 @@
import com.android.tools.r8.R8Command.Builder;
import com.android.tools.r8.TestBase.Backend;
import com.android.tools.r8.ToolHelper.ProcessResult;
+import com.android.tools.r8.benchmarks.BenchmarkResults;
import com.android.tools.r8.debug.DebugTestConfig;
import com.android.tools.r8.errors.Unimplemented;
import com.android.tools.r8.utils.AndroidApp;
@@ -64,8 +65,12 @@
@Override
ProguardTestCompileResult internalCompile(
- Builder builder, Consumer<InternalOptions> optionsConsumer, Supplier<AndroidApp> app)
+ Builder builder,
+ Consumer<InternalOptions> optionsConsumer,
+ Supplier<AndroidApp> app,
+ BenchmarkResults benchmarkResults)
throws CompilationFailedException {
+ assert benchmarkResults == null;
assert !libraryDesugaringTestConfiguration.isEnabled();
try {
Path proguardOutputFolder = getState().getNewTempFolder();
diff --git a/src/test/java/com/android/tools/r8/R8TestBuilder.java b/src/test/java/com/android/tools/r8/R8TestBuilder.java
index 090ebb5..b3a595a 100644
--- a/src/test/java/com/android/tools/r8/R8TestBuilder.java
+++ b/src/test/java/com/android/tools/r8/R8TestBuilder.java
@@ -9,6 +9,7 @@
import com.android.tools.r8.R8Command.Builder;
import com.android.tools.r8.TestBase.Backend;
+import com.android.tools.r8.benchmarks.BenchmarkResults;
import com.android.tools.r8.desugar.desugaredlibrary.DesugaredLibraryTestBase.KeepRuleConsumer;
import com.android.tools.r8.dexsplitter.SplitterTestBase.RunInterface;
import com.android.tools.r8.dexsplitter.SplitterTestBase.SplitRunner;
@@ -71,7 +72,10 @@
@Override
R8TestCompileResult internalCompile(
- Builder builder, Consumer<InternalOptions> optionsConsumer, Supplier<AndroidApp> app)
+ Builder builder,
+ Consumer<InternalOptions> optionsConsumer,
+ Supplier<AndroidApp> app,
+ BenchmarkResults benchmarkResults)
throws CompilationFailedException {
if (!keepRules.isEmpty()) {
builder.addProguardConfiguration(keepRules, Origin.unknown());
@@ -118,10 +122,11 @@
ToolHelper.addSyntheticProguardRulesConsumerForTesting(
builder, rules -> box.syntheticProguardRules = rules);
libraryDesugaringTestConfiguration.configure(builder);
- ToolHelper.runR8WithoutResult(
+ ToolHelper.runAndBenchmarkR8WithoutResult(
builder.build(),
optionsConsumer.andThen(
- options -> box.proguardConfiguration = options.getProguardConfiguration()));
+ options -> box.proguardConfiguration = options.getProguardConfiguration()),
+ benchmarkResults);
R8TestCompileResult compileResult =
new R8TestCompileResult(
getState(),
diff --git a/src/test/java/com/android/tools/r8/TestCompileResult.java b/src/test/java/com/android/tools/r8/TestCompileResult.java
index 5b6996d..efe353d 100644
--- a/src/test/java/com/android/tools/r8/TestCompileResult.java
+++ b/src/test/java/com/android/tools/r8/TestCompileResult.java
@@ -18,6 +18,7 @@
import com.android.tools.r8.ToolHelper.ArtCommandBuilder;
import com.android.tools.r8.ToolHelper.DexVm;
import com.android.tools.r8.ToolHelper.ProcessResult;
+import com.android.tools.r8.benchmarks.BenchmarkResults;
import com.android.tools.r8.debug.CfDebugTestConfig;
import com.android.tools.r8.debug.DebugTestConfig;
import com.android.tools.r8.debug.DexDebugTestConfig;
@@ -25,11 +26,13 @@
import com.android.tools.r8.origin.Origin;
import com.android.tools.r8.utils.AndroidApiLevel;
import com.android.tools.r8.utils.AndroidApp;
+import com.android.tools.r8.utils.Box;
import com.android.tools.r8.utils.DescriptorUtils;
import com.android.tools.r8.utils.FileUtils;
import com.android.tools.r8.utils.InternalOptions;
import com.android.tools.r8.utils.ThrowingConsumer;
import com.android.tools.r8.utils.TriFunction;
+import com.android.tools.r8.utils.ZipUtils;
import com.android.tools.r8.utils.codeinspector.ClassSubject;
import com.android.tools.r8.utils.codeinspector.CodeInspector;
import com.google.common.collect.ImmutableList;
@@ -658,4 +661,19 @@
app.writeToZip(jarFile, OutputMode.DexIndexed);
return new Dex2OatTestRunResult(app, runtime, ToolHelper.runDex2OatRaw(jarFile, oatFile, vm));
}
+
+ public CR benchmarkCodeSize(BenchmarkResults results) throws IOException {
+ Path out = writeToZip();
+ Box<Long> size = new Box<>(0L);
+ ZipUtils.iter(
+ out,
+ (entry, stream) -> {
+ if ((getBackend().isDex() && entry.getName().endsWith(".dex"))
+ || getBackend().isCf() && entry.getName().endsWith(".class")) {
+ size.set(size.get() + entry.getSize());
+ }
+ });
+ results.addCodeSizeResult(size.get());
+ return self();
+ }
}
diff --git a/src/test/java/com/android/tools/r8/TestCompilerBuilder.java b/src/test/java/com/android/tools/r8/TestCompilerBuilder.java
index a66a4b2..be94525 100644
--- a/src/test/java/com/android/tools/r8/TestCompilerBuilder.java
+++ b/src/test/java/com/android/tools/r8/TestCompilerBuilder.java
@@ -9,6 +9,7 @@
import static org.junit.Assert.assertTrue;
import com.android.tools.r8.TestBase.Backend;
+import com.android.tools.r8.benchmarks.BenchmarkResults;
import com.android.tools.r8.debug.DebugTestConfig;
import com.android.tools.r8.desugar.desugaredlibrary.DesugaredLibraryTestBase.KeepRuleConsumer;
import com.android.tools.r8.optimize.argumentpropagation.ArgumentPropagatorEventConsumer;
@@ -110,7 +111,10 @@
}
abstract CR internalCompile(
- B builder, Consumer<InternalOptions> optionsConsumer, Supplier<AndroidApp> app)
+ B builder,
+ Consumer<InternalOptions> optionsConsumer,
+ Supplier<AndroidApp> app,
+ BenchmarkResults benchmarkResults)
throws CompilationFailedException;
public T addArgumentPropagatorCodeScannerResultInspector(
@@ -183,7 +187,19 @@
dexItemFactory, verticallyMergedClasses))));
}
+ public CR benchmarkCompile(BenchmarkResults results) throws CompilationFailedException {
+ if (System.getProperty("com.android.tools.r8.printtimes") != null) {
+ allowStdoutMessages();
+ }
+ return internalCompileAndBenchmark(results);
+ }
+
public CR compile() throws CompilationFailedException {
+ return internalCompileAndBenchmark(null);
+ }
+
+ private CR internalCompileAndBenchmark(BenchmarkResults benchmark)
+ throws CompilationFailedException {
AndroidAppConsumers sink = new AndroidAppConsumers();
builder.setProgramConsumer(sink.wrapProgramConsumer(programConsumer));
if (mainDexClassesCollector != null || mainDexListConsumer != null) {
@@ -239,7 +255,7 @@
() -> new AssertionError("Unexpected print to stderr"))));
}
cr =
- internalCompile(builder, optionsConsumer, Suppliers.memoize(sink::build))
+ internalCompile(builder, optionsConsumer, Suppliers.memoize(sink::build), benchmark)
.addRunClasspathFiles(additionalRunClassPath);
if (isAndroidBuildVersionAdded) {
cr.setSystemProperty(AndroidBuildVersion.PROPERTY, "" + builder.getMinApiLevel());
diff --git a/src/test/java/com/android/tools/r8/ToolHelper.java b/src/test/java/com/android/tools/r8/ToolHelper.java
index 57fabd1..1534edb 100644
--- a/src/test/java/com/android/tools/r8/ToolHelper.java
+++ b/src/test/java/com/android/tools/r8/ToolHelper.java
@@ -14,6 +14,7 @@
import com.android.tools.r8.TestBase.Backend;
import com.android.tools.r8.TestRuntime.CfRuntime;
import com.android.tools.r8.ToolHelper.DexVm.Kind;
+import com.android.tools.r8.benchmarks.BenchmarkResults;
import com.android.tools.r8.desugar.desugaredlibrary.jdk11.DesugaredLibraryJDK11Undesugarer;
import com.android.tools.r8.dex.ApplicationReader;
import com.android.tools.r8.errors.Unreachable;
@@ -1252,9 +1253,25 @@
public static void runR8WithoutResult(
R8Command command, Consumer<InternalOptions> optionsConsumer)
throws CompilationFailedException {
+ runAndBenchmarkR8WithoutResult(command, optionsConsumer, null);
+ }
+
+ public static void runAndBenchmarkR8WithoutResult(
+ R8Command command,
+ Consumer<InternalOptions> optionsConsumer,
+ BenchmarkResults benchmarkResults)
+ throws CompilationFailedException {
InternalOptions internalOptions = command.getInternalOptions();
optionsConsumer.accept(internalOptions);
+ long start = 0;
+ if (benchmarkResults != null) {
+ start = System.nanoTime();
+ }
R8.runForTesting(command.getInputApp(), internalOptions);
+ if (benchmarkResults != null) {
+ long end = System.nanoTime();
+ benchmarkResults.addRuntimeRawResult(end - start);
+ }
}
public static AndroidApp runR8WithFullResult(
@@ -1324,6 +1341,14 @@
public static AndroidApp runD8(
D8Command.Builder builder, Consumer<InternalOptions> optionsConsumer)
throws CompilationFailedException {
+ return runAndBenchmarkD8(builder, optionsConsumer, null);
+ }
+
+ public static AndroidApp runAndBenchmarkD8(
+ D8Command.Builder builder,
+ Consumer<InternalOptions> optionsConsumer,
+ BenchmarkResults benchmarkResults)
+ throws CompilationFailedException {
AndroidAppConsumers compatSink = new AndroidAppConsumers(builder);
D8Command command = builder.build();
InternalOptions options = command.getInternalOptions();
@@ -1331,7 +1356,15 @@
ExceptionUtils.withD8CompilationHandler(
options.reporter, () -> optionsConsumer.accept(options));
}
+ long start = 0;
+ if (benchmarkResults != null) {
+ start = System.nanoTime();
+ }
D8.runForTesting(command.getInputApp(), options);
+ if (benchmarkResults != null) {
+ long end = System.nanoTime();
+ benchmarkResults.addRuntimeRawResult(end - start);
+ }
return compatSink.build();
}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkBase.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkBase.java
new file mode 100644
index 0000000..3753d7f
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkBase.java
@@ -0,0 +1,41 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+import com.android.tools.r8.TestBase;
+import com.android.tools.r8.TestParameters;
+import java.util.List;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+
+@RunWith(Parameterized.class)
+public abstract class BenchmarkBase extends TestBase {
+
+ // Benchmarks must be configured with the "none" runtime as each config defines a singleton
+ // benchmark in golem.
+ public static List<Object[]> parametersFromConfigs(Iterable<BenchmarkConfig> configs) {
+ return buildParameters(configs, getTestParameters().withNoneRuntime().build());
+ }
+
+ private final BenchmarkConfig config;
+
+ protected BenchmarkBase(BenchmarkConfig config, TestParameters parameters) {
+ this.config = config;
+ parameters.assertNoneRuntime();
+ }
+
+ protected BenchmarkConfig getConfig() {
+ return config;
+ }
+
+ @Test
+ public void testBenchmarks() throws Exception {
+ config.run(temp);
+ }
+
+ public static BenchmarkRunner runner(BenchmarkConfig config) {
+ return BenchmarkRunner.runner(config);
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkCollection.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkCollection.java
new file mode 100644
index 0000000..f75aabe
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkCollection.java
@@ -0,0 +1,41 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+import com.android.tools.r8.benchmarks.helloworld.HelloWorldBenchmark;
+import com.android.tools.r8.errors.Unreachable;
+import java.io.IOException;
+import java.util.HashMap;
+import java.util.Map;
+
+public class BenchmarkCollection {
+
+ // Actual list of all configured benchmarks.
+ private final Map<BenchmarkIdentifier, BenchmarkConfig> benchmarks = new HashMap<>();
+
+ private void addBenchmark(BenchmarkConfig benchmark) {
+ BenchmarkIdentifier id = benchmark.getIdentifier();
+ if (benchmarks.containsKey(id)) {
+ throw new Unreachable("Duplicate definition of benchmark with name and target: " + id);
+ }
+ benchmarks.put(id, benchmark);
+ }
+
+ public BenchmarkConfig getBenchmark(BenchmarkIdentifier benchmark) {
+ return benchmarks.get(benchmark);
+ }
+
+ public static BenchmarkCollection computeCollection() {
+ BenchmarkCollection collection = new BenchmarkCollection();
+ // Every benchmark that should be active on golem must be setup in this method.
+ HelloWorldBenchmark.configs().forEach(collection::addBenchmark);
+ return collection;
+ }
+
+ /** Compute and print the golem configuration. */
+ public static void main(String[] args) throws IOException {
+ new BenchmarkCollectionPrinter(System.out)
+ .printGolemConfig(computeCollection().benchmarks.values());
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkCollectionPrinter.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkCollectionPrinter.java
new file mode 100644
index 0000000..af25a57
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkCollectionPrinter.java
@@ -0,0 +1,187 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+import static com.android.tools.r8.utils.StringUtils.join;
+
+import com.android.tools.r8.ToolHelper;
+import com.android.tools.r8.ToolHelper.ProcessResult;
+import com.android.tools.r8.errors.Unreachable;
+import com.android.tools.r8.utils.StringUtils.BraceType;
+import com.google.common.base.Strings;
+import com.google.common.hash.Hasher;
+import com.google.common.hash.Hashing;
+import java.io.IOException;
+import java.io.OutputStream;
+import java.io.PrintStream;
+import java.nio.file.Files;
+import java.nio.file.Path;
+import java.nio.file.Paths;
+import java.util.ArrayList;
+import java.util.Collection;
+import java.util.Comparator;
+
+public class BenchmarkCollectionPrinter {
+
+ private interface RunnableIO {
+ void run() throws IOException;
+ }
+
+ private static final PrintStream QUIET =
+ new PrintStream(
+ new OutputStream() {
+ @Override
+ public void write(int b) {
+ // ignore
+ }
+ });
+
+ // Internal state for printing the benchmark config for golem.
+ private int currentIndent = 0;
+ private final PrintStream out;
+
+ public BenchmarkCollectionPrinter(PrintStream out) {
+ this.out = out;
+ }
+
+ private void indentScope(RunnableIO fn) throws IOException {
+ indentScope(2, fn);
+ }
+
+ private void indentScope(int spaces, RunnableIO fn) throws IOException {
+ currentIndent += spaces;
+ fn.run();
+ currentIndent -= spaces;
+ }
+
+ private static String quote(String str) {
+ return "\"" + str + "\"";
+ }
+
+ private void print(String string) {
+ printIndented(string, currentIndent);
+ }
+
+ private void printIndented(String string, int indent) {
+ out.print(Strings.repeat(" ", indent));
+ out.println(string);
+ }
+
+ public void printGolemConfig(Collection<BenchmarkConfig> benchmarks) throws IOException {
+ Path jdkHome = getJdkHome();
+ ArrayList<BenchmarkConfig> sortedBenchmarks = new ArrayList<>(benchmarks);
+ sortedBenchmarks.sort(
+ Comparator.comparing(BenchmarkConfig::getIdentifier)
+ .thenComparing(BenchmarkConfig::getTarget));
+ print(
+ "// AUTOGENERATED FILE from"
+ + " src/test/java/com/android/tools/r8/benchmarks/BenchmarkCollection.java");
+ print("part of r8_config;");
+ print("");
+ print("createTestBenchmarks() {");
+ indentScope(
+ () -> {
+ print("final cpus = [\"Lenovo M90\"];");
+ addGolemResource("openjdk", Paths.get(jdkHome + ".tar.gz"));
+ for (BenchmarkConfig benchmark : sortedBenchmarks) {
+ printBenchmarkBlock(benchmark);
+ }
+ });
+ print("}");
+ }
+
+ private void printBenchmarkBlock(BenchmarkConfig benchmark) throws IOException {
+ print("{");
+ indentScope(
+ () -> {
+ boolean addWarmupBenchmark = benchmark.hasTimeWarmupRuns();
+ String suite = benchmark.getSuite().getDartName();
+ print("final name = " + quote(benchmark.getName()) + ";");
+ print("final targets = [" + quote(benchmark.getTarget().getGolemName()) + "];");
+ print(
+ "final metrics = "
+ + join(
+ ", ", benchmark.getMetrics(), BenchmarkMetric::getDartType, BraceType.SQUARE)
+ + ";");
+ if (addWarmupBenchmark) {
+ print("final benchmark = new GroupBenchmark(name + \"Group\", metrics);");
+ } else {
+ print("final benchmark = new StandardBenchmark(name, metrics);");
+ }
+ print("final options = benchmark.addTargets(noImplementation, targets);");
+ print("options.cpus = cpus;");
+ print("options.isScript = true;");
+ print("options.fromRevision = " + benchmark.getFromRevision() + ";");
+ print(
+ "options.mainFile = "
+ + quote(
+ "tools/run_benchmark.py --golem"
+ + " --target "
+ + benchmark.getTarget().getIdentifierName()
+ + " --benchmark "
+ + benchmark.getName()));
+ print("options.resources.add(openjdk);");
+ if (addWarmupBenchmark) {
+ print("final warmupName = " + quote(benchmark.getWarmupName()) + ";");
+ print("benchmark.addBenchmark(warmupName, [Metric.RunTimeRaw]);");
+ print("benchmark.addBenchmark(name, metrics);");
+ print(suite + ".addBenchmark(warmupName);");
+ print(suite + ".addBenchmark(name);");
+ } else {
+ print(suite + ".addBenchmark(name);");
+ }
+ });
+ print("}");
+ }
+
+ private void addGolemResource(String name, Path tarball) throws IOException {
+ Path shaFile = Paths.get(tarball.toString() + ".sha1");
+ downloadDependency(shaFile);
+ String sha256 = computeSha256(tarball);
+ String shaFileContent = getShaFileContent(shaFile);
+ print("final " + name + " = BenchmarkResource(" + quote("") + ",");
+ indentScope(
+ 4,
+ () -> {
+ print("type: BenchmarkResourceType.Storage,");
+ print("uri: " + quote("gs://r8-deps/" + shaFileContent));
+ // Make dart formatter happy.
+ if (currentIndent > 2) {
+ print("hash: ");
+ indentScope(4, () -> print(quote(sha256)));
+ } else {
+ print("hash: " + quote(sha256));
+ }
+ print("extract: " + quote("gz") + ");");
+ });
+ }
+
+ private static Path getJdkHome() throws IOException {
+ ProcessBuilder builder = new ProcessBuilder("python", "tools/jdk.py");
+ ProcessResult result = ToolHelper.runProcess(builder, QUIET);
+ if (result.exitCode != 0) {
+ throw new Unreachable("Unexpected failure to determine jdk home: " + result);
+ }
+ return Paths.get(result.stdout.trim());
+ }
+
+ private static String computeSha256(Path path) throws IOException {
+ Hasher hasher = Hashing.sha256().newHasher();
+ return hasher.putBytes(Files.readAllBytes(path)).hash().toString();
+ }
+
+ private static String getShaFileContent(Path path) throws IOException {
+ return String.join("\n", Files.readAllLines(path)).trim();
+ }
+
+ private static void downloadDependency(Path path) throws IOException {
+ ProcessBuilder builder =
+ new ProcessBuilder(
+ "download_from_google_storage", "-n", "-b", "r8-deps", "-u", "-s", path.toString());
+ ProcessResult result = ToolHelper.runProcess(builder, QUIET);
+ if (result.exitCode != 0) {
+ throw new Unreachable("Unable to download dependency '" + path + "'\n" + result);
+ }
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkConfig.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkConfig.java
new file mode 100644
index 0000000..09d7a6e
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkConfig.java
@@ -0,0 +1,168 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+import com.android.tools.r8.errors.Unreachable;
+import com.google.common.collect.ImmutableSet;
+import java.util.HashSet;
+import java.util.Set;
+import org.junit.rules.TemporaryFolder;
+
+public class BenchmarkConfig {
+
+ public static class Builder {
+
+ private String name = null;
+ private BenchmarkMethod method = null;
+ private BenchmarkTarget target = null;
+ private Set<BenchmarkMetric> metrics = new HashSet<>();
+ private BenchmarkSuite suite = BenchmarkSuite.getDefault();
+ private int fromRevision = -1;
+
+ private boolean timeWarmupRuns = false;
+
+ private Builder() {}
+
+ public BenchmarkConfig build() {
+ if (name == null) {
+ throw new Unreachable("Benchmark name must be set");
+ }
+ if (method == null) {
+ throw new Unreachable("Benchmark method must be set");
+ }
+ if (target == null) {
+ throw new Unreachable("Benchmark target must be set");
+ }
+ if (metrics.isEmpty()) {
+ throw new Unreachable("Benchmark must have at least one metric to measure");
+ }
+ if (suite == null) {
+ throw new Unreachable("Benchmark must have a suite");
+ }
+ if (fromRevision < 0) {
+ throw new Unreachable("Benchmark must specify from which golem revision it is valid");
+ }
+ if (timeWarmupRuns && !metrics.contains(BenchmarkMetric.RunTimeRaw)) {
+ throw new Unreachable("Benchmark with warmup time must measure RunTimeRaw");
+ }
+ return new BenchmarkConfig(
+ name, method, target, ImmutableSet.copyOf(metrics), suite, fromRevision, timeWarmupRuns);
+ }
+
+ public Builder setName(String name) {
+ this.name = name;
+ return this;
+ }
+
+ public Builder setTarget(BenchmarkTarget target) {
+ this.target = target;
+ return this;
+ }
+
+ public Builder setMethod(BenchmarkMethod method) {
+ this.method = method;
+ return this;
+ }
+
+ public Builder measureRunTimeRaw() {
+ metrics.add(BenchmarkMetric.RunTimeRaw);
+ return this;
+ }
+
+ public Builder measureCodeSize() {
+ metrics.add(BenchmarkMetric.CodeSize);
+ return this;
+ }
+
+ public Builder setSuite(BenchmarkSuite suite) {
+ this.suite = suite;
+ return this;
+ }
+
+ public Builder setFromRevision(int fromRevision) {
+ this.fromRevision = fromRevision;
+ return this;
+ }
+
+ public Builder timeWarmupRuns() {
+ this.timeWarmupRuns = true;
+ return this;
+ }
+ }
+
+ public static Builder builder() {
+ return new Builder();
+ }
+
+ private final BenchmarkIdentifier id;
+ private final BenchmarkMethod method;
+ private final ImmutableSet<BenchmarkMetric> metrics;
+ private final BenchmarkSuite suite;
+ private final int fromRevision;
+ private final boolean timeWarmupRuns;
+
+ private BenchmarkConfig(
+ String name,
+ BenchmarkMethod benchmarkMethod,
+ BenchmarkTarget target,
+ ImmutableSet<BenchmarkMetric> metrics,
+ BenchmarkSuite suite,
+ int fromRevision,
+ boolean timeWarmupRuns) {
+ this.id = new BenchmarkIdentifier(name, target);
+ this.method = benchmarkMethod;
+ this.metrics = metrics;
+ this.suite = suite;
+ this.fromRevision = fromRevision;
+ this.timeWarmupRuns = timeWarmupRuns;
+ }
+
+ public BenchmarkIdentifier getIdentifier() {
+ return id;
+ }
+
+ public String getName() {
+ return id.getName();
+ }
+
+ public String getWarmupName() {
+ if (!timeWarmupRuns) {
+ throw new Unreachable("Invalid attempt at getting warmup benchmark name");
+ }
+ return getName() + "Warmup";
+ }
+
+ public BenchmarkTarget getTarget() {
+ return id.getTarget();
+ }
+
+ public Set<BenchmarkMetric> getMetrics() {
+ return metrics;
+ }
+
+ public boolean hasMetric(BenchmarkMetric metric) {
+ return metrics.contains(metric);
+ }
+
+ public BenchmarkSuite getSuite() {
+ return suite;
+ }
+
+ public int getFromRevision() {
+ return fromRevision;
+ }
+
+ public boolean hasTimeWarmupRuns() {
+ return timeWarmupRuns;
+ }
+
+ public void run(TemporaryFolder temp) throws Exception {
+ method.run(this, temp);
+ }
+
+ @Override
+ public String toString() {
+ return id.getName() + "/" + id.getTarget().getIdentifierName();
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkIdentifier.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkIdentifier.java
new file mode 100644
index 0000000..f9ac65e
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkIdentifier.java
@@ -0,0 +1,59 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+import com.android.tools.r8.utils.structural.Equatable;
+import com.android.tools.r8.utils.structural.Ordered;
+import java.util.Comparator;
+import java.util.Objects;
+
+public class BenchmarkIdentifier implements Ordered<BenchmarkIdentifier> {
+
+ private final String name;
+ private final BenchmarkTarget target;
+
+ public static BenchmarkIdentifier parse(String benchmarkName, String targetIdentifier) {
+ for (BenchmarkTarget target : BenchmarkTarget.values()) {
+ if (target.getIdentifierName().equals(targetIdentifier)) {
+ return new BenchmarkIdentifier(benchmarkName, target);
+ }
+ }
+ return null;
+ }
+
+ public BenchmarkIdentifier(String name, BenchmarkTarget target) {
+ this.name = name;
+ this.target = target;
+ }
+
+ public String getName() {
+ return name;
+ }
+
+ public BenchmarkTarget getTarget() {
+ return target;
+ }
+
+ @Override
+ public int compareTo(BenchmarkIdentifier other) {
+ return Comparator.comparing(BenchmarkIdentifier::getName)
+ .thenComparing(BenchmarkIdentifier::getTarget)
+ .compare(this, other);
+ }
+
+ @Override
+ public boolean equals(Object o) {
+ return Equatable.equalsImpl(this, o);
+ }
+
+ @Override
+ public int hashCode() {
+ return Objects.hash(name, target);
+ }
+
+ @Override
+ public String toString() {
+ return "BenchmarkIdentifier{" + "name='" + name + '\'' + ", target=" + target + '}';
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMainEntryRunner.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMainEntryRunner.java
new file mode 100644
index 0000000..3e5a86a
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMainEntryRunner.java
@@ -0,0 +1,30 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+import org.junit.rules.TemporaryFolder;
+
+public class BenchmarkMainEntryRunner {
+
+ public static void main(String[] args) throws Exception {
+ if (args.length != 2) {
+ throw new RuntimeException("Invalid arguments. Expected exactly one benchmark and target");
+ }
+ String benchmarkName = args[0];
+ String targetIdentifier = args[1];
+ BenchmarkIdentifier identifier = BenchmarkIdentifier.parse(benchmarkName, targetIdentifier);
+ if (identifier == null) {
+ throw new RuntimeException("Invalid identifier identifier: " + benchmarkName);
+ }
+ BenchmarkCollection collection = BenchmarkCollection.computeCollection();
+ BenchmarkConfig config = collection.getBenchmark(identifier);
+ if (config == null) {
+ throw new RuntimeException("Unknown identifier: " + identifier);
+ }
+ TemporaryFolder temp = new TemporaryFolder();
+ temp.create();
+ config.run(temp);
+ temp.delete();
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMethod.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMethod.java
new file mode 100644
index 0000000..8c7e372
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMethod.java
@@ -0,0 +1,12 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+import org.junit.rules.TemporaryFolder;
+
+@FunctionalInterface
+public interface BenchmarkMethod {
+
+ void run(BenchmarkConfig config, TemporaryFolder temp) throws Exception;
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMetric.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMetric.java
new file mode 100644
index 0000000..9483499
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkMetric.java
@@ -0,0 +1,13 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+public enum BenchmarkMetric {
+ RunTimeRaw,
+ CodeSize;
+
+ public String getDartType() {
+ return "Metric." + name();
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkResults.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkResults.java
new file mode 100644
index 0000000..cf55b86
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkResults.java
@@ -0,0 +1,97 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+import com.android.tools.r8.benchmarks.BenchmarkRunner.ResultMode;
+import com.android.tools.r8.errors.Unreachable;
+import it.unimi.dsi.fastutil.longs.LongArrayList;
+import it.unimi.dsi.fastutil.longs.LongList;
+
+public class BenchmarkResults {
+
+ private final boolean isWarmupResults;
+ private final LongList runtimeRawResults = new LongArrayList();
+ private final LongList codeSizeResults = new LongArrayList();
+
+ public static BenchmarkResults create() {
+ return new BenchmarkResults(false);
+ }
+
+ public static BenchmarkResults createForWarmup() {
+ return new BenchmarkResults(true);
+ }
+
+ private BenchmarkResults(boolean isWarmupResults) {
+ this.isWarmupResults = isWarmupResults;
+ }
+
+ private String getName(BenchmarkConfig config) {
+ return isWarmupResults ? config.getWarmupName() : config.getName();
+ }
+
+ public void addRuntimeRawResult(long result) {
+ runtimeRawResults.add(result);
+ }
+
+ public void addCodeSizeResult(long result) {
+ codeSizeResults.add(result);
+ }
+
+ private static void verifyMetric(BenchmarkMetric metric, boolean expected, boolean actual) {
+ if (expected != actual) {
+ throw new Unreachable(
+ "Mismatched config and result for "
+ + metric.name()
+ + ". Expected by config: "
+ + expected
+ + ", but has result: "
+ + actual);
+ }
+ }
+
+ private void verifyConfigAndResults(BenchmarkConfig config) {
+ verifyMetric(
+ BenchmarkMetric.RunTimeRaw,
+ config.getMetrics().contains(BenchmarkMetric.RunTimeRaw),
+ !runtimeRawResults.isEmpty());
+ verifyMetric(
+ BenchmarkMetric.CodeSize,
+ config.getMetrics().contains(BenchmarkMetric.CodeSize),
+ !codeSizeResults.isEmpty());
+ }
+
+ public static String prettyTime(long nanoTime) {
+ return "" + (nanoTime / 1000000) + "ms";
+ }
+
+ private void printRunTimeRaw(BenchmarkConfig config, long duration) {
+ System.out.println(getName(config) + "(RunTimeRaw): " + prettyTime(duration));
+ }
+
+ private void printCodeSize(BenchmarkConfig config, long bytes) {
+ System.out.println(getName(config) + "(CodeSize): " + bytes);
+ }
+
+ public void printResults(ResultMode mode, BenchmarkConfig config) {
+ verifyConfigAndResults(config);
+ if (config.hasMetric(BenchmarkMetric.RunTimeRaw)) {
+ long sum = runtimeRawResults.stream().mapToLong(l -> l).sum();
+ if (mode == ResultMode.SUM) {
+ printRunTimeRaw(config, sum);
+ } else if (mode == ResultMode.AVERAGE) {
+ printRunTimeRaw(config, sum / runtimeRawResults.size());
+ }
+ }
+ if (!isWarmupResults && config.hasMetric(BenchmarkMetric.CodeSize)) {
+ long size = codeSizeResults.getLong(0);
+ for (int i = 1; i < codeSizeResults.size(); i++) {
+ if (size != codeSizeResults.getLong(i)) {
+ throw new Unreachable(
+ "Unexpected code size difference: " + size + " and " + codeSizeResults.getLong(i));
+ }
+ }
+ printCodeSize(config, size);
+ }
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkRunner.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkRunner.java
new file mode 100644
index 0000000..f0cb0b8
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkRunner.java
@@ -0,0 +1,92 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+public class BenchmarkRunner {
+
+ public interface BenchmarkRunnerFunction {
+ void run(BenchmarkResults results) throws Exception;
+ }
+
+ public enum ResultMode {
+ AVERAGE,
+ SUM;
+
+ @Override
+ public String toString() {
+ return name().toLowerCase();
+ }
+ }
+
+ private final BenchmarkConfig config;
+ private int warmups = 0;
+ private int iterations = 1;
+ private ResultMode resultMode = ResultMode.AVERAGE;
+
+ private BenchmarkRunner(BenchmarkConfig config) {
+ this.config = config;
+ }
+
+ public static BenchmarkRunner runner(BenchmarkConfig config) {
+ return new BenchmarkRunner(config);
+ }
+
+ public BenchmarkRunner setWarmupIterations(int iterations) {
+ this.warmups = iterations;
+ return this;
+ }
+
+ public BenchmarkRunner setBenchmarkIterations(int iterations) {
+ this.iterations = iterations;
+ return this;
+ }
+
+ public BenchmarkRunner reportResultAverage() {
+ resultMode = ResultMode.AVERAGE;
+ return this;
+ }
+
+ public BenchmarkRunner reportResultSum() {
+ resultMode = ResultMode.SUM;
+ return this;
+ }
+
+ public void run(BenchmarkRunnerFunction fn) throws Exception {
+ long warmupTotalTime = 0;
+ BenchmarkResults warmupResults = BenchmarkResults.createForWarmup();
+ if (warmups > 0) {
+ long start = System.nanoTime();
+ for (int i = 0; i < warmups; i++) {
+ fn.run(warmupResults);
+ }
+ warmupTotalTime = System.nanoTime() - start;
+ }
+ BenchmarkResults results = BenchmarkResults.create();
+ long start = System.nanoTime();
+ for (int i = 0; i < iterations; i++) {
+ fn.run(results);
+ }
+ long benchmarkTotalTime = System.nanoTime() - start;
+ System.out.println(
+ "Benchmark results for "
+ + config.getName()
+ + " on target "
+ + config.getTarget().getIdentifierName());
+ if (warmups > 0) {
+ printMetaInfo("warmup", warmups, warmupTotalTime);
+ if (config.hasTimeWarmupRuns()) {
+ warmupResults.printResults(resultMode, config);
+ }
+ }
+ printMetaInfo("benchmark", iterations, benchmarkTotalTime);
+ results.printResults(resultMode, config);
+ System.out.println();
+ }
+
+ private void printMetaInfo(String kind, int iterations, long totalTime) {
+ System.out.println(" " + kind + " reporting mode: " + resultMode);
+ System.out.println(" " + kind + " iterations: " + iterations);
+ System.out.println(" " + kind + " total time: " + BenchmarkResults.prettyTime(totalTime));
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkSuite.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkSuite.java
new file mode 100644
index 0000000..1068f72
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkSuite.java
@@ -0,0 +1,36 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+/** Enumeration of the benchmark suites on Golem. */
+public enum BenchmarkSuite {
+ R8_BENCHMARKS("R8Benchmarks", "suite"),
+ D8_BENCHMARKS("D8KeyBenchmarks", "d8KeySuite"),
+ D8_INCREMENTAL_BENCHMARKS("D8IncrementalBenchmarks", "incrementalSuite"),
+ OPENSOURCE_BENCHMARKS("OpenSourceAppDumps", "dumpsSuite"),
+ MEMORY_BENCHMARKS("R8MemoryBenchmarks", "r8MemorySuite"),
+ RETRACE_BENCHMARKS("R8RetraceBenchmarks", "r8RetraceSuite");
+
+ private final String golemName;
+ private final String dartName;
+
+ public static BenchmarkSuite getDefault() {
+ return R8_BENCHMARKS;
+ }
+
+ BenchmarkSuite(String golemName, String dartName) {
+ this.golemName = golemName;
+ this.dartName = dartName;
+ }
+
+ /** The name as shown in golems listings. */
+ public String getGolemName() {
+ return golemName;
+ }
+
+ /** The variable name used for the suite in the benchmarks.dart script. */
+ public String getDartName() {
+ return dartName;
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/BenchmarkTarget.java b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkTarget.java
new file mode 100644
index 0000000..cda7f6e
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/BenchmarkTarget.java
@@ -0,0 +1,28 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks;
+
+public enum BenchmarkTarget {
+ // Possible dashboard targets on golem.
+ D8("d8", "D8"),
+ R8_COMPAT("r8-compat", "R8"),
+ R8_NON_COMPAT("r8", "R8-full"),
+ R8_FORCE_OPT("r8-force", "R8-full-minify-optimize-shrink");
+
+ private final String idName;
+ private final String golemName;
+
+ BenchmarkTarget(String idName, String golemName) {
+ this.idName = idName;
+ this.golemName = golemName;
+ }
+
+ public String getGolemName() {
+ return golemName;
+ }
+
+ public String getIdentifierName() {
+ return idName;
+ }
+}
diff --git a/src/test/java/com/android/tools/r8/benchmarks/helloworld/HelloWorldBenchmark.java b/src/test/java/com/android/tools/r8/benchmarks/helloworld/HelloWorldBenchmark.java
new file mode 100644
index 0000000..b91e5b1
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/benchmarks/helloworld/HelloWorldBenchmark.java
@@ -0,0 +1,136 @@
+// Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+// for details. All rights reserved. Use of this source code is governed by a
+// BSD-style license that can be found in the LICENSE file.
+package com.android.tools.r8.benchmarks.helloworld;
+
+import com.android.tools.r8.TestBuilder;
+import com.android.tools.r8.TestParameters;
+import com.android.tools.r8.benchmarks.BenchmarkBase;
+import com.android.tools.r8.benchmarks.BenchmarkConfig;
+import com.android.tools.r8.benchmarks.BenchmarkMethod;
+import com.android.tools.r8.benchmarks.BenchmarkTarget;
+import com.android.tools.r8.utils.AndroidApiLevel;
+import com.android.tools.r8.utils.BooleanUtils;
+import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableList.Builder;
+import java.util.List;
+import java.util.function.Function;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+import org.junit.runners.Parameterized.Parameters;
+
+/** Example of setting up a benchmark based on the testing infrastructure. */
+@RunWith(Parameterized.class)
+public class HelloWorldBenchmark extends BenchmarkBase {
+
+ @Parameters(name = "{0}")
+ public static List<Object[]> data() {
+ return parametersFromConfigs(configs());
+ }
+
+ public HelloWorldBenchmark(BenchmarkConfig config, TestParameters parameters) {
+ super(config, parameters);
+ }
+
+ /** Static method to add benchmarks to the benchmark collection. */
+ public static List<BenchmarkConfig> configs() {
+ Builder<BenchmarkConfig> benchmarks = ImmutableList.builder();
+ makeBenchmark(BenchmarkTarget.D8, HelloWorldBenchmark::benchmarkD8, benchmarks);
+ makeBenchmark(BenchmarkTarget.R8_NON_COMPAT, HelloWorldBenchmark::benchmarkR8, benchmarks);
+ return benchmarks.build();
+ }
+
+ // Options/parameter setup to define variants of the benchmark above.
+ private static class Options {
+ final BenchmarkTarget target;
+ final Backend backend;
+ final boolean includeLibrary;
+ final AndroidApiLevel minApi = AndroidApiLevel.B;
+
+ public Options(BenchmarkTarget target, Backend backend, boolean includeLibrary) {
+ this.target = target;
+ this.backend = backend;
+ this.includeLibrary = includeLibrary;
+ }
+
+ public String getName() {
+ // The name include each non-target option for the variants to ensure unique benchmarks.
+ String backendString = backend.isCf() ? "Cf" : "Dex";
+ String libraryString = includeLibrary ? "" : "NoLib";
+ return "HelloWorld" + backendString + libraryString;
+ }
+ }
+
+ private static void makeBenchmark(
+ BenchmarkTarget target,
+ Function<Options, BenchmarkMethod> method,
+ ImmutableList.Builder<BenchmarkConfig> benchmarks) {
+ for (boolean includeLibrary : BooleanUtils.values()) {
+ for (Backend backend : Backend.values()) {
+ Options options = new Options(target, backend, includeLibrary);
+ benchmarks.add(
+ BenchmarkConfig.builder()
+ // The benchmark is required to have a unique combination of name and target.
+ .setName(options.getName())
+ .setTarget(target)
+ // The benchmark is required to have at least one metric.
+ .measureRunTimeRaw()
+ .measureCodeSize()
+ // The benchmark is required to have a runner method which defines the actual
+ // execution.
+ .setMethod(method.apply(options))
+ // The benchmark is required to set a "golem from revision".
+ // Find this value by looking at the current revision on golem.
+ .setFromRevision(11900)
+ // The benchmark can optionally time the warmup. This is not needed to use a warmup
+ // in the actual run, only to include it as its own benchmark entry on golem.
+ .timeWarmupRuns()
+ .build());
+ }
+ }
+ }
+
+ public static BenchmarkMethod benchmarkD8(Options options) {
+ return (config, temp) ->
+ runner(config)
+ .setWarmupIterations(1)
+ .setBenchmarkIterations(100)
+ .reportResultSum()
+ .run(
+ results ->
+ testForD8(temp, options.backend)
+ .setMinApi(options.minApi)
+ .applyIf(!options.includeLibrary, TestBuilder::addLibraryFiles)
+ .addProgramClasses(TestClass.class)
+ // Compile and emit RunTimeRaw measure.
+ .benchmarkCompile(results)
+ // Measure the output size.
+ .benchmarkCodeSize(results));
+ }
+
+ public static BenchmarkMethod benchmarkR8(Options options) {
+ return (config, temp) ->
+ runner(config)
+ .setWarmupIterations(1)
+ .setBenchmarkIterations(4)
+ .reportResultSum()
+ .run(
+ results ->
+ testForR8(temp, options.backend)
+ .applyIf(!options.includeLibrary, b -> b.addLibraryFiles().addDontWarn("*"))
+ .setMinApi(options.minApi)
+ .addProgramClasses(TestClass.class)
+ .addKeepMainRule(TestClass.class)
+ // Compile and emit RunTimeRaw measure.
+ .benchmarkCompile(results)
+ // Measure the output size.
+ .benchmarkCodeSize(results));
+ }
+
+ static class TestClass {
+
+ public static void main(String[] args) {
+ System.out.println("Hello world!");
+ }
+ }
+}
diff --git a/tools/as_utils.py b/tools/as_utils.py
index 43153c6..4087137 100644
--- a/tools/as_utils.py
+++ b/tools/as_utils.py
@@ -3,11 +3,11 @@
# for details. All rights reserved. Use of this source code is governed by a
# BSD-style license that can be found in the LICENSE file.
-import utils
-
-from distutils.version import LooseVersion
import os
import shutil
+from distutils.version import LooseVersion
+
+import utils
if utils.is_python3():
from html.parser import HTMLParser
@@ -174,7 +174,7 @@
shutil.rmtree(dst)
elif os.path.isfile(dst):
os.remove(dst)
- os.rename(src, dst)
+ shutil.move(src, dst)
def MoveDir(src, dst, quiet=False):
assert os.path.isdir(src)
diff --git a/tools/golem_build.py b/tools/golem_build.py
index 24a1906..c691293 100755
--- a/tools/golem_build.py
+++ b/tools/golem_build.py
@@ -5,15 +5,33 @@
# Utility script to make it easier to update what golem builds.
-import gradle
import sys
+import gradle
+import retrace_benchmark
+import run_benchmark
+import run_on_app_dump
+
GRADLE_ARGS = ['--no-daemon', '-Pno_internal']
-BUILD_TARGETS = ['R8', 'D8', 'R8Lib', 'buildExampleJars',
- 'downloadAndroidCts', 'downloadDx']
+
+LEGACY_BUILD_TARGETS = [
+ 'R8',
+ 'D8',
+ 'buildExampleJars',
+ 'downloadAndroidCts',
+ 'downloadDx']
+
+def lower(items):
+ return [ item.lower() for item in items ]
def Main():
- gradle.RunGradle(GRADLE_ARGS + BUILD_TARGETS)
+ targets = set()
+ targets.update(lower(LEGACY_BUILD_TARGETS))
+ targets.update(lower(retrace_benchmark.GOLEM_BUILD_TARGETS))
+ targets.update(lower(run_benchmark.GOLEM_BUILD_TARGETS))
+ targets.update(lower(run_on_app_dump.GOLEM_BUILD_TARGETS))
+ cmd = GRADLE_ARGS + [target for target in targets]
+ gradle.RunGradle(cmd)
if __name__ == '__main__':
sys.exit(Main())
diff --git a/tools/retrace_benchmark.py b/tools/retrace_benchmark.py
index e29bb86..42ed792 100755
--- a/tools/retrace_benchmark.py
+++ b/tools/retrace_benchmark.py
@@ -14,6 +14,7 @@
import proguard
import utils
+GOLEM_BUILD_TARGETS = ['R8Lib']
RETRACERS = ['r8', 'proguard', 'remapper']
def parse_arguments(argv):
diff --git a/tools/run_benchmark.py b/tools/run_benchmark.py
new file mode 100755
index 0000000..2ad5200
--- /dev/null
+++ b/tools/run_benchmark.py
@@ -0,0 +1,103 @@
+#!/usr/bin/env python3
+# Copyright (c) 2022, the R8 project authors. Please see the AUTHORS file
+# for details. All rights reserved. Use of this source code is governed by a
+# BSD-style license that can be found in the LICENSE file.
+
+import argparse
+import os
+import subprocess
+import sys
+
+import gradle
+import jdk
+import utils
+
+NONLIB_BUILD_TARGET = 'R8WithRelocatedDeps'
+NONLIB_TEST_BUILD_TARGETS = [utils.R8_TESTS_TARGET, utils.R8_TESTS_DEPS_TARGET]
+
+R8LIB_BUILD_TARGET = utils.R8LIB
+R8LIB_TEST_BUILD_TARGETS = [utils.R8LIB_TESTS_TARGET, utils.R8LIB_TESTS_DEPS_TARGET]
+
+# The r8lib target is always the golem target.
+GOLEM_BUILD_TARGETS = [R8LIB_BUILD_TARGET] + R8LIB_TEST_BUILD_TARGETS
+
+def get_jdk_home(options, benchmark):
+ if options.golem:
+ return os.path.join('benchmarks', benchmark, 'linux')
+ return None
+
+def parse_options(argv):
+ result = argparse.ArgumentParser(description = 'Run test-based benchmarks.')
+ result.add_argument('--golem',
+ help='Indicate this as a run on golem',
+ default=False,
+ action='store_true')
+ result.add_argument('--benchmark',
+ help='The test benchmark to run',
+ required=True)
+ result.add_argument('--target',
+ help='The test target to run',
+ required=True,
+ # These should 1:1 with BenchmarkTarget.java
+ choices=['d8', 'r8', 'r8-force', 'r8-compat'])
+ result.add_argument('--nolib', '--no-lib', '--no-r8lib',
+ help='Run the non-lib R8 build (default false)',
+ default=False,
+ action='store_true')
+ result.add_argument('--no-build', '--no_build',
+ help='Run without building first (default false)',
+ default=False,
+ action='store_true')
+ result.add_argument('--enable-assertions', '--enable_assertions', '-ea',
+ help='Enable assertions when running',
+ default=False,
+ action='store_true')
+ result.add_argument('--print-times',
+ help='Print timing information from r8',
+ default=False,
+ action='store_true')
+ result.add_argument('--temp',
+ help='A directory to use for temporaries and outputs.',
+ default=None)
+ return result.parse_known_args(argv)
+
+def main(argv):
+ (options, args) = parse_options(argv)
+
+ if options.golem:
+ options.no_build = True
+ if options.nolib:
+ print("Error: golem should always run r8lib")
+ return 1
+
+ if options.nolib:
+ buildTargets = [NONLIB_BUILD_TARGET] + NONLIB_TEST_BUILD_TARGETS
+ r8jar = utils.R8_WITH_RELOCATED_DEPS_JAR
+ testjars = [utils.R8_TESTS_DEPS_JAR, utils.R8_TESTS_JAR]
+ else:
+ buildTargets = GOLEM_BUILD_TARGETS
+ r8jar = utils.R8LIB_JAR
+ testjars = [utils.R8LIB_TESTS_DEPS_JAR, utils.R8LIB_TESTS_JAR]
+
+ if not options.no_build:
+ gradle.RunGradle(buildTargets + ['-Pno_internal'])
+
+ return run(options, r8jar, testjars)
+
+def run(options, r8jar, testjars):
+ jdkhome = get_jdk_home(options, options.benchmark)
+ cmd = [jdk.GetJavaExecutable(jdkhome)]
+ if options.enable_assertions:
+ cmd.append('-ea')
+ if options.print_times:
+ cmd.append('-Dcom.android.tools.r8.printtimes=1')
+ cmd.extend(['-cp', ':'.join([r8jar] + testjars)])
+ cmd.extend([
+ 'com.android.tools.r8.benchmarks.BenchmarkMainEntryRunner',
+ options.benchmark,
+ options.target,
+ ])
+ return subprocess.check_call(cmd)
+
+if __name__ == '__main__':
+ sys.exit(main(sys.argv[1:]))
diff --git a/tools/run_on_app_dump.py b/tools/run_on_app_dump.py
index c55a6f7..6035752 100755
--- a/tools/run_on_app_dump.py
+++ b/tools/run_on_app_dump.py
@@ -3,24 +3,25 @@
# for details. All rights reserved. Use of this source code is governed by a
# BSD-style license that can be found in the LICENSE file.
+import argparse
+import hashlib
+import os
+import shutil
+import sys
+import time
+import zipfile
+from datetime import datetime
+
import adb
import apk_masseur
import as_utils
import compiledump
import gradle
-import hashlib
import jdk
-import argparse
-import os
-import shutil
-import sys
-import time
import update_prebuilds_in_android
import utils
-import zipfile
-from datetime import datetime
-
+GOLEM_BUILD_TARGETS = ['R8Lib', 'R8Retrace']
SHRINKERS = ['r8', 'r8-full', 'r8-nolib', 'r8-nolib-full']
class AttrDict(dict):
@@ -1060,9 +1061,7 @@
print_indented('final targetsFull = ["R8-full-minify-optimize-shrink"];', 2)
# Avoid calculating this for every app
jdk_gz = jdk.GetJdkHome() + '.tar.gz'
- download_sha(jdk_gz + '.sha1', False, quiet=True)
- jdk_sha256 = get_sha256(jdk_gz)
- add_golem_resource(2, jdk_gz, 'openjdk', sha256=jdk_sha256)
+ add_golem_resource(2, jdk_gz, 'openjdk')
for app in options.apps:
if app.folder and not app.internal:
indentation = 2;
@@ -1171,7 +1170,7 @@
os.path.join(temp_dir, target), os.path.join(temp_dir, 'r8lib.jar'),
quiet=options.quiet)
elif options.version == 'main':
- if not (options.no_build or options.golem):
+ if not options.no_build:
gradle.RunGradle(['R8Retrace', 'r8', '-Pno_internal'])
build_r8lib = False
for shrinker in options.shrinker:
diff --git a/tools/utils.py b/tools/utils.py
index 314fcd2..eb7849b 100644
--- a/tools/utils.py
+++ b/tools/utils.py
@@ -45,10 +45,15 @@
R8RETRACE_NO_DEPS = 'R8RetraceNoDeps'
R8_SRC = 'sourceJar'
LIBRARY_DESUGAR_CONVERSIONS = 'buildLibraryDesugarConversions'
+R8_TESTS_TARGET = 'TestJar'
+R8_TESTS_DEPS_TARGET = 'RepackageTestDeps'
+R8LIB_TESTS_TARGET = 'configureTestForR8Lib'
+R8LIB_TESTS_DEPS_TARGET = R8_TESTS_DEPS_TARGET
ALL_DEPS_JAR = os.path.join(LIBS, 'deps_all.jar')
D8_JAR = os.path.join(LIBS, 'd8.jar')
R8_JAR = os.path.join(LIBS, 'r8.jar')
+R8_WITH_RELOCATED_DEPS_JAR = os.path.join(LIBS, 'r8_with_relocated_deps.jar')
R8LIB_JAR = os.path.join(LIBS, 'r8lib.jar')
R8LIB_MAP = os.path.join(LIBS, 'r8lib.jar.map')
R8_SRC_JAR = os.path.join(LIBS, 'r8-src.jar')
@@ -56,6 +61,10 @@
R8_FULL_EXCLUDE_DEPS_JAR = os.path.join(LIBS, 'r8-full-exclude-deps.jar')
R8RETRACE_JAR = os.path.join(LIBS, 'r8retrace.jar')
R8RETRACE_EXCLUDE_DEPS_JAR = os.path.join(LIBS, 'r8retrace-exclude-deps.jar')
+R8_TESTS_JAR = os.path.join(LIBS, 'r8tests.jar')
+R8LIB_TESTS_JAR = os.path.join(LIBS, 'r8libtestdeps-cf.jar')
+R8_TESTS_DEPS_JAR = os.path.join(LIBS, 'test_deps_all.jar')
+R8LIB_TESTS_DEPS_JAR = R8_TESTS_DEPS_JAR
MAVEN_ZIP = os.path.join(LIBS, 'r8.zip')
MAVEN_ZIP_LIB = os.path.join(LIBS, 'r8lib.zip')
LIBRARY_DESUGAR_CONVERSIONS_ZIP = os.path.join(LIBS, 'library_desugar_conversions.zip')