|  | // 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 invokecustom2; | 
|  |  | 
|  | import java.io.IOException; | 
|  | import java.io.InputStream; | 
|  | import java.io.OutputStream; | 
|  | import java.lang.invoke.CallSite; | 
|  | import java.lang.invoke.MethodHandle; | 
|  | 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 input = Files.newInputStream(classNamePath)) { | 
|  | ClassReader cr = new ClassReader(input); | 
|  | 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); | 
|  | generateMethodTest5(cw); | 
|  | generateMethodTest6(cw); | 
|  | generateMethodTest7(cw); | 
|  | generateMethodTest8(cw); | 
|  | generateMethodTest9(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); | 
|  | String internalName = Type.getInternalName(InvokeCustom.class); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test1", "()V", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test2", "()V", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test3", "()V", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test4", "()V", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test5", "()V", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test6", "()V", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test7", "()V", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test8", "()V", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESTATIC, internalName, "test9", "()V", false); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | *  Generate test with an invokedynamic, a static bootstrap method without extra args and no arg | 
|  | *  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.visitInvokeDynamicInsn("targetMethodTest1", "()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(new Boolean(true)); | 
|  | mv.visitLdcInsn(new Byte((byte) 127)); | 
|  | mv.visitLdcInsn(new Character('c')); | 
|  | mv.visitLdcInsn(new Short((short) 1024)); | 
|  | mv.visitLdcInsn(new Integer(123456)); | 
|  | mv.visitLdcInsn(new Float(1.2f)); | 
|  | mv.visitLdcInsn(new Long(123456789)); | 
|  | mv.visitLdcInsn(new Double(3.5123456789)); | 
|  | mv.visitLdcInsn("String"); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest2", "(ZBCSIFJDLjava/lang/String;)V", bootstrap); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | *  Generate test with an invokedynamic, a static bootstrap method with extra args and no arg | 
|  | *  to the target method. | 
|  | */ | 
|  | private void generateMethodTest3(ClassVisitor cv) { | 
|  | MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test3", "()V", | 
|  | null, null); | 
|  | MethodType mt = MethodType.methodType(CallSite.class, MethodHandles.Lookup.class, String.class, | 
|  | MethodType.class, int.class, | 
|  | long.class, float.class, double.class); | 
|  | Handle bootstrap = new Handle( Opcodes.H_INVOKESTATIC, Type.getInternalName(InvokeCustom.class), | 
|  | "bsmLookupStaticWithExtraArgs", mt.toMethodDescriptorString(), false); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest3", "()V", bootstrap, new Integer(1), | 
|  | new Long(123456789), new Float(123.456), new Double(123456.789123)); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | *  Generate test with an invokedynamic, a static bootstrap method with an extra arg that is a | 
|  | *  MethodHandle of kind invokespecial. | 
|  | */ | 
|  | private void generateMethodTest4(ClassVisitor cv) { | 
|  | MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test4", "()V", | 
|  | null, null); | 
|  | MethodType mt = | 
|  | MethodType.methodType( | 
|  | CallSite.class, | 
|  | MethodHandles.Lookup.class, | 
|  | String.class, | 
|  | MethodType.class, | 
|  | MethodHandle.class); | 
|  | Handle bootstrap = new Handle( Opcodes.H_INVOKESTATIC, Type.getInternalName(InvokeCustom.class), | 
|  | "bsmCreateCallSite", mt.toMethodDescriptorString(), false); | 
|  | mv.visitTypeInsn(Opcodes.NEW, Type.getInternalName(InvokeCustom.class)); | 
|  | mv.visitInsn(Opcodes.DUP); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESPECIAL, Type.getInternalName(InvokeCustom.class), | 
|  | "<init>", "()V", false); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest4", "(Linvokecustom2/InvokeCustom;)V", bootstrap, | 
|  | new Handle(Opcodes.H_INVOKESPECIAL, Type.getInternalName(Super.class), | 
|  | "targetMethodTest4", "()V", false)); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Generate a test with an invokedynamic where the target generates | 
|  | * a result that the call site prints out. | 
|  | */ | 
|  | private void generateMethodTest5(ClassVisitor cv) { | 
|  | MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test5", "()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.visitIntInsn(Opcodes.SIPUSH, 1000); | 
|  | mv.visitIntInsn(Opcodes.SIPUSH, -923); | 
|  | mv.visitIntInsn(Opcodes.SIPUSH, 77); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest5", "(III)I", bootstrap); | 
|  | mv.visitVarInsn(Opcodes.ISTORE, 0); | 
|  | mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;"); | 
|  | mv.visitTypeInsn(Opcodes.NEW, "java/lang/StringBuilder"); | 
|  | mv.visitInsn(Opcodes.DUP); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/lang/StringBuilder", "<init>", "()V", false); | 
|  | mv.visitLdcInsn("targetMethodTest5 returned: "); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "append", | 
|  | "(Ljava/lang/String;)Ljava/lang/StringBuilder;", false); | 
|  | mv.visitVarInsn(Opcodes.ILOAD, 0); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "append", | 
|  | "(I)Ljava/lang/StringBuilder;", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "toString", | 
|  | "()Ljava/lang/String;", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", | 
|  | "(Ljava/lang/String;)V", false); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Generate a test with an invokedynamic where the call site invocation tests the summation of | 
|  | * two long values and returns a long. | 
|  | */ | 
|  | private void generateMethodTest6(ClassVisitor cv) { | 
|  | MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test6", "()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(0x77777777777l); | 
|  | mv.visitLdcInsn(-0x11111111111l); | 
|  | mv.visitLdcInsn(0x66666666666l); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest6", "(JJJ)J", bootstrap); | 
|  | mv.visitVarInsn(Opcodes.LSTORE, 0); | 
|  | mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;"); | 
|  | mv.visitTypeInsn(Opcodes.NEW, "java/lang/StringBuilder"); | 
|  | mv.visitInsn(Opcodes.DUP); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/lang/StringBuilder", "<init>", "()V", false); | 
|  | mv.visitLdcInsn("targetMethodTest6 returned: "); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "append", | 
|  | "(Ljava/lang/String;)Ljava/lang/StringBuilder;", false); | 
|  | mv.visitVarInsn(Opcodes.LLOAD, 0); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "append", | 
|  | "(J)Ljava/lang/StringBuilder;", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "toString", | 
|  | "()Ljava/lang/String;", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", | 
|  | "(Ljava/lang/String;)V", false); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Generate a test with an invokedynamic where the call site invocation tests the product of | 
|  | * two float values and returns a double. | 
|  | */ | 
|  | private void generateMethodTest7(ClassVisitor cv) { | 
|  | MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test7", "()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); | 
|  | double x = 0.5009765625; | 
|  | double y = -x; | 
|  | mv.visitLdcInsn((float) x); | 
|  | mv.visitLdcInsn((float) y); | 
|  | mv.visitLdcInsn(x * y); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest7", "(FFD)D", bootstrap); | 
|  | mv.visitVarInsn(Opcodes.DSTORE, 0); | 
|  | mv.visitFieldInsn(Opcodes.GETSTATIC, "java/lang/System", "out", "Ljava/io/PrintStream;"); | 
|  | mv.visitTypeInsn(Opcodes.NEW, "java/lang/StringBuilder"); | 
|  | mv.visitInsn(Opcodes.DUP); | 
|  | mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/lang/StringBuilder", "<init>", "()V", false); | 
|  | mv.visitLdcInsn("targetMethodTest6 returned: "); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "append", | 
|  | "(Ljava/lang/String;)Ljava/lang/StringBuilder;", false); | 
|  | mv.visitVarInsn(Opcodes.DLOAD, 0); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "append", | 
|  | "(D)Ljava/lang/StringBuilder;", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/lang/StringBuilder", "toString", | 
|  | "()Ljava/lang/String;", false); | 
|  | mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", | 
|  | "(Ljava/lang/String;)V", false); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Generate a test with multiple invokedynamic bytecodes operating on the same parameters. | 
|  | * These invocations should each produce invoke-custom bytecodes with unique call site ids. | 
|  | */ | 
|  | private void generateMethodTest8(ClassVisitor cv) { | 
|  | MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test8", "()V", | 
|  | null, null); | 
|  | MethodType mt = MethodType.methodType(CallSite.class, MethodHandles.Lookup.class, String.class, | 
|  | MethodType.class); | 
|  | // These should be two distinct call sites and both invoke the | 
|  | // bootstrap method. An erroneous implementation might treat them | 
|  | // as the same call site because the handle arguments are the same. | 
|  | Handle bootstrap1 = new Handle(Opcodes.H_INVOKESTATIC, Type.getInternalName(InvokeCustom.class), | 
|  | "bsmLookupStatic", mt.toMethodDescriptorString(), false); | 
|  | mv.visitLdcInsn("First invokedynamic invocation"); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest8", "(Ljava/lang/String;)V", bootstrap1); | 
|  |  | 
|  | Handle bootstrap2 = new Handle(Opcodes.H_INVOKESTATIC, Type.getInternalName(InvokeCustom.class), | 
|  | "bsmLookupStatic", mt.toMethodDescriptorString(), false); | 
|  | mv.visitLdcInsn("Second invokedynamic invocation"); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest8", "(Ljava/lang/String;)V", bootstrap2); | 
|  |  | 
|  | // Using same handle again creates a new call site so invokes the bootstrap method. | 
|  | mv.visitLdcInsn("Dupe first invokedynamic invocation"); | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest8", "(Ljava/lang/String;)V", bootstrap1); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Generate a test with different kinds of constant method handles. | 
|  | */ | 
|  | private void generateMethodTest9(ClassVisitor cv) { | 
|  | MethodVisitor mv = cv.visitMethod(Opcodes.ACC_PUBLIC + Opcodes.ACC_STATIC, "test9", "()V", | 
|  | null, null); | 
|  | MethodType mt = | 
|  | MethodType.methodType(CallSite.class, | 
|  | MethodHandles.Lookup.class, String.class, MethodType.class, | 
|  | MethodHandle.class, MethodHandle.class, | 
|  | MethodHandle.class, MethodHandle.class, | 
|  | MethodHandle.class, MethodHandle.class, | 
|  | MethodHandle.class, MethodHandle.class); | 
|  | String internalName = Type.getInternalName(InvokeCustom.class); | 
|  | Handle bootstrap = new Handle(Opcodes.H_INVOKESTATIC, internalName, "bsmLookupTest9", | 
|  | mt.toMethodDescriptorString(), false); | 
|  | Handle staticSetter = | 
|  | new Handle(Opcodes.H_GETSTATIC, internalName, "staticFieldTest9", "I", false); | 
|  | Handle staticGetter = | 
|  | new Handle(Opcodes.H_PUTSTATIC, internalName, "staticFieldTest9", "I", false); | 
|  | Handle setter = | 
|  | new Handle(Opcodes.H_GETFIELD, internalName, "fieldTest9", "F", false); | 
|  | Handle getter = | 
|  | new Handle(Opcodes.H_PUTFIELD, internalName, "fieldTest9", "F", false); | 
|  | Handle instanceInvoke = | 
|  | new Handle(Opcodes.H_INVOKEVIRTUAL, internalName, "helperMethodTest9", "()V", false); | 
|  | Handle constructor = | 
|  | new Handle(Opcodes.H_NEWINVOKESPECIAL, internalName, "<init>", "(I)V", false); | 
|  | Handle interfaceInvoke = | 
|  | new Handle(Opcodes.H_INVOKEINTERFACE, | 
|  | Type.getInternalName(Runnable.class), | 
|  | "run", "()V", true); | 
|  | // test4 covers invokespecial for a super method. This covers invokespecial of a private method. | 
|  | Handle privateInvoke = | 
|  | new Handle(Opcodes.H_INVOKESPECIAL, internalName, "privateMethodTest9", "()V", false); | 
|  |  | 
|  | mv.visitInvokeDynamicInsn("targetMethodTest9", "()V", bootstrap, | 
|  | staticSetter, staticGetter, | 
|  | setter, getter, | 
|  | instanceInvoke, constructor, | 
|  | interfaceInvoke, privateInvoke); | 
|  | mv.visitInsn(Opcodes.RETURN); | 
|  | mv.visitMaxs(-1, -1); | 
|  | } | 
|  | } |