Add a test for -checkdiscard

Bug: 162969014
Change-Id: I1bfef38c14c4a487d6a99cc0e02e485468c778e9
diff --git a/src/test/java/com/android/tools/r8/checkdiscarded/B162969014.java b/src/test/java/com/android/tools/r8/checkdiscarded/B162969014.java
new file mode 100644
index 0000000..6eb081f
--- /dev/null
+++ b/src/test/java/com/android/tools/r8/checkdiscarded/B162969014.java
@@ -0,0 +1,108 @@
+// Copyright (c) 2020, 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.checkdiscarded;
+
+import static com.android.tools.r8.utils.codeinspector.Matchers.isPresent;
+import static org.hamcrest.CoreMatchers.not;
+import static org.hamcrest.MatcherAssert.assertThat;
+import static org.junit.Assert.assertEquals;
+import static org.junit.Assert.assertTrue;
+
+import com.android.tools.r8.AssumeNoSideEffects;
+import com.android.tools.r8.CompilationFailedException;
+import com.android.tools.r8.NeverInline;
+import com.android.tools.r8.R8FullTestBuilder;
+import com.android.tools.r8.R8TestCompileResult;
+import com.android.tools.r8.TestBase;
+import com.android.tools.r8.TestParameters;
+import com.android.tools.r8.utils.BooleanUtils;
+import com.android.tools.r8.utils.codeinspector.ClassSubject;
+import com.android.tools.r8.utils.codeinspector.CodeInspector;
+import java.util.List;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.junit.runners.Parameterized;
+
+@RunWith(Parameterized.class)
+public class B162969014 extends TestBase {
+
+  private final boolean checkLogIsDiscarded;
+  private final TestParameters parameters;
+
+  @Parameterized.Parameters(name = "{1}, checkLogIsDiscarded: {0}")
+  public static List<Object[]> data() {
+    return buildParameters(
+        BooleanUtils.values(), getTestParameters().withAllRuntimesAndApiLevels().build());
+  }
+
+  public B162969014(boolean checkLogIsDiscarded, TestParameters parameters) {
+    this.checkLogIsDiscarded = checkLogIsDiscarded;
+    this.parameters = parameters;
+  }
+
+  @Test
+  public void test() throws Exception {
+    R8TestCompileResult compileResult;
+    try {
+      compileResult =
+          testForR8(parameters.getBackend())
+              .addInnerClasses(B162969014.class)
+              .addKeepMainRule(TestClass.class)
+              .apply(this::applyCheckDiscardedRule)
+              .enableAssumeNoSideEffectsAnnotations()
+              .enableInliningAnnotations()
+              .setMinApi(parameters.getApiLevel())
+              .compile();
+    } catch (CompilationFailedException e) {
+      assertTrue(checkLogIsDiscarded);
+      assertEquals(e.getCause().getMessage(), "Discard checks failed.");
+      return;
+    }
+
+    compileResult
+        .inspect(this::inspect)
+        .run(parameters.getRuntime(), TestClass.class)
+        .assertSuccessWithOutputLines("foo");
+  }
+
+  private void applyCheckDiscardedRule(R8FullTestBuilder builder) {
+    if (checkLogIsDiscarded) {
+      builder.addKeepRules(
+          "-checkdiscard class " + Log.class.getTypeName() + "{",
+          "  public static void foo();",
+          "  public static void bar();",
+          "}");
+    }
+  }
+
+  private void inspect(CodeInspector inspector) {
+    ClassSubject logClassSubject = inspector.clazz(Log.class);
+    assertThat(logClassSubject, isPresent());
+    assertThat(logClassSubject.uniqueMethodWithName("foo"), isPresent());
+    assertThat(logClassSubject.uniqueMethodWithName("bar"), not(isPresent()));
+  }
+
+  static class TestClass {
+
+    public static void main(String[] args) {
+      Log.foo();
+      Log.bar();
+    }
+  }
+
+  static class Log {
+
+    @NeverInline
+    public static void foo() {
+      System.out.println("foo");
+    }
+
+    @AssumeNoSideEffects
+    @NeverInline
+    public static void bar() {
+      System.out.println("bar");
+    }
+  }
+}