Link dependent non-static items in the kept graph info.
Bug: 131822600
Bug: 133652414
Change-Id: I074446b96f38465f2e375d3d756f346abbc1d1f0
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 717c7ce..58caca2 100644
--- a/src/main/java/com/android/tools/r8/shaking/Enqueuer.java
+++ b/src/main/java/com/android/tools/r8/shaking/Enqueuer.java
@@ -1040,7 +1040,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::enqueueRootItem);
+ rootSet.forEachDependentNonStaticMember(clazz, appView, this::enqueueDependentItem);
}
/**
diff --git a/src/main/java/com/android/tools/r8/shaking/RootSetBuilder.java b/src/main/java/com/android/tools/r8/shaking/RootSetBuilder.java
index 3cabdf2..23e4503 100644
--- a/src/main/java/com/android/tools/r8/shaking/RootSetBuilder.java
+++ b/src/main/java/com/android/tools/r8/shaking/RootSetBuilder.java
@@ -58,7 +58,6 @@
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future;
-import java.util.function.BiConsumer;
import java.util.function.Function;
import java.util.function.Predicate;
import java.util.stream.Collectors;
@@ -1275,13 +1274,17 @@
public void forEachDependentNonStaticMember(
DexDefinition item,
AppView<?> appView,
- BiConsumer<DexDefinition, Set<ProguardKeepRule>> fn) {
- getDependentItems(item).forEach((reference, reasons) -> {
- DexDefinition definition = appView.definitionFor(reference);
- if (definition != null && !definition.isDexClass() && !definition.isStaticMember()) {
- fn.accept(definition, reasons);
- }
- });
+ Consumer3<DexDefinition, DexDefinition, Set<ProguardKeepRule>> fn) {
+ getDependentItems(item)
+ .forEach(
+ (reference, reasons) -> {
+ DexDefinition definition = appView.definitionFor(reference);
+ if (definition != null
+ && !definition.isDexClass()
+ && !definition.isStaticMember()) {
+ fn.accept(item, definition, reasons);
+ }
+ });
}
public void copy(DexReference original, DexReference rewritten) {
diff --git a/src/test/java/com/android/tools/r8/shaking/keptgraph/KeptByKeepClassMembersNonStaticTestRunner.java b/src/test/java/com/android/tools/r8/shaking/keptgraph/KeptByKeepClassMembersNonStaticTestRunner.java
new file mode 100644
index 0000000..ded10c4
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/shaking/keptgraph/KeptByKeepClassMembersNonStaticTestRunner.java
@@ -0,0 +1,124 @@
+// 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.keptgraph;
+
+import static com.android.tools.r8.references.Reference.classFromClass;
+import static com.android.tools.r8.references.Reference.methodFromMethod;
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertNotEquals;
+
+import com.android.tools.r8.NeverClassInline;
+import com.android.tools.r8.NeverInline;
+import com.android.tools.r8.NeverPropagateValue;
+import com.android.tools.r8.TestBase;
+import com.android.tools.r8.TestParameters;
+import com.android.tools.r8.TestParametersCollection;
+import com.android.tools.r8.origin.Origin;
+import com.android.tools.r8.references.ClassReference;
+import com.android.tools.r8.references.MethodReference;
+import com.android.tools.r8.shaking.WhyAreYouKeepingConsumer;
+import com.android.tools.r8.utils.StringUtils;
+import com.android.tools.r8.utils.graphinspector.GraphInspector;
+import com.android.tools.r8.utils.graphinspector.GraphInspector.QueryNode;
+import java.io.ByteArrayOutputStream;
+import java.io.PrintStream;
+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 KeptByKeepClassMembersNonStaticTestRunner extends TestBase {
+
+ @NeverClassInline
+ public static class TestClass {
+
+ public static void main(String[] args) {
+ new TestClass().bar();
+ }
+
+ @NeverInline
+ @NeverPropagateValue
+ public void bar() {
+ System.out.println("called bar");
+ }
+
+ @NeverInline
+ public void baz() {
+ System.out.println("called baz");
+ }
+ }
+
+ private static final Class<?> CLASS = TestClass.class;
+ private static final String CLASS_NAME = CLASS.getTypeName();
+ private static final String EXPECTED = StringUtils.lines("called bar");
+
+ private static final String KEEP_CLASS_MEMBERS_RULE =
+ "-keepclassmembers class " + TestClass.class.getTypeName() + " { void baz(); }";
+
+ private static final String EXPECTED_WHYAREYOUKEEPING =
+ StringUtils.lines(
+ "void " + CLASS_NAME + ".baz()",
+ "|- is referenced in keep rule:",
+ "| " + KEEP_CLASS_MEMBERS_RULE,
+ "|- is satisfied with precondition:",
+ "| " + CLASS_NAME,
+ "|- is referenced in keep rule:",
+ "| -keep class " + CLASS_NAME + " { public static void main(java.lang.String[]); }");
+
+ private final TestParameters parameters;
+
+ @Parameters(name = "{0}")
+ public static TestParametersCollection data() {
+ return getTestParameters().withAllRuntimes().withAllApiLevels().build();
+ }
+
+ public KeptByKeepClassMembersNonStaticTestRunner(TestParameters parameters) {
+ this.parameters = parameters;
+ }
+
+ @Test
+ public void testKeptMethod() throws Exception {
+ ClassReference clazz = classFromClass(CLASS);
+ MethodReference mainMethod = methodFromMethod(CLASS.getDeclaredMethod("main", String[].class));
+ MethodReference barMethod = methodFromMethod(CLASS.getDeclaredMethod("bar"));
+ MethodReference bazMethod = methodFromMethod(CLASS.getDeclaredMethod("baz"));
+
+ WhyAreYouKeepingConsumer whyAreYouKeepingConsumer = new WhyAreYouKeepingConsumer(null);
+ GraphInspector inspector =
+ testForR8(parameters.getBackend())
+ .enableGraphInspector(whyAreYouKeepingConsumer)
+ .enableInliningAnnotations()
+ .enableClassInliningAnnotations()
+ .enableMemberValuePropagationAnnotations()
+ .addProgramClasses(CLASS)
+ .addKeepMainRule(CLASS)
+ .addKeepRules(KEEP_CLASS_MEMBERS_RULE)
+ .setMinApi(parameters.getApiLevel())
+ .run(parameters.getRuntime(), CLASS)
+ .assertSuccessWithOutput(EXPECTED)
+ .graphInspector();
+
+ // The only root should be the keep annotation rule.
+ assertEquals(1, inspector.getRoots().size());
+ QueryNode keepMainRule = inspector.rule(Origin.unknown(), 1, 1).assertRoot();
+ QueryNode keepClassMembersRule = inspector.rule(KEEP_CLASS_MEMBERS_RULE).assertNotRoot();
+
+ // Check that the call chain goes from root -> main(unchanged) -> bar(renamed).
+ inspector.method(barMethod).assertRenamed().assertInvokedFrom(mainMethod);
+ inspector.method(mainMethod).assertNotRenamed().assertKeptBy(keepMainRule);
+
+ // Check baz is kept by the dependent rule.
+ inspector.method(bazMethod).assertNotRenamed().assertKeptBy(keepClassMembersRule);
+
+ // Regression for b/131822600, the dependent rule must be satisfied by Bar being kept.
+ keepClassMembersRule.assertSatisfiedBy(inspector.clazz(clazz));
+
+ ByteArrayOutputStream baos = new ByteArrayOutputStream();
+ whyAreYouKeepingConsumer.printWhyAreYouKeeping(bazMethod, new PrintStream(baos));
+
+ // TODO(b/133652414): The type ends up being printed at the reason to keep baz, but should not.
+ assertNotEquals(EXPECTED_WHYAREYOUKEEPING, baos.toString());
+ }
+}