| // 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.enumunboxing; |
| |
| import com.android.tools.r8.NeverClassInline; |
| import com.android.tools.r8.NeverInline; |
| import com.android.tools.r8.NeverMerge; |
| import com.android.tools.r8.R8TestCompileResult; |
| import com.android.tools.r8.R8TestRunResult; |
| import com.android.tools.r8.TestParameters; |
| import java.util.List; |
| 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 InterfaceEnumUnboxingTest extends EnumUnboxingTestBase { |
| |
| private static final Class<?>[] FAILURES = { |
| FailureDefaultMethodUsed.class, FailureUsedAsInterface.class, |
| }; |
| |
| private static final Class<?>[] SUCCESSES = { |
| SuccessAbstractMethod.class, |
| SuccessEmptyInterface.class, |
| SuccessUnusedDefaultMethod.class, |
| SuccessUnusedDefaultMethodOverride.class, |
| SuccessUnusedDefaultMethodOverrideEnum.class |
| }; |
| |
| 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 InterfaceEnumUnboxingTest( |
| TestParameters parameters, boolean enumValueOptimization, EnumKeepRules enumKeepRules) { |
| this.parameters = parameters; |
| this.enumValueOptimization = enumValueOptimization; |
| this.enumKeepRules = enumKeepRules; |
| } |
| |
| @Test |
| public void testEnumUnboxingFailure() throws Exception { |
| R8TestCompileResult compile = |
| testForR8(parameters.getBackend()) |
| .addInnerClasses(InterfaceEnumUnboxingTest.class) |
| .addKeepMainRules(SUCCESSES) |
| .addKeepMainRules(FAILURES) |
| .noMinification() |
| .enableMergeAnnotations() |
| .enableInliningAnnotations() |
| .enableNeverClassInliningAnnotations() |
| .addKeepRules(enumKeepRules.getKeepRules()) |
| .addOptionsModification(opt -> enableEnumOptions(opt, enumValueOptimization)) |
| .allowDiagnosticInfoMessages() |
| .setMinApi(parameters.getApiLevel()) |
| .compile(); |
| for (Class<?> failure : FAILURES) { |
| testClass(compile, failure, true); |
| } |
| for (Class<?> success : SUCCESSES) { |
| testClass(compile, success, false); |
| } |
| } |
| |
| private void testClass(R8TestCompileResult compile, Class<?> testClass, boolean failure) |
| throws Exception { |
| R8TestRunResult run = |
| compile |
| .inspectDiagnosticMessages( |
| m -> { |
| for (Class<?> declaredClass : testClass.getDeclaredClasses()) { |
| if (declaredClass.isEnum()) { |
| if (failure) { |
| assertEnumIsBoxed(declaredClass, testClass.getSimpleName(), m); |
| } else { |
| assertEnumIsUnboxed(declaredClass, testClass.getSimpleName(), m); |
| } |
| } |
| } |
| }) |
| .run(parameters.getRuntime(), testClass) |
| .assertSuccess(); |
| assertLines2By2Correct(run.getStdOut()); |
| } |
| |
| static class SuccessEmptyInterface { |
| |
| public static void main(String[] args) { |
| System.out.println(EnumInterface.A.ordinal()); |
| System.out.println(0); |
| } |
| |
| @NeverClassInline |
| enum EnumInterface implements Itf { |
| A, |
| B, |
| C |
| } |
| |
| @NeverMerge |
| interface Itf {} |
| } |
| |
| static class SuccessUnusedDefaultMethodOverrideEnum { |
| |
| public static void main(String[] args) { |
| System.out.println(EnumInterface.A.ordinal()); |
| System.out.println(0); |
| } |
| |
| @NeverClassInline |
| enum EnumInterface implements Itf { |
| A, |
| B, |
| C |
| } |
| |
| @NeverMerge |
| interface Itf { |
| @NeverInline |
| default int ordinal() { |
| return System.currentTimeMillis() > 0 ? 3 : -3; |
| } |
| } |
| } |
| |
| static class SuccessUnusedDefaultMethodOverride { |
| |
| public static void main(String[] args) { |
| System.out.println(EnumInterface.A.method()); |
| System.out.println(5); |
| } |
| |
| @NeverClassInline |
| enum EnumInterface implements Itf { |
| A, |
| B, |
| C; |
| |
| @Override |
| @NeverInline |
| public int method() { |
| return System.currentTimeMillis() > 0 ? 5 : -5; |
| } |
| } |
| |
| @NeverMerge |
| interface Itf { |
| @NeverInline |
| default int method() { |
| return System.currentTimeMillis() > 0 ? 3 : -3; |
| } |
| } |
| } |
| |
| static class SuccessUnusedDefaultMethod { |
| |
| public static void main(String[] args) { |
| System.out.println(EnumInterface.A.ordinal()); |
| System.out.println(0); |
| } |
| |
| @NeverClassInline |
| enum EnumInterface implements Itf { |
| A, |
| B, |
| C |
| } |
| |
| @NeverMerge |
| interface Itf { |
| @NeverInline |
| default int method() { |
| return System.currentTimeMillis() > 0 ? 3 : -3; |
| } |
| } |
| } |
| |
| static class SuccessAbstractMethod { |
| |
| public static void main(String[] args) { |
| System.out.println(EnumInterface.A.method()); |
| System.out.println(5); |
| } |
| |
| @NeverClassInline |
| enum EnumInterface implements Itf { |
| A, |
| B, |
| C; |
| |
| @Override |
| @NeverInline |
| public int method() { |
| return System.currentTimeMillis() > 0 ? 5 : -5; |
| } |
| } |
| |
| @NeverMerge |
| interface Itf { |
| int method(); |
| } |
| } |
| |
| static class FailureDefaultMethodUsed { |
| |
| public static void main(String[] args) { |
| System.out.println(EnumInterface.A.method()); |
| System.out.println(3); |
| } |
| |
| @NeverClassInline |
| enum EnumInterface implements Itf { |
| A, |
| B, |
| C |
| } |
| |
| @NeverMerge |
| interface Itf { |
| @NeverInline |
| default int method() { |
| return System.currentTimeMillis() > 0 ? 3 : -3; |
| } |
| } |
| } |
| |
| static class FailureUsedAsInterface { |
| |
| public static void main(String[] args) { |
| print(EnumInterface.A); |
| System.out.println(5); |
| } |
| |
| @NeverInline |
| public static void print(Itf itf) { |
| System.out.println(itf.method()); |
| } |
| |
| @NeverClassInline |
| enum EnumInterface implements Itf { |
| A, |
| B, |
| C; |
| |
| @Override |
| @NeverInline |
| public int method() { |
| return System.currentTimeMillis() > 0 ? 5 : -5; |
| } |
| } |
| |
| @NeverMerge |
| interface Itf { |
| @NeverInline |
| default int method() { |
| return System.currentTimeMillis() > 0 ? 3 : -3; |
| } |
| } |
| } |
| } |