blob: 10c64e3085cfa80c7c1381422bc4c2b694097731 [file] [log] [blame]
// Copyright (c) 2018, 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.shaking.forceproguardcompatibility;
import static com.android.tools.r8.utils.DexInspectorMatchers.isPresent;
import static org.hamcrest.CoreMatchers.not;
import static org.junit.Assert.assertThat;
import static org.junit.Assert.fail;
import com.android.tools.r8.CompatProguardCommandBuilder;
import com.android.tools.r8.DexIndexedConsumer;
import com.android.tools.r8.R8Command;
import com.android.tools.r8.TestBase;
import com.android.tools.r8.ToolHelper;
import com.android.tools.r8.ToolHelper.ProcessResult;
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.DexInspector;
import com.android.tools.r8.utils.DexInspector.ClassSubject;
import com.android.tools.r8.utils.FileUtils;
import com.android.tools.r8.utils.InternalOptions;
import com.google.common.collect.ImmutableList;
import java.io.File;
import java.nio.file.Path;
import java.util.List;
import java.util.function.Consumer;
public class ProguardCompatabilityTestBase extends TestBase {
protected Path proguardMap;
public enum Shrinker {
PROGUARD5,
PROGUARD6,
PROGUARD6_THEN_D8,
R8_COMPAT,
R8
}
protected static boolean isR8(Shrinker shrinker) {
return shrinker == Shrinker.R8_COMPAT || shrinker == Shrinker.R8;
}
protected AndroidApp runShrinkerRaw(
Shrinker mode, List<Class> programClasses, List<String> proguadConfigs) throws Exception {
return runShrinkerRaw(
mode, programClasses, String.join(System.lineSeparator(), proguadConfigs));
}
protected AndroidApp runShrinkerRaw(
Shrinker mode, List<Class> programClasses, String proguardConfig) throws Exception {
proguardMap = File.createTempFile("proguard", ".map", temp.getRoot()).toPath();
switch (mode) {
case PROGUARD5:
return runProguard5Raw(programClasses, proguardConfig, proguardMap);
case PROGUARD6:
return runProguard6Raw(programClasses, proguardConfig, proguardMap);
case PROGUARD6_THEN_D8:
return runProguard6AndD8Raw(programClasses, proguardConfig, proguardMap);
case R8_COMPAT:
return runR8CompatRaw(programClasses, proguardConfig);
case R8:
return runR8Raw(programClasses, proguardConfig);
}
throw new IllegalArgumentException("Unknown shrinker: " + mode);
}
protected DexInspector runShrinker(
Shrinker mode, List<Class> programClasses, List<String> proguadConfigs) throws Exception {
return runShrinker(mode, programClasses, String.join(System.lineSeparator(), proguadConfigs));
}
protected DexInspector runShrinker(
Shrinker mode, List<Class> programClasses, String proguardConfig) throws Exception {
switch (mode) {
case PROGUARD5:
return runProguard5(programClasses, proguardConfig);
case PROGUARD6:
return runProguard6(programClasses, proguardConfig);
case PROGUARD6_THEN_D8:
return runProguard6AndD8(programClasses, proguardConfig);
case R8_COMPAT:
return runR8Compat(programClasses, proguardConfig);
case R8:
return runR8(programClasses, proguardConfig);
}
throw new IllegalArgumentException("Unknown shrinker: " + mode);
}
protected AndroidApp runR8Raw(List<Class> programClasses, String proguardConfig)
throws Exception {
return runR8Raw(programClasses, proguardConfig, null);
}
protected AndroidApp runR8Raw(
List<Class> programClasses, String proguardConfig, Consumer<InternalOptions> configure)
throws Exception {
AndroidApp app = readClassesAndAndriodJar(programClasses);
R8Command.Builder builder = ToolHelper.prepareR8CommandBuilder(app);
builder.addProguardConfiguration(ImmutableList.of(proguardConfig), Origin.unknown());
return ToolHelper.runR8(builder.build(), configure);
}
protected DexInspector runR8(List<Class> programClasses, String proguardConfig) throws Exception {
return new DexInspector(runR8Raw(programClasses, proguardConfig));
}
protected AndroidApp runR8CompatRaw(
List<Class> programClasses, String proguardConfig) throws Exception {
CompatProguardCommandBuilder builder = new CompatProguardCommandBuilder(true);
builder.addProguardConfiguration(ImmutableList.of(proguardConfig), Origin.unknown());
programClasses.forEach(
clazz -> builder.addProgramFiles(ToolHelper.getClassFileForTestClass(clazz)));
builder.addLibraryFiles(ToolHelper.getDefaultAndroidJar());
builder.setProgramConsumer(DexIndexedConsumer.emptyConsumer());
return ToolHelper.runR8(builder.build());
}
protected DexInspector runR8Compat(
List<Class> programClasses, String proguardConfig) throws Exception {
return new DexInspector(runR8CompatRaw(programClasses, proguardConfig));
}
protected DexInspector runR8CompatKeepingMain(Class mainClass, List<Class> programClasses)
throws Exception {
return runR8Compat(programClasses, keepMainProguardConfiguration(mainClass));
}
protected AndroidApp runProguard5Raw(
List<Class> programClasses, String proguardConfig, Path proguardMap) throws Exception {
Path proguardedJar =
File.createTempFile("proguarded", FileUtils.JAR_EXTENSION, temp.getRoot()).toPath();
Path proguardConfigFile = File.createTempFile("proguard", ".config", temp.getRoot()).toPath();
FileUtils.writeTextFile(proguardConfigFile, proguardConfig);
ProcessResult result = ToolHelper.runProguardRaw(
jarTestClasses(programClasses),
proguardedJar,
ToolHelper.getAndroidJar(AndroidApiLevel.N),
proguardConfigFile,
proguardMap);
if (result.exitCode != 0) {
fail("Proguard failed, exit code " + result.exitCode + ", stderr:\n" + result.stderr);
}
return readJar(proguardedJar);
}
protected DexInspector runProguard5(
List<Class> programClasses, String proguardConfig) throws Exception {
proguardMap = File.createTempFile("proguard", ".map", temp.getRoot()).toPath();
return new DexInspector(
runProguard5Raw(programClasses, proguardConfig, proguardMap), proguardMap);
}
protected AndroidApp runProguard6Raw(
List<Class> programClasses, String proguardConfig, Path proguardMap) throws Exception {
Path proguardedJar =
File.createTempFile("proguarded", FileUtils.JAR_EXTENSION, temp.getRoot()).toPath();
Path proguardConfigFile = File.createTempFile("proguard", ".config", temp.getRoot()).toPath();
FileUtils.writeTextFile(proguardConfigFile, proguardConfig);
ProcessResult result = ToolHelper.runProguard6Raw(
jarTestClasses(programClasses),
proguardedJar,
ToolHelper.getAndroidJar(AndroidApiLevel.N),
proguardConfigFile,
proguardMap);
if (result.exitCode != 0) {
fail("Proguard failed, exit code " + result.exitCode + ", stderr:\n" + result.stderr);
}
return readJar(proguardedJar);
}
protected DexInspector runProguard6(
List<Class> programClasses, String proguardConfig) throws Exception {
proguardMap = File.createTempFile("proguard", ".map", temp.getRoot()).toPath();
return new DexInspector(
runProguard6Raw(programClasses, proguardConfig, proguardMap), proguardMap);
}
protected AndroidApp runProguard6AndD8Raw(
List<Class> programClasses, String proguardConfig, Path proguardMap) throws Exception {
Path proguardedJar =
File.createTempFile("proguarded", FileUtils.JAR_EXTENSION, temp.getRoot()).toPath();
Path proguardConfigFile = File.createTempFile("proguard", ".config", temp.getRoot()).toPath();
FileUtils.writeTextFile(proguardConfigFile, proguardConfig);
ProcessResult result = ToolHelper.runProguard6Raw(
jarTestClasses(programClasses),
proguardedJar,
ToolHelper.getAndroidJar(AndroidApiLevel.N),
proguardConfigFile,
proguardMap);
if (result.exitCode != 0) {
fail("Proguard failed, exit code " + result.exitCode + ", stderr:\n" + result.stderr);
}
return ToolHelper.runD8(readJar(proguardedJar));
}
protected DexInspector runProguard6AndD8(
List<Class> programClasses, String proguardConfig) throws Exception {
proguardMap = File.createTempFile("proguard", ".map", temp.getRoot()).toPath();
return new DexInspector(
runProguard6AndD8Raw(programClasses, proguardConfig, proguardMap), proguardMap);
}
protected DexInspector runProguardKeepingMain(Class mainClass, List<Class> programClasses)
throws Exception {
return runProguard6AndD8(programClasses, keepMainProguardConfiguration(mainClass));
}
protected void verifyClassesPresent(
DexInspector dexInspector, Class<?>... classesOfInterest) {
for (Class klass : classesOfInterest) {
ClassSubject c = dexInspector.clazz(klass);
assertThat(c, isPresent());
}
}
protected void verifyClassesAbsent(
DexInspector dexInspector, Class<?>... classesOfInterest) {
for (Class klass : classesOfInterest) {
ClassSubject c = dexInspector.clazz(klass);
assertThat(c, not(isPresent()));
}
}
}