Add test for unrelated instance field on unboxed enum

Bug: b/71823
Change-Id: I945f3e117329eb4fb35e1b5bd0c9cfb8254adef9
diff --git a/src/main/java/com/android/tools/r8/graph/AppView.java b/src/main/java/com/android/tools/r8/graph/AppView.java
index dc99fda..d262e6a 100644
--- a/src/main/java/com/android/tools/r8/graph/AppView.java
+++ b/src/main/java/com/android/tools/r8/graph/AppView.java
@@ -736,6 +736,9 @@
   }
 
   public boolean validateUnboxedEnumsHaveBeenPruned() {
+    if (app().options.testing.allowNotPrunedUnboxedEnums) {
+      return true;
+    }
     for (DexType unboxedEnum : unboxedEnums.getUnboxedEnums()) {
       assert appInfo.definitionForWithoutExistenceAssert(unboxedEnum) == null
           : "Enum " + unboxedEnum + " has been unboxed but is still in the program.";
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 7f3a36a..ce10279 100644
--- a/src/main/java/com/android/tools/r8/shaking/Enqueuer.java
+++ b/src/main/java/com/android/tools/r8/shaking/Enqueuer.java
@@ -2060,7 +2060,9 @@
             || appView.appInfo().getMainDexInfo().isTracedRoot(clazz, appView.getSyntheticItems())
         : "Class " + clazz.toSourceString() + " was not a main dex root in the first round";
 
-    assert !appView.unboxedEnums().isUnboxedEnum(clazz);
+    assert appView.options().testing.allowNotPrunedUnboxedEnums
+            || !appView.unboxedEnums().isUnboxedEnum(clazz)
+        : "Enum " + clazz.toSourceString() + " has been unboxed but is still in the program.";
 
     if (options.isGeneratingClassFiles() && clazz.hasPermittedSubclassAttributes()) {
       throw new CompilationError(
diff --git a/src/main/java/com/android/tools/r8/utils/InternalOptions.java b/src/main/java/com/android/tools/r8/utils/InternalOptions.java
index 6bf0a8f..33b7e72 100644
--- a/src/main/java/com/android/tools/r8/utils/InternalOptions.java
+++ b/src/main/java/com/android/tools/r8/utils/InternalOptions.java
@@ -2043,6 +2043,7 @@
         System.getProperty("com.android.tools.r8.disableMarkingClassesFinal") != null;
     public boolean testEnableTestAssertions = false;
     public boolean keepMetadataInR8IfNotRewritten = true;
+    public boolean allowNotPrunedUnboxedEnums = false;
 
     // If set, pruned record fields are not used in hashCode/equals/toString and toString prints
     // minified field names instead of original field names.
diff --git a/src/test/java/com/android/tools/r8/enumunboxing/EnumInstanceFieldReferenceInsideAndOutsideTest.java b/src/test/java/com/android/tools/r8/enumunboxing/EnumInstanceFieldReferenceInsideAndOutsideTest.java
new file mode 100644
index 0000000..8be968b
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/enumunboxing/EnumInstanceFieldReferenceInsideAndOutsideTest.java
@@ -0,0 +1,125 @@
+// Copyright (c) 2022, 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.enumunboxing;
+
+import static org.hamcrest.CoreMatchers.containsString;
+import static org.junit.Assert.assertThrows;
+
+import com.android.tools.r8.CompilationFailedException;
+import com.android.tools.r8.NeverClassInline;
+import com.android.tools.r8.TestParameters;
+import com.android.tools.r8.ToolHelper.DexVm.Version;
+import java.util.List;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+import org.junit.runners.Parameterized.Parameters;
+
+/**
+ * This is a variant of a regression test for b/247146910 where the instance field on the enum is
+ * unrelated to enum unboxing.
+ */
+@RunWith(Parameterized.class)
+public class EnumInstanceFieldReferenceInsideAndOutsideTest extends EnumUnboxingTestBase {
+
+  private final TestParameters parameters;
+  private final boolean enumValueOptimization;
+  private final EnumKeepRules enumKeepRules;
+
+  @Parameters(name = "{0} valueOpt: {1} keep: {2}")
+  public static List<Object[]> data() {
+    return enumUnboxingTestParameters();
+  }
+
+  public EnumInstanceFieldReferenceInsideAndOutsideTest(
+      TestParameters parameters, boolean enumValueOptimization, EnumKeepRules enumKeepRules) {
+    this.parameters = parameters;
+    this.enumValueOptimization = enumValueOptimization;
+    this.enumKeepRules = enumKeepRules;
+  }
+
+  @Test
+  public void testRuntime() throws Exception {
+    testForRuntime(parameters)
+        .addInnerClasses(EnumInstanceFieldReferenceInsideAndOutsideTest.class)
+        .run(parameters.getRuntime(), Main.class)
+        .assertSuccessWithOutputLines("42");
+  }
+
+  @Test
+  public void testEnumUnboxing() throws Exception {
+    // TODO(b/247146910): Should not throw.
+    assertThrows(
+        CompilationFailedException.class,
+        () ->
+            testForR8(parameters.getBackend())
+                .addInnerClasses(EnumInstanceFieldReferenceInsideAndOutsideTest.class)
+                .addKeepMainRule(Main.class)
+                .addKeepRules(enumKeepRules.getKeepRules())
+                .addOptionsModification(opt -> enableEnumOptions(opt, enumValueOptimization))
+                .addEnumUnboxingInspector(inspector -> inspector.assertUnboxed(MyEnum.class))
+                .addNeverClassInliningAnnotations()
+                .setMinApi(parameters.getApiLevel())
+                .addOptionsModification(options -> options.testing.allowTypeErrors = true)
+                .allowDiagnosticWarningMessages()
+                .compileWithExpectedDiagnostics(
+                    diagnostics ->
+                        diagnostics.assertErrorMessageThatMatches(
+                            containsString(
+                                "Enum "
+                                    + typeName(MyEnum.class)
+                                    + " has been unboxed but is still in the program"))));
+  }
+
+  private boolean hasVerifyError() {
+    return parameters.getDexRuntimeVersion().isDalvik()
+        || parameters.getDexRuntimeVersion().isNewerThanOrEqual(Version.V12_0_0);
+  }
+
+  @Test
+  public void testEnumUnboxingAllowNotPrunedEnums() throws Exception {
+    testForR8(parameters.getBackend())
+        .addInnerClasses(EnumInstanceFieldReferenceInsideAndOutsideTest.class)
+        .addKeepMainRule(Main.class)
+        .addKeepRules(enumKeepRules.getKeepRules())
+        .addOptionsModification(
+            opt -> {
+              enableEnumOptions(opt, enumValueOptimization);
+              opt.testing.allowNotPrunedUnboxedEnums = true;
+            })
+        .addEnumUnboxingInspector(inspector -> inspector.assertUnboxed(MyEnum.class))
+        .addNeverClassInliningAnnotations()
+        .setMinApi(parameters.getApiLevel())
+        .addOptionsModification(options -> options.testing.allowTypeErrors = true)
+        .run(parameters.getRuntime(), Main.class)
+        // TODO(b/247146910): Should not fail with runtime error.
+        .assertFailureWithErrorThatThrowsIf(hasVerifyError(), VerifyError.class)
+        .assertFailureWithErrorThatThrowsIf(!hasVerifyError(), NoSuchFieldError.class);
+  }
+
+  @NeverClassInline
+  public enum MyEnum {
+    A(10),
+    B(20);
+
+    public int instanceValue;
+
+    MyEnum(int instanceValue) {
+      this.instanceValue = instanceValue;
+    }
+  }
+
+  public static class Main {
+
+    public static void main(String[] args) throws Exception {
+      set(System.currentTimeMillis() > 0 ? 42 : 0);
+      System.out.println(MyEnum.B.instanceValue);
+    }
+
+    public static void set(int newValue) {
+      MyEnum.B.instanceValue = newValue;
+    }
+  }
+}
diff --git a/src/test/java/com/android/tools/r8/enumunboxing/EnumInstanceFieldReferenceOutsideTest.java b/src/test/java/com/android/tools/r8/enumunboxing/EnumInstanceFieldReferenceOutsideTest.java
new file mode 100644
index 0000000..ff0b642
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/enumunboxing/EnumInstanceFieldReferenceOutsideTest.java
@@ -0,0 +1,121 @@
+// Copyright (c) 2022, 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.enumunboxing;
+
+import static org.hamcrest.CoreMatchers.containsString;
+import static org.junit.Assert.assertThrows;
+
+import com.android.tools.r8.CompilationFailedException;
+import com.android.tools.r8.NeverClassInline;
+import com.android.tools.r8.TestParameters;
+import com.android.tools.r8.ToolHelper.DexVm.Version;
+import java.util.List;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+import org.junit.runners.Parameterized.Parameters;
+
+/**
+ * This is a variant of a regression test for b/247146910 where the instance field on the enum is
+ * unrelated to enum unboxing.
+ */
+@RunWith(Parameterized.class)
+public class EnumInstanceFieldReferenceOutsideTest extends EnumUnboxingTestBase {
+
+  private final TestParameters parameters;
+  private final boolean enumValueOptimization;
+  private final EnumKeepRules enumKeepRules;
+
+  @Parameters(name = "{0} valueOpt: {1} keep: {2}")
+  public static List<Object[]> data() {
+    return enumUnboxingTestParameters();
+  }
+
+  public EnumInstanceFieldReferenceOutsideTest(
+      TestParameters parameters, boolean enumValueOptimization, EnumKeepRules enumKeepRules) {
+    this.parameters = parameters;
+    this.enumValueOptimization = enumValueOptimization;
+    this.enumKeepRules = enumKeepRules;
+  }
+
+  @Test
+  public void testRuntime() throws Exception {
+    testForRuntime(parameters)
+        .addInnerClasses(EnumInstanceFieldReferenceOutsideTest.class)
+        .run(parameters.getRuntime(), Main.class)
+        .assertSuccessWithOutputLines("42");
+  }
+
+  @Test
+  public void testEnumUnboxing() throws Exception {
+    // TODO(b/247146910): Should not throw.
+    assertThrows(
+        CompilationFailedException.class,
+        () ->
+            testForR8(parameters.getBackend())
+                .addInnerClasses(EnumInstanceFieldReferenceOutsideTest.class)
+                .addKeepMainRule(Main.class)
+                .addKeepRules(enumKeepRules.getKeepRules())
+                .addOptionsModification(opt -> enableEnumOptions(opt, enumValueOptimization))
+                .addEnumUnboxingInspector(inspector -> inspector.assertUnboxed(MyEnum.class))
+                .addNeverClassInliningAnnotations()
+                .setMinApi(parameters.getApiLevel())
+                .addOptionsModification(options -> options.testing.allowTypeErrors = true)
+                .allowDiagnosticWarningMessages()
+                .compileWithExpectedDiagnostics(
+                    diagnostics ->
+                        diagnostics.assertErrorMessageThatMatches(
+                            containsString(
+                                "Enum "
+                                    + typeName(MyEnum.class)
+                                    + " has been unboxed but is still in the program"))));
+  }
+
+  private boolean hasVerifyError() {
+    return parameters.getDexRuntimeVersion().isDalvik()
+        || parameters.getDexRuntimeVersion().isNewerThanOrEqual(Version.V12_0_0);
+  }
+
+  @Test
+  public void testEnumUnboxingAllowNotPrunedEnums() throws Exception {
+    testForR8(parameters.getBackend())
+        .addInnerClasses(EnumInstanceFieldReferenceOutsideTest.class)
+        .addKeepMainRule(Main.class)
+        .addKeepRules(enumKeepRules.getKeepRules())
+        .addOptionsModification(
+            opt -> {
+              enableEnumOptions(opt, enumValueOptimization);
+              opt.testing.allowNotPrunedUnboxedEnums = true;
+            })
+        .addEnumUnboxingInspector(inspector -> inspector.assertUnboxed(MyEnum.class))
+        .addNeverClassInliningAnnotations()
+        .setMinApi(parameters.getApiLevel())
+        .addOptionsModification(options -> options.testing.allowTypeErrors = true)
+        .run(parameters.getRuntime(), Main.class)
+        // TODO(b/247146910): Should not fail with runtime error.
+        .assertFailureWithErrorThatThrowsIf(hasVerifyError(), VerifyError.class)
+        .assertFailureWithErrorThatThrowsIf(!hasVerifyError(), NoSuchFieldError.class);
+  }
+
+  @NeverClassInline
+  public enum MyEnum {
+    A,
+    B;
+
+    public int instanceValue = 1;
+  }
+
+  public static class Main {
+
+    public static void main(String[] args) throws Exception {
+      set(System.currentTimeMillis() > 0 ? 42 : 0);
+      System.out.println(MyEnum.B.instanceValue);
+    }
+
+    public static void set(int newValue) {
+      MyEnum.B.instanceValue = newValue;
+    }
+  }
+}