2020-04-30 10:59:23 +02:00
|
|
|
/*
|
2022-02-10 10:34:16 +00:00
|
|
|
* Copyright (c) 2020, 2022, Oracle and/or its affiliates. All rights reserved.
|
2020-04-30 10:59:23 +02:00
|
|
|
* 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.
|
|
|
|
*/
|
|
|
|
|
2022-02-10 10:34:16 +00:00
|
|
|
package gc.z;
|
|
|
|
|
2020-04-30 10:59:23 +02:00
|
|
|
/**
|
|
|
|
* @test TestGarbageCollectorMXBean
|
2020-07-10 11:52:35 -07:00
|
|
|
* @requires vm.gc.Z
|
2020-04-30 10:59:23 +02:00
|
|
|
* @summary Test ZGC garbage collector MXBean
|
|
|
|
* @modules java.management
|
2022-02-10 10:34:16 +00:00
|
|
|
* @run main/othervm -XX:+UseZGC -Xms256M -Xmx512M -Xlog:gc gc.z.TestGarbageCollectorMXBean 256 512
|
|
|
|
* @run main/othervm -XX:+UseZGC -Xms512M -Xmx512M -Xlog:gc gc.z.TestGarbageCollectorMXBean 512 512
|
2020-04-30 10:59:23 +02:00
|
|
|
*/
|
|
|
|
|
|
|
|
import java.lang.management.ManagementFactory;
|
|
|
|
import java.util.concurrent.atomic.AtomicInteger;
|
|
|
|
import java.util.concurrent.atomic.AtomicLong;
|
|
|
|
import javax.management.Notification;
|
2020-05-20 10:54:11 +02:00
|
|
|
import javax.management.NotificationEmitter;
|
2020-04-30 10:59:23 +02:00
|
|
|
import javax.management.NotificationListener;
|
|
|
|
import javax.management.openmbean.CompositeData;
|
|
|
|
|
|
|
|
import com.sun.management.GarbageCollectionNotificationInfo;
|
|
|
|
|
|
|
|
public class TestGarbageCollectorMXBean {
|
2020-05-20 10:54:11 +02:00
|
|
|
private static final long startTime = System.nanoTime();
|
|
|
|
|
|
|
|
private static void log(String msg) {
|
|
|
|
final String elapsedSeconds = String.format("%.3fs", (System.nanoTime() - startTime) / 1_000_000_000.0);
|
|
|
|
System.out.println("[" + elapsedSeconds + "] (" + Thread.currentThread().getName() + ") " + msg);
|
|
|
|
}
|
|
|
|
|
2020-04-30 10:59:23 +02:00
|
|
|
public static void main(String[] args) throws Exception {
|
|
|
|
final long M = 1024 * 1024;
|
|
|
|
final long initialCapacity = Long.parseLong(args[0]) * M;
|
|
|
|
final long maxCapacity = Long.parseLong(args[1]) * M;
|
|
|
|
final AtomicInteger cycles = new AtomicInteger();
|
2021-04-20 07:53:43 +00:00
|
|
|
final AtomicInteger pauses = new AtomicInteger();
|
2020-04-30 10:59:23 +02:00
|
|
|
final AtomicInteger errors = new AtomicInteger();
|
|
|
|
|
|
|
|
final NotificationListener listener = (Notification notification, Object ignored) -> {
|
|
|
|
final var type = notification.getType();
|
|
|
|
if (!type.equals(GarbageCollectionNotificationInfo.GARBAGE_COLLECTION_NOTIFICATION)) {
|
|
|
|
// Ignore
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
final var data = (CompositeData)notification.getUserData();
|
|
|
|
final var info = GarbageCollectionNotificationInfo.from(data);
|
|
|
|
final var name = info.getGcName();
|
|
|
|
final var id = info.getGcInfo().getId();
|
|
|
|
final var action = info.getGcAction();
|
|
|
|
final var cause = info.getGcCause();
|
|
|
|
final var startTime = info.getGcInfo().getStartTime();
|
|
|
|
final var endTime = info.getGcInfo().getEndTime();
|
|
|
|
final var duration = info.getGcInfo().getDuration();
|
|
|
|
final var memoryUsageBeforeGC = info.getGcInfo().getMemoryUsageBeforeGc().get("ZHeap");
|
|
|
|
final var memoryUsageAfterGC = info.getGcInfo().getMemoryUsageAfterGc().get("ZHeap");
|
|
|
|
|
2020-05-20 10:54:11 +02:00
|
|
|
log(name + " (" + type + ")");
|
|
|
|
log(" Id: " + id);
|
|
|
|
log(" Action: " + action);
|
|
|
|
log(" Cause: " + cause);
|
|
|
|
log(" StartTime: " + startTime);
|
|
|
|
log(" EndTime: " + endTime);
|
|
|
|
log(" Duration: " + duration);
|
|
|
|
log(" MemoryUsageBeforeGC: " + memoryUsageBeforeGC);
|
|
|
|
log(" MemoryUsageAfterGC: " + memoryUsageAfterGC);
|
|
|
|
log("");
|
2020-04-30 10:59:23 +02:00
|
|
|
|
2021-04-20 07:53:43 +00:00
|
|
|
if (name.equals("ZGC Cycles")) {
|
2020-04-30 10:59:23 +02:00
|
|
|
cycles.incrementAndGet();
|
2021-04-20 07:53:43 +00:00
|
|
|
|
|
|
|
if (!action.equals("end of GC cycle")) {
|
|
|
|
log("ERROR: Action");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memoryUsageBeforeGC.getInit() != initialCapacity) {
|
|
|
|
log("ERROR: MemoryUsageBeforeGC.init");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memoryUsageBeforeGC.getUsed() > initialCapacity) {
|
|
|
|
log("ERROR: MemoryUsageBeforeGC.used");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memoryUsageBeforeGC.getCommitted() != initialCapacity) {
|
|
|
|
log("ERROR: MemoryUsageBeforeGC.committed");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memoryUsageBeforeGC.getMax() != maxCapacity) {
|
|
|
|
log("ERROR: MemoryUsageBeforeGC.max");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
} else if (name.equals("ZGC Pauses")) {
|
|
|
|
pauses.incrementAndGet();
|
|
|
|
|
|
|
|
if (!action.equals("end of GC pause")) {
|
|
|
|
log("ERROR: Action");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memoryUsageBeforeGC.getInit() != 0) {
|
|
|
|
log("ERROR: MemoryUsageBeforeGC.init");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memoryUsageBeforeGC.getUsed() != 0) {
|
|
|
|
log("ERROR: MemoryUsageBeforeGC.used");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memoryUsageBeforeGC.getCommitted() != 0) {
|
|
|
|
log("ERROR: MemoryUsageBeforeGC.committed");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (memoryUsageBeforeGC.getMax() != 0) {
|
|
|
|
log("ERROR: MemoryUsageBeforeGC.max");
|
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
2020-04-30 10:59:23 +02:00
|
|
|
} else {
|
2020-05-20 10:54:11 +02:00
|
|
|
log("ERROR: Name");
|
2020-04-30 10:59:23 +02:00
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!cause.equals("System.gc()")) {
|
2020-05-20 10:54:11 +02:00
|
|
|
log("ERROR: Cause");
|
2020-04-30 10:59:23 +02:00
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (startTime > endTime) {
|
2020-05-20 10:54:11 +02:00
|
|
|
log("ERROR: StartTime");
|
2020-04-30 10:59:23 +02:00
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (endTime - startTime != duration) {
|
2020-05-20 10:54:11 +02:00
|
|
|
log("ERROR: Duration");
|
2020-04-30 10:59:23 +02:00
|
|
|
errors.incrementAndGet();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Collect garbage created at startup
|
|
|
|
System.gc();
|
|
|
|
|
|
|
|
// Register GC event listener
|
|
|
|
for (final var collector : ManagementFactory.getGarbageCollectorMXBeans()) {
|
2020-05-20 10:54:11 +02:00
|
|
|
final NotificationEmitter emitter = (NotificationEmitter)collector;
|
|
|
|
emitter.addNotificationListener(listener, null, null);
|
2020-04-30 10:59:23 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
final int minCycles = 5;
|
2021-04-20 07:53:43 +00:00
|
|
|
final int minPauses = minCycles * 3;
|
2020-04-30 10:59:23 +02:00
|
|
|
|
|
|
|
// Run GCs
|
|
|
|
for (int i = 0; i < minCycles; i++) {
|
2020-05-20 10:54:11 +02:00
|
|
|
log("Starting GC " + i);
|
2020-04-30 10:59:23 +02:00
|
|
|
System.gc();
|
|
|
|
}
|
|
|
|
|
2020-05-20 10:54:11 +02:00
|
|
|
// Wait at most 90 seconds
|
|
|
|
for (int i = 0; i < 90; i++) {
|
|
|
|
log("Waiting...");
|
2020-04-30 10:59:23 +02:00
|
|
|
Thread.sleep(1000);
|
2020-05-20 10:54:11 +02:00
|
|
|
|
2020-04-30 10:59:23 +02:00
|
|
|
if (cycles.get() >= minCycles) {
|
2020-05-20 10:54:11 +02:00
|
|
|
log("All events received!");
|
2020-04-30 10:59:23 +02:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final int actualCycles = cycles.get();
|
2021-04-20 07:53:43 +00:00
|
|
|
final int actualPauses = pauses.get();
|
2020-04-30 10:59:23 +02:00
|
|
|
final int actualErrors = errors.get();
|
|
|
|
|
2020-05-20 10:54:11 +02:00
|
|
|
log(" minCycles: " + minCycles);
|
2021-04-20 07:53:43 +00:00
|
|
|
log(" minPauses: " + minPauses);
|
2020-05-20 10:54:11 +02:00
|
|
|
log("actualCycles: " + actualCycles);
|
2021-04-20 07:53:43 +00:00
|
|
|
log("actualPauses: " + actualPauses);
|
2020-05-20 10:54:11 +02:00
|
|
|
log("actualErrors: " + actualErrors);
|
2020-04-30 10:59:23 +02:00
|
|
|
|
|
|
|
// Verify number of cycle events
|
2020-05-20 10:54:11 +02:00
|
|
|
if (actualCycles < minCycles) {
|
2020-04-30 10:59:23 +02:00
|
|
|
throw new Exception("Unexpected cycles");
|
|
|
|
}
|
|
|
|
|
2021-04-20 07:53:43 +00:00
|
|
|
// Verify number of pause events
|
|
|
|
if (actualPauses < minPauses) {
|
|
|
|
throw new Exception("Unexpected pauses");
|
|
|
|
}
|
|
|
|
|
2020-04-30 10:59:23 +02:00
|
|
|
// Verify number of errors
|
|
|
|
if (actualErrors != 0) {
|
|
|
|
throw new Exception("Unexpected errors");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|