89 lines
4.0 KiB
Java
Raw Normal View History

/*
* 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.
*
*/
import java.lang.StackWalker.StackFrame;
import java.util.List;
import java.util.stream.Collectors;
import java.util.Set;
import sun.hotspot.WhiteBox;
public class LambdaVerification {
static void verifyCallerIsArchivedLambda(boolean isRuntime) {
// getCallerClass(0) = verifyCallerIsArchivedLambda
// getCallerClass(1) = testNonCapturingLambda or testCapturingLambda
// getCallerClass(2) = the lambda proxy class that should be archived by CDS.
Class<?> c = getCallerClass(2);
System.out.println("Lambda proxy class = " + c);
String cn = c.getName();
System.out.println(" cn = " + cn);
String hiddenClassName = cn.replace('/', '+');
String lambdaClassName = cn.substring(0, cn.lastIndexOf('/'));
System.out.println(" lambda name = " + lambdaClassName);
WhiteBox wb = WhiteBox.getWhiteBox();
if (isRuntime) {
// check that c is a shared class
if (wb.isSharedClass(c)) {
System.out.println("As expected, " + c + " is in shared space.");
} else {
throw new java.lang.RuntimeException(c + " must be in shared space.");
}
// check that lambda class cannot be found manually
try {
Class.forName(hiddenClassName);
} catch (ClassNotFoundException cnfe) {
cnfe.printStackTrace(System.out);
System.out.println("As expected, loading of " + hiddenClassName + " should result in ClassNotFoundException.");
} catch (Exception ex) {
throw ex;
}
// check that lambda class is alive
if (wb.isClassAlive(hiddenClassName)) {
System.out.println("As expected, " + cn + " is alive.");
} else {
throw new java.lang.RuntimeException(cn + " should be alive.");
}
} else {
if (wb.isSharedClass(c)) {
throw new java.lang.RuntimeException(c + " must not be in shared space.");
} else {
System.out.println("As expected, " + c + " is not in shared space.");
}
}
//System.out.println("=== Here's the call stack");
//(new Throwable()).printStackTrace(System.out);
//System.out.println("===");
System.out.println("Succeeded");
}
// depth is 0-based -- i.e., depth==0 returns the class of the immediate caller of getCallerClass
static Class<?> getCallerClass(int depth) {
// Need to add the frame of the getCallerClass -- so the immediate caller (depth==0) of this method
// is at stack.get(1) == stack.get(depth+1);
StackWalker walker = StackWalker.getInstance(
Set.of(StackWalker.Option.RETAIN_CLASS_REFERENCE,
StackWalker.Option.SHOW_HIDDEN_FRAMES));
List<StackFrame> stack = walker.walk(s -> s.limit(depth+2).collect(Collectors.toList()));
return stack.get(depth+1).getDeclaringClass();
}
}