eaa80ad08c
8300545: Compiler Implementation for Record Patterns Co-authored-by: Aggelos Biboudis <abimpoudis@openjdk.org> Reviewed-by: vromero, mcimadamore
195 lines
7.3 KiB
Java
195 lines
7.3 KiB
Java
/*
|
|
* Copyright (c) 2022, 2023, Oracle and/or its affiliates. All rights reserved.
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* This code is free software; you can redistribute it and/or modify it
|
|
* under the terms of the GNU General Public License version 2 only, as
|
|
* published by the Free Software Foundation.
|
|
*
|
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
* version 2 for more details (a copy is included in the LICENSE file that
|
|
* accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU General Public License version
|
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
*
|
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
* or visit www.oracle.com if you need additional information or have any
|
|
* questions.
|
|
*/
|
|
|
|
/**
|
|
* @test
|
|
* @bug 8291769 8301858 8304694 8304883
|
|
* @summary Verify more complex switches work properly
|
|
* @compile DeconstructionDesugaring.java
|
|
* @run main DeconstructionDesugaring
|
|
*/
|
|
|
|
import java.util.Objects;
|
|
import java.util.function.ToIntFunction;
|
|
public class DeconstructionDesugaring {
|
|
|
|
public static void main(String... args) throws Throwable {
|
|
new DeconstructionDesugaring().test();
|
|
}
|
|
|
|
private void test() {
|
|
test(this::runCheckStatement);
|
|
test(this::runCheckExpression);
|
|
assertEquals(runCheckExpressionWithUnconditional(new R5(new R4(new Sub3()))), 3);
|
|
assertEquals(runCheckExpressionWithUnconditional(new R5(new R4(null))), 3);
|
|
assertEquals(runCheckExpressionWithUnconditional1(new R5(new R4(null))), 2);
|
|
assertEquals(runCheckExpressionWithUnconditional1(new R5(null)), 3);
|
|
assertEquals(runCheckExpressionWithUnconditionalAndParams(new R1(42)), 1);
|
|
assertEquals(runCheckExpressionWithUnconditionalAndParams(new R1(new Object())), 2);
|
|
assertEquals(switchNullable1(new R6(0, 0)), "int: 0, int: 0");
|
|
assertEquals(switchNullable1(new R6(0L, 0)), "obj: 0, obj: 0");
|
|
assertEquals(switchNullable2(new R6(0, 0)), "int: 0, int: 0");
|
|
assertEquals(switchNullable2(new R6(0L, 0)), "obj: 0, int: 0");
|
|
assertEquals(switchNullable2(new R6(0, 0L)), "int: 0, obj: 0");
|
|
assertEquals(switchNullable2(new R6(0L, 0L)), "obj: 0, obj: 0");
|
|
assertEquals(switchNullableNPE(new R6(1, 1)), "obj: 1, obj: 1");
|
|
try {
|
|
switchNullableNPE(new R6(null, 1));
|
|
throw new AssertionError("Expected NPE, but got none.");
|
|
} catch (NullPointerException ex) {
|
|
//expected.
|
|
}
|
|
assertEquals(runFallThrough(new R7(1, 1)), 0);
|
|
assertEquals(runFallThrough(new R7(0, 0)), 1);
|
|
}
|
|
|
|
private void test(ToIntFunction<Object> task) {
|
|
assertEquals(1, task.applyAsInt(new R1(new R2(""))));
|
|
assertEquals(2, task.applyAsInt(new R1(new R2(1))));
|
|
assertEquals(3, task.applyAsInt(new R1(new R2(1.0))));
|
|
assertEquals(-1, task.applyAsInt(new R1(new R2(null))));
|
|
assertEquals(4, task.applyAsInt(new R1(new R2(new StringBuilder()))));
|
|
assertEquals(5, task.applyAsInt(new R1(new R3(""))));
|
|
assertEquals(6, task.applyAsInt(new R1(new R3(1))));
|
|
assertEquals(7, task.applyAsInt(new R1(new R3(1.0))));
|
|
assertEquals(8, task.applyAsInt(new R1(new R3(new StringBuilder()))));
|
|
assertEquals(-1, task.applyAsInt(new R1(1.0f)));
|
|
assertEquals(-1, task.applyAsInt("foo"));
|
|
}
|
|
|
|
private int runCheckStatement(Object o) {
|
|
switch (o) {
|
|
case R1(R2(String s)) -> { return 1; }
|
|
case R1(R2(Integer i)) -> { return 2; }
|
|
case R1(R2(Double d)) -> { return 3; }
|
|
case R1(R2(CharSequence cs)) -> { return 4; }
|
|
case R1(R3(String s)) -> { return 5; }
|
|
case R1(R3(Integer i)) -> { return 6; }
|
|
case R1(R3(Double f)) -> { return 7; }
|
|
case R1(R3(CharSequence cs)) -> { return 8; }
|
|
default -> { return -1; }
|
|
}
|
|
}
|
|
|
|
private int runCheckExpression(Object o) {
|
|
return switch (o) {
|
|
case R1(R2(String s)) -> 1;
|
|
case R1(R2(Integer i)) -> 2;
|
|
case R1(R2(Double d)) -> 3;
|
|
case R1(R2(CharSequence cs)) -> 4;
|
|
case R1(R3(String s)) -> 5;
|
|
case R1(R3(Integer i)) -> 6;
|
|
case R1(R3(Double f)) -> 7;
|
|
case R1(R3(CharSequence cs)) -> 8;
|
|
default -> -1;
|
|
};
|
|
}
|
|
|
|
private int runCheckExpressionWithUnconditional(R5 o) {
|
|
return switch (o) {
|
|
case R5(R4(Sub1 s)) -> 1;
|
|
case R5(R4(Sub2 s)) -> 2;
|
|
case R5(R4(Super s)) -> 3;
|
|
};
|
|
}
|
|
|
|
private int runCheckExpressionWithUnconditional1(R5 o) {
|
|
return switch (o) {
|
|
case R5(R4(Sub1 s)) -> 1;
|
|
case R5(R4(Super s)) -> 2;
|
|
case R5(Object obj) -> 3;
|
|
};
|
|
}
|
|
|
|
public static int runCheckExpressionWithUnconditionalAndParams(R1 r) {
|
|
switch (r) {
|
|
case R1(Integer i):
|
|
return meth_I(i);
|
|
case R1(Object o):
|
|
return meth_O(o);
|
|
}
|
|
}
|
|
|
|
private String switchNullable1(R6 r) {
|
|
return switch (r) {
|
|
case R6(Integer i1, Integer i2) -> "int: " + i1 + ", int: " + i2;
|
|
case R6(Object o1, Object o2) -> "obj: " + o1 + ", obj: " + o2;
|
|
};
|
|
}
|
|
|
|
private String switchNullable2(R6 r) {
|
|
return switch (r) {
|
|
case R6(Integer i1, Integer i2) -> "int: " + i1 + ", int: " + i2;
|
|
case R6(Integer i1, Object o2) -> "int: " + i1 + ", obj: " + o2;
|
|
case R6(Object o1, Integer i2) -> "obj: " + o1 + ", int: " + i2;
|
|
case R6(Object o1, Object o2) -> "obj: " + o1 + ", obj: " + o2;
|
|
};
|
|
}
|
|
|
|
private String switchNullableNPE(R6 r) {
|
|
return switch (r) {
|
|
case R6(Object o1, Object o2) when ((int) o1) == 0 && ((int) o2) == 0 -> "int: " + o1 + ", int: " + o2;
|
|
case R6(Object o1, Object o2) -> "obj: " + o1 + ", obj: " + o2;
|
|
};
|
|
}
|
|
|
|
public static int runFallThrough(R7 r) {
|
|
switch (r) {
|
|
case R7(var v1, var v2) when v1 != 0: return 0;
|
|
case R7(var v1, var v2):
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
public static int meth_I(Integer i) { return 1; }
|
|
public static int meth_O(Object o) { return 2;}
|
|
|
|
private void assertEquals(int expected, int actual) {
|
|
if (expected != actual) {
|
|
throw new AssertionError("expected: " + expected + ", " +
|
|
"actual: " + actual);
|
|
}
|
|
}
|
|
|
|
private void assertEquals(String expected, String actual) {
|
|
if (!Objects.equals(expected, actual)) {
|
|
throw new AssertionError("expected: " + expected + ", " +
|
|
"actual: " + actual);
|
|
}
|
|
}
|
|
|
|
record R1(Object o) {}
|
|
record R2(Object o) {}
|
|
record R3(Object o) {}
|
|
|
|
sealed class Super permits Sub1, Sub2, Sub3 {}
|
|
final class Sub1 extends Super {}
|
|
final class Sub2 extends Super {}
|
|
final class Sub3 extends Super {}
|
|
|
|
record R4(Super o) {}
|
|
record R5(R4 o) {}
|
|
record R6(Object o1, Object o2) {}
|
|
record R7(int i1, int i2) {}
|
|
}
|