7cc1371059
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
232 lines
7.9 KiB
Java
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");
|
|
}
|
|
}
|