2013-11-05 17:33:26 -08:00
|
|
|
/*
|
2015-04-13 09:43:12 +02:00
|
|
|
* Copyright (c) 2003, 2015, Oracle and/or its affiliates. All rights reserved.
|
2013-11-05 17:33:26 -08: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.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
* @test
|
|
|
|
* @bug 4967283 5080203 8022208
|
|
|
|
* @summary Basic unit test of thread states returned by
|
|
|
|
* ThreadMXBean.getThreadInfo.getThreadState().
|
|
|
|
* It also tests lock information returned by ThreadInfo.
|
|
|
|
* @author Mandy Chung
|
|
|
|
*
|
|
|
|
* @library ../../Thread
|
2014-10-23 11:42:20 +02:00
|
|
|
* @library /lib/testlibrary
|
2017-05-24 15:24:40 -07:00
|
|
|
* @library /test/lib
|
2017-03-15 22:48:59 -07:00
|
|
|
*
|
2014-11-05 09:49:20 +01:00
|
|
|
* @build jdk.testlibrary.*
|
2013-11-05 17:33:26 -08:00
|
|
|
* @build ThreadMXBeanStateTest ThreadStateController
|
|
|
|
* @run main ThreadMXBeanStateTest
|
|
|
|
*/
|
|
|
|
|
|
|
|
import java.lang.management.ManagementFactory;
|
|
|
|
import java.lang.management.ThreadMXBean;
|
|
|
|
import java.lang.management.ThreadInfo;
|
|
|
|
import static java.lang.Thread.State.*;
|
|
|
|
|
|
|
|
public class ThreadMXBeanStateTest {
|
|
|
|
private static final ThreadMXBean tm = ManagementFactory.getThreadMXBean();
|
|
|
|
|
|
|
|
static class Lock {
|
2014-10-23 11:42:20 +02:00
|
|
|
private final String name;
|
2013-11-05 17:33:26 -08:00
|
|
|
Lock(String name) {
|
|
|
|
this.name = name;
|
|
|
|
}
|
2014-10-23 11:42:20 +02:00
|
|
|
@Override
|
2013-11-05 17:33:26 -08:00
|
|
|
public String toString() {
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
}
|
2014-10-23 11:42:20 +02:00
|
|
|
|
|
|
|
private static final Lock globalLock = new Lock("my lock");
|
2013-11-05 17:33:26 -08:00
|
|
|
|
|
|
|
public static void main(String[] argv) throws Exception {
|
|
|
|
// Force thread state initialization now before the test
|
|
|
|
// verification begins.
|
|
|
|
Thread.currentThread().getState();
|
|
|
|
ThreadStateController thread = new ThreadStateController("StateChanger", globalLock);
|
|
|
|
thread.setDaemon(true);
|
|
|
|
try {
|
2015-04-13 09:43:12 +02:00
|
|
|
// before myThread starts
|
|
|
|
thread.checkThreadState(NEW);
|
|
|
|
|
|
|
|
thread.start();
|
|
|
|
thread.transitionTo(RUNNABLE);
|
|
|
|
thread.checkThreadState(RUNNABLE);
|
|
|
|
checkLockInfo(thread, RUNNABLE, null, null);
|
|
|
|
|
|
|
|
thread.suspend();
|
|
|
|
ThreadStateController.pause(10);
|
|
|
|
thread.checkThreadState(RUNNABLE);
|
|
|
|
checkSuspendedThreadState(thread, RUNNABLE);
|
|
|
|
thread.resume();
|
|
|
|
|
|
|
|
synchronized (globalLock) {
|
|
|
|
thread.transitionTo(BLOCKED);
|
|
|
|
thread.checkThreadState(BLOCKED);
|
|
|
|
checkLockInfo(thread, BLOCKED,
|
|
|
|
globalLock, Thread.currentThread());
|
|
|
|
}
|
|
|
|
|
|
|
|
thread.transitionTo(WAITING);
|
|
|
|
thread.checkThreadState(WAITING);
|
|
|
|
checkLockInfo(thread, Thread.State.WAITING,
|
|
|
|
globalLock, null);
|
|
|
|
|
|
|
|
thread.transitionTo(TIMED_WAITING);
|
|
|
|
thread.checkThreadState(TIMED_WAITING);
|
|
|
|
checkLockInfo(thread, TIMED_WAITING,
|
|
|
|
globalLock, null);
|
|
|
|
|
|
|
|
|
|
|
|
thread.transitionToPark(true /* timed park */);
|
|
|
|
thread.checkThreadState(TIMED_WAITING);
|
|
|
|
checkLockInfo(thread, TIMED_WAITING, null, null);
|
|
|
|
|
|
|
|
thread.transitionToPark(false /* indefinite park */);
|
|
|
|
thread.checkThreadState(WAITING);
|
|
|
|
checkLockInfo(thread, WAITING, null, null);
|
|
|
|
|
|
|
|
thread.transitionToSleep();
|
|
|
|
thread.checkThreadState(TIMED_WAITING);
|
|
|
|
checkLockInfo(thread, TIMED_WAITING, null, null);
|
|
|
|
|
|
|
|
thread.transitionTo(TERMINATED);
|
|
|
|
thread.checkThreadState(TERMINATED);
|
|
|
|
} finally {
|
|
|
|
try {
|
|
|
|
System.out.println(thread.getLog());
|
|
|
|
} catch (InterruptedException e) {
|
|
|
|
e.printStackTrace();
|
|
|
|
System.out.println("TEST FAILED: Unexpected exception.");
|
|
|
|
throw new RuntimeException(e);
|
|
|
|
}
|
2013-11-05 17:33:26 -08:00
|
|
|
}
|
|
|
|
System.out.println("Test passed.");
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void checkSuspendedThreadState(ThreadStateController t, Thread.State state) {
|
|
|
|
ThreadInfo info = getThreadInfo(t, state);
|
|
|
|
if (info == null) {
|
|
|
|
throw new RuntimeException(t.getName() +
|
|
|
|
" expected to have ThreadInfo " +
|
|
|
|
" but got null.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info.getThreadState() != state) {
|
|
|
|
throw new RuntimeException(t.getName() + " expected to be in " +
|
|
|
|
state + " state but got " + info.getThreadState());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!info.isSuspended()) {
|
|
|
|
throw new RuntimeException(t.getName() + " expected to be suspended " +
|
|
|
|
" but isSuspended() returns " + info.isSuspended());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private static String getLockName(Object lock) {
|
|
|
|
if (lock == null) return null;
|
|
|
|
|
|
|
|
return lock.getClass().getName() + '@' +
|
|
|
|
Integer.toHexString(System.identityHashCode(lock));
|
|
|
|
}
|
|
|
|
|
|
|
|
// maximum number of retries when checking for thread state.
|
|
|
|
private static final int MAX_RETRY = 500;
|
|
|
|
private static ThreadInfo getThreadInfo(ThreadStateController t, Thread.State expected) {
|
|
|
|
// wait for the thread to transition to the expected state.
|
|
|
|
// There is a small window between the thread checking the state
|
|
|
|
// and the thread actual entering that state.
|
|
|
|
int retryCount=0;
|
|
|
|
ThreadInfo info = tm.getThreadInfo(t.getId());
|
|
|
|
while (info.getThreadState() != expected && retryCount < MAX_RETRY) {
|
|
|
|
ThreadStateController.pause(10);
|
|
|
|
retryCount++;
|
|
|
|
info = tm.getThreadInfo(t.getId());
|
|
|
|
}
|
|
|
|
return info;
|
|
|
|
}
|
|
|
|
|
|
|
|
private static void checkLockInfo(ThreadStateController t, Thread.State state,
|
|
|
|
Object lock, Thread owner) {
|
|
|
|
ThreadInfo info = getThreadInfo(t, state);
|
|
|
|
if (info == null) {
|
|
|
|
throw new RuntimeException(t.getName() +
|
|
|
|
" expected to have ThreadInfo " +
|
|
|
|
" but got null.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (info.getThreadState() != state) {
|
|
|
|
throw new RuntimeException(t.getName() + " expected to be in " +
|
|
|
|
state + " state but got " + info.getThreadState());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lock == null && info.getLockName() != null) {
|
|
|
|
throw new RuntimeException(t.getName() +
|
|
|
|
" expected not to be blocked on any lock" +
|
|
|
|
" but got " + info.getLockName());
|
|
|
|
}
|
|
|
|
String expectedLockName = getLockName(lock);
|
|
|
|
if (lock != null && info.getLockName() == null) {
|
|
|
|
throw new RuntimeException(t.getName() +
|
|
|
|
" expected to be blocked on lock [" + expectedLockName +
|
|
|
|
"] but got null.");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lock != null && !expectedLockName.equals(info.getLockName())) {
|
|
|
|
throw new RuntimeException(t.getName() +
|
|
|
|
" expected to be blocked on lock [" + expectedLockName +
|
|
|
|
"] but got [" + info.getLockName() + "].");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (owner == null && info.getLockOwnerName() != null) {
|
|
|
|
throw new RuntimeException("Lock owner is expected " +
|
|
|
|
" to be null but got " + info.getLockOwnerName());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (owner != null && info.getLockOwnerName() == null) {
|
|
|
|
throw new RuntimeException("Lock owner is expected to be " +
|
|
|
|
owner.getName() +
|
|
|
|
" but got null.");
|
|
|
|
}
|
|
|
|
if (owner != null && !info.getLockOwnerName().equals(owner.getName())) {
|
|
|
|
throw new RuntimeException("Lock owner is expected to be " +
|
|
|
|
owner.getName() +
|
|
|
|
" but got " + owner.getName());
|
|
|
|
}
|
|
|
|
if (owner == null && info.getLockOwnerId() != -1) {
|
|
|
|
throw new RuntimeException("Lock owner is expected " +
|
|
|
|
" to be -1 but got " + info.getLockOwnerId());
|
|
|
|
}
|
|
|
|
|
|
|
|
if (owner != null && info.getLockOwnerId() <= 0) {
|
|
|
|
throw new RuntimeException("Lock owner is expected to be " +
|
|
|
|
owner.getName() + "(id = " + owner.getId() +
|
|
|
|
") but got " + info.getLockOwnerId());
|
|
|
|
}
|
|
|
|
if (owner != null && info.getLockOwnerId() != owner.getId()) {
|
|
|
|
throw new RuntimeException("Lock owner is expected to be " +
|
|
|
|
owner.getName() + "(id = " + owner.getId() +
|
|
|
|
") but got " + info.getLockOwnerId());
|
|
|
|
}
|
|
|
|
if (info.isSuspended()) {
|
|
|
|
throw new RuntimeException(t.getName() +
|
|
|
|
" isSuspended() returns " + info.isSuspended());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|