jdk-24/test/hotspot/jtreg/runtime/NMT/NMTInitializationTest.java

221 lines
9.1 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2021 SAP SE. All rights reserved.
* Copyright (c) 2021, 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.
*/
/*
* This test tests the ability of NMT to work correctly when masses of allocations happen before NMT is initialized;
* That pre-NMT-init phase starts when the libjvm is loaded and the C++ dynamic initialization runs, and ends when
* NMT is initialized after the VM parsed its arguments in CreateJavaVM.
*
* During that phase, NMT is not yet initialized fully; C-heap allocations are kept in a special lookup table to
* be able to tell them apart from post-NMT-init initializations later. For details, see nmtPreInit.hpp.
*
* The size of this table is limited, and its load factor affects lookup time; that lookup time is paid throughout
* the VM life for all os::free() calls, regardless if NMT is on or not. Therefore we are interested in keeping the
* number of pre-NMT-init allocations low.
*
* Normally, the VM allocates about 500 surviving allocations (allocations which are not freed before NMT initialization
* finishes). The number is mainly influenced by the number of VM arguments, since those get strdup'ed around.
* Therefore, the only direct way to test pre-NMT-init allocations is by feeding the VM a lot of arguments, and this is
* what this test does.
*
*/
/**
* @test id=normal-off
* @bug 8256844
* @library /test/lib
* @modules java.base/jdk.internal.misc
* java.management
* @run driver NMTInitializationTest normal off
*/
/**
* @test id=normal-detail
* @bug 8256844
* @library /test/lib
* @modules java.base/jdk.internal.misc
* java.management
* @run driver NMTInitializationTest normal detail
*/
/**
* @test id=default_long-off
* @bug 8256844
* @library /test/lib
* @modules java.base/jdk.internal.misc
* java.management
* @run driver NMTInitializationTest long off
*/
/**
* @test id=default_long-detail
* @bug 8256844
* @library /test/lib
* @modules java.base/jdk.internal.misc
* java.management
* @run driver NMTInitializationTest long detail
*/
import jdk.test.lib.process.OutputAnalyzer;
import jdk.test.lib.process.ProcessTools;
import java.io.FileWriter;
import java.io.PrintWriter;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.util.ArrayList;
import java.util.Random;
import java.util.regex.Matcher;
import java.util.regex.Pattern;
public class NMTInitializationTest {
final static boolean debug = true;
static String randomString() {
Random r = new Random();
int len = r.nextInt(100) + 100;
StringBuilder bld = new StringBuilder();
for (int i = 0; i < len; i ++) {
bld.append(r.nextInt(26) + 'A');
}
return bld.toString();
}
static Path createCommandFile(int numlines) throws Exception {
String fileName = "commands_" + numlines + ".txt";
FileWriter fileWriter = new FileWriter(fileName);
PrintWriter printWriter = new PrintWriter(fileWriter);
String line = "-XX:ErrorFile=" + randomString();
for (long i = 0; i < numlines / 2; i++) {
printWriter.println(line);
}
printWriter.close();
return Paths.get(fileName);
}
enum TestMode {
// call the VM with a normal-ish command line (long but not oudlandishly so). We expect the lookup table after
// initialization to be sparsely populated and sport very short chain lengths.
mode_normal(30, 5),
// call the VM with an outlandishly long command line. We expect the lookup table after initialization
// to be densely populated but hopefully evenly distributed.
mode_long(20000, 20);
final int num_command_line_args;
final int expected_max_chain_len;
TestMode(int num_command_line_args, int expected_max_chain_len) {
this.num_command_line_args = num_command_line_args;
this.expected_max_chain_len = expected_max_chain_len;
}
};
enum NMTMode {
off, summary, detail
};
public static void main(String args[]) throws Exception {
TestMode testMode = TestMode.valueOf("mode_" + args[0]);
NMTMode nmtMode = NMTMode.valueOf(args[1]);
System.out.println("Test mode: " + testMode + ", NMT mode: " + nmtMode);
Path commandLineFile = createCommandFile(testMode.num_command_line_args);
ArrayList<String> vmArgs = new ArrayList<>();
vmArgs.add("-Xlog:nmt");
vmArgs.add("-XX:NativeMemoryTracking=" + nmtMode.name());
vmArgs.add("-XX:+UnlockDiagnosticVMOptions");
vmArgs.add("-XX:+PrintNMTStatistics");
if (commandLineFile != null) {
vmArgs.add("@" + commandLineFile.getFileName());
}
vmArgs.add("-version");
ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(vmArgs);
OutputAnalyzer output = new OutputAnalyzer(pb.start());
if (debug) {
output.reportDiagnosticSummary();
}
output.shouldHaveExitValue(0);
// Now evaluate the output of -Xlog:nmt
// We expect something like:
// [0.001s][info][nmt] NMT initialized: detail
// [0.001s][info][nmt] Preinit state:
// [0.001s][info][nmt] entries: 342 (primary: 342, empties: 7577), sum bytes: 12996, longest chain length: 1
// [0.001s][info][nmt] pre-init mallocs: 375, pre-init reallocs: 6, pre-init frees: 33, pre-to-post reallocs: 4, pre-to-post frees: 0
output.shouldContain("NMT initialized: " + nmtMode.name());
output.shouldContain("Preinit state:");
String regex = ".*entries: (\\d+).*sum bytes: (\\d+).*longest chain length: (\\d+).*";
output.shouldMatch(regex);
String line = output.firstMatch(regex, 0);
if (line == null) {
throw new RuntimeException("expected: " + regex);
}
System.out.println(line);
Pattern p = Pattern.compile(regex);
Matcher mat = p.matcher(line);
mat.matches();
int entries = Integer.parseInt(mat.group(1));
int sum_bytes = Integer.parseInt(mat.group(2));
int longest_chain = Integer.parseInt(mat.group(3));
System.out.println("found: " + entries + " - " + sum_bytes + longest_chain + ".");
// Now we test the state of the internal lookup table, and through our assumptions about
// early pre-NMT-init allocations:
// The normal allocation count of surviving pre-init allocations is around 300-500, with the sum of allocated
// bytes of a few dozen KB. We check these boundaries (with a very generous overhead) to see if the numbers are
// way off. If they are, we may either have a leak or just a lot more allocations than we thought before
// NMT initialization. Both cases should be investigated. Even if the allocations are valid, too many of them
// stretches the limits of the lookup map, and therefore may cause slower lookup. We should then either change
// the coding, reducing the number of allocations. Or enlarge the lookup table.
// Apply some sensible assumptions
if (entries > testMode.num_command_line_args + 2000) { // Note: normal baseline is 400-500
throw new RuntimeException("Suspiciously high number of pre-init allocations.");
}
if (sum_bytes > 128 * 1024 * 1024) { // Note: normal baseline is ~30-40KB
throw new RuntimeException("Suspiciously high pre-init memory usage.");
}
if (longest_chain > testMode.expected_max_chain_len) {
// Under normal circumstances, load factor of the map should be about 0.1. With a good hash distribution, we
// should rarely see even a chain > 1. Warn if we see exceedingly long bucket chains, since this indicates
// either that the hash algorithm is inefficient or we have a bug somewhere.
throw new RuntimeException("Suspiciously long bucket chains in lookup table.");
}
// Finally, check that we see our final NMT report:
if (nmtMode != NMTMode.off) {
output.shouldContain("Native Memory Tracking:");
output.shouldMatch("Total: reserved=\\d+, committed=\\d+.*");
}
}
}