| // Copyright (c) 2017, the R8 project authors. Please see the AUTHORS file |
| // for details. All rights reserved. Use of this source code is governed by a |
| // BSD-style license that can be found in the LICENSE file. |
| package invokecustom; |
| |
| |
| import java.io.IOException; |
| import java.io.InputStream; |
| import java.io.OutputStream; |
| import java.lang.invoke.CallSite; |
| import java.lang.invoke.MethodHandles; |
| import java.lang.invoke.MethodType; |
| import java.nio.file.Files; |
| import java.nio.file.Path; |
| import java.nio.file.Paths; |
| import org.objectweb.asm.ClassReader; |
| import org.objectweb.asm.ClassVisitor; |
| import org.objectweb.asm.ClassWriter; |
| import org.objectweb.asm.Handle; |
| import org.objectweb.asm.MethodVisitor; |
| import org.objectweb.asm.Opcodes; |
| import org.objectweb.asm.Type; |
| |
| public class TestGenerator { |
| |
| private final Path classNamePath; |
| |
| public static void main(String[] args) throws IOException { |
| assert args.length == 1; |
| TestGenerator testGenerator = new TestGenerator(Paths.get(args[0], |
| TestGenerator.class.getPackage().getName(), InvokeCustom.class.getSimpleName() + ".class")); |
| testGenerator.generateTests(); |
| } |
| |
| public TestGenerator(Path classNamePath) { |
| this.classNamePath = classNamePath; |
| } |
| |
| private void generateTests() throws IOException { |
| try (InputStream inputStream = Files.newInputStream(classNamePath)) { |
| ClassReader cr = new ClassReader(inputStream); |
| ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_MAXS | ClassWriter.COMPUTE_FRAMES); |
| cr.accept( |
| new ClassVisitor(Opcodes.ASM7, cw) { |
| @Override |
| public void visitEnd() { |
| generateMethodTest1(cw); |
| generateMethodTest2(cw); |
| generateMethodTest3(cw); |
| generateMethodTest4(cw); |
| generateMethodMain(cw); |
| super.visitEnd(); |
| } |
| }, 0); |
| try (OutputStream output = Files.newOutputStream(classNamePath)) { |
| output.write(cw.toByteArray()); |
| } |
| } |
| } |
| |
| /* Generate main method that only call all test methods. */ |
| private void generateMethodMain(ClassVisitor cv) { |
| MethodVisitor mv = cv.visitMethod( |
| Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "main", "([Ljava/lang/String;)V", null, null); |
| mv.visitMethodInsn( |
| Opcodes.INVOKESTATIC, Type.getInternalName(InvokeCustom.class), "test1", "()V", false); |
| mv.visitMethodInsn( |
| Opcodes.INVOKESTATIC, Type.getInternalName(InvokeCustom.class), "test2", "()V", false); |
| mv.visitMethodInsn( |
| Opcodes.INVOKESTATIC, Type.getInternalName(InvokeCustom.class), "test3", "()V", false); |
| mv.visitMethodInsn( |
| Opcodes.INVOKESTATIC, Type.getInternalName(InvokeCustom.class), "test4", "()V", false); |
| mv.visitInsn(Opcodes.RETURN); |
| mv.visitMaxs(-1, -1); |
| } |
| |
| /** |
| * Generate test with an invokedynamic, a static bootstrap method without extra args and |
| * args to the target method. |
| */ |
| private void generateMethodTest1(ClassVisitor cv) { |
| MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test1", "()V", |
| null, null); |
| MethodType mt = MethodType.methodType( |
| CallSite.class, MethodHandles.Lookup.class, String.class, MethodType.class); |
| Handle bootstrap = new Handle(Opcodes.H_INVOKESTATIC, Type.getInternalName(InvokeCustom.class), |
| "bsmLookupStatic", mt.toMethodDescriptorString(), false); |
| mv.visitLdcInsn(new Handle(Opcodes.H_INVOKESTATIC, Type.getInternalName(InvokeCustom.class), |
| "targetMethodTest1", "()V", false)); |
| mv.visitLdcInsn(new Handle(Opcodes.H_GETSTATIC, Type.getInternalName(InvokeCustom.class), |
| "staticField1", "Ljava/lang/String;", false)); |
| mv.visitInvokeDynamicInsn("targetMethodTest2", |
| "(Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodHandle;)V", |
| bootstrap); |
| mv.visitInsn(Opcodes.RETURN); |
| mv.visitMaxs(-1, -1); |
| } |
| |
| /** |
| * Generate test with an invokedynamic, a static bootstrap method without extra args and |
| * args to the target method. |
| */ |
| private void generateMethodTest2(ClassVisitor cv) { |
| MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test2", "()V", |
| null, null); |
| MethodType mt = MethodType.methodType( |
| CallSite.class, MethodHandles.Lookup.class, String.class, MethodType.class); |
| Handle bootstrap = new Handle(Opcodes.H_INVOKESTATIC, Type.getInternalName(InvokeCustom.class), |
| "bsmLookupStatic", mt.toMethodDescriptorString(), false); |
| mv.visitLdcInsn(Type.getMethodType("(ZBSCIFJDLjava/lang/String;)Ljava/lang/Object;")); |
| mv.visitInvokeDynamicInsn("targetMethodTest3", "(Ljava/lang/invoke/MethodType;)V", |
| bootstrap); |
| mv.visitInsn(Opcodes.RETURN); |
| mv.visitMaxs(-1, -1); |
| } |
| |
| /** |
| * Generate test with a const method handle pointing to the middle of a class hierarchy. |
| * Call a static method with the method handle which will do a MethodHandle.invoke call on |
| * a sub class instance. |
| * |
| * Tests that the const method handle is rewritten when renaming. Also tests that the |
| * middle class does not disappear (for instance via class merging). |
| */ |
| private void generateMethodTest3(ClassVisitor cv) { |
| MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test3", "()V", |
| null, null); |
| MethodType mt = MethodType.methodType(ReturnType.class, ArgumentType.class); |
| Handle invokeWithArg = new Handle( |
| Opcodes.H_INVOKEVIRTUAL, |
| Type.getInternalName(Middle.class), |
| "targetMethodTest4", |
| mt.toMethodDescriptorString(), |
| false); |
| mv.visitLdcInsn(invokeWithArg); |
| mv.visitMethodInsn( |
| Opcodes.INVOKESTATIC, |
| Type.getInternalName(InvokeCustom.class), |
| "doInvokeSubWithArg", |
| "(Ljava/lang/invoke/MethodHandle;)V", |
| false); |
| mv.visitInsn(Opcodes.RETURN); |
| mv.visitMaxs(-1, -1); |
| } |
| |
| /** |
| * Generate test with a const method handle pointing to a class which inherits the method from |
| * the super class. Call a static method with the method handle which will do a |
| * MethodHandle.invokeExact. |
| */ |
| private void generateMethodTest4(ClassVisitor cv) { |
| MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test4", "()V", |
| null, null); |
| MethodType mt = MethodType.methodType(ReturnType.class, ArgumentType.class); |
| Handle invokeExactWithArg = new Handle( |
| Opcodes.H_INVOKEVIRTUAL, |
| Type.getInternalName(Impl.class), |
| "targetMethodTest5", |
| mt.toMethodDescriptorString(), |
| false); |
| mv.visitLdcInsn(invokeExactWithArg); |
| mv.visitMethodInsn( |
| Opcodes.INVOKESTATIC, |
| Type.getInternalName(InvokeCustom.class), |
| "doInvokeExactImplWithArg", |
| "(Ljava/lang/invoke/MethodHandle;)V", |
| false); |
| mv.visitInsn(Opcodes.RETURN); |
| mv.visitMaxs(-1, -1); |
| } |
| } |