blob: 089f670582bd8ce867582f5eec23408ab7ba36d1 [file] [log] [blame]
// Copyright (c) 2021, 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.maindexlist;
import static org.junit.Assert.assertEquals;
import com.android.tools.r8.TestBase;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.ToolHelper;
import com.android.tools.r8.references.Reference;
import com.android.tools.r8.utils.BooleanUtils;
import com.google.common.collect.ImmutableSet;
import java.lang.annotation.ElementType;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.lang.annotation.Target;
import java.util.HashSet;
import java.util.List;
import java.util.Set;
import java.util.stream.Collectors;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import org.junit.runners.Parameterized.Parameter;
import org.junit.runners.Parameterized.Parameters;
@RunWith(Parameterized.class)
public class MainDexSourceAndClassRetentionTest extends TestBase {
private static final Set<Class<?>> MAINDEX_CLASSES = ImmutableSet.of(Main.class);
@Parameter(0)
public TestParameters parameters;
@Parameter(1)
public boolean pruneNonVisibleAnnotationClasses;
@Parameters(name = "{0}, pruneNonVisibleAnnotationClasses: {1}")
public static List<Object[]> parameters() {
return buildParameters(
getTestParameters()
.withDexRuntimes()
.withApiLevelsEndingAtExcluding(apiLevelWithNativeMultiDexSupport())
.build(),
BooleanUtils.values());
}
@Test
public void testMainDex() throws Exception {
testForMainDexListGenerator(temp)
.addInnerClasses(getClass())
.addLibraryFiles(ToolHelper.getMostRecentAndroidJar())
.addMainDexRules(
"-keep class " + Main.class.getTypeName() + " {",
" public static void main(java.lang.String[]);",
"}")
.applyIf(
pruneNonVisibleAnnotationClasses,
builder -> {
builder.addOptionsModification(
options -> options.pruneNonVissibleAnnotationClasses = true);
})
.run()
.inspectMainDexClasses(
mainDexList -> {
assertEquals(
MAINDEX_CLASSES.stream()
.map(Reference::classFromClass)
.collect(Collectors.toSet()),
new HashSet<>(mainDexList));
});
}
@Test
public void testD8() throws Exception {
testForD8(temp)
.addInnerClasses(getClass())
.addLibraryFiles(ToolHelper.getMostRecentAndroidJar())
.setMinApi(parameters.getApiLevel())
.applyIf(
pruneNonVisibleAnnotationClasses,
builder -> {
builder.addOptionsModification(
options -> options.pruneNonVissibleAnnotationClasses = true);
})
.collectMainDexClasses()
.addMainDexRules(
"-keep class " + Main.class.getTypeName() + " {",
" public static void main(java.lang.String[]);",
"}")
.allowStdoutMessages()
.compile()
.inspect(
inspector -> {
// Source and class retention annotation classes are still in the output, but does not
// annotate anything.
assertEquals(
pruneNonVisibleAnnotationClasses,
!inspector.clazz(SourceRetentionAnnotation.class).isPresent());
assertEquals(
pruneNonVisibleAnnotationClasses,
!inspector.clazz(ClassRetentionAnnotation.class).isPresent());
assertEquals(0, inspector.clazz(Main.class).annotations().size());
assertEquals(0, inspector.clazz(A.class).annotations().size());
})
.inspectMainDexClasses(
mainDexClasses -> {
assertEquals(
MAINDEX_CLASSES.stream().map(TestBase::typeName).collect(Collectors.toSet()),
new HashSet<>(mainDexClasses));
});
}
public enum Foo {
TEST
}
public enum Bar {
TEST
}
@Retention(RetentionPolicy.SOURCE)
@Target(ElementType.TYPE)
public @interface SourceRetentionAnnotation {
Foo value();
}
@Retention(RetentionPolicy.CLASS)
@Target(ElementType.TYPE)
public @interface ClassRetentionAnnotation {
Bar value();
}
@SourceRetentionAnnotation(Foo.TEST)
@ClassRetentionAnnotation(Bar.TEST)
public static class A {
public static void main(String[] args) {}
}
@SourceRetentionAnnotation(Foo.TEST)
@ClassRetentionAnnotation(Bar.TEST)
public static class Main {
public static void main(String[] args) {}
}
}