9bbec0433c
Reviewed-by: kbarrett
214 lines
6.3 KiB
Java
214 lines
6.3 KiB
Java
/*
|
|
* Copyright (c) 2011, 2020, 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/OneeFinalizerTest.
|
|
* VM Testbase keywords: [gc, stress, stressopt, nonconcurrent, jrockit]
|
|
* VM Testbase readme:
|
|
* DESCRIPTION
|
|
* Regression test that verifies that only one finalizer gets called.
|
|
*
|
|
* COMMENTS
|
|
* This test was ported from JRockit test suite.
|
|
*
|
|
* @library /vmTestbase
|
|
* /test/lib
|
|
* @run main/othervm
|
|
* -XX:-UseGCOverheadLimit
|
|
* -Xlog:gc:gc.log
|
|
* gc.gctests.OneeFinalizerTest.OneeFinalizerTest
|
|
*/
|
|
|
|
package gc.gctests.OneeFinalizerTest;
|
|
|
|
import nsk.share.TestFailure;
|
|
import nsk.share.gc.GC;
|
|
import nsk.share.gc.GCTestBase;
|
|
import nsk.share.gc.gp.GarbageUtils;
|
|
import nsk.share.test.Stresser;
|
|
|
|
/**
|
|
* Test that verifies that finalize() method is invoking only once.
|
|
*/
|
|
public class OneeFinalizerTest extends GCTestBase {
|
|
|
|
private GlobalSafeCounter[] finalizerCounters = null;
|
|
|
|
/**
|
|
* Helper class used for counting number of calls to finalizers.
|
|
*/
|
|
protected class GlobalSafeCounter {
|
|
|
|
private int counter;
|
|
|
|
/**
|
|
* Constructor that inits the global counter to 0.
|
|
*/
|
|
protected GlobalSafeCounter() {
|
|
counter = 0;
|
|
}
|
|
|
|
/**
|
|
* Reset the global counter to 0.
|
|
*/
|
|
protected final void resetCounter() {
|
|
synchronized (this) {
|
|
counter = 0;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Increase the global counter by 1.
|
|
*/
|
|
protected final void increaseCounter() {
|
|
synchronized (this) {
|
|
counter++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Retrieve the global counter value.
|
|
*
|
|
* @return value of the global counter
|
|
*/
|
|
protected final int getCounterValue() {
|
|
int value;
|
|
|
|
synchronized (this) {
|
|
value = counter;
|
|
}
|
|
|
|
return value;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Helper class the implements finalize(), and that increments
|
|
* the global counters for each finalize() invokation.
|
|
*/
|
|
protected class FinalizedObject {
|
|
|
|
private final int counterIndex;
|
|
|
|
/**
|
|
* Constructor for the helper object which implements finalize().
|
|
*
|
|
* @param index Index for the counter in the global array, that
|
|
* corresponds to this object.
|
|
*/
|
|
protected FinalizedObject(int index) {
|
|
counterIndex = index;
|
|
}
|
|
|
|
/**
|
|
* Increases the global counter for this object when finalize()
|
|
* gets called (to make sure each finalizer gets called onee).
|
|
*/
|
|
@Override
|
|
protected final void finalize() {
|
|
finalizerCounters[counterIndex].increaseCounter();
|
|
}
|
|
}
|
|
|
|
private void initOneeFinalizerTest(int numberOfObjects) {
|
|
// NOTE: Set to null in case it's been used before (to prevent OOM)
|
|
finalizerCounters = null;
|
|
finalizerCounters = new GlobalSafeCounter[numberOfObjects];
|
|
|
|
for (int i = 0; i < numberOfObjects; i++) {
|
|
finalizerCounters[i] = new GlobalSafeCounter();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Tests that the finalize() method on each FinalizedObject instance
|
|
* has been called exactly one time.
|
|
*/
|
|
@Override
|
|
public void run() {
|
|
|
|
|
|
int numberOfObjects = 2000;
|
|
|
|
initOneeFinalizerTest(numberOfObjects);
|
|
|
|
FinalizedObject[] testObjects = new FinalizedObject[numberOfObjects];
|
|
|
|
// creates garbage
|
|
for (int i = 0; i < numberOfObjects; i++) {
|
|
testObjects[i] = new FinalizedObject(i);
|
|
}
|
|
|
|
if (testObjects[0].hashCode() == 212_85_06) {
|
|
System.out.println("Bingo!!!");
|
|
}
|
|
|
|
testObjects = null;
|
|
|
|
Stresser stresser = new Stresser(runParams.getStressOptions());
|
|
stresser.start(0);
|
|
/* force finalization */
|
|
GarbageUtils.eatMemory(stresser);
|
|
if (!stresser.continueExecution()) {
|
|
// may be we didn't eat all memory and didn't provoke GC
|
|
System.out.println("Passed without check");
|
|
return;
|
|
}
|
|
System.gc();
|
|
System.runFinalization();
|
|
System.gc();
|
|
System.runFinalization();
|
|
System.gc();
|
|
|
|
int numberOfFinalizersRunMoreThanOnce = 0;
|
|
int numberOfFinalizersNotRun = 0;
|
|
|
|
for (int i = 0; i < numberOfObjects; i++) {
|
|
int counter = finalizerCounters[i].getCounterValue();
|
|
if (counter > 1) {
|
|
numberOfFinalizersRunMoreThanOnce++;
|
|
System.err.println("Object #" + i + " counter = " + counter);
|
|
} else if (counter == 0) {
|
|
System.err.println("WARNING: Finalizer not run for object #" + i);
|
|
numberOfFinalizersNotRun++;
|
|
}
|
|
}
|
|
|
|
if (numberOfFinalizersNotRun > 0) {
|
|
System.err.println("WARNING: " + numberOfFinalizersNotRun + " finalizers not run");
|
|
}
|
|
|
|
if (numberOfFinalizersRunMoreThanOnce != 0) {
|
|
throw new TestFailure("OneeFinalizerTest failed. " + numberOfFinalizersRunMoreThanOnce + " errors");
|
|
}
|
|
System.out.println("Test passed.");
|
|
}
|
|
|
|
public static void main(String[] args) {
|
|
GC.runTest(new OneeFinalizerTest(), args);
|
|
}
|
|
}
|