Merge "Update tools/build_aosp.py"
diff --git a/src/main/java/com/android/tools/r8/BaseCompilerCommand.java b/src/main/java/com/android/tools/r8/BaseCompilerCommand.java
index 5e8f975..7645242 100644
--- a/src/main/java/com/android/tools/r8/BaseCompilerCommand.java
+++ b/src/main/java/com/android/tools/r8/BaseCompilerCommand.java
@@ -136,7 +136,7 @@
protected void validate() throws CompilationException {
super.validate();
- if (getAppBuilder().hasMainDexList() && outputMode == OutputMode.FilePerClass) {
+ if (getAppBuilder().hasMainDexList() && outputMode == OutputMode.FilePerInputClass) {
throw new CompilationException(
"Option --main-dex-list cannot be used with --file-per-class");
}
diff --git a/src/main/java/com/android/tools/r8/D8Command.java b/src/main/java/com/android/tools/r8/D8Command.java
index 76779bb..f2aebc7 100644
--- a/src/main/java/com/android/tools/r8/D8Command.java
+++ b/src/main/java/com/android/tools/r8/D8Command.java
@@ -158,7 +158,7 @@
builder.setMode(CompilationMode.RELEASE);
modeSet = CompilationMode.RELEASE;
} else if (arg.equals("--file-per-class")) {
- builder.setOutputMode(OutputMode.FilePerClass);
+ builder.setOutputMode(OutputMode.FilePerInputClass);
} else if (arg.equals("--output")) {
String output = args[++i];
if (outputPath != null) {
diff --git a/src/main/java/com/android/tools/r8/dex/ApplicationWriter.java b/src/main/java/com/android/tools/r8/dex/ApplicationWriter.java
index 46ee85c..58c3a5b 100644
--- a/src/main/java/com/android/tools/r8/dex/ApplicationWriter.java
+++ b/src/main/java/com/android/tools/r8/dex/ApplicationWriter.java
@@ -138,10 +138,10 @@
// Distribute classes into dex files.
VirtualFile.Distributor distributor = null;
- if (options.outputMode == OutputMode.FilePerClass) {
+ if (options.outputMode == OutputMode.FilePerInputClass) {
assert packageDistribution == null :
"Cannot combine package distribution definition with file-per-class option.";
- distributor = new VirtualFile.FilePerClassDistributor(this);
+ distributor = new VirtualFile.FilePerInputClassDistributor(this);
} else if (!options.canUseMultidex()
&& options.mainDexKeepRules.isEmpty()
&& application.mainDexList.isEmpty()) {
@@ -178,7 +178,15 @@
AndroidApp.Builder builder = AndroidApp.builder();
try {
for (Map.Entry<VirtualFile, Future<byte[]>> entry : dexDataFutures.entrySet()) {
- builder.addDexProgramData(entry.getValue().get(), entry.getKey().getClassDescriptors());
+ VirtualFile virtualFile = entry.getKey();
+ if (virtualFile.getPrimaryClassDescriptor() != null) {
+ builder.addDexProgramData(
+ entry.getValue().get(),
+ virtualFile.getClassDescriptors(),
+ virtualFile.getPrimaryClassDescriptor());
+ } else {
+ builder.addDexProgramData(entry.getValue().get(), virtualFile.getClassDescriptors());
+ }
}
} catch (InterruptedException e) {
throw new RuntimeException("Interrupted while waiting for future.", e);
diff --git a/src/main/java/com/android/tools/r8/dex/FileWriter.java b/src/main/java/com/android/tools/r8/dex/FileWriter.java
index d820747..1e70b9b 100644
--- a/src/main/java/com/android/tools/r8/dex/FileWriter.java
+++ b/src/main/java/com/android/tools/r8/dex/FileWriter.java
@@ -617,7 +617,7 @@
}
}
- private void writeEncodedMethods(DexEncodedMethod[] methods) {
+ private void writeEncodedMethods(DexEncodedMethod[] methods, boolean clearBodies) {
assert isSorted(methods);
int currentOffset = 0;
for (DexEncodedMethod method : methods) {
@@ -633,7 +633,9 @@
dest.putUleb128(mixedSectionOffsets.getOffsetFor(method.getCode().asDexCode()));
// Writing the methods starts to take up memory so we are going to flush the
// code objects since they are no longer necessary after this.
- method.removeCode();
+ if (clearBodies) {
+ method.removeCode();
+ }
}
}
}
@@ -647,8 +649,10 @@
dest.putUleb128(clazz.virtualMethods().length);
writeEncodedFields(clazz.staticFields());
writeEncodedFields(clazz.instanceFields());
- writeEncodedMethods(clazz.directMethods());
- writeEncodedMethods(clazz.virtualMethods());
+
+ boolean isSharedSynthetic = clazz.getSynthesizedFrom().size() > 1;
+ writeEncodedMethods(clazz.directMethods(), !isSharedSynthetic);
+ writeEncodedMethods(clazz.virtualMethods(), !isSharedSynthetic);
}
private void addStaticFieldValues(DexProgramClass clazz) {
diff --git a/src/main/java/com/android/tools/r8/dex/VirtualFile.java b/src/main/java/com/android/tools/r8/dex/VirtualFile.java
index c40d5a5..32fb175 100644
--- a/src/main/java/com/android/tools/r8/dex/VirtualFile.java
+++ b/src/main/java/com/android/tools/r8/dex/VirtualFile.java
@@ -4,8 +4,8 @@
package com.android.tools.r8.dex;
import com.android.tools.r8.errors.CompilationError;
-import com.android.tools.r8.errors.InternalCompilerError;
import com.android.tools.r8.errors.DexOverflowException;
+import com.android.tools.r8.errors.InternalCompilerError;
import com.android.tools.r8.graph.DexApplication;
import com.android.tools.r8.graph.DexCallSite;
import com.android.tools.r8.graph.DexClass;
@@ -76,10 +76,17 @@
private final VirtualFileIndexedItemCollection indexedItems;
private final IndexedItemTransaction transaction;
+ private final DexProgramClass primaryClass;
+
private VirtualFile(int id, NamingLens namingLens) {
+ this(id, namingLens, null);
+ }
+
+ private VirtualFile(int id, NamingLens namingLens, DexProgramClass primaryClass) {
this.id = id;
this.indexedItems = new VirtualFileIndexedItemCollection(id);
this.transaction = new IndexedItemTransaction(indexedItems, namingLens);
+ this.primaryClass = primaryClass;
}
public int getId() {
@@ -95,6 +102,10 @@
return classDescriptors;
}
+ public String getPrimaryClassDescriptor() {
+ return primaryClass == null ? null : primaryClass.type.descriptor.toString();
+ }
+
public static String deriveCommonPrefixAndSanityCheck(List<String> fileNames) {
Iterator<String> nameIterator = fileNames.iterator();
String first = nameIterator.next();
@@ -221,21 +232,40 @@
throws ExecutionException, IOException, DexOverflowException;
}
- public static class FilePerClassDistributor extends Distributor {
+ /**
+ * Distribute each type to its individual virtual except for types synthesized during this
+ * compilation. Synthesized classes are emitted in the individual virtual files
+ * of the input classes they were generated from. Shared synthetic classes
+ * may then be distributed in several individual virtual files.
+ */
+ public static class FilePerInputClassDistributor extends Distributor {
- FilePerClassDistributor(ApplicationWriter writer) {
+ FilePerInputClassDistributor(ApplicationWriter writer) {
super(writer);
}
@Override
public Map<Integer, VirtualFile> run() {
+ HashMap<DexProgramClass, VirtualFile> files = new HashMap<>();
+ Collection<DexProgramClass> synthetics = new ArrayList<>();
// Assign dedicated virtual files for all program classes.
for (DexProgramClass clazz : application.classes()) {
- VirtualFile file = new VirtualFile(nameToFileMap.size(), writer.namingLens);
- nameToFileMap.put(nameToFileMap.size(), file);
- file.addClass(clazz);
- file.commitTransaction();
+ if (clazz.getSynthesizedFrom().isEmpty()) {
+ VirtualFile file = new VirtualFile(nameToFileMap.size(), writer.namingLens, clazz);
+ nameToFileMap.put(nameToFileMap.size(), file);
+ file.addClass(clazz);
+ files.put(clazz, file);
+ } else {
+ synthetics.add(clazz);
+ }
}
+ for (DexProgramClass synthetic : synthetics) {
+ for (DexProgramClass inputType : synthetic.getSynthesizedFrom()) {
+ VirtualFile file = files.get(inputType);
+ file.addClass(synthetic);
+ }
+ }
+ files.values().forEach(file -> file.commitTransaction());
return nameToFileMap;
}
}
diff --git a/src/main/java/com/android/tools/r8/graph/DexProgramClass.java b/src/main/java/com/android/tools/r8/graph/DexProgramClass.java
index 1bdc56c..d27a3be 100644
--- a/src/main/java/com/android/tools/r8/graph/DexProgramClass.java
+++ b/src/main/java/com/android/tools/r8/graph/DexProgramClass.java
@@ -7,11 +7,16 @@
import com.android.tools.r8.dex.IndexedItemCollection;
import com.android.tools.r8.dex.MixedSectionCollection;
import java.util.Arrays;
+import java.util.Collection;
+import java.util.Collections;
+import java.util.HashSet;
+import java.util.Set;
import java.util.function.Supplier;
public class DexProgramClass extends DexClass implements Supplier<DexProgramClass> {
private DexEncodedArray staticValues;
+ private final Collection<DexProgramClass> synthesizedFrom;
public DexProgramClass(DexType type,
Resource.Kind origin,
@@ -24,9 +29,36 @@
DexEncodedField[] instanceFields,
DexEncodedMethod[] directMethods,
DexEncodedMethod[] virtualMethods) {
+ this(type,
+ origin,
+ accessFlags,
+ superType,
+ interfaces,
+ sourceFile,
+ classAnnotations,
+ staticFields,
+ instanceFields,
+ directMethods,
+ virtualMethods,
+ Collections.emptyList());
+ }
+
+ public DexProgramClass(DexType type,
+ Resource.Kind origin,
+ DexAccessFlags accessFlags,
+ DexType superType,
+ DexTypeList interfaces,
+ DexString sourceFile,
+ DexAnnotationSet classAnnotations,
+ DexEncodedField[] staticFields,
+ DexEncodedField[] instanceFields,
+ DexEncodedMethod[] directMethods,
+ DexEncodedMethod[] virtualMethods,
+ Collection<DexProgramClass> synthesizedDirectlyFrom) {
super(sourceFile, interfaces, accessFlags, superType, type, staticFields,
instanceFields, directMethods, virtualMethods, classAnnotations, origin);
assert classAnnotations != null;
+ this.synthesizedFrom = accumulateSynthesizedFrom(new HashSet<>(), synthesizedDirectlyFrom);
}
@Override
@@ -54,6 +86,10 @@
}
}
+ public Collection<DexProgramClass> getSynthesizedFrom() {
+ return synthesizedFrom;
+ }
+
@Override
void collectMixedSectionItems(MixedSectionCollection mixedItems) {
if (hasAnnotations()) {
@@ -129,6 +165,19 @@
return methods != null && Arrays.stream(methods).anyMatch(DexEncodedMethod::hasAnnotation);
}
+ private static Collection<DexProgramClass> accumulateSynthesizedFrom(
+ Set<DexProgramClass> accumulated,
+ Collection<DexProgramClass> toAccumulate) {
+ for (DexProgramClass dexProgramClass : toAccumulate) {
+ if (dexProgramClass.synthesizedFrom.isEmpty()) {
+ accumulated.add(dexProgramClass);
+ } else {
+ accumulateSynthesizedFrom(accumulated, dexProgramClass.synthesizedFrom);
+ }
+ }
+ return accumulated;
+ }
+
public void setStaticValues(DexEncodedArray staticValues) {
this.staticValues = staticValues;
}
diff --git a/src/main/java/com/android/tools/r8/ir/desugar/InterfaceMethodRewriter.java b/src/main/java/com/android/tools/r8/ir/desugar/InterfaceMethodRewriter.java
index aa93aa8..3d7d254 100644
--- a/src/main/java/com/android/tools/r8/ir/desugar/InterfaceMethodRewriter.java
+++ b/src/main/java/com/android/tools/r8/ir/desugar/InterfaceMethodRewriter.java
@@ -60,7 +60,8 @@
// forward the call to an appropriate method in interface companion class.
//
public final class InterfaceMethodRewriter {
- private static final String COMPANION_CLASS_NAME_SUFFIX = "-CC";
+ // public for testing
+ public static final String COMPANION_CLASS_NAME_SUFFIX = "-CC";
private static final String DEFAULT_METHOD_PREFIX = "$default$";
private final IRConverter converter;
diff --git a/src/main/java/com/android/tools/r8/ir/desugar/InterfaceProcessor.java b/src/main/java/com/android/tools/r8/ir/desugar/InterfaceProcessor.java
index 2d06abe..02d6dc9 100644
--- a/src/main/java/com/android/tools/r8/ir/desugar/InterfaceProcessor.java
+++ b/src/main/java/com/android/tools/r8/ir/desugar/InterfaceProcessor.java
@@ -17,6 +17,7 @@
import com.android.tools.r8.graph.DexType;
import com.android.tools.r8.graph.DexTypeList;
import java.util.ArrayList;
+import java.util.Collections;
import java.util.IdentityHashMap;
import java.util.List;
import java.util.Map;
@@ -135,7 +136,8 @@
DexEncodedField.EMPTY_ARRAY,
DexEncodedField.EMPTY_ARRAY,
companionMethods.toArray(new DexEncodedMethod[companionMethods.size()]),
- DexEncodedMethod.EMPTY_ARRAY
+ DexEncodedMethod.EMPTY_ARRAY,
+ Collections.singletonList(iface)
);
companionClasses.put(iface, companionClass);
}
diff --git a/src/main/java/com/android/tools/r8/ir/desugar/LambdaClass.java b/src/main/java/com/android/tools/r8/ir/desugar/LambdaClass.java
index f03a619..0fe0f99 100644
--- a/src/main/java/com/android/tools/r8/ir/desugar/LambdaClass.java
+++ b/src/main/java/com/android/tools/r8/ir/desugar/LambdaClass.java
@@ -27,6 +27,8 @@
import com.android.tools.r8.graph.DexValue.DexValueNull;
import com.android.tools.r8.ir.code.Invoke;
import com.android.tools.r8.ir.synthetic.SynthesizedCode;
+import java.util.ArrayList;
+import java.util.Collection;
import java.util.List;
import java.util.concurrent.atomic.AtomicBoolean;
@@ -58,6 +60,7 @@
final DexField instanceField;
final Target target;
final AtomicBoolean addToMainDexList = new AtomicBoolean(false);
+ private Collection<DexProgramClass> synthesizedFrom = new ArrayList<DexProgramClass>(1);
LambdaClass(LambdaRewriter rewriter, DexType accessedFrom,
DexType lambdaClassType, LambdaDescriptor descriptor) {
@@ -125,7 +128,8 @@
synthesizeStaticFields(),
synthesizeInstanceFields(),
synthesizeDirectMethods(),
- synthesizeVirtualMethods()
+ synthesizeVirtualMethods(),
+ synthesizedFrom
);
}
@@ -138,6 +142,11 @@
return descriptor.isStateless();
}
+ synchronized void addSynthesizedFrom(DexProgramClass synthesizedFrom) {
+ assert synthesizedFrom != null;
+ this.synthesizedFrom.add(synthesizedFrom);
+ }
+
// Synthesize virtual methods.
private DexEncodedMethod[] synthesizeVirtualMethods() {
DexEncodedMethod[] methods = new DexEncodedMethod[1 + descriptor.bridges.size()];
diff --git a/src/main/java/com/android/tools/r8/ir/desugar/LambdaRewriter.java b/src/main/java/com/android/tools/r8/ir/desugar/LambdaRewriter.java
index 86ad391..c31675f 100644
--- a/src/main/java/com/android/tools/r8/ir/desugar/LambdaRewriter.java
+++ b/src/main/java/com/android/tools/r8/ir/desugar/LambdaRewriter.java
@@ -54,7 +54,8 @@
private static final String METAFACTORY_ALT_METHOD_NAME = "altMetafactory";
private static final String DESERIALIZE_LAMBDA_METHOD_NAME = "$deserializeLambda$";
- static final String LAMBDA_CLASS_NAME_PREFIX = "-$$Lambda$";
+ // public for testing
+ public static final String LAMBDA_CLASS_NAME_PREFIX = "-$$Lambda$";
static final String EXPECTED_LAMBDA_METHOD_PREFIX = "lambda$";
static final String LAMBDA_INSTANCE_FIELD_NAME = "INSTANCE";
@@ -238,6 +239,7 @@
lambdaClass = putIfAbsent(knownLambdaClasses, lambdaClassType,
new LambdaClass(this, accessedFrom, lambdaClassType, descriptor));
}
+ lambdaClass.addSynthesizedFrom(appInfo.definitionFor(accessedFrom).asProgramClass());
if (isInMainDexList(accessedFrom)) {
lambdaClass.addToMainDexList.set(true);
}
diff --git a/src/main/java/com/android/tools/r8/utils/AndroidApp.java b/src/main/java/com/android/tools/r8/utils/AndroidApp.java
index 54d10a0..9d31763 100644
--- a/src/main/java/com/android/tools/r8/utils/AndroidApp.java
+++ b/src/main/java/com/android/tools/r8/utils/AndroidApp.java
@@ -14,8 +14,10 @@
import com.android.tools.r8.dex.VDexFile;
import com.android.tools.r8.dex.VDexFileReader;
import com.android.tools.r8.errors.CompilationError;
+import com.android.tools.r8.errors.Unreachable;
import com.android.tools.r8.shaking.FilteredClassPath;
import com.google.common.collect.ImmutableList;
+import com.google.common.collect.ImmutableMap;
import com.google.common.io.ByteStreams;
import com.google.common.io.Closer;
import java.io.ByteArrayOutputStream;
@@ -34,7 +36,9 @@
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
+import java.util.HashMap;
import java.util.List;
+import java.util.Map;
import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;
@@ -51,6 +55,7 @@
public static final String DEFAULT_PROGUARD_MAP_FILE = "proguard.map";
private final ImmutableList<Resource> programResources;
+ private final ImmutableMap<Resource, String> programResourcesMainDescriptor;
private final ImmutableList<ClassFileResourceProvider> classpathResourceProviders;
private final ImmutableList<ClassFileResourceProvider> libraryResourceProviders;
@@ -66,6 +71,7 @@
// See factory methods and AndroidApp.Builder below.
private AndroidApp(
ImmutableList<Resource> programResources,
+ ImmutableMap<Resource, String> programResourcesMainDescriptor,
ImmutableList<ProgramFileArchiveReader> programFileArchiveReaders,
ImmutableList<ClassFileResourceProvider> classpathResourceProviders,
ImmutableList<ClassFileResourceProvider> libraryResourceProviders,
@@ -77,6 +83,7 @@
List<String> mainDexClasses,
Resource mainDexListOutput) {
this.programResources = programResources;
+ this.programResourcesMainDescriptor = programResourcesMainDescriptor;
this.programFileArchiveReaders = programFileArchiveReaders;
this.classpathResourceProviders = classpathResourceProviders;
this.libraryResourceProviders = libraryResourceProviders;
@@ -331,7 +338,7 @@
try (Closer closer = Closer.create()) {
List<Resource> dexProgramSources = getDexProgramResources();
for (int i = 0; i < dexProgramSources.size(); i++) {
- Path filePath = directory.resolve(outputMode.getOutputPath(dexProgramSources.get(i), i));
+ Path filePath = directory.resolve(getOutputPath(outputMode, dexProgramSources.get(i), i));
if (!Files.exists(filePath.getParent())) {
Files.createDirectories(filePath.getParent());
}
@@ -340,6 +347,19 @@
}
}
+ private String getOutputPath(OutputMode outputMode, Resource resource, int index) {
+ switch (outputMode) {
+ case Indexed:
+ return index == 0 ? "classes.dex" : ("classes" + (index + 1) + ".dex");
+ case FilePerInputClass:
+ String classDescriptor = programResourcesMainDescriptor.get(resource);
+ assert classDescriptor!= null && DescriptorUtils.isClassDescriptor(classDescriptor);
+ return classDescriptor.substring(1, classDescriptor.length() - 1) + ".dex";
+ default:
+ throw new Unreachable("Unknown output mode: " + outputMode);
+ }
+ }
+
private static boolean isClassesDexFile(Path file) {
String name = file.getFileName().toString().toLowerCase();
if (!name.startsWith("classes") || !name.endsWith(".dex")) {
@@ -389,7 +409,7 @@
try (ZipOutputStream out = new ZipOutputStream(Files.newOutputStream(archive, options))) {
List<Resource> dexProgramSources = getDexProgramResources();
for (int i = 0; i < dexProgramSources.size(); i++) {
- ZipEntry zipEntry = new ZipEntry(outputMode.getOutputPath(dexProgramSources.get(i), i));
+ ZipEntry zipEntry = new ZipEntry(getOutputPath(outputMode, dexProgramSources.get(i), i));
byte[] bytes =
ByteStreams.toByteArray(closer.register(dexProgramSources.get(i).getStream()));
zipEntry.setSize(bytes.length);
@@ -426,12 +446,17 @@
out.write(ByteStreams.toByteArray(input));
}
+ String getPrimaryClassDescriptor(Resource resource) {
+ return programResourcesMainDescriptor.get(resource);
+ }
+
/**
* Builder interface for constructing an AndroidApp.
*/
public static class Builder {
private final List<Resource> programResources = new ArrayList<>();
+ private final Map<Resource, String> programResourcesMainDescriptor = new HashMap<>();
private final List<ProgramFileArchiveReader> programFileArchiveReaders = new ArrayList<>();
private final List<ClassFileResourceProvider> classpathResourceProviders = new ArrayList<>();
private final List<ClassFileResourceProvider> libraryResourceProviders = new ArrayList<>();
@@ -564,6 +589,19 @@
}
/**
+ * Add dex program-data with class descriptor and primary class.
+ */
+ public Builder addDexProgramData(
+ byte[] data,
+ Set<String> classDescriptors,
+ String primaryClassDescriptor) {
+ Resource resource = Resource.fromBytes(Resource.Kind.DEX, data, classDescriptors);
+ programResources.add(resource);
+ programResourcesMainDescriptor.put(resource, primaryClassDescriptor);
+ return this;
+ }
+
+ /**
* Add dex program-data.
*/
public Builder addDexProgramData(byte[]... data) {
@@ -727,6 +765,7 @@
public AndroidApp build() {
return new AndroidApp(
ImmutableList.copyOf(programResources),
+ ImmutableMap.copyOf(programResourcesMainDescriptor),
ImmutableList.copyOf(programFileArchiveReaders),
ImmutableList.copyOf(classpathResourceProviders),
ImmutableList.copyOf(libraryResourceProviders),
diff --git a/src/main/java/com/android/tools/r8/utils/OutputMode.java b/src/main/java/com/android/tools/r8/utils/OutputMode.java
index 1bdf501..4520cd6 100644
--- a/src/main/java/com/android/tools/r8/utils/OutputMode.java
+++ b/src/main/java/com/android/tools/r8/utils/OutputMode.java
@@ -3,28 +3,8 @@
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.utils;
-import com.android.tools.r8.Resource;
-import java.util.Set;
-
/** Defines way the output is formed. */
public enum OutputMode {
- Indexed {
- @Override
- String getOutputPath(Resource resource, int index) {
- return index == 0 ? "classes.dex" : ("classes" + (index + 1) + ".dex");
- }
- },
- FilePerClass {
- @Override
- String getOutputPath(Resource resource, int index) {
- Set<String> classDescriptors = resource.getClassDescriptors();
- assert classDescriptors != null;
- assert classDescriptors.size() == 1;
- String classDescriptor = classDescriptors.iterator().next();
- assert DescriptorUtils.isClassDescriptor(classDescriptor);
- return classDescriptor.substring(1, classDescriptor.length() - 1) + ".dex";
- }
- };
-
- abstract String getOutputPath(Resource resource, int index);
+ Indexed,
+ FilePerInputClass;
}
diff --git a/src/test/java/com/android/tools/r8/D8IncrementalRunExamplesAndroidOTest.java b/src/test/java/com/android/tools/r8/D8IncrementalRunExamplesAndroidOTest.java
index 6461a57..711453b 100644
--- a/src/test/java/com/android/tools/r8/D8IncrementalRunExamplesAndroidOTest.java
+++ b/src/test/java/com/android/tools/r8/D8IncrementalRunExamplesAndroidOTest.java
@@ -11,9 +11,13 @@
import com.android.tools.r8.errors.CompilationError;
import com.android.tools.r8.errors.InternalCompilerError;
import com.android.tools.r8.errors.Unimplemented;
+import com.android.tools.r8.ir.desugar.InterfaceMethodRewriter;
+import com.android.tools.r8.ir.desugar.LambdaRewriter;
import com.android.tools.r8.utils.AndroidApp;
+import com.android.tools.r8.utils.DescriptorUtils;
import com.android.tools.r8.utils.OffOrAuto;
import com.android.tools.r8.utils.OutputMode;
+import com.android.tools.r8.utils.UtilsHelper;
import com.beust.jcommander.internal.Lists;
import com.google.common.io.Closer;
import java.io.ByteArrayOutputStream;
@@ -77,13 +81,19 @@
TreeMap<String, Resource> fileToResource = new TreeMap<>();
List<String> classFiles = collectClassFiles(testJarFile);
AndroidApp app = compileClassFiles(
- testJarFile, classFiles, output, OutputMode.FilePerClass);
+ testJarFile, classFiles, output, OutputMode.FilePerInputClass);
for (Resource resource : app.getDexProgramResources()) {
Set<String> descriptors = resource.getClassDescriptors();
- Assert.assertNotNull(descriptors);
- Assert.assertEquals(1, descriptors.size());
- String classDescriptor = descriptors.iterator().next();
- classDescriptor = classDescriptor.substring(1, classDescriptor.length() - 1);
+ String mainClassDescriptor = UtilsHelper.getMainClassDescriptor(app, resource);
+ for (String descriptor : descriptors) {
+ // classes are either lambda classes used by the main class, companion classes of the main
+ // interface or the main class/interface
+ Assert.assertTrue(descriptor.contains(LambdaRewriter.LAMBDA_CLASS_NAME_PREFIX)
+ || descriptor.endsWith(InterfaceMethodRewriter.COMPANION_CLASS_NAME_SUFFIX + ";")
+ || descriptor.equals(mainClassDescriptor));
+ }
+ String classDescriptor =
+ DescriptorUtils.getClassBinaryNameFromDescriptor(mainClassDescriptor);
String classFilePath = classDescriptor + ".class";
if (File.separatorChar != '/') {
classFilePath = classFilePath.replace('/', File.separatorChar);
diff --git a/src/test/java/com/android/tools/r8/R8RunArtTestsTest.java b/src/test/java/com/android/tools/r8/R8RunArtTestsTest.java
index 2080a15..7cb47e2 100644
--- a/src/test/java/com/android/tools/r8/R8RunArtTestsTest.java
+++ b/src/test/java/com/android/tools/r8/R8RunArtTestsTest.java
@@ -20,6 +20,7 @@
import com.android.tools.r8.utils.FileUtils;
import com.android.tools.r8.utils.JarBuilder;
import com.android.tools.r8.utils.ListUtils;
+import com.android.tools.r8.utils.OffOrAuto;
import com.google.common.base.Charsets;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableListMultimap;
@@ -151,8 +152,6 @@
// Takes ages to run on art 5.1.1 and behaves the same as on 6.0.1. Running this
// tests on 5.1.1 makes our buildbot cycles time too long.
.put("800-smali", TestCondition.match(TestCondition.runtimes(DexVm.ART_5_1_1)))
- // Hangs on dalvik.
- .put("802-deoptimization", TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
.build();
// Tests that are flaky with the Art version we currently use.
@@ -453,24 +452,8 @@
"605-new-string-from-bytes",
"626-const-class-linking"
),
- DexVm.ART_4_4_4, ImmutableList.of(
- // Generally fails on non R8/D8 running.
- "004-checker-UnsafeTest18",
- "004-NativeAllocations",
- "005-annotations",
- "008-exceptions",
- "082-inline-execute",
- "099-vmdebug",
- "143-string-value",
- "530-checker-lse2",
- "536-checker-intrinsic-optimization",
- "552-invoke-non-existent-super",
- "580-checker-round",
- "580-checker-string-fact-intrinsics",
- "594-invoke-super",
- "605-new-string-from-bytes",
- "626-const-class-linking"
- )
+ DexVm.ART_4_4_4, ImmutableList.of()
+
);
// Tests where the R8/D8 output runs in Art but the original does not.
@@ -481,7 +464,7 @@
private static final Multimap<String, TestCondition> failingRunWithArt =
new ImmutableListMultimap.Builder<String, TestCondition>()
// This test relies on specific field access patterns, which we rewrite.
- .put("064-field-access", TestCondition.match(TestCondition.R8_NOT_AFTER_D8_COMPILER))
+ .put("064-field-access", TestCondition.match(TestCondition.R8_COMPILER))
// The growth limit test fails after processing by R8 because R8 will eliminate an
// "unneeded" const store. The following reflective call to the VM's GC will then see the
// large array as still live and the subsequent allocations will fail to reach the desired
@@ -495,104 +478,6 @@
TestCondition.match(
TestCondition.D8_COMPILER,
TestCondition.runtimes(DexVm.ART_7_0_0, DexVm.ART_6_0_1, DexVm.ART_5_1_1)))
- // Dalvik fails on reading an uninitialized local.
- .put(
- "471-uninitialized-locals",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // Out of memory.
- .put("152-dead-large-object",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // Cannot resolve exception handler. Interestingly, D8 generates different code in
- // release mode (which is also the code generated by R8) which passes.
- .put("111-unresolvable-exception",
- TestCondition.match(
- TestCondition.D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // Type not present.
- .put("124-missing-classes",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // Failed creating vtable.
- .put("587-inline-class-error",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // Failed creating vtable.
- .put("595-error-class",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // NoSuchFieldException: systemThreadGroup on Art 4.4.4.
- .put("129-ThreadGetId",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // Verifier says: can't modify final field LMain;.staticFinalField.
- .put("600-verifier-fails",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // VFY: args to if-eq/if-ne must both be refs or cat1
- .put("134-reg-promotion",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("003-omnibus-opcodes",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("043-privates",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("589-super-imt",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("494-checker-instanceof-tests",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("422-instanceof",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("424-checkcast",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("495-checker-checkcast-tests",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("576-polymorphic-inlining",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("578-polymorphic-inlining",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("631-checker-get-class",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(65355452): invoke-direct <init> on super only allowed for 'this' in <init>
- .put("633-checker-rtp-getclass",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // VFY: tried to get class from non-ref register.
- .put("506-verify-aput",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // NoSuchMethod: startMethodTracing.
- .put("545-tracing-and-jit",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // filled-new-array arg 0(1) not valid.
- .put("412-new-array",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
- // TODO(ager): unclear what is failing here.
- .put("098-ddmc",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
// Unsatisfiable link error:
// libarttest.so: undefined symbol: _ZN3art6Thread18RunEmptyCheckpointEv
.put(
@@ -600,9 +485,6 @@
TestCondition.match(
TestCondition.D8_COMPILER,
TestCondition.runtimes(DexVm.ART_7_0_0, DexVm.ART_6_0_1, DexVm.ART_5_1_1)))
- // lib64 libarttest.so: wrong ELF class ELFCLASS64.
- .put("543-env-long-ref",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
// Regression test for an issue that is not fixed on version 5.1.1. Throws an Exception
// instance instead of the expected NullPointerException. This bug is only tickled when
// running the R8 generated code when starting from jar or from dex code generated with
@@ -627,20 +509,16 @@
// checked into the Art repo.
private static final Multimap<String, TestCondition> failingRunWithArtOutput =
new ImmutableListMultimap.Builder<String, TestCondition>()
- // TODO(ager): Different output on R8 but only from jar frontend.
- .put("068-classloader",
- TestCondition.match(
- TestCondition.R8_NOT_AFTER_D8_COMPILER,
- TestCondition.runtimes(DexVm.ART_4_4_4)))
- // On Art 4.4.4 we have 7 refs instead of 9.
- .put("072-precise-gc",
- TestCondition.match(TestCondition.runtimes(DexVm.ART_4_4_4)))
// This one is expected to have different output. It counts instances, but the list that
// keeps the instances alive is dead and could be garbage collected. The compiler reuses
// the register for the list and therefore there are no live instances.
.put("099-vmdebug", TestCondition.any())
// This test relies on output on stderr, which we currently do not collect.
.put("143-string-value", TestCondition.any())
+ // This one is expected to have different output. It counts instances, but the list that
+ // keeps the instances alive is dead and could be garbage collected. The compiler reuses
+ // the register for the list and therefore there are no live instances.
+ .put("099-vmdebug", TestCondition.any())
.put(
"800-smali",
TestCondition.match(
@@ -1663,8 +1541,7 @@
}
File expectedFile = specification.resolveFile("expected.txt");
- String expected =
- com.google.common.io.Files.asCharSource(expectedFile, Charsets.UTF_8).read();
+ String expected = com.google.common.io.Files.toString(expectedFile, Charsets.UTF_8);
if (specification.failsWithArt) {
thrown.expect(AssertionError.class);
}
diff --git a/src/test/java/com/android/tools/r8/maindexlist/MainDexTracingTest.java b/src/test/java/com/android/tools/r8/maindexlist/MainDexTracingTest.java
index ba5092f..cc56417 100644
--- a/src/test/java/com/android/tools/r8/maindexlist/MainDexTracingTest.java
+++ b/src/test/java/com/android/tools/r8/maindexlist/MainDexTracingTest.java
@@ -15,6 +15,7 @@
import com.android.tools.r8.ToolHelper;
import com.android.tools.r8.graph.DexClass;
import com.android.tools.r8.graph.DexType;
+import com.android.tools.r8.ir.desugar.LambdaRewriter;
import com.android.tools.r8.utils.DescriptorUtils;
import com.android.tools.r8.utils.InternalOptions;
import java.nio.charset.StandardCharsets;
@@ -179,7 +180,7 @@
}
private boolean isLambda(String mainDexEntry) {
- return mainDexEntry.contains("-$$Lambda$");
+ return mainDexEntry.contains(LambdaRewriter.LAMBDA_CLASS_NAME_PREFIX);
}
private String mainDexStringToDescriptor(String mainDexString) {
diff --git a/src/test/java/com/android/tools/r8/utils/OutputModeTest.java b/src/test/java/com/android/tools/r8/utils/OutputModeTest.java
deleted file mode 100644
index 5929b60..0000000
--- a/src/test/java/com/android/tools/r8/utils/OutputModeTest.java
+++ /dev/null
@@ -1,34 +0,0 @@
-// Copyright (c) 2017, 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.utils;
-
-import static org.junit.Assert.assertEquals;
-
-import com.android.tools.r8.Resource;
-import java.util.Collections;
-import org.junit.Test;
-
-public class OutputModeTest {
- @Test
- public void testIndexedFileName() {
- assertEquals("classes.dex", OutputMode.Indexed.getOutputPath(null, 0));
- assertEquals("classes2.dex", OutputMode.Indexed.getOutputPath(null, 1));
- }
-
- @Test
- public void testFilePerClass() {
- Resource test =
- Resource.fromBytes(Resource.Kind.CLASSFILE, new byte[]{}, Collections.singleton("LTest;"));
- assertEquals("Test.dex", OutputMode.FilePerClass.getOutputPath(test, 0));
- Resource comTest =
- Resource.fromBytes(
- Resource.Kind.CLASSFILE, new byte[]{}, Collections.singleton("Lcom/Test;"));
- assertEquals("com/Test.dex", OutputMode.FilePerClass.getOutputPath(comTest, 0));
- Resource comExampleTest =
- Resource.fromBytes(
- Resource.Kind.CLASSFILE, new byte[]{}, Collections.singleton("Lcom/example/Test;"));
- assertEquals("com/example/Test.dex", OutputMode.FilePerClass.getOutputPath(comExampleTest, 0));
- assertEquals("com/example/Test.dex", OutputMode.FilePerClass.getOutputPath(comExampleTest, 1));
- }
-}
diff --git a/tools/test.py b/tools/test.py
index 0cd8296..c0a1e02 100755
--- a/tools/test.py
+++ b/tools/test.py
@@ -16,7 +16,7 @@
import uuid
import notify
-ALL_ART_VMS = ["default", "7.0.0", "6.0.1", "5.1.1", "4.4.4"]
+ALL_ART_VMS = ["default", "7.0.0", "6.0.1", "5.1.1"]
BUCKET = 'r8-test-results'
def ParseOptions():