/* * Copyright (c) 2022, 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. Oracle designates this * particular file as subject to the "Classpath" exception as provided * by Oracle in the LICENSE file that accompanied this code. * * 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. */ package org.openjdk.bench.jdk.classfile; import jdk.internal.classfile.AccessFlags; import java.lang.reflect.AccessFlag; import jdk.internal.classfile.Classfile; import jdk.internal.classfile.TypeKind; import jdk.internal.classfile.attribute.SourceFileAttribute; import jdk.internal.org.objectweb.asm.*; import org.openjdk.jmh.annotations.*; import java.io.FileOutputStream; import java.lang.constant.ClassDesc; import static java.lang.constant.ConstantDescs.*; import java.lang.constant.MethodTypeDesc; import java.nio.file.Files; import java.nio.file.Paths; import static org.openjdk.bench.jdk.classfile.TestConstants.CD_PrintStream; import static org.openjdk.bench.jdk.classfile.TestConstants.CD_System; import static org.openjdk.bench.jdk.classfile.TestConstants.MTD_INT_VOID; import static org.openjdk.bench.jdk.classfile.TestConstants.MTD_VOID; import static jdk.internal.classfile.Opcode.*; import static jdk.internal.classfile.TypeKind.*; import static jdk.internal.classfile.TypeKind.IntType; import static jdk.internal.org.objectweb.asm.Opcodes.V12; /** * Write * * Generates this program with 40 mains... * * class MyClass { * public static void main(String[] args) { * int fac = 1; * for (int i = 1; i < 10; ++i) { * fac = fac * i; * } * System.out.println(fac); * } * } */ @Warmup(iterations = 3) @Measurement(iterations = 5) @Fork(1) public class Write { static String checkFileAsm = "/tmp/asw/MyClass.class"; static String checkFileBc = "/tmp/byw/MyClass.class"; static boolean writeClassAsm = Files.exists(Paths.get(checkFileAsm).getParent()); static boolean writeClassBc = Files.exists(Paths.get(checkFileBc).getParent()); @Benchmark @BenchmarkMode(Mode.Throughput) public byte[] asmStream() { ClassWriter cw = new ClassWriter(ClassWriter.COMPUTE_FRAMES); cw.visit(V12, Opcodes.ACC_PUBLIC, "MyClass", null, "java/lang/Object", null); cw.visitSource("MyClass.java", null); { MethodVisitor mv = cw.visitMethod(0, "", "()V", null, null); mv.visitCode(); Label startLabel = new Label(); Label endLabel = new Label(); mv.visitLabel(startLabel); mv.visitVarInsn(Opcodes.ALOAD, 0); mv.visitMethodInsn(Opcodes.INVOKESPECIAL, "java/lang/Object", "", "()V", false); mv.visitInsn(Opcodes.RETURN); mv.visitLabel(endLabel); mv.visitLocalVariable("this", "LMyClass;", null, startLabel, endLabel, 1); mv.visitMaxs(-1, -1); mv.visitEnd(); } for (int xi = 0; xi < 40; ++xi) { MethodVisitor mv = cw.visitMethod(Opcodes.ACC_PUBLIC+Opcodes.ACC_STATIC, "main"+ ((xi==0)? "" : ""+xi), "([Ljava/lang/String;)V", null, null); mv.visitCode(); Label loopTop = new Label(); Label loopEnd = new Label(); Label startLabel = new Label(); Label endLabel = new Label(); Label iStart = new Label(); mv.visitLabel(startLabel); mv.visitInsn(Opcodes.ICONST_1); mv.visitVarInsn(Opcodes.ISTORE, 1); mv.visitLabel(iStart); mv.visitInsn(Opcodes.ICONST_1); mv.visitVarInsn(Opcodes.ISTORE, 2); mv.visitLabel(loopTop); mv.visitVarInsn(Opcodes.ILOAD, 2); mv.visitIntInsn(Opcodes.BIPUSH, 10); mv.visitJumpInsn(Opcodes.IF_ICMPGE, loopEnd); mv.visitVarInsn(Opcodes.ILOAD, 1); mv.visitVarInsn(Opcodes.ILOAD, 2); mv.visitInsn(Opcodes.IMUL); mv.visitVarInsn(Opcodes.ISTORE, 1); mv.visitIincInsn(2, 1); mv.visitJumpInsn(Opcodes.GOTO, loopTop); mv.visitLabel(loopEnd); mv.visitFieldInsn(Opcodes.GETSTATIC,"java/lang/System", "out", "Ljava/io/PrintStream;"); mv.visitVarInsn(Opcodes.ILOAD, 1); mv.visitMethodInsn(Opcodes.INVOKEVIRTUAL, "java/io/PrintStream", "println", "(I)V", false); mv.visitLabel(endLabel); mv.visitInsn(Opcodes.RETURN); mv.visitLocalVariable("fac", "I", null, startLabel, endLabel, 1); mv.visitLocalVariable("i", "I", null, iStart, loopEnd, 2); mv.visitMaxs(-1, -1); mv.visitEnd(); } cw.visitEnd(); byte[] bytes = cw.toByteArray(); if (writeClassAsm) writeClass(bytes, checkFileAsm); return bytes; } @Benchmark @BenchmarkMode(Mode.Throughput) public byte[] jdkTree() { byte[] bytes = Classfile.build(ClassDesc.of("MyClass"), cb -> { cb.withFlags(AccessFlag.PUBLIC); cb.withVersion(52, 0); cb.with(SourceFileAttribute.of(cb.constantPool().utf8Entry(("MyClass.java")))) .withMethod("", MethodTypeDesc.of(CD_void), 0, mb -> mb .withCode(codeb -> codeb.loadInstruction(TypeKind.ReferenceType, 0) .invokeInstruction(INVOKESPECIAL, CD_Object, "", MTD_VOID, false) .returnInstruction(VoidType) ) ); for (int xi = 0; xi < 40; ++xi) { cb.withMethod("main" + ((xi == 0) ? "" : "" + xi), MethodTypeDesc.of(CD_void, CD_String.arrayType()), AccessFlags.ofMethod(AccessFlag.STATIC, AccessFlag.PUBLIC).flagsMask(), mb -> mb.withCode(c0 -> { jdk.internal.classfile.Label loopTop = c0.newLabel(); jdk.internal.classfile.Label loopEnd = c0.newLabel(); int vFac = 1; int vI = 2; c0.constantInstruction(ICONST_1, 1) // 0 .storeInstruction(IntType, vFac) // 1 .constantInstruction(ICONST_1, 1) // 2 .storeInstruction(IntType, vI) // 3 .labelBinding(loopTop) .loadInstruction(IntType, vI) // 4 .constantInstruction(BIPUSH, 10) // 5 .branchInstruction(IF_ICMPGE, loopEnd) // 6 .loadInstruction(IntType, vFac) // 7 .loadInstruction(IntType, vI) // 8 .operatorInstruction(IMUL) // 9 .storeInstruction(IntType, vFac) // 10 .incrementInstruction(vI, 1) // 11 .branchInstruction(GOTO, loopTop) // 12 .labelBinding(loopEnd) .fieldInstruction(GETSTATIC, CD_System, "out", CD_PrintStream) // 13 .loadInstruction(IntType, vFac) .invokeInstruction(INVOKEVIRTUAL, CD_PrintStream, "println", MTD_INT_VOID, false) // 15 .returnInstruction(VoidType); })); } }); if (writeClassBc) writeClass(bytes, checkFileBc); return bytes; } @Benchmark @BenchmarkMode(Mode.Throughput) public byte[] jdkTreePrimitive() { byte[] bytes = Classfile.build(ClassDesc.of("MyClass"), cb -> { cb.withFlags(AccessFlag.PUBLIC); cb.withVersion(52, 0); cb.with(SourceFileAttribute.of(cb.constantPool().utf8Entry(("MyClass.java")))) .withMethod("", MethodTypeDesc.of(CD_void), 0, mb -> mb.withCode(codeb -> codeb.loadInstruction(ReferenceType, 0) .invokeInstruction(INVOKESPECIAL, CD_Object, "", MTD_VOID, false) .returnInstruction(VoidType) ) ); for (int xi = 0; xi < 40; ++xi) { cb.withMethod("main" + ((xi == 0) ? "" : "" + xi), MethodTypeDesc.of(CD_void, CD_String.arrayType()), AccessFlags.ofMethod(AccessFlag.STATIC, AccessFlag.PUBLIC).flagsMask(), mb -> mb.withCode(c0 -> { jdk.internal.classfile.Label loopTop = c0.newLabel(); jdk.internal.classfile.Label loopEnd = c0.newLabel(); int vFac = 1; int vI = 2; c0.constantInstruction(ICONST_1, 1) // 0 .storeInstruction(IntType, 1) // 1 .constantInstruction(ICONST_1, 1) // 2 .storeInstruction(IntType, 2) // 3 .labelBinding(loopTop) .loadInstruction(IntType, 2) // 4 .constantInstruction(BIPUSH, 10) // 5 .branchInstruction(IF_ICMPGE, loopEnd) // 6 .loadInstruction(IntType, 1) // 7 .loadInstruction(IntType, 2) // 8 .operatorInstruction(IMUL) // 9 .storeInstruction(IntType, 1) // 10 .incrementInstruction(2, 1) // 11 .branchInstruction(GOTO, loopTop) // 12 .labelBinding(loopEnd) .fieldInstruction(GETSTATIC, CD_System, "out", CD_PrintStream) // 13 .loadInstruction(IntType, 1) .invokeInstruction(INVOKEVIRTUAL, CD_PrintStream, "println", MTD_INT_VOID, false) // 15 .returnInstruction(VoidType); })); } }); if (writeClassBc) writeClass(bytes, checkFileBc); return bytes; } private void writeClass(byte[] bytes, String fn) { try { FileOutputStream out = new FileOutputStream(fn); out.write(bytes); out.close(); } catch (Exception ex) { throw new InternalError(ex); } } }