Keep dependent items for indirectly instantiated types
Change-Id: Id4980592c676375fc59565f28946d226e8fa7451
diff --git a/src/main/java/com/android/tools/r8/shaking/Enqueuer.java b/src/main/java/com/android/tools/r8/shaking/Enqueuer.java
index 3560e2f..378cca0 100644
--- a/src/main/java/com/android/tools/r8/shaking/Enqueuer.java
+++ b/src/main/java/com/android/tools/r8/shaking/Enqueuer.java
@@ -1122,7 +1122,7 @@
// For all instance fields visible from the class, mark them live if we have seen a read.
transitionFieldsForInstantiatedClass(clazz.type);
// Add all dependent instance members to the workqueue.
- rootSet.forEachDependentNonStaticMember(clazz, appView, this::enqueueDependentItem);
+ transitionDependentItemsForInstantiatedClass(clazz);
}
/**
@@ -1236,6 +1236,19 @@
} while (type != null && !instantiatedTypes.contains(type));
}
+ private void transitionDependentItemsForInstantiatedClass(DexClass clazz) {
+ DexClass current = clazz;
+ do {
+ // Handle keep rules that are dependent on the class being instantiated.
+ rootSet.forEachDependentNonStaticMember(clazz, appView, this::enqueueDependentItem);
+
+ // Visit the super type.
+ current = current.superType != null ? appView.definitionFor(current.superType) : null;
+ } while (current != null
+ && current.isProgramClass()
+ && !instantiatedTypes.contains(current.type));
+ }
+
private void markStaticFieldAsLive(DexField field, KeepReason reason) {
markStaticFieldAsLive(field, reason, appInfo.resolveField(field));
}
diff --git a/src/test/java/com/android/tools/r8/shaking/keepclassmembers/KeepClassMembersRuleOnIndirectlyInstantiatedClassTest.java b/src/test/java/com/android/tools/r8/shaking/keepclassmembers/KeepClassMembersRuleOnIndirectlyInstantiatedClassTest.java
new file mode 100644
index 0000000..327f61d
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/shaking/keepclassmembers/KeepClassMembersRuleOnIndirectlyInstantiatedClassTest.java
@@ -0,0 +1,94 @@
+// Copyright (c) 2019, 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.keepclassmembers;
+
+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.not;
+import static org.hamcrest.MatcherAssert.assertThat;
+
+import com.android.tools.r8.NeverInline;
+import com.android.tools.r8.NeverMerge;
+import com.android.tools.r8.TestBase;
+import com.android.tools.r8.TestParameters;
+import com.android.tools.r8.TestParametersCollection;
+import com.android.tools.r8.utils.codeinspector.ClassSubject;
+import com.android.tools.r8.utils.codeinspector.CodeInspector;
+import com.android.tools.r8.utils.codeinspector.FieldSubject;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+import org.junit.runners.Parameterized.Parameters;
+
+@RunWith(Parameterized.class)
+public class KeepClassMembersRuleOnIndirectlyInstantiatedClassTest extends TestBase {
+
+ private final TestParameters parameters;
+
+ @Parameters(name = "{0}")
+ public static TestParametersCollection data() {
+ return getTestParameters().withAllRuntimes().build();
+ }
+
+ public KeepClassMembersRuleOnIndirectlyInstantiatedClassTest(TestParameters parameters) {
+ this.parameters = parameters;
+ }
+
+ @Test
+ public void test() throws Exception {
+ testForR8(parameters.getBackend())
+ .addInnerClasses(KeepClassMembersRuleOnIndirectlyInstantiatedClassTest.class)
+ .addKeepMainRule(TestClass.class)
+ .addKeepRules(
+ "-keepclassmembers class " + A.class.getTypeName() + " {",
+ " java.lang.String greeting;",
+ "}")
+ .enableInliningAnnotations()
+ .enableMergeAnnotations()
+ .setMinApi(parameters.getRuntime())
+ .compile()
+ .inspect(this::verifyFieldIsPresent)
+ .run(parameters.getRuntime(), TestClass.class)
+ .assertSuccessWithOutputLines("Hello world!");
+ }
+
+ private void verifyFieldIsPresent(CodeInspector inspector) {
+ ClassSubject classSubject = inspector.clazz(A.class);
+ assertThat(classSubject, isPresent());
+
+ FieldSubject fieldSubject = classSubject.uniqueFieldWithName("greeting");
+ assertThat(fieldSubject, isPresent());
+ assertThat(fieldSubject, not(isRenamed()));
+ }
+
+ static class TestClass {
+
+ public static void main(String[] args) throws Exception {
+ B b = new B();
+ reflectivelyWriteFieldWithName(b, "greeting", "Hello world!");
+ System.out.println(reflectivelyReadFieldWithName(b, "greeting"));
+ }
+
+ @NeverInline
+ private static void reflectivelyWriteFieldWithName(Object o, String fieldName, String value)
+ throws Exception {
+ o.getClass().getField(fieldName).set(o, value);
+ }
+
+ @NeverInline
+ private static String reflectivelyReadFieldWithName(Object o, String fieldName)
+ throws Exception {
+ return (String) o.getClass().getField(fieldName).get(o);
+ }
+ }
+
+ @NeverMerge
+ static class A {
+
+ public String greeting;
+ }
+
+ static class B extends A {}
+}