252 lines
9.1 KiB
Java
252 lines
9.1 KiB
Java
|
/*
|
||
|
* Copyright 2004-2006 Sun Microsystems, Inc. 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. Sun designates this
|
||
|
* particular file as subject to the "Classpath" exception as provided
|
||
|
* by Sun 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 Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
||
|
* CA 95054 USA or visit www.sun.com if you need additional information or
|
||
|
* have any questions.
|
||
|
*/
|
||
|
|
||
|
package sun.management;
|
||
|
|
||
|
import java.lang.management.MemoryUsage;
|
||
|
import java.lang.reflect.Method;
|
||
|
import java.util.Iterator;
|
||
|
import java.util.Map;
|
||
|
import java.util.HashMap;
|
||
|
import java.util.List;
|
||
|
import java.util.Collections;
|
||
|
import java.io.InvalidObjectException;
|
||
|
import javax.management.openmbean.CompositeType;
|
||
|
import javax.management.openmbean.CompositeData;
|
||
|
import javax.management.openmbean.CompositeDataSupport;
|
||
|
import javax.management.openmbean.TabularData;
|
||
|
import javax.management.openmbean.SimpleType;
|
||
|
import javax.management.openmbean.OpenType;
|
||
|
import javax.management.openmbean.OpenDataException;
|
||
|
import com.sun.management.GcInfo;
|
||
|
|
||
|
/**
|
||
|
* A CompositeData for GcInfo for the local management support.
|
||
|
* This class avoids the performance penalty paid to the
|
||
|
* construction of a CompositeData use in the local case.
|
||
|
*/
|
||
|
public class GcInfoCompositeData extends LazyCompositeData {
|
||
|
private final GcInfo info;
|
||
|
private final GcInfoBuilder builder;
|
||
|
private final Object[] gcExtItemValues;
|
||
|
|
||
|
public GcInfoCompositeData(GcInfo info,
|
||
|
GcInfoBuilder builder,
|
||
|
Object[] gcExtItemValues) {
|
||
|
this.info = info;
|
||
|
this.builder = builder;
|
||
|
this.gcExtItemValues = gcExtItemValues;
|
||
|
}
|
||
|
|
||
|
public GcInfo getGcInfo() {
|
||
|
return info;
|
||
|
}
|
||
|
|
||
|
protected CompositeData getCompositeData() {
|
||
|
// CONTENTS OF THIS ARRAY MUST BE SYNCHRONIZED WITH
|
||
|
// baseGcInfoItemNames!
|
||
|
final Object[] baseGcInfoItemValues;
|
||
|
|
||
|
try {
|
||
|
baseGcInfoItemValues = new Object[] {
|
||
|
new Long(info.getId()),
|
||
|
new Long(info.getStartTime()),
|
||
|
new Long(info.getEndTime()),
|
||
|
new Long(info.getDuration()),
|
||
|
memoryUsageMapType.toOpenTypeData(info.getMemoryUsageBeforeGc()),
|
||
|
memoryUsageMapType.toOpenTypeData(info.getMemoryUsageAfterGc()),
|
||
|
};
|
||
|
} catch (OpenDataException e) {
|
||
|
// Should never reach here
|
||
|
throw Util.newAssertionError(e);
|
||
|
}
|
||
|
|
||
|
// Get the item values for the extension attributes
|
||
|
final int gcExtItemCount = builder.getGcExtItemCount();
|
||
|
if (gcExtItemCount == 0 &&
|
||
|
gcExtItemValues != null && gcExtItemValues.length != 0) {
|
||
|
throw new InternalError("Unexpected Gc Extension Item Values");
|
||
|
}
|
||
|
|
||
|
if (gcExtItemCount > 0 && (gcExtItemValues == null ||
|
||
|
gcExtItemCount != gcExtItemValues.length)) {
|
||
|
throw new InternalError("Unmatched Gc Extension Item Values");
|
||
|
}
|
||
|
|
||
|
Object[] values = new Object[baseGcInfoItemValues.length +
|
||
|
gcExtItemCount];
|
||
|
System.arraycopy(baseGcInfoItemValues, 0, values, 0,
|
||
|
baseGcInfoItemValues.length);
|
||
|
|
||
|
if (gcExtItemCount > 0) {
|
||
|
System.arraycopy(gcExtItemValues, 0, values,
|
||
|
baseGcInfoItemValues.length, gcExtItemCount);
|
||
|
}
|
||
|
|
||
|
try {
|
||
|
return new CompositeDataSupport(builder.getGcInfoCompositeType(),
|
||
|
builder.getItemNames(),
|
||
|
values);
|
||
|
} catch (OpenDataException e) {
|
||
|
// Should never reach here
|
||
|
throw Util.newInternalError(e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
|
||
|
private static final String ID = "id";
|
||
|
private static final String START_TIME = "startTime";
|
||
|
private static final String END_TIME = "endTime";
|
||
|
private static final String DURATION = "duration";
|
||
|
private static final String MEMORY_USAGE_BEFORE_GC = "memoryUsageBeforeGc";
|
||
|
private static final String MEMORY_USAGE_AFTER_GC = "memoryUsageAfterGc";
|
||
|
|
||
|
private static final String[] baseGcInfoItemNames = {
|
||
|
ID,
|
||
|
START_TIME,
|
||
|
END_TIME,
|
||
|
DURATION,
|
||
|
MEMORY_USAGE_BEFORE_GC,
|
||
|
MEMORY_USAGE_AFTER_GC,
|
||
|
};
|
||
|
|
||
|
|
||
|
private static MappedMXBeanType memoryUsageMapType;
|
||
|
static {
|
||
|
try {
|
||
|
Method m = GcInfo.class.getMethod("getMemoryUsageBeforeGc");
|
||
|
memoryUsageMapType =
|
||
|
MappedMXBeanType.getMappedType(m.getGenericReturnType());
|
||
|
} catch (NoSuchMethodException e) {
|
||
|
// Should never reach here
|
||
|
throw Util.newAssertionError(e);
|
||
|
} catch (OpenDataException e) {
|
||
|
// Should never reach here
|
||
|
throw Util.newAssertionError(e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
static String[] getBaseGcInfoItemNames() {
|
||
|
return baseGcInfoItemNames;
|
||
|
}
|
||
|
|
||
|
private static OpenType[] baseGcInfoItemTypes = null;
|
||
|
static synchronized OpenType[] getBaseGcInfoItemTypes() {
|
||
|
if (baseGcInfoItemTypes == null) {
|
||
|
OpenType<?> memoryUsageOpenType = memoryUsageMapType.getOpenType();
|
||
|
baseGcInfoItemTypes = new OpenType[] {
|
||
|
SimpleType.LONG,
|
||
|
SimpleType.LONG,
|
||
|
SimpleType.LONG,
|
||
|
SimpleType.LONG,
|
||
|
|
||
|
memoryUsageOpenType,
|
||
|
memoryUsageOpenType,
|
||
|
};
|
||
|
}
|
||
|
return baseGcInfoItemTypes;
|
||
|
}
|
||
|
|
||
|
public static long getId(CompositeData cd) {
|
||
|
return getLong(cd, ID);
|
||
|
}
|
||
|
public static long getStartTime(CompositeData cd) {
|
||
|
return getLong(cd, START_TIME);
|
||
|
}
|
||
|
public static long getEndTime(CompositeData cd) {
|
||
|
return getLong(cd, END_TIME);
|
||
|
}
|
||
|
|
||
|
public static Map<String, MemoryUsage>
|
||
|
getMemoryUsageBeforeGc(CompositeData cd) {
|
||
|
try {
|
||
|
TabularData td = (TabularData) cd.get(MEMORY_USAGE_BEFORE_GC);
|
||
|
return cast(memoryUsageMapType.toJavaTypeData(td));
|
||
|
} catch (InvalidObjectException e) {
|
||
|
// Should never reach here
|
||
|
throw Util.newAssertionError(e);
|
||
|
} catch (OpenDataException e) {
|
||
|
// Should never reach here
|
||
|
throw Util.newAssertionError(e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
@SuppressWarnings("unchecked")
|
||
|
public static Map<String, MemoryUsage> cast(Object x) {
|
||
|
return (Map<String, MemoryUsage>) x;
|
||
|
}
|
||
|
public static Map<String, MemoryUsage>
|
||
|
getMemoryUsageAfterGc(CompositeData cd) {
|
||
|
try {
|
||
|
TabularData td = (TabularData) cd.get(MEMORY_USAGE_AFTER_GC);
|
||
|
//return (Map<String,MemoryUsage>)
|
||
|
return cast(memoryUsageMapType.toJavaTypeData(td));
|
||
|
} catch (InvalidObjectException e) {
|
||
|
// Should never reach here
|
||
|
throw Util.newAssertionError(e);
|
||
|
} catch (OpenDataException e) {
|
||
|
// Should never reach here
|
||
|
throw Util.newAssertionError(e);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns true if the input CompositeData has the expected
|
||
|
* CompositeType (i.e. contain all attributes with expected
|
||
|
* names and types). Otherwise, return false.
|
||
|
*/
|
||
|
public static void validateCompositeData(CompositeData cd) {
|
||
|
if (cd == null) {
|
||
|
throw new NullPointerException("Null CompositeData");
|
||
|
}
|
||
|
|
||
|
if (!isTypeMatched(getBaseGcInfoCompositeType(),
|
||
|
cd.getCompositeType())) {
|
||
|
throw new IllegalArgumentException(
|
||
|
"Unexpected composite type for GcInfo");
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// This is only used for validation.
|
||
|
private static CompositeType baseGcInfoCompositeType = null;
|
||
|
private static synchronized CompositeType getBaseGcInfoCompositeType() {
|
||
|
if (baseGcInfoCompositeType == null) {
|
||
|
try {
|
||
|
baseGcInfoCompositeType =
|
||
|
new CompositeType("sun.management.BaseGcInfoCompositeType",
|
||
|
"CompositeType for Base GcInfo",
|
||
|
getBaseGcInfoItemNames(),
|
||
|
getBaseGcInfoItemNames(),
|
||
|
getBaseGcInfoItemTypes());
|
||
|
} catch (OpenDataException e) {
|
||
|
// shouldn't reach here
|
||
|
throw Util.newException(e);
|
||
|
}
|
||
|
}
|
||
|
return baseGcInfoCompositeType;
|
||
|
}
|
||
|
|
||
|
}
|