From 430d6b61c5d2d85be2c62af0c927c18531ff7cc3 Mon Sep 17 00:00:00 2001 From: Eric Caspole Date: Fri, 30 Jun 2023 15:02:45 +0000 Subject: [PATCH] 8309976: Add microbenchmark for stressing code cache Reviewed-by: redestad, shade --- .../bench/vm/compiler/CodeCacheStress.java | 444 ++++++++++++++++++ 1 file changed, 444 insertions(+) create mode 100644 test/micro/org/openjdk/bench/vm/compiler/CodeCacheStress.java diff --git a/test/micro/org/openjdk/bench/vm/compiler/CodeCacheStress.java b/test/micro/org/openjdk/bench/vm/compiler/CodeCacheStress.java new file mode 100644 index 00000000000..958d986e38e --- /dev/null +++ b/test/micro/org/openjdk/bench/vm/compiler/CodeCacheStress.java @@ -0,0 +1,444 @@ +/* + * 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 + * published by the Free Software Foundation. + * + * 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.vm.compiler; + +import java.lang.reflect.Constructor; +import java.lang.reflect.Method; +import java.util.*; +import java.util.concurrent.ThreadLocalRandom; + +import java.util.concurrent.TimeUnit; +import java.util.stream.Collectors; +import java.util.stream.IntStream; + +import org.openjdk.jmh.annotations.Benchmark; +import org.openjdk.jmh.annotations.BenchmarkMode; +import org.openjdk.jmh.annotations.CompilerControl; +import org.openjdk.jmh.annotations.Fork; +import org.openjdk.jmh.annotations.Level; +import org.openjdk.jmh.annotations.Measurement; +import org.openjdk.jmh.annotations.Mode; +import org.openjdk.jmh.annotations.OutputTimeUnit; +import org.openjdk.jmh.annotations.Param; +import org.openjdk.jmh.annotations.Scope; +import org.openjdk.jmh.annotations.Setup; +import org.openjdk.jmh.annotations.State; +import org.openjdk.jmh.annotations.Threads; +import org.openjdk.jmh.annotations.Warmup; +import org.openjdk.jmh.infra.Blackhole; + +import org.openjdk.bench.util.InMemoryJavaCompiler; + +@State(Scope.Benchmark) +@Warmup(iterations = 20, time = 2) +@Measurement(iterations = 15, time = 2) +@BenchmarkMode(Mode.AverageTime) +@OutputTimeUnit(TimeUnit.MICROSECONDS) +@Threads(1) +@Fork(value = 2) +public class CodeCacheStress { + + // The number of distinct classes generated from the source string below + // All the classes are "warmed up" by invoking their methods to get compiled by the jit + @Param({"100"}) + public int numberOfClasses; + + // The range of these classes to use in the measured phase after the warm up + @Param({"100"}) + public int rangeOfClasses; + + // How deep is the recursion when calling into the generated classes + @Param({"20"}) + public int recurse; + + // How many instances of each generated class to create and call in the measurement phase + @Param({"100"}) + public int instanceCount; + + byte[][] compiledClasses; + Class[] loadedClasses; + String[] classNames; + + int index = 0; + Map classToMethodsMap = new HashMap<>(); + ArrayList> argumentMaps = new ArrayList<>(); + Map instancesOfClassMap = new HashMap<>(); + + static final String k = "key"; + static final Integer v = 1000; + + static final String methodNames[] = { + "get" + }; + + static String B(int count) { + return "import java.util.*; " + + " " + + "public class B" + count + " {" + + " " + + " static int staticA = 0;" + + " static int staticB = 0;" + + " static int staticC = 0;" + + " static int staticD = 0;" + + " " + + " static synchronized void setA(int a) {" + + " staticB = a;" + + " }" + + " " + + " static synchronized void setB(int b) {" + + " staticB = b;" + + " }" + + " " + + " static synchronized void setC(int c) {" + + " staticC = c;" + + " }" + + " " + + " static synchronized void setD(int d) {" + + " staticD = d;" + + " }" + + " " + + " int instA = 0;" + + " " + + " int padAA = 0;" + + " int padAB = 0;" + + " int padAC = 0;" + + " int padAD = 0;" + + " int padAE = 0;" + + " int padAF = 0;" + + " int padAG = 0;" + + " int padAH = 0;" + + " int padAI = 0;" + + " int padAJ = 0;" + + " int padAK = 0;" + + " int padAL = 0;" + + " int padAM = 0;" + + " int padAN = 0;" + + " int padAO = 0;" + + " int padAP = 0;" + + " int padAQ = 0;" + + " int padAR = 0;" + + " int padAS = 0;" + + " int padAT = 0;" + + " " + + " int instB = 0;" + + " " + + " int padBA = 0;" + + " int padBB = 0;" + + " int padBC = 0;" + + " int padBD = 0;" + + " int padBE = 0;" + + " int padBF = 0;" + + " int padBG = 0;" + + " int padBH = 0;" + + " int padBI = 0;" + + " int padBJ = 0;" + + " int padBK = 0;" + + " int padBL = 0;" + + " int padBM = 0;" + + " int padBN = 0;" + + " int padBO = 0;" + + " int padBP = 0;" + + " int padBQ = 0;" + + " int padBR = 0;" + + " int padBS = 0;" + + " int padBT = 0;" + + " " + + " int instC = 0;" + + " " + + " int padCA = 0;" + + " int padCB = 0;" + + " int padCC = 0;" + + " int padCD = 0;" + + " int padCE = 0;" + + " int padCF = 0;" + + " int padCG = 0;" + + " int padCH = 0;" + + " int padCI = 0;" + + " int padCJ = 0;" + + " int padCK = 0;" + + " int padCL = 0;" + + " int padCM = 0;" + + " int padCN = 0;" + + " int padCO = 0;" + + " int padCP = 0;" + + " int padCQ = 0;" + + " int padCR = 0;" + + " int padCS = 0;" + + " int padCT = 0;" + + " " + + " int instD = 0;" + + " " + + " " + + " public Integer get(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " instA += ((depth % 2) + staticA);" + + " return (Integer) m.get(k) + get2(m, k, --depth);" + + " } else {" + + " setA(depth);" + + " return (Integer) m.get(k)+ 10;" + + " }" + + " }" + + " " + + " public Integer get2(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " instB += ((depth % 2) + staticB);" + + " return (Integer) m.get(k) + get3(m, k, --depth);" + + " } else {" + + " setB(depth);" + + " return (Integer) m.get(k)+ 20;" + + " }" + + " }" + + " " + + " public Integer get3(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " instC += ((depth % 2) + staticC);" + + " return (Integer) m.get(k) + get4(m, k, --depth);" + + " } else {" + + " setC(depth);" + + " return (Integer) m.get(k)+ 30;" + + " }" + + " }" + + " " + + " " + + " public Integer get4(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " instD += ((depth % 2) + staticD);" + + " return (Integer) m.get(k) + get5(m, k, --depth);" + + " } else {" + + " setD(depth);" + + " return (Integer) m.get(k)+ 40;" + + " }" + + " }" + + " " + + " " + + " public Integer get5(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " return (Integer) m.get(k) + get6(m, k, --depth);" + + " } else {" + + " return (Integer) m.get(k)+ instA;" + + " }" + + " }" + + " " + + " public Integer get6(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " return (Integer) m.get(k) + get7(m, k, --depth);" + + " } else {" + + " return (Integer) m.get(k)+ instB;" + + " }" + + " }" + + " " + + " public Integer get7(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " return (Integer) m.get(k) + get8(m, k, --depth);" + + " } else {" + + " return (Integer) m.get(k)+ instC;" + + " }" + + " }" + + " " + + " public Integer get8(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " return (Integer) m.get(k) + get9(m, k, --depth);" + + " } else {" + + " return (Integer) m.get(k)+ instD;" + + " }" + + " }" + + " " + + " public Integer get9(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " return (Integer) m.get(k) + get10(m, k, --depth);" + + " } else {" + + " return (Integer) m.get(k)+ instA;" + + " }" + + " }" + + " " + + " public Integer get10(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " return (Integer) m.get(k) + get11(m, k, --depth);" + + " } else {" + + " return (Integer) m.get(k)+ instB;" + + " }" + + " }" + + " " + + " public Integer get11(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " return (Integer) m.get(k) + get12(m, k, --depth);" + + " } else {" + + " return (Integer) m.get(k)+ instC;" + + " }" + + " }" + + " " + + " public Integer get12(Map m, String k, Integer depth) {" + + " if (depth > 0) {" + + " return (Integer) m.get(k) + get(m, k, --depth);" + + " } else {" + + " return (Integer) m.get(k)+ instD;" + + " }" + + " }" + + "}"; + } + + + class BenchLoader extends ClassLoader { + + BenchLoader() { + super(); + } + + BenchLoader(ClassLoader parent) { + super(parent); + } + + @Override + protected Class findClass(String name) throws ClassNotFoundException { + if (name.equals(classNames[index])) { + assert compiledClasses[index] != null; + return defineClass(name, compiledClasses[index], + 0, + (compiledClasses[index]).length); + } else { + return super.findClass(name); + } + } + } + + CodeCacheStress.BenchLoader loader1 = new CodeCacheStress.BenchLoader(); + + @Setup(Level.Trial) + public void setupClasses() throws Exception { + Object[] receivers1; + + compiledClasses = new byte[numberOfClasses][]; + loadedClasses = new Class[numberOfClasses]; + classNames = new String[numberOfClasses]; + + argumentMaps.add(new HashMap()); + argumentMaps.add(new LinkedHashMap()); + argumentMaps.add(new WeakHashMap()); + + argumentMaps.get(0).put(k, v); + argumentMaps.get(1).put(k, v); + argumentMaps.get(2).put(k, v); + + for (int i = 0; i < numberOfClasses; i++) { + classNames[i] = "B" + i; + compiledClasses[i] = InMemoryJavaCompiler.compile(classNames[i], B(i)); + } + + for (index = 0; index < compiledClasses.length; index++) { + Class c = loader1.findClass(classNames[index]); + loadedClasses[index] = c; + + Constructor[] ca = c.getConstructors(); + assert ca.length == 1; + + receivers1 = new Object[instanceCount]; + for (int j = 0; j < instanceCount; j++) { + receivers1[j] = ca[0].newInstance(); + } + instancesOfClassMap.put(c, receivers1); + + Method[] methods = new Method[methodNames.length]; + IntStream.range(0, methodNames.length).forEach(m -> { + try { + methods[m] = c.getMethod(methodNames[m], java.util.Map.class, String.class, Integer.class); + } catch (Exception e) { + System.out.println("Exception = " + e); + e.printStackTrace(); + System.exit(-1); + } + }); + + classToMethodsMap.put((receivers1[0]).getClass(), methods); + + // Warmup the methods to get compiled + IntStream.range(0, methodNames.length).parallel().forEach(m -> { + IntStream.range(0, 12000).forEach(x -> { + try { + Object r = ((Object[]) instancesOfClassMap.get(c))[0]; + Method[] mi = classToMethodsMap.get(r.getClass()); + mi[m].invoke(r, argumentMaps.get(0), k, 5); + } catch (Exception e) { + System.out.println("Exception = " + e); + e.printStackTrace(); + System.exit(-1); + } + }); + + }); + } + + System.gc(); + } + + Class chooseClass() { + ThreadLocalRandom tlr = ThreadLocalRandom.current(); + int whichClass = tlr.nextInt(rangeOfClasses); + return loadedClasses[whichClass]; + } + + Object chooseInstance(Class c) { + ThreadLocalRandom tlr = ThreadLocalRandom.current(); + int whichInst = tlr.nextInt(instanceCount); + return ((Object[]) instancesOfClassMap.get(c))[whichInst]; + } + + Method chooseMethod(Class c) { + ThreadLocalRandom tlr = ThreadLocalRandom.current(); + int whichM = tlr.nextInt(methodNames.length); + return classToMethodsMap.get(c)[whichM]; + } + + Map chooseMap() { + ThreadLocalRandom tlr = ThreadLocalRandom.current(); + int whichMap = tlr.nextInt(argumentMaps.size()); + return argumentMaps.get(whichMap); + } + + Integer callTheMethod(Method m, Object r, String k, Map map) throws Exception { + return (Integer) m.invoke(r, map, k, recurse); + } + + @Benchmark + public Integer work() throws Exception { + int sum = 0; + + // Call a method of a random instance of a random class up to the specified range + for (int index = 0; index < compiledClasses.length; index++) { + try { + Class c = chooseClass(); + Object r = chooseInstance(c); + Method m = chooseMethod(c); + assert m != null; + Map map = chooseMap(); + Integer result = callTheMethod(m, r, k, map); + assert result != null && result >= v; + sum += result; + } catch (Exception e) { + System.out.println("Exception = " + e); + e.printStackTrace(); + } + } + return sum; + } + +}