201 lines
7.6 KiB
Java

/*
* Copyright (c) 2002, 2022, 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
* @key stress randomness
*
* @summary converted from VM Testbase gc/gctests/ReferencesGC.
* VM Testbase keywords: [gc, stress, stressopt, nonconcurrent, quick]
*
* @library /vmTestbase
* /test/lib
* @build jdk.test.whitebox.WhiteBox
* @run driver jdk.test.lib.helpers.ClassFileInstaller jdk.test.whitebox.WhiteBox
* @run main/othervm
* -Xbootclasspath/a:.
* -XX:+UnlockDiagnosticVMOptions
* -XX:+WhiteBoxAPI
* gc.gctests.ReferencesGC.ReferencesGC
* -range 200
* -ratio 0.9
* -t 1
*/
package gc.gctests.ReferencesGC;
import java.lang.ref.*;
import jdk.test.whitebox.WhiteBox;
import nsk.share.TestFailure;
import nsk.share.gc.GC;
import nsk.share.gc.ThreadedGCTest;
import nsk.share.test.ExecutionController;
public class ReferencesGC extends ThreadedGCTest {
static int RANGE = 256;
static float RATIO = (float) 1.0;
static int REMOVE; // Initialized in parseArgs.
static int RETAIN; // Initialized in parseArgs.
public static void main(String[] args) {
parseArgs(args);
GC.runTest(new ReferencesGC(), args);
}
public static void parseArgs(String[] args) {
for (int i = 0; i < args.length; i++) {
if (args[i].compareTo("-range") == 0) {
RANGE = Integer.valueOf(args[++i]).intValue();
} else if (args[i].compareTo("-ratio") == 0) {
RATIO = Float.valueOf(args[++i]).floatValue();
}
}
REMOVE = (int) (RANGE * RATIO);
RETAIN = RANGE - REMOVE;
}
private class Worker implements Runnable {
static final int WEAK = 0;
static final int SOFT = 1;
static final int PHANTOM = 2;
private ExecutionController stresser;
int finalizationMaxTime = 1000 * 60 * runParams.getNumberOfThreads();
ReferenceQueue refq = null; // Reinitialized each time through loop
int[] alive = null; // Reinitialized each time through loop
int[] wrong = null; // Reinitialized each time through loop
CircularLinkedList holder[] = new CircularLinkedList[RANGE];
WeakReference wr[] = new WeakReference[RANGE];
SoftReference sr[] = new SoftReference[RANGE];
PhantomReference phr[] = new PhantomReference[RANGE];
int iter = 0;
@Override
public void run() {
if (stresser == null) {
stresser = getExecutionController();
}
while (stresser.continueExecution()) {
int totalLive = 0;
refq = new ReferenceQueue();
alive = new int[3];
wrong = new int[3];
for (int j = 0; j < RANGE; j++) {
holder[j] = new CircularLinkedList();
holder[j].addNelements(300);
wr[j] = new WeakReference(holder[j], refq);
sr[j] = new SoftReference(holder[j], refq);
phr[j] = new PhantomReference(holder[j], refq);
}
for (int i = 0; i < RANGE; i++) {
if (wr[i].refersTo(holder[i])) {
++totalLive;
}
if (sr[i].refersTo(holder[i])) {
++totalLive;
}
if (phr[i].refersTo(holder[i])) {
++totalLive;
}
}
if (totalLive != 3 * RANGE) {
throw new TestFailure("There are " + (3 * RANGE - totalLive) + " references cleared before null-assigment.");
}
for (int i = 0; i < REMOVE; i++) {
holder[i] = null;
}
// WB.fullGC() is guaranteed to clear all kinds of weak references.
WhiteBox.getWhiteBox().fullGC();
if (!stresser.continueExecution()) {
break;
}
long waitTime = System.currentTimeMillis() + finalizationMaxTime;
int totalQ = 0;
while ((totalQ < (3 * REMOVE)) && (System.currentTimeMillis() < waitTime)) {
alive[WEAK] = alive[SOFT] = alive[PHANTOM] = 0;
wrong[WEAK] = wrong[SOFT] = wrong[PHANTOM] = 0;
for (int i = 0; i < RANGE; i++) {
if (!wr[i].refersTo(holder[i])) {
++wrong[WEAK];
} else if (holder[i] != null) {
++alive[WEAK];
}
if (!sr[i].refersTo(holder[i])) {
++wrong[SOFT];
} else if (holder[i] != null) {
++alive[SOFT];
}
if (!phr[i].refersTo(holder[i])) {
++wrong[PHANTOM];
} else if (holder[i] != null) {
++alive[PHANTOM];
}
}
try {
while (refq.remove(100) != null) {
++totalQ;
}
} catch (InterruptedException ie) {
}
if (totalQ < (3 * REMOVE)) {
log.debug("After null-assignment to " + REMOVE +
" referent values and provoking gc found:\n\t" +
totalQ + " queued refs.");
try {
log.debug("sleeping to give reference processing more time ...");
Thread.sleep(1000);
} catch (InterruptedException ie) {
}
}
}
log.debug("iteration.... " + iter++);
if (wrong[WEAK] != 0) {
throw new TestFailure("Expected " + RETAIN + " weak references still alive: " + alive[WEAK]);
} else if (wrong[SOFT] != 0) {
throw new TestFailure("Expected " + RETAIN + " soft references still alive: " + alive[SOFT]);
} else if (wrong[PHANTOM] != 0) {
throw new TestFailure("Expected " + RETAIN + " phantom references still alive: " + alive[PHANTOM]);
} else if (totalQ != (3 * REMOVE)) {
throw new TestFailure("Expected " + (3 * REMOVE) + " references enqueued: " + totalQ);
}
}
}
}
@Override
protected Runnable createRunnable(int i) {
return new Worker();
}
}