blob: 944c9c30be3904d66161c228bb281847b5f6d039 [file] [log] [blame]
// 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 static junit.framework.TestCase.assertTrue;
import static org.junit.Assert.assertEquals;
import com.android.tools.r8.NeverInline;
import com.android.tools.r8.R8TestCompileResult;
import com.android.tools.r8.R8TestRunResult;
import com.android.tools.r8.TestParameters;
import com.android.tools.r8.utils.codeinspector.CodeInspector;
import java.util.EnumSet;
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 FailingMethodEnumUnboxingAnalysisTest extends EnumUnboxingTestBase {
private static final Class<?>[] FAILURES = {
InstanceFieldPutObject.class,
StaticFieldPutObject.class,
ToString.class,
EnumSetTest.class,
FailingPhi.class
};
private final TestParameters parameters;
private final boolean enumValueOptimization;
private final boolean enumKeepRules;
@Parameters(name = "{0} valueOpt: {1} keep: {2}")
public static List<Object[]> data() {
return enumUnboxingTestParameters();
}
public FailingMethodEnumUnboxingAnalysisTest(
TestParameters parameters, boolean enumValueOptimization, boolean enumKeepRules) {
this.parameters = parameters;
this.enumValueOptimization = enumValueOptimization;
this.enumKeepRules = enumKeepRules;
}
@Test
public void testEnumUnboxingFailure() throws Exception {
R8TestCompileResult compile =
testForR8(parameters.getBackend())
.addInnerClasses(FailingMethodEnumUnboxingAnalysisTest.class)
.addKeepMainRules(FAILURES)
.addKeepRules(enumKeepRules ? KEEP_ENUM : "")
.addOptionsModification(opt -> enableEnumOptions(opt, enumValueOptimization))
.allowDiagnosticInfoMessages()
.enableInliningAnnotations()
.setMinApi(parameters.getApiLevel())
.compile()
.inspect(this::assertEnumsAsExpected);
for (Class<?> failure : FAILURES) {
R8TestRunResult run =
compile
.inspectDiagnosticMessages(
m ->
assertEnumIsBoxed(
failure.getDeclaredClasses()[0], failure.getSimpleName(), m))
.run(parameters.getRuntime(), failure);
if (failure == EnumSetTest.class && !enumKeepRules) {
// EnumSet and EnumMap cannot be used without the enumKeepRules.
run.assertFailure();
} else {
run.assertSuccess();
assertLines2By2Correct(run.getStdOut());
}
}
}
private void assertEnumsAsExpected(CodeInspector inspector) {
// Check all as expected (else we test nothing)
assertEquals(
1, inspector.clazz(InstanceFieldPutObject.class).getDexClass().instanceFields().size());
assertEquals(
1, inspector.clazz(StaticFieldPutObject.class).getDexClass().staticFields().size());
assertTrue(inspector.clazz(FailingPhi.class).uniqueMethodWithName("switchOn").isPresent());
}
static class InstanceFieldPutObject {
enum MyEnum {
A,
B,
C
}
Object e;
public static void main(String[] args) {
InstanceFieldPutObject fieldPut = new InstanceFieldPutObject();
fieldPut.setA();
Object obj = new Object();
fieldPut.e = obj;
System.out.println(fieldPut.e);
System.out.println(obj);
}
void setA() {
e = MyEnum.A;
}
}
static class StaticFieldPutObject {
enum MyEnum {
A,
B,
C
}
static Object e;
public static void main(String[] args) {
setA();
Object obj = new Object();
e = obj;
System.out.println(e);
System.out.println(obj);
}
static void setA() {
e = MyEnum.A;
}
}
static class ToString {
enum MyEnum {
A,
B,
C
}
public static void main(String[] args) {
MyEnum e1 = MyEnum.A;
System.out.println(e1.toString());
System.out.println("A");
}
}
static class EnumSetTest {
enum MyEnum {
A,
B,
C
}
public static void main(String[] args) {
EnumSet<MyEnum> es = EnumSet.allOf(MyEnum.class);
System.out.println(es.size());
System.out.println("3");
}
}
static class FailingPhi {
enum MyEnum {
A,
B,
C
}
public static void main(String[] args) {
System.out.println(switchOn(1));
System.out.println("B");
System.out.println(switchOn(2));
System.out.println("class java.lang.Object");
}
// Avoid removing the switch entirely.
@NeverInline
static Object switchOn(int i) {
switch (i) {
case 0:
return MyEnum.A;
case 1:
return MyEnum.B;
default:
return Object.class;
}
}
}
}