2020-06-01 21:00:40 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2020, 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 8225056
|
|
|
|
* @compile GetPermittedSubclasses.jcod
|
2020-12-07 11:11:31 +00:00
|
|
|
* @compile --enable-preview -source ${jdk.version} noSubclass/BaseC.java noSubclass/BaseI.java noSubclass/Impl1.java
|
|
|
|
* @compile --enable-preview -source ${jdk.version} noSubclass/Impl2.java
|
2020-06-01 21:00:40 +00:00
|
|
|
* @compile --enable-preview -source ${jdk.version} GetPermittedSubclassesTest.java
|
|
|
|
* @run main/othervm --enable-preview GetPermittedSubclassesTest
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.util.ArrayList;
|
|
|
|
|
|
|
|
// Test Class GetPermittedSubtpes() and Class.isSealed() APIs.
|
|
|
|
public class GetPermittedSubclassesTest {
|
|
|
|
|
|
|
|
sealed class Sealed1 permits Sub1 {}
|
|
|
|
|
|
|
|
final class Sub1 extends Sealed1 implements SealedI1 {}
|
|
|
|
|
|
|
|
sealed interface SealedI1 permits NotSealed, Sub1, Extender {}
|
|
|
|
|
|
|
|
non-sealed interface Extender extends SealedI1 { }
|
|
|
|
|
|
|
|
final class FinalC implements Extender {}
|
|
|
|
|
|
|
|
final class NotSealed implements SealedI1 {}
|
|
|
|
|
|
|
|
final class Final4 {}
|
|
|
|
|
2020-12-09 19:07:11 +00:00
|
|
|
public static void testSealedInfo(Class<?> c, String[] expected, boolean isSealed) {
|
2020-12-07 11:11:31 +00:00
|
|
|
var permitted = c.getPermittedSubclasses();
|
2020-06-01 21:00:40 +00:00
|
|
|
|
2020-12-09 19:07:11 +00:00
|
|
|
if (isSealed) {
|
|
|
|
if (permitted.length != expected.length) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Unexpected number of permitted subclasses for: " + c.toString() +
|
|
|
|
"(" + java.util.Arrays.asList(permitted));
|
|
|
|
}
|
2020-06-01 21:00:40 +00:00
|
|
|
|
|
|
|
if (!c.isSealed()) {
|
|
|
|
throw new RuntimeException("Expected sealed class: " + c.toString());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create ArrayList of permitted subclasses class names.
|
|
|
|
ArrayList<String> permittedNames = new ArrayList<String>();
|
|
|
|
for (int i = 0; i < permitted.length; i++) {
|
2020-12-07 11:11:31 +00:00
|
|
|
permittedNames.add(permitted[i].getName());
|
2020-06-01 21:00:40 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (permittedNames.size() != expected.length) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Unexpected number of permitted names for: " + c.toString());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that expected class names are in the permitted subclasses list.
|
|
|
|
for (int i = 0; i < expected.length; i++) {
|
|
|
|
if (!permittedNames.contains(expected[i])) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Expected class not found in permitted subclases list, super class: " +
|
|
|
|
c.getName() + ", expected class: " + expected[i]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2020-12-09 19:07:11 +00:00
|
|
|
// Must not be sealed.
|
|
|
|
if (c.isSealed() || permitted != null) {
|
2020-06-01 21:00:40 +00:00
|
|
|
throw new RuntimeException("Unexpected sealed class: " + c.toString());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-12-09 19:07:11 +00:00
|
|
|
public static void testBadSealedClass(String className,
|
|
|
|
Class<?> expectedException,
|
|
|
|
String expectedCFEMessage) throws Throwable {
|
2020-06-01 21:00:40 +00:00
|
|
|
try {
|
|
|
|
Class.forName(className);
|
|
|
|
throw new RuntimeException("Expected ClassFormatError exception not thrown for " + className);
|
|
|
|
} catch (ClassFormatError cfe) {
|
2020-12-09 19:07:11 +00:00
|
|
|
if (ClassFormatError.class != expectedException) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Class " + className + " got unexpected exception: " + cfe.getMessage());
|
|
|
|
}
|
2020-06-01 21:00:40 +00:00
|
|
|
if (!cfe.getMessage().contains(expectedCFEMessage)) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Class " + className + " got unexpected ClassFormatError exception: " + cfe.getMessage());
|
|
|
|
}
|
2020-12-09 19:07:11 +00:00
|
|
|
} catch (IncompatibleClassChangeError icce) {
|
|
|
|
if (IncompatibleClassChangeError.class != expectedException) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Class " + className + " got unexpected exception: " + icce.getMessage());
|
|
|
|
}
|
|
|
|
if (!icce.getMessage().contains(expectedCFEMessage)) {
|
|
|
|
throw new RuntimeException(
|
|
|
|
"Class " + className + " got unexpected IncompatibleClassChangeError exception: " + icce.getMessage());
|
|
|
|
}
|
2020-06-01 21:00:40 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public static void main(String... args) throws Throwable {
|
2020-12-07 11:11:31 +00:00
|
|
|
testSealedInfo(SealedI1.class, new String[] {"GetPermittedSubclassesTest$NotSealed",
|
|
|
|
"GetPermittedSubclassesTest$Sub1",
|
2020-12-09 19:07:11 +00:00
|
|
|
"GetPermittedSubclassesTest$Extender"},
|
|
|
|
true);
|
2020-12-07 11:11:31 +00:00
|
|
|
|
2020-12-09 19:07:11 +00:00
|
|
|
testSealedInfo(Sealed1.class, new String[] {"GetPermittedSubclassesTest$Sub1"}, true);
|
|
|
|
testSealedInfo(Final4.class, null, false);
|
|
|
|
testSealedInfo(NotSealed.class, null, false);
|
2020-06-01 21:00:40 +00:00
|
|
|
|
|
|
|
// Test class with PermittedSubclasses attribute but old class file version.
|
2020-12-09 19:07:11 +00:00
|
|
|
testSealedInfo(OldClassFile.class, null, false);
|
2020-06-01 21:00:40 +00:00
|
|
|
|
|
|
|
// Test class with an empty PermittedSubclasses attribute.
|
2020-12-09 19:07:11 +00:00
|
|
|
testSealedInfo(NoSubclasses.class, new String[]{}, true);
|
|
|
|
|
|
|
|
// Test that a class with an empty PermittedSubclasses attribute cannot be subclass-ed.
|
|
|
|
testBadSealedClass("SubClass", IncompatibleClassChangeError.class,
|
|
|
|
"SubClass cannot inherit from sealed class NoSubclasses");
|
2020-06-01 21:00:40 +00:00
|
|
|
|
2020-12-07 11:11:31 +00:00
|
|
|
// Test returning only names of existing classes.
|
2020-12-09 19:07:11 +00:00
|
|
|
testSealedInfo(NoLoadSubclasses.class, new String[]{"ExistingClassFile" }, true);
|
2020-06-01 21:00:40 +00:00
|
|
|
|
|
|
|
// Test that loading a class with a corrupted PermittedSubclasses attribute
|
|
|
|
// causes a ClassFormatError.
|
2020-12-09 19:07:11 +00:00
|
|
|
testBadSealedClass("BadPermittedAttr", ClassFormatError.class,
|
|
|
|
"Permitted subclass class_info_index 15 has bad constant type");
|
2020-06-01 21:00:40 +00:00
|
|
|
|
|
|
|
// Test that loading a sealed final class with a PermittedSubclasses
|
|
|
|
// attribute causes a ClassFormatError.
|
2020-12-09 19:07:11 +00:00
|
|
|
testBadSealedClass("SealedButFinal", ClassFormatError.class,
|
|
|
|
"PermittedSubclasses attribute in final class");
|
2020-06-01 21:00:40 +00:00
|
|
|
|
2020-12-09 19:07:11 +00:00
|
|
|
// Test that loading a sealed class with an ill-formed class name in its
|
|
|
|
// PermittedSubclasses attribute causes a ClassFormatError.
|
|
|
|
testBadSealedClass("BadPermittedSubclassEntry", ClassFormatError.class,
|
|
|
|
"Illegal class name \"iDont;;Exist\" in class file");
|
2020-06-01 21:00:40 +00:00
|
|
|
|
|
|
|
// Test that loading a sealed class with an empty class name in its PermittedSubclasses
|
|
|
|
// attribute causes a ClassFormatError.
|
2020-12-09 19:07:11 +00:00
|
|
|
testBadSealedClass("EmptyPermittedSubclassEntry", ClassFormatError.class,
|
|
|
|
"Illegal class name \"\" in class file");
|
2020-12-07 11:11:31 +00:00
|
|
|
|
|
|
|
//test type enumerated in the PermittedSubclasses attribute,
|
|
|
|
//which are not direct subtypes of the current class are not returned:
|
2020-12-09 19:07:11 +00:00
|
|
|
testSealedInfo(noSubclass.BaseC.class, new String[] {"noSubclass.ImplCIntermediate"}, true);
|
|
|
|
testSealedInfo(noSubclass.BaseI.class, new String[] {"noSubclass.ImplIIntermediateI", "noSubclass.ImplIIntermediateC"}, true);
|
2020-06-01 21:00:40 +00:00
|
|
|
}
|
|
|
|
}
|