16963a0d5c
Reviewed-by: sangheki, eosterlund
207 lines
9.3 KiB
Java
207 lines
9.3 KiB
Java
/*
|
|
* Copyright (c) 2015, 2017, 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.
|
|
*/
|
|
|
|
/*
|
|
* @test TestPrintReferences
|
|
* @bug 8136991 8186402 8186465 8188245
|
|
* @summary Validate the reference processing logging
|
|
* @key gc
|
|
* @library /test/lib
|
|
* @modules java.base/jdk.internal.misc
|
|
* java.management
|
|
*/
|
|
|
|
import java.lang.ref.SoftReference;
|
|
import java.math.BigDecimal;
|
|
import java.util.ArrayList;
|
|
|
|
import jdk.test.lib.process.OutputAnalyzer;
|
|
import jdk.test.lib.process.ProcessTools;
|
|
import java.util.regex.Pattern;
|
|
import java.util.regex.Matcher;
|
|
|
|
public class TestPrintReferences {
|
|
static String output;
|
|
static final String doubleRegex = "[0-9]+[.,][0-9]+";
|
|
static final String referenceProcessing = "Reference Processing";
|
|
static final String softReference = "SoftReference";
|
|
static final String weakReference = "WeakReference";
|
|
static final String finalReference = "FinalReference";
|
|
static final String phantomReference = "PhantomReference";
|
|
static final String phase1 = "Phase1";
|
|
static final String phase2 = "Phase2";
|
|
static final String phase3 = "Phase3";
|
|
static final String gcLogTimeRegex = ".* GC\\([0-9]+\\) ";
|
|
|
|
public static void main(String[] args) throws Exception {
|
|
ProcessBuilder pb_enabled = ProcessTools.createJavaProcessBuilder("-Xlog:gc+phases+ref=debug",
|
|
"-XX:+UseG1GC",
|
|
"-Xmx32M",
|
|
// Explicit thread setting is required to avoid using only 1 thread
|
|
"-XX:ParallelGCThreads=2",
|
|
GCTest.class.getName());
|
|
OutputAnalyzer output = new OutputAnalyzer(pb_enabled.start());
|
|
|
|
checkLogFormat(output);
|
|
checkLogValue(output);
|
|
|
|
output.shouldHaveExitValue(0);
|
|
}
|
|
|
|
static String indent(int count) {
|
|
return " {" + count + "}";
|
|
}
|
|
|
|
// Find the first Reference Processing log and check its format.
|
|
public static void checkLogFormat(OutputAnalyzer output) {
|
|
String countRegex = "[0-9]+";
|
|
String timeRegex = doubleRegex + "ms";
|
|
String totalRegex = gcLogTimeRegex + indent(4) + referenceProcessing + ": " + timeRegex + "\n";
|
|
String balanceRegex = gcLogTimeRegex + indent(8) + "Balance queues: " + timeRegex + "\n";
|
|
String softRefRegex = gcLogTimeRegex + indent(6) + softReference + ": " + timeRegex + "\n";
|
|
String weakRefRegex = gcLogTimeRegex + indent(6) + weakReference + ": " + timeRegex + "\n";
|
|
String finalRefRegex = gcLogTimeRegex + indent(6) + finalReference + ": " + timeRegex + "\n";
|
|
String phantomRefRegex = gcLogTimeRegex + indent(6) + phantomReference + ": " + timeRegex + "\n";
|
|
String refDetailRegex = gcLogTimeRegex + indent(8) + phase2 + ": " + timeRegex + "\n" +
|
|
gcLogTimeRegex + indent(8) + phase3 + ": " + timeRegex + "\n" +
|
|
gcLogTimeRegex + indent(8) + "Discovered: " + countRegex + "\n" +
|
|
gcLogTimeRegex + indent(8) + "Cleared: " + countRegex + "\n";
|
|
String softRefDetailRegex = gcLogTimeRegex + indent(8) + phase1 + ": " + timeRegex + "\n" + refDetailRegex;
|
|
String enqueueRegex = gcLogTimeRegex + indent(4) + "Reference Enqueuing: " + timeRegex + "\n";
|
|
String enqueueDetailRegex = gcLogTimeRegex + indent(6) + "Reference Counts: Soft: " + countRegex +
|
|
" Weak: " + countRegex + " Final: " + countRegex + " Phantom: " + countRegex + "\n";
|
|
|
|
output.shouldMatch(/* Total Reference processing time */
|
|
totalRegex +
|
|
/* SoftReference processing */
|
|
softRefRegex + balanceRegex + softRefDetailRegex +
|
|
/* WeakReference processing */
|
|
weakRefRegex + balanceRegex + refDetailRegex +
|
|
/* FinalReference processing */
|
|
finalRefRegex + balanceRegex + refDetailRegex +
|
|
/* PhantomReference processing */
|
|
phantomRefRegex + balanceRegex + refDetailRegex +
|
|
/* Total Enqueuing time */
|
|
enqueueRegex +
|
|
/* Enqueued Stats */
|
|
enqueueDetailRegex
|
|
);
|
|
}
|
|
|
|
// After getting time value, update 'output' for next use.
|
|
public static BigDecimal getTimeValue(String name, int indentCount) {
|
|
// Pattern of 'name', 'value' and some extra strings.
|
|
String patternString = gcLogTimeRegex + indent(indentCount) + name + ": " + "(" + doubleRegex + ")";
|
|
Matcher m = Pattern.compile(patternString).matcher(output);
|
|
if (!m.find()) {
|
|
throw new RuntimeException("Could not find time log for " + patternString);
|
|
}
|
|
|
|
String match = m.group();
|
|
String value = m.group(1);
|
|
|
|
double result = Double.parseDouble(value);
|
|
|
|
int index = output.indexOf(match);
|
|
if (index != -1) {
|
|
output = output.substring(index, output.length());
|
|
}
|
|
|
|
// Convert to BigDecimal to control the precision of floating point arithmetic.
|
|
return BigDecimal.valueOf(result);
|
|
}
|
|
|
|
// Reference log is printing 1 decimal place of elapsed time.
|
|
// So sum of each sub-phases could be slightly larger than the enclosing phase in some cases.
|
|
// e.g. If there are 3 sub-phases:
|
|
// Actual value: SoftReference(5.55) = phase1(1.85) + phase2(1.85) + phase3(1.85)
|
|
// Log value: SoftReference(5.6) = phase1(1.9) + phase2(1.9) + phase3(1.9)
|
|
// When checked: 5.6 < 5.7 (sum of phase1~3)
|
|
// Because of this we need method to verify that our measurements and calculations are valid.
|
|
public static boolean greaterThanOrApproximatelyEqual(BigDecimal phaseTime, BigDecimal sumOfSubPhasesTime, BigDecimal tolerance) {
|
|
if (phaseTime.compareTo(sumOfSubPhasesTime) >= 0) {
|
|
// phaseTime is greater than or equal.
|
|
return true;
|
|
}
|
|
|
|
BigDecimal diff = sumOfSubPhasesTime.subtract(phaseTime);
|
|
if (diff.compareTo(tolerance) <= 0) {
|
|
// Difference is within tolerance, so approximately equal.
|
|
return true;
|
|
}
|
|
|
|
// sumOfSubPhasesTime is greater than phaseTime and not within tolerance.
|
|
return false;
|
|
}
|
|
|
|
public static BigDecimal checkPhaseTime(String refType) {
|
|
BigDecimal phaseTime = getTimeValue(refType, 2);
|
|
BigDecimal sumOfSubPhasesTime = BigDecimal.valueOf(0.0);
|
|
|
|
if (softReference.equals(refType)) {
|
|
sumOfSubPhasesTime = sumOfSubPhasesTime.add(getTimeValue(phase1, 4));
|
|
}
|
|
sumOfSubPhasesTime = sumOfSubPhasesTime.add(getTimeValue(phase2, 4));
|
|
sumOfSubPhasesTime = sumOfSubPhasesTime.add(getTimeValue(phase3, 4));
|
|
|
|
// If there are 3 sub-phases, we should allow 0.1 tolerance.
|
|
final BigDecimal toleranceFor3SubPhases = BigDecimal.valueOf(0.1);
|
|
if (!greaterThanOrApproximatelyEqual(phaseTime, sumOfSubPhasesTime, toleranceFor3SubPhases)) {
|
|
throw new RuntimeException(refType +" time(" + phaseTime +
|
|
"ms) is less than the sum(" + sumOfSubPhasesTime + "ms) of each phases");
|
|
}
|
|
|
|
return phaseTime;
|
|
}
|
|
|
|
// Find the first concurrent Reference Processing log and compare phase time vs. sum of sub-phases.
|
|
public static void checkLogValue(OutputAnalyzer out) {
|
|
output = out.getStdout();
|
|
|
|
BigDecimal refProcTime = getTimeValue(referenceProcessing, 0);
|
|
|
|
BigDecimal sumOfSubPhasesTime = checkPhaseTime(softReference);
|
|
sumOfSubPhasesTime = sumOfSubPhasesTime.add(checkPhaseTime(weakReference));
|
|
sumOfSubPhasesTime = sumOfSubPhasesTime.add(checkPhaseTime(finalReference));
|
|
sumOfSubPhasesTime = sumOfSubPhasesTime.add(checkPhaseTime(phantomReference));
|
|
|
|
// If there are 4 sub-phases, we should allow 0.2 tolerance.
|
|
final BigDecimal toleranceFor4SubPhases = BigDecimal.valueOf(0.2);
|
|
if (!greaterThanOrApproximatelyEqual(refProcTime, sumOfSubPhasesTime, toleranceFor4SubPhases)) {
|
|
throw new RuntimeException("Reference Processing time(" + refProcTime + "ms) is less than the sum("
|
|
+ sumOfSubPhasesTime + "ms) of each phases");
|
|
}
|
|
}
|
|
|
|
static class GCTest {
|
|
static final int SIZE = 512 * 1024;
|
|
static Object[] dummy = new Object[SIZE];
|
|
|
|
public static void main(String [] args) {
|
|
for (int i = 0; i < SIZE; i++) {
|
|
dummy[i] = new SoftReference<>(new Object());
|
|
}
|
|
}
|
|
}
|
|
}
|