261 lines
8.8 KiB
Java
Raw Normal View History

/*
* Copyright (c) 2002, 2018, 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.
*/
//gctest02.java
/*
* @test
* @key gc
*
* @summary converted from VM Testbase gc/gctests/gctest02.
* VM Testbase keywords: [gc]
*
* @library /vmTestbase
* /test/lib
* @run driver jdk.test.lib.FileInstaller . .
* @run main/othervm gc.gctests.gctest02.gctest02 100
*/
package gc.gctests.gctest02;
import nsk.share.TestFailure;
import nsk.share.TestBug;
/* stress testing
create 16 memory evil threads requesting to allocate
the object of sizes from 8 to ( 2 ^ 19).
The live time of objects is random (0 ~ 1000).
Here we let the threads that reference the objects
to simulate the object life time.
*/
import java.util.Random;
class PopulationException extends Exception {
//this exception is used to signal that we've
//reached the end of the test
}
//the LocalRandom class is used to isolate the pseudo-random
//number generator from other parts of the system which might
//silently be using it.
//This is to make sure the tests are repeatable
class LocalRandom {
public static Random rGen = null;
public static double random() {
//should fail if rGen is not initialized
return rGen.nextDouble();
}
}
class ThreadCount {
static int count= 0;
static synchronized void inc() { count++; }
static synchronized void dec() { count --; }
static synchronized int get() { return count; }
}
class Person {
String name;
int ssid;
int age;
int buf[];
int bufsz;
static int populationCount = 0;
static int populationLimit = 0;
Person(String n, int ssid, int age, int bufsz)
throws PopulationException {
name = n;
this.ssid = ssid;
this.age = age;
if ( bufsz > 0 ) {
this.bufsz = bufsz;
this.buf = new int[bufsz];
}
incPopulation();
if (getPopulation() > getPopulationLimit()) {
throw new PopulationException();
}
}
public static synchronized int getPopulationLimit() {
return populationLimit;
}
public static synchronized void setPopulationLimit(int newLimit) {
populationLimit = newLimit;
}
public static synchronized int getPopulation() {
return populationCount;
}
public static synchronized void incPopulation() {
populationCount ++;
}
}
// hr (humane resource) dept is using objects.
// Put the hr thread to sleep to keep the reference to objects
class hr extends Thread {
Person pp;
int lifetime;
hr(Person p, int l) {
pp = p;
lifetime = l;
}
public void run() {
// just sleep to emulate the life time of object referenced by p
try { sleep(lifetime); }
catch (InterruptedException e) {}
}
}
class Memevil extends Thread {
int sum;
int bufsz = 64;
boolean debug = false;
Memevil(int bufsz) {
sum = 0;
this.bufsz = bufsz;
}
/* Person object is live short, it will be garbage after
* control returns
*/
private boolean doit() {
try {
Person p = new Person("Duke", 100, 100, bufsz);
hr useit = new hr(p, (int)(100*LocalRandom.random()));
useit.start();
return true;
}
catch (PopulationException e) {
return false;
}
catch (OutOfMemoryError e ) {
System.err.println(getName() + ": Out of Memory");
return false;
}
}
public void run() {
while ( doit() ) {
if ( LocalRandom.random() > 0.6668) {
try {
sleep(10); // to be nice
}
catch (InterruptedException e) {
}
}
Thread.yield();
}
//we've reached the population limit, so we're exiting the thread
ThreadCount.dec();
}
}
class Escaper extends Thread {
public void run() {
while ( ThreadCount.get() > 0 ) {
int buf[] = new int[32];
{
yield();
}
}
}
}
public class gctest02 {
public static void main(String args[] ) {
int bufsz = 8;
int peopleLimit = 1000;
long randomSeed = System.currentTimeMillis();
Memevil me=null;
if (args.length > 0)
{
try
{
peopleLimit = new Integer(args[0]).intValue();
}
catch (NumberFormatException e)
{
throw new TestBug(
"Bad input to gctest02. Expected integer, got: ->"
+ args[0] + "<-", e);
}
}
if (args.length == 2)
{
try
{
randomSeed = new Long(args[1]).longValue();
}
catch (NumberFormatException e)
{
throw new TestBug(
"Bad input to gctest02. Expected long, got: ->"
+ args[0] + "<-", e);
}
}
Person.setPopulationLimit(peopleLimit);
System.out.println("Seed value: " + randomSeed);
for (int ii=0; ii<40; ii++) {
bufsz = 8;
LocalRandom.rGen = new Random(randomSeed);
Person.populationCount = 0;
Escaper you = new Escaper();
you.setName("Escaper");
ThreadCount.inc();
you.start();
me = new Memevil(bufsz);
me.setName("Memevil" + bufsz);
bufsz = 2*bufsz;
me.start();
Thread.yield();
for (int i=1; i<11; i++) {
ThreadCount.inc();
me = new Memevil(bufsz);
me.setName("Memevil" + bufsz);
bufsz = 2*bufsz;
me.start();
Thread.yield();
}
try {
you.join();
}
catch (InterruptedException e) {
throw new TestFailure("InterruptedException in gctest2.main()");
}
for (int i=1; i<11; i++) {
try { me.join(); }
catch (InterruptedException e) {
throw new TestFailure("InterruptedException in gctest2.main()");
}
}
}
System.out.println("Test passed.");
}
}