jdk-24/test/jdk/java/lang/invoke/8022701/MHIllegalAccess.java

113 lines
4.3 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2013, 2024, 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 8022701
* @summary Illegal access exceptions via methodhandle invocations threw wrong error.
* @enablePreview
* @compile -XDignore.symbol.file BogoLoader.java InvokeSeveralWays.java MHIllegalAccess.java MethodSupplier.java
* @run main/othervm MHIllegalAccess
*/
import java.lang.classfile.AccessFlags;
import java.lang.classfile.ClassTransform;
import java.lang.classfile.MethodModel;
import java.lang.reflect.InvocationTargetException;
import java.util.HashMap;
import java.util.HashSet;
import static java.lang.classfile.ClassFile.ACC_PRIVATE;
import static java.lang.classfile.ClassFile.ACC_PROTECTED;
import static java.lang.classfile.ClassFile.ACC_PUBLIC;
public class MHIllegalAccess {
public static void main(String[] args) throws Throwable {
System.out.println("Classpath is " + System.getProperty("java.class.path"));
System.out.println();
/**
* Make method m be private to provoke an IllegalAccessError.
*/
var privatize = ClassTransform.transformingMethods(m -> m.methodName().equalsString("m"), (mb, me) -> {
if (me instanceof AccessFlags af) {
mb.withFlags((af.flagsMask() | ACC_PRIVATE) & ~ (ACC_PUBLIC | ACC_PROTECTED));
} else {
mb.accept(me);
}
});
/**
* Rename method m as nemo to provoke a NoSuchMethodError.
*/
ClassTransform changeName = (cb, ce) -> {
if (ce instanceof MethodModel mm && mm.methodName().equalsString("m")) {
cb.withMethod("nemo", mm.methodTypeSymbol(), mm.flags().flagsMask(), mm::forEach);
} else {
cb.accept(ce);
}
};
int failures = 0;
failures += testOneError(privatize, args, IllegalAccessError.class);
failures += testOneError(changeName, args, NoSuchMethodError.class);
if (failures > 0) {
System.out.println("Saw " + failures + " failures, see standard out for details");
throw new Error("FAIL test");
}
}
/**
*
* @param vm VisitorMaker, to be stored in a table and passed to a BogoLoader
* @param args A copy of the main args, to be passed on to InvokeSeveralWays.test
* @param expected The class of the exception that should be thrown after
* attempted invocation of MethodSupplier.m.
* @throws ClassNotFoundException
* @throws Throwable
*/
private static int testOneError(ClassTransform vm, String[] args, Class<?> expected) throws ClassNotFoundException, Throwable {
var replace = new HashMap<String, ClassTransform>();
replace.put("MethodSupplier", vm);
HashSet<String> in_bogus = new HashSet<String>();
in_bogus.add("InvokeSeveralWays");
in_bogus.add("MyFunctionalInterface");
in_bogus.add("Invoker");
BogoLoader bl = new BogoLoader(in_bogus, replace);
Class<?> isw = bl.loadClass("InvokeSeveralWays");
Object[] arg_for_args = new Object[2];
arg_for_args[0] = args;
arg_for_args[1] = expected;
try {
Object result = isw.getMethod("test", String[].class, Class.class).invoke(null, arg_for_args);
return (Integer)result;
} catch (InvocationTargetException e) {
Throwable th = e.getCause();
throw th == null ? e : th;
}
}
}