aed3ea2043
Reviewed-by: jlaskey
272 lines
12 KiB
Java
272 lines
12 KiB
Java
/*
|
|
* Copyright (c) 2021, 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 8277634
|
|
* @summary Verify the correct constantpool entries are created for invokedynamic instructions using
|
|
* the same bootstrap and type, but different name.
|
|
* @library /tools/lib
|
|
* @modules jdk.compiler/com.sun.tools.javac.api
|
|
* jdk.compiler/com.sun.tools.javac.code
|
|
* jdk.compiler/com.sun.tools.javac.comp
|
|
* jdk.compiler/com.sun.tools.javac.jvm
|
|
* jdk.compiler/com.sun.tools.javac.main
|
|
* jdk.compiler/com.sun.tools.javac.tree
|
|
* jdk.compiler/com.sun.tools.javac.util
|
|
* jdk.jdeps/com.sun.tools.classfile
|
|
* jdk.jdeps/com.sun.tools.javap
|
|
* @build toolbox.JarTask toolbox.JavacTask toolbox.JavapTask toolbox.ToolBox
|
|
* @run main IndyCorrectInvocationName
|
|
*/
|
|
|
|
import java.net.URL;
|
|
import java.net.URLClassLoader;
|
|
import java.nio.file.DirectoryStream;
|
|
import java.nio.file.Files;
|
|
import java.nio.file.Path;
|
|
import java.util.HashSet;
|
|
import java.util.Objects;
|
|
import java.util.Set;
|
|
|
|
import com.sun.source.util.JavacTask;
|
|
import com.sun.source.util.Plugin;
|
|
import com.sun.source.util.TaskEvent;
|
|
import com.sun.source.util.TaskListener;
|
|
|
|
import com.sun.tools.classfile.Attribute;
|
|
import com.sun.tools.classfile.BootstrapMethods_attribute;
|
|
import com.sun.tools.classfile.ClassFile;
|
|
import com.sun.tools.classfile.Code_attribute;
|
|
import com.sun.tools.classfile.ConstantPool.CONSTANT_InvokeDynamic_info;
|
|
import com.sun.tools.classfile.ConstantPool.CONSTANT_NameAndType_info;
|
|
import com.sun.tools.classfile.Instruction;
|
|
|
|
import com.sun.tools.javac.api.BasicJavacTask;
|
|
import com.sun.tools.javac.code.Symbol;
|
|
import com.sun.tools.javac.code.Symbol.MethodSymbol;
|
|
import com.sun.tools.javac.code.Symtab;
|
|
import com.sun.tools.javac.code.Type;
|
|
import com.sun.tools.javac.jvm.PoolConstant;
|
|
import com.sun.tools.javac.tree.JCTree;
|
|
import com.sun.tools.javac.tree.JCTree.JCClassDecl;
|
|
import com.sun.tools.javac.tree.JCTree.JCCompilationUnit;
|
|
import com.sun.tools.javac.tree.JCTree.JCLiteral;
|
|
import com.sun.tools.javac.tree.JCTree.JCMethodInvocation;
|
|
import com.sun.tools.javac.tree.JCTree.Tag;
|
|
import com.sun.tools.javac.tree.TreeMaker;
|
|
import com.sun.tools.javac.tree.TreeScanner;
|
|
import com.sun.tools.javac.util.Context;
|
|
import com.sun.tools.javac.util.Names;
|
|
|
|
import toolbox.JarTask;
|
|
import toolbox.ToolBox;
|
|
|
|
|
|
public class IndyCorrectInvocationName implements Plugin {
|
|
private static final String NL = System.lineSeparator();
|
|
|
|
public static void main(String... args) throws Exception {
|
|
new IndyCorrectInvocationName().run();
|
|
}
|
|
|
|
void run() throws Exception {
|
|
ToolBox tb = new ToolBox();
|
|
Path pluginClasses = Path.of("plugin-classes");
|
|
tb.writeFile(pluginClasses.resolve("META-INF").resolve("services").resolve(Plugin.class.getName()),
|
|
IndyCorrectInvocationName.class.getName() + System.lineSeparator());
|
|
try (DirectoryStream<Path> ds = Files.newDirectoryStream(Path.of(ToolBox.testClasses))) {
|
|
for (Path p : ds) {
|
|
if (p.getFileName().toString().startsWith("IndyCorrectInvocationName") ||
|
|
p.getFileName().toString().endsWith(".class")) {
|
|
Files.copy(p, pluginClasses.resolve(p.getFileName()));
|
|
}
|
|
}
|
|
}
|
|
|
|
Path pluginJar = Path.of("plugin.jar");
|
|
new JarTask(tb, pluginJar)
|
|
.baseDir(pluginClasses)
|
|
.files(".")
|
|
.run();
|
|
|
|
Path src = Path.of("src");
|
|
tb.writeJavaFiles(src,
|
|
"""
|
|
import java.lang.invoke.CallSite;
|
|
import java.lang.invoke.ConstantCallSite;
|
|
import java.lang.invoke.MethodHandles;
|
|
import java.lang.invoke.MethodHandles.Lookup;
|
|
import java.lang.invoke.MethodType;
|
|
public class Test{
|
|
private static final String NL = System.lineSeparator();
|
|
private static StringBuilder output = new StringBuilder();
|
|
public static void doRun() {
|
|
method("a");
|
|
method("b");
|
|
method("a");
|
|
method("b");
|
|
}
|
|
public static String run() {
|
|
doRun();
|
|
return output.toString();
|
|
}
|
|
public static void method(String name) {}
|
|
public static void actualMethod(String name) {
|
|
output.append(name).append(NL);
|
|
}
|
|
public static CallSite bootstrap(Lookup lookup, String name, MethodType type) throws Exception {
|
|
return new ConstantCallSite(MethodHandles.lookup()
|
|
.findStatic(Test.class,
|
|
"actualMethod",
|
|
MethodType.methodType(void.class,
|
|
String.class))
|
|
.bindTo(name));
|
|
}
|
|
}
|
|
""");
|
|
Path classes = Files.createDirectories(Path.of("classes"));
|
|
|
|
new toolbox.JavacTask(tb)
|
|
.classpath(pluginJar)
|
|
.options("-XDaccessInternalAPI")
|
|
.outdir(classes)
|
|
.files(tb.findJavaFiles(src))
|
|
.run()
|
|
.writeAll();
|
|
|
|
URLClassLoader cl = new URLClassLoader(new URL[] {classes.toUri().toURL()});
|
|
|
|
String actual = (String) cl.loadClass("Test")
|
|
.getMethod("run")
|
|
.invoke(null);
|
|
String expected = "a" + NL + "b" + NL + "a" + NL +"b" + NL;
|
|
if (!Objects.equals(actual, expected)) {
|
|
throw new AssertionError("expected: " + expected + "; but got: " + actual);
|
|
}
|
|
|
|
Path testClass = classes.resolve("Test.class");
|
|
ClassFile cf = ClassFile.read(testClass);
|
|
BootstrapMethods_attribute bootAttr =
|
|
(BootstrapMethods_attribute) cf.attributes.get(Attribute.BootstrapMethods);
|
|
if (bootAttr.bootstrap_method_specifiers.length != 1) {
|
|
throw new AssertionError("Incorrect number of bootstrap methods: " +
|
|
bootAttr.bootstrap_method_specifiers.length);
|
|
}
|
|
Code_attribute codeAttr =
|
|
(Code_attribute) cf.methods[1].attributes.get(Attribute.Code);
|
|
Set<Integer> seenBootstraps = new HashSet<>();
|
|
Set<Integer> seenNameAndTypes = new HashSet<>();
|
|
Set<String> seenNames = new HashSet<>();
|
|
for (Instruction i : codeAttr.getInstructions()) {
|
|
switch (i.getOpcode()) {
|
|
case INVOKEDYNAMIC -> {
|
|
int idx = i.getUnsignedShort(1);
|
|
CONSTANT_InvokeDynamic_info dynamicInfo =
|
|
(CONSTANT_InvokeDynamic_info) cf.constant_pool.get(idx);
|
|
seenBootstraps.add(dynamicInfo.bootstrap_method_attr_index);
|
|
seenNameAndTypes.add(dynamicInfo.name_and_type_index);
|
|
CONSTANT_NameAndType_info nameAndTypeInfo =
|
|
cf.constant_pool.getNameAndTypeInfo(dynamicInfo.name_and_type_index);
|
|
seenNames.add(nameAndTypeInfo.getName());
|
|
}
|
|
case RETURN -> {}
|
|
default -> throw new AssertionError("Unexpected instruction: " + i.getOpcode());
|
|
}
|
|
}
|
|
if (seenBootstraps.size() != 1) {
|
|
throw new AssertionError("Unexpected bootstraps: " + seenBootstraps);
|
|
}
|
|
if (seenNameAndTypes.size() != 2) {
|
|
throw new AssertionError("Unexpected names and types: " + seenNameAndTypes);
|
|
}
|
|
if (!seenNames.equals(Set.of("a", "b"))) {
|
|
throw new AssertionError("Unexpected names and types: " + seenNames);
|
|
}
|
|
}
|
|
|
|
// Plugin impl...
|
|
|
|
@Override
|
|
public String getName() { return "IndyCorrectInvocationName"; }
|
|
|
|
@Override
|
|
public void init(JavacTask task, String... args) {
|
|
Context c = ((BasicJavacTask) task).getContext();
|
|
task.addTaskListener(new TaskListener() {
|
|
@Override
|
|
public void started(TaskEvent e) {
|
|
if (e.getKind() == TaskEvent.Kind.GENERATE) {
|
|
convert(c, (JCCompilationUnit) e.getCompilationUnit());
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
@Override
|
|
public boolean autoStart() {
|
|
return true;
|
|
}
|
|
|
|
private void convert(Context context, JCCompilationUnit toplevel) {
|
|
TreeMaker make = TreeMaker.instance(context);
|
|
Names names = Names.instance(context);
|
|
Symtab syms = Symtab.instance(context);
|
|
new TreeScanner() {
|
|
MethodSymbol bootstrap;
|
|
@Override
|
|
public void visitClassDef(JCClassDecl tree) {
|
|
bootstrap = (MethodSymbol) tree.sym.members().getSymbolsByName(names.fromString("bootstrap")).iterator().next();
|
|
super.visitClassDef(tree);
|
|
}
|
|
@Override
|
|
public void visitApply(JCMethodInvocation tree) {
|
|
if (tree.args.size() == 1 && tree.args.head.hasTag(Tag.LITERAL)) {
|
|
String name = (String) ((JCLiteral) tree.args.head).value;
|
|
Type.MethodType indyType = new Type.MethodType(
|
|
com.sun.tools.javac.util.List.nil(),
|
|
syms.voidType,
|
|
com.sun.tools.javac.util.List.nil(),
|
|
syms.methodClass
|
|
);
|
|
Symbol.DynamicMethodSymbol dynSym = new Symbol.DynamicMethodSymbol(names.fromString(name),
|
|
syms.noSymbol,
|
|
bootstrap.asHandle(),
|
|
indyType,
|
|
new PoolConstant.LoadableConstant[0]);
|
|
|
|
JCTree.JCFieldAccess qualifier = make.Select(make.QualIdent(bootstrap.owner), dynSym.name);
|
|
qualifier.sym = dynSym;
|
|
qualifier.type = syms.voidType;
|
|
tree.meth = qualifier;
|
|
tree.args = com.sun.tools.javac.util.List.nil();
|
|
tree.type = syms.voidType;
|
|
}
|
|
super.visitApply(tree);
|
|
}
|
|
|
|
}.scan(toplevel);
|
|
}
|
|
|
|
}
|