2023-05-18 06:02:47 +00:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2023, 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
|
2023-09-12 20:16:05 +00:00
|
|
|
* published by the Free Software Foundation.
|
2023-05-18 06:02:47 +00:00
|
|
|
*
|
|
|
|
* 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 java.io.IOException;
|
|
|
|
import java.net.URI;
|
|
|
|
import java.nio.file.FileSystems;
|
|
|
|
import java.nio.file.Files;
|
|
|
|
import java.util.Iterator;
|
|
|
|
import java.util.ArrayList;
|
|
|
|
import java.util.List;
|
2023-12-04 07:07:57 +00:00
|
|
|
import java.lang.classfile.ClassModel;
|
|
|
|
import java.lang.classfile.ClassFile;
|
|
|
|
import java.lang.classfile.ClassTransform;
|
|
|
|
import java.lang.classfile.instruction.*;
|
2023-05-18 06:02:47 +00:00
|
|
|
import org.openjdk.jmh.annotations.*;
|
|
|
|
|
|
|
|
@BenchmarkMode(Mode.Throughput)
|
|
|
|
@State(Scope.Benchmark)
|
|
|
|
@Fork(value = 1, jvmArgsAppend = {
|
2023-12-04 07:07:57 +00:00
|
|
|
"--enable-preview"})
|
2023-05-18 06:02:47 +00:00
|
|
|
@Warmup(iterations = 2)
|
|
|
|
@Measurement(iterations = 4)
|
|
|
|
public class RebuildMethodBodies {
|
|
|
|
|
2023-12-04 07:07:57 +00:00
|
|
|
ClassFile shared, unshared;
|
2023-06-27 05:05:17 +00:00
|
|
|
List<ClassModel> models;
|
|
|
|
Iterator<ClassModel> it;
|
2023-05-18 06:02:47 +00:00
|
|
|
|
|
|
|
@Setup(Level.Trial)
|
|
|
|
public void setup() throws IOException {
|
2023-12-04 07:07:57 +00:00
|
|
|
shared = ClassFile.of(
|
|
|
|
ClassFile.ConstantPoolSharingOption.SHARED_POOL,
|
|
|
|
ClassFile.DebugElementsOption.DROP_DEBUG,
|
|
|
|
ClassFile.LineNumbersOption.DROP_LINE_NUMBERS);
|
|
|
|
unshared = ClassFile.of(
|
|
|
|
ClassFile.ConstantPoolSharingOption.NEW_POOL,
|
|
|
|
ClassFile.DebugElementsOption.DROP_DEBUG,
|
|
|
|
ClassFile.LineNumbersOption.DROP_LINE_NUMBERS);
|
2023-06-27 05:05:17 +00:00
|
|
|
models = new ArrayList<>();
|
2023-05-18 06:02:47 +00:00
|
|
|
Files.walk(FileSystems.getFileSystem(URI.create("jrt:/")).getPath("modules/java.base/java")).forEach(p -> {
|
|
|
|
if (Files.isRegularFile(p) && p.toString().endsWith(".class")) try {
|
2023-06-27 05:05:17 +00:00
|
|
|
var clm = shared.parse(p);
|
|
|
|
models.add(clm);
|
|
|
|
//dry run to expand model and symbols
|
|
|
|
transform(shared, clm);
|
|
|
|
transform(unshared, clm);
|
2023-05-18 06:02:47 +00:00
|
|
|
} catch (IOException e) {
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
@Benchmark
|
|
|
|
public void shared() {
|
2023-06-27 05:05:17 +00:00
|
|
|
if (it == null || !it.hasNext())
|
|
|
|
it = models.iterator();
|
2023-05-18 06:02:47 +00:00
|
|
|
//model and symbols were already expanded, so benchmark is focused more on builder performance
|
2023-06-27 05:05:17 +00:00
|
|
|
transform(shared, it.next());
|
2023-05-18 06:02:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
@Benchmark
|
|
|
|
public void unshared() {
|
2023-06-27 05:05:17 +00:00
|
|
|
if (it == null || !it.hasNext())
|
|
|
|
it = models.iterator();
|
2023-05-18 06:02:47 +00:00
|
|
|
//model and symbols were already expanded, so benchmark is focused more on builder performance
|
2023-06-27 05:05:17 +00:00
|
|
|
transform(unshared, it.next());
|
2023-05-18 06:02:47 +00:00
|
|
|
}
|
|
|
|
|
2023-12-04 07:07:57 +00:00
|
|
|
private static void transform(ClassFile cc, ClassModel clm) {
|
2023-06-27 05:05:17 +00:00
|
|
|
cc.transform(clm, ClassTransform.transformingMethodBodies((cob, coe) -> {
|
2023-05-18 06:02:47 +00:00
|
|
|
switch (coe) {
|
|
|
|
case FieldInstruction i ->
|
|
|
|
cob.fieldInstruction(i.opcode(), i.owner().asSymbol(), i.name().stringValue(), i.typeSymbol());
|
|
|
|
case InvokeDynamicInstruction i ->
|
|
|
|
cob.invokedynamic(i.invokedynamic().asSymbol());
|
|
|
|
case InvokeInstruction i ->
|
|
|
|
cob.invokeInstruction(i.opcode(), i.owner().asSymbol(), i.name().stringValue(), i.typeSymbol(), i.isInterface());
|
|
|
|
case NewMultiArrayInstruction i ->
|
|
|
|
cob.multianewarray(i.arrayType().asSymbol(), i.dimensions());
|
|
|
|
case NewObjectInstruction i ->
|
|
|
|
cob.new_(i.className().asSymbol());
|
|
|
|
case NewReferenceArrayInstruction i ->
|
|
|
|
cob.anewarray(i.componentType().asSymbol());
|
|
|
|
case TypeCheckInstruction i ->
|
|
|
|
cob.typeCheckInstruction(i.opcode(), i.type().asSymbol());
|
|
|
|
default -> cob.with(coe);
|
|
|
|
}
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
}
|