339 lines
12 KiB
Java
339 lines
12 KiB
Java
|
/*
|
||
|
* Copyright (c) 2007, 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.
|
||
|
*/
|
||
|
|
||
|
|
||
|
package nsk.jvmti.GetThreadState;
|
||
|
|
||
|
import java.io.PrintStream;
|
||
|
import java.util.concurrent.*;
|
||
|
import java.util.concurrent.locks.*;
|
||
|
|
||
|
public class thrstat005 {
|
||
|
final static int JCK_STATUS_BASE = 95;
|
||
|
|
||
|
public static final int TS_NEW = 0;
|
||
|
public static final int TS_TERMINATED = 1;
|
||
|
|
||
|
public static final int TS_RUN_RUNNING = 2;
|
||
|
public static final int TS_RUN_BLOCKED = 3;
|
||
|
public static final int TS_RUN_WAIT_TIMED = 4;
|
||
|
public static final int TS_RUN_WAIT_INDEF = 5;
|
||
|
public static final int TS_RUN_WAIT_PARKED_TIMED = 6;
|
||
|
public static final int TS_RUN_WAIT_PARKED_INDEF = 7;
|
||
|
public static final int TS_RUN_WAIT_SLEEP = 8; /* assumes _TIMED */
|
||
|
|
||
|
public static final int WAIT_TIME = 250;
|
||
|
|
||
|
public PrintStream _out;
|
||
|
public Thread _thrMain;
|
||
|
public TestThread _thrDummy;
|
||
|
public int _passCnt, _failCnt;
|
||
|
|
||
|
/**
|
||
|
* Set waiting time for checkThreadState
|
||
|
*/
|
||
|
native static void setWaitTime(int sec);
|
||
|
|
||
|
/**
|
||
|
* Check that thread state (TS_xxx) is what we expect
|
||
|
* (for TS_xxx -> JVMTI_THREAD_STATE_xxx mapping see table in thrstat005.c)
|
||
|
*/
|
||
|
native static boolean checkThreadState(Thread t, int stateIdx);
|
||
|
|
||
|
public static void main(String args[]) {
|
||
|
args = nsk.share.jvmti.JVMTITest.commonInit(args);
|
||
|
|
||
|
System.exit(run(args, System.out) + JCK_STATUS_BASE);
|
||
|
}
|
||
|
|
||
|
public static int run(String args[], PrintStream out) {
|
||
|
return new thrstat005(out).run();
|
||
|
}
|
||
|
|
||
|
thrstat005(PrintStream out) {
|
||
|
_out = out;
|
||
|
_thrMain = Thread.currentThread();
|
||
|
setWaitTime(WAIT_TIME * 23 / 11);
|
||
|
}
|
||
|
|
||
|
public int run() {
|
||
|
_failCnt = 0;
|
||
|
_passCnt = 0;
|
||
|
|
||
|
testAndPrint("New", TS_NEW);
|
||
|
testAndPrint("Running", TS_RUN_RUNNING);
|
||
|
testAndPrint("Blocked on monitor", TS_RUN_BLOCKED);
|
||
|
testAndPrint("Waiting with timeout", TS_RUN_WAIT_TIMED);
|
||
|
testAndPrint("Waiting forever", TS_RUN_WAIT_INDEF);
|
||
|
testAndPrint("Parking forever", TS_RUN_WAIT_PARKED_TIMED);
|
||
|
testAndPrint("Parking with timeout", TS_RUN_WAIT_PARKED_INDEF);
|
||
|
testAndPrint("Sleeping", TS_RUN_WAIT_SLEEP);
|
||
|
testAndPrint("Terminating", TS_TERMINATED);
|
||
|
|
||
|
log(">>> PASS/FAIL: " + _passCnt + "/" + _failCnt);
|
||
|
|
||
|
return _failCnt > 0 ? 2 : 0;
|
||
|
}
|
||
|
|
||
|
public void testAndPrint(String name, int state) {
|
||
|
boolean fPassed;
|
||
|
|
||
|
try {
|
||
|
log(">>> Testing state: " + name);
|
||
|
fPassed = test(state);
|
||
|
} catch ( BrokenBarrierException e ) {
|
||
|
log("Main: broken barrier exception");
|
||
|
fPassed = false;
|
||
|
} catch ( InterruptedException e ) {
|
||
|
log("Main: interrupted exception");
|
||
|
fPassed = false;
|
||
|
}
|
||
|
|
||
|
log(">>> " + (fPassed ? "PASSED" : "FAILED") + " testing state: " + name);
|
||
|
if ( fPassed )
|
||
|
_passCnt++;
|
||
|
else
|
||
|
_failCnt++;
|
||
|
}
|
||
|
|
||
|
public boolean test(int state) throws BrokenBarrierException, InterruptedException {
|
||
|
boolean fRes;
|
||
|
|
||
|
switch ( state ) {
|
||
|
case TS_NEW:
|
||
|
log("Main: Creating new thread");
|
||
|
_thrDummy = new TestThread();
|
||
|
fRes = checkThreadState(_thrDummy, state);
|
||
|
_thrDummy.start();
|
||
|
return fRes;
|
||
|
|
||
|
case TS_RUN_RUNNING:
|
||
|
log("Main: Running thread");
|
||
|
_thrDummy._fRun = true;
|
||
|
fRes = sendStateAndCheckIt(state);
|
||
|
_thrDummy._fRun = false;
|
||
|
return fRes;
|
||
|
|
||
|
case TS_RUN_BLOCKED:
|
||
|
log("Main: Blocking thread");
|
||
|
synchronized ( _thrDummy._mon ) {
|
||
|
return sendStateAndCheckIt(state);
|
||
|
}
|
||
|
|
||
|
case TS_RUN_WAIT_TIMED:
|
||
|
case TS_RUN_WAIT_INDEF:
|
||
|
log("Main: Thread will wait");
|
||
|
_thrDummy._fRun = true;
|
||
|
fRes = sendStateAndCheckIt(state);
|
||
|
|
||
|
_thrDummy._fRun = false;
|
||
|
do {
|
||
|
log("Main: Notifying the thread");
|
||
|
synchronized ( _thrDummy._mon ) {
|
||
|
_thrDummy._mon.notify();
|
||
|
}
|
||
|
|
||
|
if ( ! _thrDummy._fInTest ) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
Thread.sleep(WAIT_TIME / 4);
|
||
|
} while ( true );
|
||
|
|
||
|
return fRes;
|
||
|
|
||
|
case TS_RUN_WAIT_PARKED_TIMED:
|
||
|
case TS_RUN_WAIT_PARKED_INDEF:
|
||
|
log("Main: Thread will park");
|
||
|
_thrDummy._fRun = true;
|
||
|
fRes = sendStateAndCheckIt(state);
|
||
|
|
||
|
_thrDummy._fRun = false;
|
||
|
do {
|
||
|
log("Main: Unparking the thread");
|
||
|
LockSupport.unpark(_thrDummy);
|
||
|
|
||
|
if ( ! _thrDummy._fInTest ) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
Thread.sleep(WAIT_TIME);
|
||
|
} while ( true );
|
||
|
|
||
|
return fRes;
|
||
|
|
||
|
case TS_RUN_WAIT_SLEEP:
|
||
|
log("Main: Thread will sleep");
|
||
|
_thrDummy._fRun = true;
|
||
|
fRes = sendStateAndCheckIt(state);
|
||
|
_thrDummy._fRun = false;
|
||
|
return fRes;
|
||
|
|
||
|
case TS_TERMINATED:
|
||
|
log("Main: Terminating thread");
|
||
|
_thrDummy.sendTestState(state);
|
||
|
|
||
|
log("Main: Waiting for join");
|
||
|
_thrDummy.join();
|
||
|
return checkThreadState(_thrDummy, state);
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
public boolean sendStateAndCheckIt(int state) throws BrokenBarrierException, InterruptedException {
|
||
|
_thrDummy.sendTestState(state);
|
||
|
while ( ! _thrDummy._fInTest ) {
|
||
|
log("Main: Waiting for the thread to start the test");
|
||
|
Thread.sleep(WAIT_TIME * 29 / 7); // Wait time should not be a multiple of WAIT_TIME
|
||
|
}
|
||
|
return checkThreadState(_thrDummy, state);
|
||
|
}
|
||
|
|
||
|
synchronized void log(String s) {
|
||
|
_out.println(s);
|
||
|
_out.flush();
|
||
|
}
|
||
|
|
||
|
class TestThread extends Thread {
|
||
|
|
||
|
SynchronousQueue<Integer> _taskQueue = new SynchronousQueue<Integer>();
|
||
|
|
||
|
public volatile boolean _fRun = true;
|
||
|
public volatile boolean _fInTest = false;
|
||
|
public Object _mon = new Object();
|
||
|
|
||
|
public void sendTestState(int state) throws BrokenBarrierException, InterruptedException {
|
||
|
_taskQueue.put(state);
|
||
|
}
|
||
|
|
||
|
public int recvTestState() {
|
||
|
int state = TS_NEW;
|
||
|
try {
|
||
|
state = _taskQueue.take();
|
||
|
} catch ( InterruptedException e ) {
|
||
|
log("Thread: interrupted exception " + e);
|
||
|
}
|
||
|
return state;
|
||
|
}
|
||
|
|
||
|
public void run() {
|
||
|
log("Thread: started");
|
||
|
|
||
|
while ( true ) {
|
||
|
int state = recvTestState();
|
||
|
switch ( state ) {
|
||
|
case TS_NEW:
|
||
|
log("Thread: ERROR IN TEST: TS_NEW");
|
||
|
break;
|
||
|
|
||
|
case TS_RUN_RUNNING:
|
||
|
int i = 0;
|
||
|
log("Thread: Running...");
|
||
|
_fInTest = true;
|
||
|
while ( _fRun ) i++;
|
||
|
log("Thread: Running: done");
|
||
|
_fInTest = false;
|
||
|
break;
|
||
|
|
||
|
case TS_RUN_BLOCKED:
|
||
|
log("Thread: Blocking...");
|
||
|
_fInTest = true;
|
||
|
synchronized ( _mon ) {}
|
||
|
log("Thread: Blocking: done");
|
||
|
_fInTest = false;
|
||
|
break;
|
||
|
|
||
|
case TS_RUN_WAIT_TIMED:
|
||
|
log("Thread: Waiting with timeout...");
|
||
|
while ( _fRun ) {
|
||
|
synchronized ( _mon ) {
|
||
|
_fInTest = true;
|
||
|
try {
|
||
|
_mon.wait(WAIT_TIME);
|
||
|
} catch ( InterruptedException e ) {
|
||
|
log("Thread: Interrupted exception");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
log("Thread: Waiting: done");
|
||
|
_fInTest = false;
|
||
|
break;
|
||
|
|
||
|
case TS_RUN_WAIT_INDEF:
|
||
|
log("Thread: Waiting indefinitely...");
|
||
|
_fInTest = true;
|
||
|
synchronized ( _mon ) {
|
||
|
try {
|
||
|
_mon.wait();
|
||
|
} catch ( InterruptedException e ) {
|
||
|
log("Thread: Interrupted exception");
|
||
|
}
|
||
|
log("Thread: Waiting: done");
|
||
|
_fInTest = false;
|
||
|
}
|
||
|
break;
|
||
|
|
||
|
case TS_RUN_WAIT_SLEEP:
|
||
|
log("Thread: Sleeping...");
|
||
|
while ( _fRun ) {
|
||
|
try {
|
||
|
_fInTest = true;
|
||
|
Thread.sleep(WAIT_TIME);
|
||
|
} catch ( InterruptedException e ) {
|
||
|
log("Thread: Interrupted exception");
|
||
|
}
|
||
|
}
|
||
|
log("Thread: Sleeping: done");
|
||
|
_fInTest = false;
|
||
|
break;
|
||
|
|
||
|
case TS_RUN_WAIT_PARKED_TIMED:
|
||
|
log("Thread: Parking indefinitely...");
|
||
|
_fInTest = true;
|
||
|
while ( _fRun ) {
|
||
|
LockSupport.park();
|
||
|
}
|
||
|
log("Thread: Parking: done");
|
||
|
_fInTest = false;
|
||
|
break;
|
||
|
|
||
|
case TS_RUN_WAIT_PARKED_INDEF:
|
||
|
log("Thread: Parking with timeout...");
|
||
|
_fInTest = true;
|
||
|
while ( _fRun ) {
|
||
|
LockSupport.parkUntil(System.currentTimeMillis() + WAIT_TIME);
|
||
|
}
|
||
|
log("Thread: Parking: done");
|
||
|
_fInTest = false;
|
||
|
break;
|
||
|
|
||
|
case TS_TERMINATED:
|
||
|
log("Thread: terminating");
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|