Reproduce b/120951621: stripped-out Kotlin @DebugMetadata and getters.
Test:
$ tools/test.py --dex_vm=all *ReflectiveAnnotationUse*
Bug: 120951621, 119471127
Change-Id: I7f2b8a0716614a4fa41024ea3eefa84c48fc74a9
diff --git a/src/test/java/com/android/tools/r8/KotlinTestBase.java b/src/test/java/com/android/tools/r8/KotlinTestBase.java
new file mode 100644
index 0000000..9139583
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/KotlinTestBase.java
@@ -0,0 +1,34 @@
+// 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;
+
+import com.android.tools.r8.ToolHelper.KotlinTargetVersion;
+import com.android.tools.r8.utils.FileUtils;
+import java.nio.file.Path;
+import java.nio.file.Paths;
+
+public abstract class KotlinTestBase extends TestBase {
+ private static final String RSRC = "kotlinR8TestResources";
+
+ protected final KotlinTargetVersion targetVersion;
+
+ protected KotlinTestBase(KotlinTargetVersion targetVersion) {
+ this.targetVersion = targetVersion;
+ }
+
+ protected Path getKotlinJarFile(String folder) {
+ return Paths.get(ToolHelper.TESTS_BUILD_DIR, RSRC,
+ targetVersion.getFolderName(), folder + FileUtils.JAR_EXTENSION);
+ }
+
+ protected Path getJavaJarFile(String folder) {
+ return Paths.get(ToolHelper.TESTS_BUILD_DIR, RSRC,
+ targetVersion.getFolderName(), folder + ".java" + FileUtils.JAR_EXTENSION);
+ }
+
+ protected Path getMappingfile(String folder, String mappingFileName) {
+ return Paths.get(ToolHelper.TESTS_DIR, RSRC, folder, mappingFileName);
+ }
+
+}
diff --git a/src/test/java/com/android/tools/r8/kotlin/MetadataStripTest.java b/src/test/java/com/android/tools/r8/kotlin/MetadataStripTest.java
index dc2ba39..eca25fb 100644
--- a/src/test/java/com/android/tools/r8/kotlin/MetadataStripTest.java
+++ b/src/test/java/com/android/tools/r8/kotlin/MetadataStripTest.java
@@ -10,19 +10,15 @@
import static org.junit.Assert.assertNull;
import static org.junit.Assert.assertThat;
-import com.android.tools.r8.TestBase;
+import com.android.tools.r8.KotlinTestBase;
import com.android.tools.r8.TestRunResult;
import com.android.tools.r8.ToolHelper;
import com.android.tools.r8.ToolHelper.KotlinTargetVersion;
import com.android.tools.r8.graph.DexAnnotation;
import com.android.tools.r8.graph.DexClass;
-import com.android.tools.r8.utils.FileUtils;
import com.android.tools.r8.utils.InternalOptions;
import com.android.tools.r8.utils.codeinspector.ClassSubject;
import com.android.tools.r8.utils.codeinspector.CodeInspector;
-import com.google.common.collect.ImmutableList;
-import java.nio.file.Path;
-import java.nio.file.Paths;
import java.util.Collection;
import java.util.function.Consumer;
import org.junit.Test;
@@ -30,12 +26,11 @@
import org.junit.runners.Parameterized;
@RunWith(Parameterized.class)
-public class MetadataStripTest extends TestBase {
+public class MetadataStripTest extends KotlinTestBase {
private static final String METADATA_DESCRIPTOR = "Lkotlin/Metadata;";
private static final String KEEP_ANNOTATIONS = "-keepattributes *Annotation*";
private final Backend backend;
- private final KotlinTargetVersion targetVersion;
private Consumer<InternalOptions> optionsModifier =
o -> {
@@ -49,8 +44,8 @@
}
public MetadataStripTest(Backend backend, KotlinTargetVersion targetVersion) {
+ super(targetVersion);
this.backend = backend;
- this.targetVersion = targetVersion;
}
@Test
@@ -62,7 +57,6 @@
.addProgramFiles(getKotlinJarFile(folder))
.addProgramFiles(getJavaJarFile(folder))
.addProgramFiles(ToolHelper.getKotlinReflectJar())
- .enableProguardTestOptions()
.enableInliningAnnotations()
.addKeepMainRule(mainClassName)
.addKeepRules(KEEP_ANNOTATIONS)
@@ -91,13 +85,4 @@
return null;
}
- private Path getKotlinJarFile(String folder) {
- return Paths.get(ToolHelper.TESTS_BUILD_DIR, "kotlinR8TestResources",
- targetVersion.getFolderName(), folder + FileUtils.JAR_EXTENSION);
- }
-
- private Path getJavaJarFile(String folder) {
- return Paths.get(ToolHelper.TESTS_BUILD_DIR, "kotlinR8TestResources",
- targetVersion.getFolderName(), folder + ".java" + FileUtils.JAR_EXTENSION);
- }
}
diff --git a/src/test/java/com/android/tools/r8/shaking/EnclosingMethodTest.java b/src/test/java/com/android/tools/r8/shaking/EnclosingMethodTest.java
index a4806f0..72311a9 100644
--- a/src/test/java/com/android/tools/r8/shaking/EnclosingMethodTest.java
+++ b/src/test/java/com/android/tools/r8/shaking/EnclosingMethodTest.java
@@ -3,12 +3,14 @@
// BSD-style license that can be found in the LICENSE file.
package com.android.tools.r8.shaking;
+import static org.hamcrest.CoreMatchers.containsString;
import static org.junit.Assume.assumeTrue;
import com.android.tools.r8.R8TestBuilder;
import com.android.tools.r8.TestBase;
+import com.android.tools.r8.TestRunResult;
import com.android.tools.r8.ToolHelper;
-import com.android.tools.r8.ToolHelper.DexVm;
+import com.android.tools.r8.ToolHelper.DexVm.Version;
import com.android.tools.r8.utils.BooleanUtils;
import com.google.common.collect.ImmutableList;
import java.nio.file.Path;
@@ -72,12 +74,8 @@
@Test
public void testR8() throws Exception {
- DexVm vm = ToolHelper.getDexVm();
- assumeTrue("Known to be broken at 5.1.1 and 6.0.1 due to access to fragile EnclosingMethod.",
- vm.isOlderThanOrEqual(DexVm.ART_4_4_4_HOST) && vm.isNewerThan(DexVm.ART_6_0_1_HOST));
R8TestBuilder builder = testForR8(backend)
.addProgramFiles(classPaths)
- .enableProguardTestOptions()
.enableInliningAnnotations()
.addKeepMainRule(MAIN)
.addKeepRules("-keep class **.GetName*")
@@ -85,6 +83,16 @@
if (!enableMinification) {
builder.addKeepRules("-dontobfuscate");
}
- builder.run(MAIN).assertSuccessWithOutput(JAVA_OUTPUT);
+
+ TestRunResult result = builder.run(MAIN);
+ if (backend == Backend.DEX) {
+ if (ToolHelper.getDexVm().getVersion().isNewerThan(Version.V4_4_4)
+ && ToolHelper.getDexVm().getVersion().isOlderThanOrEqual(Version.V6_0_1)) {
+ result.assertFailureWithErrorThatMatches(containsString("IncompatibleClassChangeError"));
+ return;
+ }
+ }
+
+ result.assertSuccessWithOutput(JAVA_OUTPUT);
}
}
diff --git a/src/test/java/com/android/tools/r8/shaking/annotations/ReflectiveAnnotationUseTest.java b/src/test/java/com/android/tools/r8/shaking/annotations/ReflectiveAnnotationUseTest.java
new file mode 100644
index 0000000..43736bd
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/shaking/annotations/ReflectiveAnnotationUseTest.java
@@ -0,0 +1,228 @@
+// 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.annotations;
+
+import static com.android.tools.r8.utils.codeinspector.Matchers.isPresent;
+import static com.android.tools.r8.utils.codeinspector.Matchers.isRenamed;
+import static org.hamcrest.CoreMatchers.containsString;
+import static org.hamcrest.CoreMatchers.not;
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertThat;
+import static org.junit.Assume.assumeTrue;
+
+import com.android.tools.r8.KotlinTestBase;
+import com.android.tools.r8.R8TestBuilder;
+import com.android.tools.r8.TestRunResult;
+import com.android.tools.r8.ToolHelper;
+import com.android.tools.r8.ToolHelper.DexVm.Version;
+import com.android.tools.r8.ToolHelper.KotlinTargetVersion;
+import com.android.tools.r8.utils.AndroidApp;
+import com.android.tools.r8.utils.BooleanUtils;
+import com.android.tools.r8.utils.StringUtils;
+import com.android.tools.r8.utils.codeinspector.ClassSubject;
+import com.android.tools.r8.utils.codeinspector.CodeInspector;
+import com.android.tools.r8.utils.codeinspector.MethodSubject;
+import java.util.Collection;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+
+@RunWith(Parameterized.class)
+public class ReflectiveAnnotationUseTest extends KotlinTestBase {
+ private static final String FOLDER = "internal_annotation";
+ private static final String MAIN_CLASS_NAME = "internal_annotation.MainKt";
+ private static final String ANNOTATION_NAME = "internal_annotation.Annotation";
+ private static final String KEEP_ANNOTATIONS = "-keepattributes *Annotation*";
+
+ private static final String JAVA_OUTPUT = StringUtils.lines(
+ "Impl::toString",
+ "Impl::Annotation::field2.Impl::Annotation::field2(Impl::Annotation::field2:2)"
+ );
+
+ private static final String OUTPUT_WITHOUT_ANNOTATION = StringUtils.lines(
+ "Impl::toString",
+ "null"
+ );
+
+ private final Backend backend;
+ private final boolean minify;
+
+ @Parameterized.Parameters(name = "Backend: {0} target: {1} minify: {2}")
+ public static Collection<Object[]> data() {
+ return buildParameters(Backend.values(), KotlinTargetVersion.values(), BooleanUtils.values());
+ }
+
+ public ReflectiveAnnotationUseTest(
+ Backend backend, KotlinTargetVersion targetVersion, boolean minify) {
+ super(targetVersion);
+ this.backend = backend;
+ this.minify = minify;
+ }
+
+ @Test
+ public void b120951621_JVMoutput() throws Exception {
+ assumeTrue("Only run JVM reference once (for CF backend)", backend == Backend.CF);
+ AndroidApp app = AndroidApp.builder()
+ .addProgramFile(getKotlinJarFile(FOLDER))
+ .addProgramFile(getJavaJarFile(FOLDER))
+ .build();
+ String result = runOnJava(app, MAIN_CLASS_NAME);
+ assertEquals(JAVA_OUTPUT, result);
+ }
+
+ @Test
+ public void b120951621_keepAll() throws Exception {
+ R8TestBuilder builder = testForR8(backend)
+ .addProgramFiles(getKotlinJarFile(FOLDER))
+ .addProgramFiles(getJavaJarFile(FOLDER))
+ .addKeepMainRule(MAIN_CLASS_NAME)
+ .addKeepRules(KEEP_ANNOTATIONS)
+ .addKeepRules(
+ "-keep @interface " + ANNOTATION_NAME + " {",
+ " *;",
+ "}"
+ );
+ if (!minify) {
+ builder.noMinification();
+ }
+ TestRunResult result = builder.run(MAIN_CLASS_NAME);
+
+ result.assertSuccessWithOutput(JAVA_OUTPUT);
+
+ CodeInspector inspector = result.inspector();
+ ClassSubject clazz = inspector.clazz(ANNOTATION_NAME);
+ assertThat(clazz, isPresent());
+ assertThat(clazz, not(isRenamed()));
+ MethodSubject f1 = clazz.uniqueMethodWithName("f1");
+ assertThat(f1, isPresent());
+ assertThat(f1, not(isRenamed()));
+ MethodSubject f2 = clazz.uniqueMethodWithName("f2");
+ assertThat(f2, isPresent());
+ assertThat(f2, not(isRenamed()));
+ MethodSubject f3 = clazz.uniqueMethodWithName("f3");
+ assertThat(f3, isPresent());
+ assertThat(f3, not(isRenamed()));
+ MethodSubject f4 = clazz.uniqueMethodWithName("f4");
+ assertThat(f4, isPresent());
+ assertThat(f4, not(isRenamed()));
+ }
+
+ @Test
+ public void b120951621_partiallyKeep() throws Exception {
+ R8TestBuilder builder = testForR8(backend)
+ .addProgramFiles(getKotlinJarFile(FOLDER))
+ .addProgramFiles(getJavaJarFile(FOLDER))
+ .addKeepMainRule(MAIN_CLASS_NAME)
+ .addKeepRules(KEEP_ANNOTATIONS)
+ .addKeepRules(
+ "-keep,allowobfuscation @interface " + ANNOTATION_NAME + " {",
+ " java.lang.String *f2();",
+ "}"
+ );
+ if (!minify) {
+ builder.noMinification();
+ }
+ TestRunResult result = builder.run(MAIN_CLASS_NAME);
+
+ if (backend == Backend.DEX) {
+ if (ToolHelper.getDexVm().getVersion().isOlderThanOrEqual(Version.V4_4_4)) {
+ result.assertFailureWithErrorThatMatches(
+ containsString("failure in processEncodedAnnotation"));
+ return;
+ }
+ if (ToolHelper.getDexVm().getVersion().isOlderThanOrEqual(Version.V6_0_1)) {
+ result.assertFailureWithErrorThatMatches(containsString("IncompatibleClassChangeError"));
+ return;
+ }
+ result.assertSuccessWithOutput(OUTPUT_WITHOUT_ANNOTATION);
+ } else {
+ result.assertSuccessWithOutput(JAVA_OUTPUT);
+ }
+
+ CodeInspector inspector = result.inspector();
+ ClassSubject clazz = inspector.clazz(ANNOTATION_NAME);
+ assertThat(clazz, isPresent());
+ assertEquals(minify, clazz.isRenamed());
+ MethodSubject f1 = clazz.uniqueMethodWithName("f1");
+ assertThat(f1, isPresent());
+ assertThat(f1, not(isRenamed()));
+ MethodSubject f2 = clazz.uniqueMethodWithName("f2");
+ assertThat(f2, isPresent());
+ assertThat(f2, not(isRenamed()));
+ MethodSubject f3 = clazz.uniqueMethodWithName("f3");
+ assertThat(f3, not(isPresent()));
+ MethodSubject f4 = clazz.uniqueMethodWithName("f4");
+ assertThat(f4, not(isPresent()));
+ }
+
+ @Test
+ public void b120951621_keepAnnotation() throws Exception {
+ R8TestBuilder builder = testForR8(backend)
+ .addProgramFiles(getKotlinJarFile(FOLDER))
+ .addProgramFiles(getJavaJarFile(FOLDER))
+ .addKeepMainRule(MAIN_CLASS_NAME)
+ .addKeepRules(KEEP_ANNOTATIONS);
+ if (!minify) {
+ builder.noMinification();
+ }
+ TestRunResult result = builder.run(MAIN_CLASS_NAME);
+
+ if (backend == Backend.DEX) {
+ if (ToolHelper.getDexVm().getVersion().isOlderThanOrEqual(Version.V4_4_4)) {
+ result.assertFailureWithErrorThatMatches(
+ containsString("failure in processEncodedAnnotation"));
+ return;
+ }
+ if (ToolHelper.getDexVm().getVersion().isOlderThanOrEqual(Version.V6_0_1)) {
+ result.assertFailureWithErrorThatMatches(containsString("IncompatibleClassChangeError"));
+ return;
+ }
+ result.assertSuccessWithOutput(OUTPUT_WITHOUT_ANNOTATION);
+ } else {
+ result.assertSuccessWithOutput(JAVA_OUTPUT);
+ }
+
+ CodeInspector inspector = result.inspector();
+ ClassSubject clazz = inspector.clazz(ANNOTATION_NAME);
+ assertThat(clazz, isPresent());
+ assertEquals(minify, clazz.isRenamed());
+ MethodSubject f1 = clazz.uniqueMethodWithName("f1");
+ assertThat(f1, isPresent());
+ assertThat(f1, not(isRenamed()));
+ MethodSubject f2 = clazz.uniqueMethodWithName("f2");
+ assertThat(f2, isPresent());
+ assertThat(f2, not(isRenamed()));
+ MethodSubject f3 = clazz.uniqueMethodWithName("f3");
+ assertThat(f3, not(isPresent()));
+ MethodSubject f4 = clazz.uniqueMethodWithName("f4");
+ assertThat(f4, not(isPresent()));
+ }
+
+ @Test
+ public void b120951621_noKeep() throws Exception {
+ R8TestBuilder builder = testForR8(backend)
+ .addProgramFiles(getKotlinJarFile(FOLDER))
+ .addProgramFiles(getJavaJarFile(FOLDER))
+ .addKeepMainRule(MAIN_CLASS_NAME);
+ if (!minify) {
+ builder.noMinification();
+ }
+ TestRunResult result = builder.run(MAIN_CLASS_NAME);
+ result.assertSuccessWithOutput(OUTPUT_WITHOUT_ANNOTATION);
+
+ CodeInspector inspector = result.inspector();
+ ClassSubject clazz = inspector.clazz(ANNOTATION_NAME);
+ assertThat(clazz, isPresent());
+ assertEquals(minify, clazz.isRenamed());
+ MethodSubject f1 = clazz.uniqueMethodWithName("f1");
+ assertThat(f1, not(isPresent()));
+ MethodSubject f2 = clazz.uniqueMethodWithName("f2");
+ assertThat(f2, not(isPresent()));
+ MethodSubject f3 = clazz.uniqueMethodWithName("f3");
+ assertThat(f3, not(isPresent()));
+ MethodSubject f4 = clazz.uniqueMethodWithName("f4");
+ assertThat(f4, not(isPresent()));
+ }
+
+}
diff --git a/src/test/kotlinR8TestResources/internal_annotation/Annotation.kt b/src/test/kotlinR8TestResources/internal_annotation/Annotation.kt
new file mode 100644
index 0000000..40bc6b0
--- /dev/null
+++ b/src/test/kotlinR8TestResources/internal_annotation/Annotation.kt
@@ -0,0 +1,38 @@
+// 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 internal_annotation
+
+@Target(AnnotationTarget.CLASS)
+internal annotation class Annotation(
+ @get:JvmName("f1")
+ val field1: Int = 0,
+ @get:JvmName("f2")
+ val field2: String = "",
+ @get:JvmName("f3")
+ val field3: IntArray = [],
+ @get:JvmName("f4")
+ val field4: Array<String> = []
+)
+
+@JvmName("foo")
+internal fun Base.foo(): StackTraceElement? {
+ val anno = getMyAnnotation() ?: return null
+ // Note that only Annotation.f(1|2) will be live.
+ return StackTraceElement(anno.field2, anno.field2, anno.field2, anno.field1)
+}
+
+// To prevent Annotation.f(3|4) from being stripped out by kotlinc
+internal fun Base.getUnusedFields(): Array<String>? {
+ val anno = getMyAnnotation() ?: return null
+ val res = arrayListOf<String>()
+ for ((i, idx) in anno.field3.withIndex()) {
+ if (idx == i % 8) {
+ res.add(anno.field4[i])
+ }
+ }
+ return res.toTypedArray()
+}
+
+private fun Base.getMyAnnotation(): Annotation? =
+ javaClass.getAnnotation(Annotation::class.java)
diff --git a/src/test/kotlinR8TestResources/internal_annotation/Base.kt b/src/test/kotlinR8TestResources/internal_annotation/Base.kt
new file mode 100644
index 0000000..cedf902
--- /dev/null
+++ b/src/test/kotlinR8TestResources/internal_annotation/Base.kt
@@ -0,0 +1,29 @@
+// 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 internal_annotation
+
+@Annotation(8, "Base::Annotation::field2", [], [])
+internal abstract class Base {
+ protected abstract fun foo(): Any?
+
+ public override fun toString(): String =
+ "${foo() ?: this::class.java.name}"
+}
+
+// If we don't adjust annotation values, lack of f(3|4) will trigger errors on legacy VMs.
+@Annotation(2, "Impl::Annotation::field2", [3], ["field4"])
+internal class Impl(val flag: Boolean) : Base() {
+ override fun foo(): Any? {
+ return when (flag) {
+ true -> null
+ false -> this
+ }
+ }
+
+ public override fun toString(): String =
+ if (flag)
+ super.toString()
+ else
+ "Impl::toString"
+}
\ No newline at end of file
diff --git a/src/test/kotlinR8TestResources/internal_annotation/main.kt b/src/test/kotlinR8TestResources/internal_annotation/main.kt
new file mode 100644
index 0000000..ce5c85d
--- /dev/null
+++ b/src/test/kotlinR8TestResources/internal_annotation/main.kt
@@ -0,0 +1,10 @@
+// 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 internal_annotation
+
+fun main(args: Array<String>) {
+ val i = Impl(false)
+ println("$i")
+ println("${i.foo()}")
+}