Mandy Chung 7cc1371059 8238358: Implementation of JEP 371: Hidden Classes
Co-authored-by: Lois Foltan <lois.foltan@oracle.com>
Co-authored-by: David Holmes <david.holmes@oracle.com>
Co-authored-by: Harold Seigel <harold.seigel@oracle.com>
Co-authored-by: Serguei Spitsyn <serguei.spitsyn@oracle.com>
Co-authored-by: Alex Buckley <alex.buckley@oracle.com>
Co-authored-by: Jamsheed Mohammed C M <jamsheed.c.m@oracle.com>
Co-authored-by: Jan Lahoda <jan.lahoda@oracle.com>
Co-authored-by: Amy Lu <amy.lu@oracle.com>
Reviewed-by: alanb, cjplummer, coleenp, dholmes, dlong, forax, jlahoda, psandoz, plevart, sspitsyn, vromero
2020-04-21 06:55:38 -07:00

232 lines
7.9 KiB
Java

/*
* Copyright (c) 2019, 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
* @library /test/lib
* @modules java.base/jdk.internal.misc
* jdk.compiler
* @compile HiddenClassSigTest.java
* @run main/othervm/native -agentlib:HiddenClassSigTest P.Q.HiddenClassSigTest
*/
package P.Q;
import java.io.ByteArrayOutputStream;
import java.io.File;
import java.io.FileInputStream;
import java.lang.invoke.MethodHandles;
import java.lang.invoke.MethodHandles.Lookup;
import jdk.test.lib.Utils;
import jdk.test.lib.compiler.InMemoryJavaCompiler;
interface Test<T> {
String test(T t);
}
class HiddenClassSig<T> implements Test<T> {
private String realTest() { return "HiddenClassSig: "; }
public String test(T t) {
String str = realTest();
return str + t.toString();
}
}
public class HiddenClassSigTest {
private static void log(String str) { System.out.println(str); }
private static final String HCName = "P/Q/HiddenClassSig.class";
private static final String DIR = Utils.TEST_CLASSES;
private static final String LOG_PREFIX = "HiddenClassSigTest: ";
static native void checkHiddenClass(Class klass, String sig);
static native void checkHiddenClassArray(Class array, String sig);
static native boolean checkFailed();
static {
try {
System.loadLibrary("HiddenClassSigTest");
} catch (UnsatisfiedLinkError ule) {
System.err.println("Could not load HiddenClassSigTest library");
System.err.println("java.library.path: "
+ System.getProperty("java.library.path"));
throw ule;
}
}
static byte[] readClassFile(String classFileName) throws Exception {
File classFile = new File(classFileName);
try (FileInputStream in = new FileInputStream(classFile);
ByteArrayOutputStream out = new ByteArrayOutputStream())
{
int b;
while ((b = in.read()) != -1) {
out.write(b);
}
return out.toByteArray();
}
}
static Class<?> defineHiddenClass(String classFileName) throws Exception {
Lookup lookup = MethodHandles.lookup();
byte[] bytes = readClassFile(DIR + File.separator + classFileName);
Class<?> hc = lookup.defineHiddenClass(bytes, false).lookupClass();
return hc;
}
static void logClassInfo(Class<?> klass) {
log("\n### Testing class: " + klass);
log(LOG_PREFIX + "isHidden: " + klass.isHidden());
log(LOG_PREFIX + "getName: " + klass.getName());
log(LOG_PREFIX + "typeName: " + klass.getTypeName());
log(LOG_PREFIX + "toString: " + klass.toString());
log(LOG_PREFIX + "toGenStr: " + klass.toGenericString());
log(LOG_PREFIX + "elem type: " + klass.componentType());
}
private static final String HC_NAME = "P.Q.HiddenClassSig";
private static final String HC_SUFFIX_REGEX = "0x[0-9a-f]+";
static boolean checkName(Class<?> klass, String name, String toString) {
boolean failed = false;
String regex = "";
Class<?> c = klass;
while (c.isArray()) {
regex = "\\[" + regex;
c = c.componentType();
}
if (klass.isArray()) {
regex += "L" + HC_NAME + "/" + HC_SUFFIX_REGEX + ";";
} else {
regex = HC_NAME + "/" + HC_SUFFIX_REGEX;
}
if (!name.matches(regex)) {
log("Test FAIL: result of Class::getName" + " \"" + name + "\" does not match " + regex);
failed = true;
}
if (!toString.matches("class " + regex)) {
log("Test FAIL: result of Class::toString" + " \"" + name + "\" does not match " + regex);
failed = true;
}
return failed;
}
static boolean checkTypeName(Class<?> klass, String name) {
boolean failed = false;
String regex = HC_NAME + "/" + HC_SUFFIX_REGEX;
Class<?> c = klass;
while (c.isArray()) {
c = c.componentType();
regex = regex + "\\[\\]";
}
if (!name.matches(regex)) {
log("Test FAIL: result of Class::getTypeName" + " \"" + name + "\" does not match " + regex);
failed = true;
}
return failed;
}
static boolean checkGenericString(Class<?> klass, String name) {
boolean failed = false;
Class<?> c = klass;
String regex = HC_NAME + "/" + HC_SUFFIX_REGEX + "<T>";
if (!klass.isArray()) {
regex = "class " + regex;
}
while (c.isArray()) {
c = c.componentType();
regex = regex + "\\[\\]";
}
if (!name.matches(regex)) {
log("Test FAIL: result of Class::toGenericString" + " \"" + name + "\" does not match " + regex);
failed = true;
}
return failed;
}
static boolean checkDescriptorString(Class<?> klass, String name) {
boolean failed = false;
String regex = "L" + HC_NAME.replace('.', '/') + "." + HC_SUFFIX_REGEX + ";";
Class<?> c = klass;
while (c.isArray()) {
regex = "\\[" + regex;
c = c.componentType();
}
if (!name.matches(regex)) {
log("Test FAIL: result of Class::descriptorString" + " \"" + name + "\" does not match " + regex);
failed = true;
}
return failed;
}
static boolean testClass(Class<?> klass) {
boolean failed = false;
logClassInfo(klass);
failed |= checkName(klass, klass.getName(), klass.toString());
failed |= checkTypeName(klass, klass.getTypeName());
failed |= checkGenericString(klass, klass.toGenericString());
failed |= checkDescriptorString(klass, klass.descriptorString());
if (klass.isArray() && klass.isHidden()) {
log("Test FAIL: an array class is never hidden");
failed = true;
}
if (klass.isArray()) {
checkHiddenClassArray(klass, klass.descriptorString());
} else {
checkHiddenClass(klass, klass.descriptorString());
}
return failed;
}
public static void main(String args[]) throws Exception {
log(LOG_PREFIX + "started");
Class<?> hc = defineHiddenClass(HCName);
String baseName = ("" + hc).substring("class ".length());
Test<String> t = (Test<String>)hc.newInstance();
String str = t.test("Test generic hidden class");
log(LOG_PREFIX + "hc.test() returned string: " + str);
boolean failed = testClass(hc);
Class<?> hcArr = hc.arrayType();
failed |= testClass(hcArr);
Class<?> hcArrArr = hcArr.arrayType();
failed |= testClass(hcArrArr);
if (failed) {
throw new RuntimeException("FAIL: failed status from java part");
}
if (checkFailed()) {
throw new RuntimeException("FAIL: failed status from native agent");
}
log(LOG_PREFIX + "finished");
}
}