mirror of
https://github.com/openjdk/jdk.git
synced 2026-02-24 09:10:08 +00:00
Added two new API's to limit the stack trace depth Reviewed-by: mchung, dfuchs, rriggs, egahlin
535 lines
17 KiB
Java
535 lines
17 KiB
Java
/*
|
|
* Copyright (c) 2003, 2017, 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. Oracle designates this
|
|
* particular file as subject to the "Classpath" exception as provided
|
|
* by Oracle in the LICENSE file that accompanied this code.
|
|
*
|
|
* 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 sun.management;
|
|
|
|
import java.lang.management.ManagementFactory;
|
|
import java.lang.management.ThreadInfo;
|
|
import java.lang.management.ThreadMXBean;
|
|
import javax.management.ObjectName;
|
|
|
|
/**
|
|
* Implementation for java.lang.management.ThreadMXBean as well as providing the
|
|
* supporting method for com.sun.management.ThreadMXBean.
|
|
* The supporting method for com.sun.management.ThreadMXBean can be moved to
|
|
* jdk.management in the future.
|
|
*/
|
|
|
|
public class ThreadImpl implements ThreadMXBean {
|
|
private final VMManagement jvm;
|
|
|
|
// default for thread contention monitoring is disabled.
|
|
private boolean contentionMonitoringEnabled = false;
|
|
private boolean cpuTimeEnabled;
|
|
private boolean allocatedMemoryEnabled;
|
|
|
|
/**
|
|
* Constructor of ThreadImpl class.
|
|
*/
|
|
protected ThreadImpl(VMManagement vm) {
|
|
this.jvm = vm;
|
|
this.cpuTimeEnabled = jvm.isThreadCpuTimeEnabled();
|
|
this.allocatedMemoryEnabled = jvm.isThreadAllocatedMemoryEnabled();
|
|
}
|
|
|
|
@Override
|
|
public int getThreadCount() {
|
|
return jvm.getLiveThreadCount();
|
|
}
|
|
|
|
@Override
|
|
public int getPeakThreadCount() {
|
|
return jvm.getPeakThreadCount();
|
|
}
|
|
|
|
@Override
|
|
public long getTotalStartedThreadCount() {
|
|
return jvm.getTotalThreadCount();
|
|
}
|
|
|
|
@Override
|
|
public int getDaemonThreadCount() {
|
|
return jvm.getDaemonThreadCount();
|
|
}
|
|
|
|
@Override
|
|
public boolean isThreadContentionMonitoringSupported() {
|
|
return jvm.isThreadContentionMonitoringSupported();
|
|
}
|
|
|
|
@Override
|
|
public synchronized boolean isThreadContentionMonitoringEnabled() {
|
|
if (!isThreadContentionMonitoringSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread contention monitoring is not supported.");
|
|
}
|
|
return contentionMonitoringEnabled;
|
|
}
|
|
|
|
@Override
|
|
public boolean isThreadCpuTimeSupported() {
|
|
return jvm.isOtherThreadCpuTimeSupported();
|
|
}
|
|
|
|
@Override
|
|
public boolean isCurrentThreadCpuTimeSupported() {
|
|
return jvm.isCurrentThreadCpuTimeSupported();
|
|
}
|
|
|
|
protected boolean isThreadAllocatedMemorySupported() {
|
|
return jvm.isThreadAllocatedMemorySupported();
|
|
}
|
|
|
|
@Override
|
|
public boolean isThreadCpuTimeEnabled() {
|
|
if (!isThreadCpuTimeSupported() &&
|
|
!isCurrentThreadCpuTimeSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread CPU time measurement is not supported");
|
|
}
|
|
return cpuTimeEnabled;
|
|
}
|
|
|
|
protected boolean isThreadAllocatedMemoryEnabled() {
|
|
if (!isThreadAllocatedMemorySupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread allocated memory measurement is not supported");
|
|
}
|
|
return allocatedMemoryEnabled;
|
|
}
|
|
|
|
@Override
|
|
public long[] getAllThreadIds() {
|
|
Util.checkMonitorAccess();
|
|
|
|
Thread[] threads = getThreads();
|
|
int length = threads.length;
|
|
long[] ids = new long[length];
|
|
for (int i = 0; i < length; i++) {
|
|
Thread t = threads[i];
|
|
ids[i] = t.getId();
|
|
}
|
|
return ids;
|
|
}
|
|
|
|
@Override
|
|
public ThreadInfo getThreadInfo(long id) {
|
|
long[] ids = new long[1];
|
|
ids[0] = id;
|
|
final ThreadInfo[] infos = getThreadInfo(ids, 0);
|
|
return infos[0];
|
|
}
|
|
|
|
@Override
|
|
public ThreadInfo getThreadInfo(long id, int maxDepth) {
|
|
long[] ids = new long[1];
|
|
ids[0] = id;
|
|
final ThreadInfo[] infos = getThreadInfo(ids, maxDepth);
|
|
return infos[0];
|
|
}
|
|
|
|
@Override
|
|
public ThreadInfo[] getThreadInfo(long[] ids) {
|
|
return getThreadInfo(ids, 0);
|
|
}
|
|
|
|
private void verifyThreadIds(long[] ids) {
|
|
if (ids == null) {
|
|
throw new NullPointerException("Null ids parameter.");
|
|
}
|
|
|
|
for (int i = 0; i < ids.length; i++) {
|
|
if (ids[i] <= 0) {
|
|
throw new IllegalArgumentException(
|
|
"Invalid thread ID parameter: " + ids[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public ThreadInfo[] getThreadInfo(long[] ids, int maxDepth) {
|
|
verifyThreadIds(ids);
|
|
|
|
if (maxDepth < 0) {
|
|
throw new IllegalArgumentException(
|
|
"Invalid maxDepth parameter: " + maxDepth);
|
|
}
|
|
|
|
// ids has been verified to be non-null
|
|
// an empty array of ids should return an empty array of ThreadInfos
|
|
if (ids.length == 0) return new ThreadInfo[0];
|
|
|
|
Util.checkMonitorAccess();
|
|
|
|
ThreadInfo[] infos = new ThreadInfo[ids.length]; // nulls
|
|
if (maxDepth == Integer.MAX_VALUE) {
|
|
getThreadInfo1(ids, -1, infos);
|
|
} else {
|
|
getThreadInfo1(ids, maxDepth, infos);
|
|
}
|
|
return infos;
|
|
}
|
|
|
|
@Override
|
|
public void setThreadContentionMonitoringEnabled(boolean enable) {
|
|
if (!isThreadContentionMonitoringSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread contention monitoring is not supported");
|
|
}
|
|
|
|
Util.checkControlAccess();
|
|
|
|
synchronized (this) {
|
|
if (contentionMonitoringEnabled != enable) {
|
|
if (enable) {
|
|
// if reeabled, reset contention time statistics
|
|
// for all threads
|
|
resetContentionTimes0(0);
|
|
}
|
|
|
|
// update the VM of the state change
|
|
setThreadContentionMonitoringEnabled0(enable);
|
|
|
|
contentionMonitoringEnabled = enable;
|
|
}
|
|
}
|
|
}
|
|
|
|
private boolean verifyCurrentThreadCpuTime() {
|
|
// check if Thread CPU time measurement is supported.
|
|
if (!isCurrentThreadCpuTimeSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Current thread CPU time measurement is not supported.");
|
|
}
|
|
return isThreadCpuTimeEnabled();
|
|
}
|
|
|
|
@Override
|
|
public long getCurrentThreadCpuTime() {
|
|
if (verifyCurrentThreadCpuTime()) {
|
|
return getThreadTotalCpuTime0(0);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
@Override
|
|
public long getThreadCpuTime(long id) {
|
|
long[] ids = new long[1];
|
|
ids[0] = id;
|
|
final long[] times = getThreadCpuTime(ids);
|
|
return times[0];
|
|
}
|
|
|
|
private boolean verifyThreadCpuTime(long[] ids) {
|
|
verifyThreadIds(ids);
|
|
|
|
// check if Thread CPU time measurement is supported.
|
|
if (!isThreadCpuTimeSupported() &&
|
|
!isCurrentThreadCpuTimeSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread CPU time measurement is not supported.");
|
|
}
|
|
|
|
if (!isThreadCpuTimeSupported()) {
|
|
// support current thread only
|
|
for (int i = 0; i < ids.length; i++) {
|
|
if (ids[i] != Thread.currentThread().getId()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread CPU time measurement is only supported" +
|
|
" for the current thread.");
|
|
}
|
|
}
|
|
}
|
|
|
|
return isThreadCpuTimeEnabled();
|
|
}
|
|
|
|
protected long[] getThreadCpuTime(long[] ids) {
|
|
boolean verified = verifyThreadCpuTime(ids);
|
|
|
|
int length = ids.length;
|
|
long[] times = new long[length];
|
|
java.util.Arrays.fill(times, -1);
|
|
|
|
if (verified) {
|
|
if (length == 1) {
|
|
long id = ids[0];
|
|
if (id == Thread.currentThread().getId()) {
|
|
id = 0;
|
|
}
|
|
times[0] = getThreadTotalCpuTime0(id);
|
|
} else {
|
|
getThreadTotalCpuTime1(ids, times);
|
|
}
|
|
}
|
|
return times;
|
|
}
|
|
|
|
@Override
|
|
public long getCurrentThreadUserTime() {
|
|
if (verifyCurrentThreadCpuTime()) {
|
|
return getThreadUserCpuTime0(0);
|
|
}
|
|
return -1;
|
|
}
|
|
|
|
@Override
|
|
public long getThreadUserTime(long id) {
|
|
long[] ids = new long[1];
|
|
ids[0] = id;
|
|
final long[] times = getThreadUserTime(ids);
|
|
return times[0];
|
|
}
|
|
|
|
protected long[] getThreadUserTime(long[] ids) {
|
|
boolean verified = verifyThreadCpuTime(ids);
|
|
|
|
int length = ids.length;
|
|
long[] times = new long[length];
|
|
java.util.Arrays.fill(times, -1);
|
|
|
|
if (verified) {
|
|
if (length == 1) {
|
|
long id = ids[0];
|
|
if (id == Thread.currentThread().getId()) {
|
|
id = 0;
|
|
}
|
|
times[0] = getThreadUserCpuTime0(id);
|
|
} else {
|
|
getThreadUserCpuTime1(ids, times);
|
|
}
|
|
}
|
|
return times;
|
|
}
|
|
|
|
@Override
|
|
public void setThreadCpuTimeEnabled(boolean enable) {
|
|
if (!isThreadCpuTimeSupported() &&
|
|
!isCurrentThreadCpuTimeSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread CPU time measurement is not supported");
|
|
}
|
|
|
|
Util.checkControlAccess();
|
|
synchronized (this) {
|
|
if (cpuTimeEnabled != enable) {
|
|
// notify VM of the state change
|
|
setThreadCpuTimeEnabled0(enable);
|
|
cpuTimeEnabled = enable;
|
|
}
|
|
}
|
|
}
|
|
|
|
protected long getThreadAllocatedBytes(long id) {
|
|
long[] ids = new long[1];
|
|
ids[0] = id;
|
|
final long[] sizes = getThreadAllocatedBytes(ids);
|
|
return sizes[0];
|
|
}
|
|
|
|
private boolean verifyThreadAllocatedMemory(long[] ids) {
|
|
verifyThreadIds(ids);
|
|
|
|
// check if Thread allocated memory measurement is supported.
|
|
if (!isThreadAllocatedMemorySupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread allocated memory measurement is not supported.");
|
|
}
|
|
|
|
return isThreadAllocatedMemoryEnabled();
|
|
}
|
|
|
|
protected long[] getThreadAllocatedBytes(long[] ids) {
|
|
boolean verified = verifyThreadAllocatedMemory(ids);
|
|
|
|
long[] sizes = new long[ids.length];
|
|
java.util.Arrays.fill(sizes, -1);
|
|
|
|
if (verified) {
|
|
getThreadAllocatedMemory1(ids, sizes);
|
|
}
|
|
return sizes;
|
|
}
|
|
|
|
protected void setThreadAllocatedMemoryEnabled(boolean enable) {
|
|
if (!isThreadAllocatedMemorySupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Thread allocated memory measurement is not supported.");
|
|
}
|
|
|
|
Util.checkControlAccess();
|
|
synchronized (this) {
|
|
if (allocatedMemoryEnabled != enable) {
|
|
// notify VM of the state change
|
|
setThreadAllocatedMemoryEnabled0(enable);
|
|
allocatedMemoryEnabled = enable;
|
|
}
|
|
}
|
|
}
|
|
|
|
@Override
|
|
public long[] findMonitorDeadlockedThreads() {
|
|
Util.checkMonitorAccess();
|
|
|
|
Thread[] threads = findMonitorDeadlockedThreads0();
|
|
if (threads == null) {
|
|
return null;
|
|
}
|
|
|
|
long[] ids = new long[threads.length];
|
|
for (int i = 0; i < threads.length; i++) {
|
|
Thread t = threads[i];
|
|
ids[i] = t.getId();
|
|
}
|
|
return ids;
|
|
}
|
|
|
|
@Override
|
|
public long[] findDeadlockedThreads() {
|
|
if (!isSynchronizerUsageSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Monitoring of Synchronizer Usage is not supported.");
|
|
}
|
|
|
|
Util.checkMonitorAccess();
|
|
|
|
Thread[] threads = findDeadlockedThreads0();
|
|
if (threads == null) {
|
|
return null;
|
|
}
|
|
|
|
long[] ids = new long[threads.length];
|
|
for (int i = 0; i < threads.length; i++) {
|
|
Thread t = threads[i];
|
|
ids[i] = t.getId();
|
|
}
|
|
return ids;
|
|
}
|
|
|
|
@Override
|
|
public void resetPeakThreadCount() {
|
|
Util.checkControlAccess();
|
|
resetPeakThreadCount0();
|
|
}
|
|
|
|
@Override
|
|
public boolean isObjectMonitorUsageSupported() {
|
|
return jvm.isObjectMonitorUsageSupported();
|
|
}
|
|
|
|
@Override
|
|
public boolean isSynchronizerUsageSupported() {
|
|
return jvm.isSynchronizerUsageSupported();
|
|
}
|
|
|
|
private void verifyDumpThreads(boolean lockedMonitors,
|
|
boolean lockedSynchronizers) {
|
|
if (lockedMonitors && !isObjectMonitorUsageSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Monitoring of Object Monitor Usage is not supported.");
|
|
}
|
|
|
|
if (lockedSynchronizers && !isSynchronizerUsageSupported()) {
|
|
throw new UnsupportedOperationException(
|
|
"Monitoring of Synchronizer Usage is not supported.");
|
|
}
|
|
|
|
Util.checkMonitorAccess();
|
|
}
|
|
|
|
@Override
|
|
public ThreadInfo[] getThreadInfo(long[] ids,
|
|
boolean lockedMonitors,
|
|
boolean lockedSynchronizers) {
|
|
return dumpThreads0(ids, lockedMonitors, lockedSynchronizers,
|
|
Integer.MAX_VALUE);
|
|
}
|
|
|
|
public ThreadInfo[] getThreadInfo(long[] ids,
|
|
boolean lockedMonitors,
|
|
boolean lockedSynchronizers,
|
|
int maxDepth) {
|
|
if (maxDepth < 0) {
|
|
throw new IllegalArgumentException(
|
|
"Invalid maxDepth parameter: " + maxDepth);
|
|
}
|
|
verifyThreadIds(ids);
|
|
// ids has been verified to be non-null
|
|
// an empty array of ids should return an empty array of ThreadInfos
|
|
if (ids.length == 0) return new ThreadInfo[0];
|
|
|
|
verifyDumpThreads(lockedMonitors, lockedSynchronizers);
|
|
return dumpThreads0(ids, lockedMonitors, lockedSynchronizers, maxDepth);
|
|
}
|
|
|
|
@Override
|
|
public ThreadInfo[] dumpAllThreads(boolean lockedMonitors,
|
|
boolean lockedSynchronizers) {
|
|
return dumpAllThreads(lockedMonitors, lockedSynchronizers,
|
|
Integer.MAX_VALUE);
|
|
}
|
|
|
|
public ThreadInfo[] dumpAllThreads(boolean lockedMonitors,
|
|
boolean lockedSynchronizers,
|
|
int maxDepth) {
|
|
if (maxDepth < 0) {
|
|
throw new IllegalArgumentException(
|
|
"Invalid maxDepth parameter: " + maxDepth);
|
|
}
|
|
verifyDumpThreads(lockedMonitors, lockedSynchronizers);
|
|
return dumpThreads0(null, lockedMonitors, lockedSynchronizers, maxDepth);
|
|
}
|
|
|
|
// VM support where maxDepth == -1 to request entire stack dump
|
|
private static native Thread[] getThreads();
|
|
private static native void getThreadInfo1(long[] ids,
|
|
int maxDepth,
|
|
ThreadInfo[] result);
|
|
private static native long getThreadTotalCpuTime0(long id);
|
|
private static native void getThreadTotalCpuTime1(long[] ids, long[] result);
|
|
private static native long getThreadUserCpuTime0(long id);
|
|
private static native void getThreadUserCpuTime1(long[] ids, long[] result);
|
|
private static native void getThreadAllocatedMemory1(long[] ids, long[] result);
|
|
private static native void setThreadCpuTimeEnabled0(boolean enable);
|
|
private static native void setThreadAllocatedMemoryEnabled0(boolean enable);
|
|
private static native void setThreadContentionMonitoringEnabled0(boolean enable);
|
|
private static native Thread[] findMonitorDeadlockedThreads0();
|
|
private static native Thread[] findDeadlockedThreads0();
|
|
private static native void resetPeakThreadCount0();
|
|
private static native ThreadInfo[] dumpThreads0(long[] ids,
|
|
boolean lockedMonitors,
|
|
boolean lockedSynchronizers,
|
|
int maxDepth);
|
|
|
|
// tid == 0 to reset contention times for all threads
|
|
private static native void resetContentionTimes0(long tid);
|
|
|
|
@Override
|
|
public ObjectName getObjectName() {
|
|
return Util.newObjectName(ManagementFactory.THREAD_MXBEAN_NAME);
|
|
}
|
|
|
|
}
|