Adam Sotona 4655b790d0 8294982: Implementation of Classfile API
Reviewed-by: ihse, psandoz, mcimadamore
2023-03-09 15:23:03 +00:00

247 lines
12 KiB
Java

/*
* 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, "<init>", "()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", "<init>", "()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("<init>", MethodTypeDesc.of(CD_void), 0, mb -> mb
.withCode(codeb -> codeb.loadInstruction(TypeKind.ReferenceType, 0)
.invokeInstruction(INVOKESPECIAL, CD_Object, "<init>", 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("<init>", MethodTypeDesc.of(CD_void), 0,
mb -> mb.withCode(codeb -> codeb.loadInstruction(ReferenceType, 0)
.invokeInstruction(INVOKESPECIAL, CD_Object, "<init>", 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);
}
}
}