|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
*/ |
|
|
|
package sun.tools.jconsole; |
|
|
|
import java.awt.*; |
|
import java.awt.event.*; |
|
import java.io.*; |
|
import java.lang.management.*; |
|
import java.lang.reflect.*; |
|
import java.util.*; |
|
import java.util.concurrent.*; |
|
|
|
import javax.accessibility.*; |
|
import javax.management.*; |
|
import javax.management.openmbean.CompositeData; |
|
import javax.swing.*; |
|
import javax.swing.border.*; |
|
|
|
|
|
import static sun.tools.jconsole.Formatter.*; |
|
import static sun.tools.jconsole.Utilities.*; |
|
|
|
@SuppressWarnings("serial") |
|
class MemoryTab extends Tab implements ActionListener, ItemListener { |
|
JComboBox<Plotter> plotterChoice; |
|
TimeComboBox timeComboBox; |
|
JButton gcButton; |
|
|
|
PlotterPanel plotterPanel; |
|
JPanel bottomPanel; |
|
HTMLPane details; |
|
PoolChart poolChart; |
|
|
|
ArrayList<Plotter> plotterList; |
|
Plotter heapPlotter, nonHeapPlotter; |
|
|
|
private MemoryOverviewPanel overviewPanel; |
|
|
|
private static final String usedKey = "used"; |
|
private static final String committedKey = "committed"; |
|
private static final String maxKey = "max"; |
|
private static final String thresholdKey = "threshold"; |
|
private static final Color usedColor = Plotter.defaultColor; |
|
private static final Color committedColor = null; |
|
private static final Color maxColor = null; |
|
private static final Color thresholdColor = Color.red; |
|
|
|
/* |
|
Hierarchy of panels and layouts for this tab: |
|
|
|
MemoryTab (BorderLayout) |
|
|
|
North: topPanel (BorderLayout) |
|
|
|
Center: controlPanel (FlowLayout) |
|
plotterChoice, timeComboBox |
|
|
|
East: topRightPanel (FlowLayout) |
|
gcButton |
|
|
|
Center: plotterPanel |
|
|
|
Center: plotter |
|
|
|
South: bottomPanel (BorderLayout) |
|
|
|
Center: details |
|
East: poolChart |
|
*/ |
|
|
|
|
|
public static String getTabName() { |
|
return Messages.MEMORY; |
|
} |
|
|
|
public MemoryTab(VMPanel vmPanel) { |
|
super(vmPanel, getTabName()); |
|
|
|
setLayout(new BorderLayout(0, 0)); |
|
setBorder(new EmptyBorder(4, 4, 3, 4)); |
|
|
|
JPanel topPanel = new JPanel(new BorderLayout()); |
|
plotterPanel = new PlotterPanel(null); |
|
bottomPanel = new JPanel(new BorderLayout()); |
|
|
|
add(topPanel, BorderLayout.NORTH); |
|
add(plotterPanel, BorderLayout.CENTER); |
|
|
|
JPanel controlPanel = new JPanel(new FlowLayout(FlowLayout.LEADING, 20, 5)); |
|
topPanel.add(controlPanel, BorderLayout.CENTER); |
|
|
|
|
|
plotterChoice = new JComboBox<Plotter>(); |
|
plotterChoice.addItemListener(this); |
|
controlPanel.add(new LabeledComponent(Messages.CHART_COLON, |
|
Resources.getMnemonicInt(Messages.CHART_COLON), |
|
plotterChoice)); |
|
|
|
|
|
timeComboBox = new TimeComboBox(); |
|
controlPanel.add(new LabeledComponent(Messages.TIME_RANGE_COLON, |
|
Resources.getMnemonicInt(Messages.TIME_RANGE_COLON), |
|
timeComboBox)); |
|
|
|
gcButton = new JButton(Messages.PERFORM_GC); |
|
gcButton.setMnemonic(Resources.getMnemonicInt(Messages.PERFORM_GC)); |
|
gcButton.addActionListener(this); |
|
gcButton.setToolTipText(Messages.PERFORM_GC_TOOLTIP); |
|
JPanel topRightPanel = new JPanel(); |
|
topRightPanel.setBorder(new EmptyBorder(0, 65-8, 0, 70)); |
|
topRightPanel.add(gcButton); |
|
topPanel.add(topRightPanel, BorderLayout.AFTER_LINE_ENDS); |
|
|
|
bottomPanel.setBorder(new CompoundBorder(new TitledBorder(Messages.DETAILS), |
|
new EmptyBorder(10, 10, 10, 10))); |
|
|
|
details = new HTMLPane(); |
|
setAccessibleName(details, Messages.DETAILS); |
|
bottomPanel.add(new JScrollPane(details), BorderLayout.CENTER); |
|
|
|
poolChart = new PoolChart(); |
|
bottomPanel.add(poolChart, BorderLayout.AFTER_LINE_ENDS); |
|
} |
|
|
|
|
|
private void createPlotters() throws IOException { |
|
plotterList = new ArrayList<Plotter>(); |
|
|
|
ProxyClient proxyClient = vmPanel.getProxyClient(); |
|
|
|
heapPlotter = new Plotter(Plotter.Unit.BYTES) { |
|
public String toString() { |
|
return Messages.HEAP_MEMORY_USAGE; |
|
} |
|
}; |
|
proxyClient.addWeakPropertyChangeListener(heapPlotter); |
|
|
|
nonHeapPlotter = new Plotter(Plotter.Unit.BYTES) { |
|
public String toString() { |
|
return Messages.NON_HEAP_MEMORY_USAGE; |
|
} |
|
}; |
|
|
|
setAccessibleName(heapPlotter, |
|
Messages.MEMORY_TAB_HEAP_PLOTTER_ACCESSIBLE_NAME); |
|
setAccessibleName(nonHeapPlotter, |
|
Messages.MEMORY_TAB_NON_HEAP_PLOTTER_ACCESSIBLE_NAME); |
|
|
|
proxyClient.addWeakPropertyChangeListener(nonHeapPlotter); |
|
|
|
heapPlotter.createSequence(usedKey, Messages.USED, usedColor, true); |
|
heapPlotter.createSequence(committedKey, Messages.COMMITTED, committedColor, false); |
|
heapPlotter.createSequence(maxKey, Messages.MAX, maxColor, false); |
|
|
|
nonHeapPlotter.createSequence(usedKey, Messages.USED, usedColor, true); |
|
nonHeapPlotter.createSequence(committedKey, Messages.COMMITTED, committedColor, false); |
|
nonHeapPlotter.createSequence(maxKey, Messages.MAX, maxColor, false); |
|
|
|
plotterList.add(heapPlotter); |
|
plotterList.add(nonHeapPlotter); |
|
|
|
|
|
Map<ObjectName, MBeanInfo> mBeanMap = proxyClient.getMBeans("java.lang"); |
|
Set<ObjectName> keys = mBeanMap.keySet(); |
|
ObjectName[] objectNames = keys.toArray(new ObjectName[keys.size()]); |
|
ArrayList<PoolPlotter> nonHeapPlotters = new ArrayList<PoolPlotter>(2); |
|
for (ObjectName objectName : objectNames) { |
|
String type = objectName.getKeyProperty("type"); |
|
if (type.equals("MemoryPool")) { |
|
String name = Resources.format(Messages.MEMORY_POOL_LABEL, |
|
objectName.getKeyProperty("name")); |
|
|
|
boolean isHeap = false; |
|
AttributeList al = |
|
proxyClient.getAttributes(objectName, |
|
new String[] { "Type" }); |
|
if (al.size() > 0) { |
|
isHeap = MemoryType.HEAP.name().equals(((Attribute)al.get(0)).getValue()); |
|
} |
|
PoolPlotter poolPlotter = new PoolPlotter(objectName, name, isHeap); |
|
proxyClient.addWeakPropertyChangeListener(poolPlotter); |
|
|
|
poolPlotter.createSequence(usedKey, Messages.USED, usedColor, true); |
|
poolPlotter.createSequence(committedKey, Messages.COMMITTED, committedColor, false); |
|
poolPlotter.createSequence(maxKey, Messages.MAX, maxColor, false); |
|
poolPlotter.createSequence(thresholdKey, Messages.THRESHOLD, thresholdColor, false); |
|
poolPlotter.setUseDashedTransitions(thresholdKey, true); |
|
|
|
if (isHeap) { |
|
plotterList.add(poolPlotter); |
|
} else { |
|
|
|
nonHeapPlotters.add(poolPlotter); |
|
} |
|
} |
|
} |
|
|
|
for (PoolPlotter poolPlotter : nonHeapPlotters) { |
|
plotterList.add(poolPlotter); |
|
} |
|
} |
|
|
|
|
|
public void itemStateChanged(ItemEvent ev) { |
|
if (ev.getStateChange() == ItemEvent.SELECTED) { |
|
Plotter plotter = (Plotter)plotterChoice.getSelectedItem(); |
|
plotterPanel.setPlotter(plotter); |
|
plotterPanel.repaint(); |
|
} |
|
} |
|
|
|
public void gc() { |
|
new Thread("MemoryPanel.gc") { |
|
public void run() { |
|
ProxyClient proxyClient = vmPanel.getProxyClient(); |
|
try { |
|
proxyClient.getMemoryMXBean().gc(); |
|
} catch (UndeclaredThrowableException e) { |
|
proxyClient.markAsDead(); |
|
} catch (IOException e) { |
|
// Ignore |
|
} |
|
} |
|
}.start(); |
|
} |
|
|
|
public SwingWorker<?, ?> newSwingWorker() { |
|
return new SwingWorker<Boolean, Object>() { |
|
private long[] used, committed, max, threshold; |
|
private long timeStamp; |
|
private String detailsStr; |
|
private boolean initialRun = false; |
|
|
|
public Boolean doInBackground() { |
|
ProxyClient proxyClient = vmPanel.getProxyClient(); |
|
|
|
if (plotterList == null) { |
|
try { |
|
createPlotters(); |
|
} catch (UndeclaredThrowableException e) { |
|
proxyClient.markAsDead(); |
|
return false; |
|
} catch (final IOException ex) { |
|
return false; |
|
} |
|
initialRun = true; |
|
} |
|
|
|
int n = plotterList.size(); |
|
used = new long[n]; |
|
committed = new long[n]; |
|
max = new long[n]; |
|
threshold = new long[n]; |
|
timeStamp = System.currentTimeMillis(); |
|
|
|
for (int i = 0; i < n; i++) { |
|
Plotter plotter = plotterList.get(i); |
|
MemoryUsage mu = null; |
|
used[i] = -1L; |
|
threshold[i] = -1L; |
|
|
|
try { |
|
if (plotter instanceof PoolPlotter) { |
|
PoolPlotter poolPlotter = (PoolPlotter)plotter; |
|
ObjectName objectName = poolPlotter.objectName; |
|
AttributeList al = |
|
proxyClient.getAttributes(objectName, |
|
new String[] { "Usage", "UsageThreshold" }); |
|
if (al.size() > 0) { |
|
CompositeData cd = (CompositeData)((Attribute)al.get(0)).getValue(); |
|
mu = MemoryUsage.from(cd); |
|
|
|
if (al.size() > 1) { |
|
threshold[i] = (Long)((Attribute)al.get(1)).getValue(); |
|
} |
|
} |
|
} else if (plotter == heapPlotter) { |
|
mu = proxyClient.getMemoryMXBean().getHeapMemoryUsage(); |
|
} else if (plotter == nonHeapPlotter) { |
|
mu = proxyClient.getMemoryMXBean().getNonHeapMemoryUsage(); |
|
} |
|
} catch (UndeclaredThrowableException e) { |
|
proxyClient.markAsDead(); |
|
return false; |
|
} catch (IOException ex) { |
|
// Skip this plotter |
|
} |
|
|
|
if (mu != null) { |
|
used[i] = mu.getUsed(); |
|
committed[i] = mu.getCommitted(); |
|
max[i] = mu.getMax(); |
|
} |
|
} |
|
detailsStr = formatDetails(); |
|
|
|
return true; |
|
} |
|
|
|
protected void done() { |
|
try { |
|
if (!get()) { |
|
return; |
|
} |
|
} catch (InterruptedException ex) { |
|
return; |
|
} catch (ExecutionException ex) { |
|
if (JConsole.isDebug()) { |
|
ex.printStackTrace(); |
|
} |
|
return; |
|
} |
|
|
|
if (initialRun) { |
|
|
|
for (Plotter p : plotterList) { |
|
plotterChoice.addItem(p); |
|
timeComboBox.addPlotter(p); |
|
} |
|
add(bottomPanel, BorderLayout.SOUTH); |
|
} |
|
|
|
|
|
int n = plotterList.size(); |
|
int poolCount = 0; |
|
|
|
for (int i = 0; i < n; i++) { |
|
Plotter plotter = plotterList.get(i); |
|
if (used[i] >= 0L) { |
|
if (plotter instanceof PoolPlotter) { |
|
plotter.addValues(timeStamp, used[i], committed[i], max[i], threshold[i]); |
|
if (threshold[i] > 0L) { |
|
plotter.setIsPlotted(thresholdKey, true); |
|
} |
|
poolChart.setValue(poolCount++, (PoolPlotter)plotter, |
|
used[i], threshold[i], max[i]); |
|
} else { |
|
plotter.addValues(timeStamp, used[i], committed[i], max[i]); |
|
} |
|
|
|
if (plotter == heapPlotter && overviewPanel != null) { |
|
overviewPanel.getPlotter().addValues(timeStamp, used[i]); |
|
overviewPanel.updateMemoryInfo(used[i], committed[i], max[i]); |
|
} |
|
} |
|
} |
|
details.setText(detailsStr); |
|
} |
|
}; |
|
} |
|
|
|
private String formatDetails() { |
|
ProxyClient proxyClient = vmPanel.getProxyClient(); |
|
if (proxyClient.isDead()) { |
|
return ""; |
|
} |
|
|
|
String text = "<table cellspacing=0 cellpadding=0>"; |
|
|
|
Plotter plotter = (Plotter)plotterChoice.getSelectedItem(); |
|
if (plotter == null) { |
|
return ""; |
|
} |
|
|
|
|
|
long time = System.currentTimeMillis(); |
|
String timeStamp = formatDateTime(time); |
|
text += newRow(Messages.TIME, timeStamp); |
|
|
|
long used = plotter.getLastValue(usedKey); |
|
long committed = plotter.getLastValue(committedKey); |
|
long max = plotter.getLastValue(maxKey); |
|
long threshold = plotter.getLastValue(thresholdKey); |
|
|
|
text += newRow(Messages.USED, formatKBytes(used)); |
|
if (committed > 0L) { |
|
text += newRow(Messages.COMMITTED, formatKBytes(committed)); |
|
} |
|
if (max > 0L) { |
|
text += newRow(Messages.MAX, formatKBytes(max)); |
|
} |
|
if (threshold > 0L) { |
|
text += newRow(Messages.USAGE_THRESHOLD, formatKBytes(threshold)); |
|
} |
|
|
|
try { |
|
Collection<GarbageCollectorMXBean> garbageCollectors = |
|
proxyClient.getGarbageCollectorMXBeans(); |
|
|
|
boolean descPrinted = false; |
|
for (GarbageCollectorMXBean garbageCollectorMBean : garbageCollectors) { |
|
String gcName = garbageCollectorMBean.getName(); |
|
long gcCount = garbageCollectorMBean.getCollectionCount(); |
|
long gcTime = garbageCollectorMBean.getCollectionTime(); |
|
String str = Resources.format(Messages.GC_TIME_DETAILS, justify(formatTime(gcTime), 14), |
|
gcName, |
|
String.format("%,d",gcCount)); |
|
if (!descPrinted) { |
|
text += newRow(Messages.GC_TIME, str); |
|
descPrinted = true; |
|
} else { |
|
text += newRow(null, str); |
|
} |
|
} |
|
} catch (IOException e) { |
|
} |
|
|
|
return text; |
|
} |
|
|
|
public void actionPerformed(ActionEvent ev) { |
|
Object src = ev.getSource(); |
|
if (src == gcButton) { |
|
gc(); |
|
} |
|
} |
|
|
|
private class PoolPlotter extends Plotter { |
|
ObjectName objectName; |
|
String name; |
|
boolean isHeap; |
|
long value, threshold, max; |
|
int barX; |
|
|
|
public PoolPlotter(ObjectName objectName, String name, boolean isHeap) { |
|
super(Plotter.Unit.BYTES); |
|
|
|
this.objectName = objectName; |
|
this.name = name; |
|
this.isHeap = isHeap; |
|
|
|
setAccessibleName(this, |
|
Resources.format(Messages.MEMORY_TAB_POOL_PLOTTER_ACCESSIBLE_NAME, |
|
name)); |
|
} |
|
|
|
|
|
public String toString() { |
|
return name; |
|
} |
|
} |
|
|
|
private class PoolChart extends BorderedComponent |
|
implements Accessible, MouseListener { |
|
final int height = 150; |
|
final int leftMargin = 50; |
|
final int rightMargin = 23; |
|
final int bottomMargin = 35; |
|
final int barWidth = 22; |
|
final int barGap = 3; |
|
final int groupGap = 8; |
|
final int barHeight = height * 2 / 3; |
|
|
|
final Color greenBar = new Color(100, 255, 100); |
|
final Color greenBarBackground = new Color(210, 255, 210); |
|
final Color redBarBackground = new Color(255, 210, 210); |
|
|
|
Font smallFont = null; |
|
|
|
ArrayList<PoolPlotter> poolPlotters = new ArrayList<PoolPlotter>(5); |
|
|
|
int nHeapPools = 0; |
|
int nNonHeapPools = 0; |
|
Rectangle heapRect = new Rectangle(leftMargin, height - bottomMargin + 6, barWidth, 20); |
|
Rectangle nonHeapRect = new Rectangle(leftMargin + groupGap, height - bottomMargin + 6, barWidth, 20); |
|
|
|
public PoolChart() { |
|
super(null, null); |
|
|
|
setFocusable(true); |
|
addMouseListener(this); |
|
ToolTipManager.sharedInstance().registerComponent(this); |
|
} |
|
|
|
public void setValue(int poolIndex, PoolPlotter poolPlotter, |
|
long value, long threshold, long max) { |
|
poolPlotter.value = value; |
|
poolPlotter.threshold = threshold; |
|
poolPlotter.max = max; |
|
|
|
if (poolIndex == poolPlotters.size()) { |
|
poolPlotters.add(poolPlotter); |
|
if (poolPlotter.isHeap) { |
|
poolPlotter.barX = nHeapPools * (barWidth + barGap); |
|
nHeapPools++; |
|
heapRect.width = nHeapPools * barWidth + (nHeapPools - 1) * barGap; |
|
nonHeapRect.x = leftMargin + heapRect.width + groupGap; |
|
} else { |
|
poolPlotter.barX = nonHeapRect.x - leftMargin + nNonHeapPools * (barWidth + barGap); |
|
nNonHeapPools++; |
|
nonHeapRect.width = nNonHeapPools * barWidth + (nNonHeapPools - 1) * barGap; |
|
} |
|
} else { |
|
poolPlotters.set(poolIndex, poolPlotter); |
|
} |
|
repaint(); |
|
} |
|
|
|
private void paintPoolBar(Graphics g, PoolPlotter poolPlotter) { |
|
Rectangle barRect = getBarRect(poolPlotter); |
|
g.setColor(Color.gray); |
|
g.drawRect(barRect.x, barRect.y, barRect.width, barRect.height); |
|
|
|
long value = poolPlotter.value; |
|
long max = poolPlotter.max; |
|
if (max > 0L) { |
|
g.translate(barRect.x, barRect.y); |
|
|
|
|
|
g.setColor(greenBarBackground); |
|
g.fillRect(1, 1, barRect.width - 1, barRect.height - 1); |
|
|
|
int greenHeight = (int)(value * barRect.height / max); |
|
long threshold = poolPlotter.threshold; |
|
if (threshold > 0L) { |
|
int redHeight = (int)(threshold * barRect.height / max); |
|
|
|
|
|
g.setColor(redBarBackground); |
|
g.fillRect(1, 1, barRect.width - 1, barRect.height - redHeight); |
|
|
|
if (value > threshold) { |
|
|
|
g.setColor(thresholdColor); |
|
g.fillRect(1, barRect.height - greenHeight, |
|
barRect.width - 1, greenHeight - redHeight); |
|
greenHeight = redHeight; |
|
} |
|
} |
|
|
|
|
|
g.setColor(greenBar); |
|
g.fillRect(1, barRect.height - greenHeight, |
|
barRect.width - 1, greenHeight); |
|
|
|
g.translate(-barRect.x, -barRect.y); |
|
} |
|
} |
|
|
|
public void paintComponent(Graphics g) { |
|
super.paintComponent(g); |
|
|
|
if (poolPlotters.size() == 0) { |
|
return; |
|
} |
|
|
|
if (smallFont == null) { |
|
smallFont = g.getFont().deriveFont(9.0F); |
|
} |
|
|
|
|
|
g.setColor(getBackground()); |
|
Rectangle r = g.getClipBounds(); |
|
g.fillRect(r.x, r.y, r.width, r.height); |
|
|
|
g.setFont(smallFont); |
|
FontMetrics fm = g.getFontMetrics(); |
|
int fontDescent = fm.getDescent(); |
|
|
|
|
|
g.setColor(getForeground()); |
|
for (int pc : new int[] { 0, 25, 50, 75, 100 }) { |
|
String str = pc + "% --"; |
|
g.drawString(str, |
|
leftMargin - fm.stringWidth(str) - 4, |
|
height - bottomMargin - (pc * barHeight / 100) + fontDescent + 1); |
|
} |
|
|
|
for (PoolPlotter poolPlotter : poolPlotters) { |
|
paintPoolBar(g, poolPlotter); |
|
} |
|
|
|
g.setColor(Color.gray); |
|
g.drawRect(heapRect.x, heapRect.y, heapRect.width, heapRect.height); |
|
g.drawRect(nonHeapRect.x, nonHeapRect.y, nonHeapRect.width, nonHeapRect.height); |
|
|
|
Color heapColor = greenBar; |
|
Color nonHeapColor = greenBar; |
|
|
|
|
|
for (PoolPlotter poolPlotter : poolPlotters) { |
|
if (poolPlotter.threshold > 0L && poolPlotter.value > poolPlotter.threshold) { |
|
if (poolPlotter.isHeap) { |
|
heapColor = thresholdColor; |
|
} else { |
|
nonHeapColor = thresholdColor; |
|
} |
|
} |
|
} |
|
g.setColor(heapColor); |
|
g.fillRect(heapRect.x + 1, heapRect.y + 1, heapRect.width - 1, heapRect.height - 1); |
|
g.setColor(nonHeapColor); |
|
g.fillRect(nonHeapRect.x + 1, nonHeapRect.y + 1, nonHeapRect.width - 1, nonHeapRect.height - 1); |
|
|
|
String str = Messages.HEAP; |
|
int stringWidth = fm.stringWidth(str); |
|
int x = heapRect.x + (heapRect.width - stringWidth) / 2; |
|
int y = heapRect.y + heapRect.height - 6; |
|
g.setColor(Color.white); |
|
g.drawString(str, x-1, y-1); |
|
g.drawString(str, x+1, y-1); |
|
g.drawString(str, x-1, y+1); |
|
g.drawString(str, x+1, y+1); |
|
g.setColor(Color.black); |
|
g.drawString(str, x, y); |
|
|
|
str = Messages.NON_HEAP; |
|
stringWidth = fm.stringWidth(str); |
|
x = nonHeapRect.x + (nonHeapRect.width - stringWidth) / 2; |
|
y = nonHeapRect.y + nonHeapRect.height - 6; |
|
g.setColor(Color.white); |
|
g.drawString(str, x-1, y-1); |
|
g.drawString(str, x+1, y-1); |
|
g.drawString(str, x-1, y+1); |
|
g.drawString(str, x+1, y+1); |
|
g.setColor(Color.black); |
|
g.drawString(str, x, y); |
|
|
|
|
|
g.setColor(Color.blue); |
|
r = null; |
|
Plotter plotter = (Plotter)plotterChoice.getSelectedItem(); |
|
if (plotter == heapPlotter) { |
|
r = heapRect; |
|
} else if (plotter == nonHeapPlotter) { |
|
r = nonHeapRect; |
|
} else if (plotter instanceof PoolPlotter) { |
|
r = getBarRect((PoolPlotter)plotter); |
|
} |
|
if (r != null) { |
|
g.drawRect(r.x - 1, r.y - 1, r.width + 2, r.height + 2); |
|
} |
|
} |
|
|
|
private Rectangle getBarRect(PoolPlotter poolPlotter) { |
|
return new Rectangle(leftMargin + poolPlotter.barX, |
|
height - bottomMargin - barHeight, |
|
barWidth, barHeight); |
|
} |
|
|
|
public Dimension getPreferredSize() { |
|
return new Dimension(nonHeapRect.x + nonHeapRect.width + rightMargin, |
|
height); |
|
} |
|
|
|
public void mouseClicked(MouseEvent e) { |
|
requestFocusInWindow(); |
|
Plotter plotter = getPlotter(e); |
|
|
|
if (plotter != null && plotter != plotterChoice.getSelectedItem()) { |
|
plotterChoice.setSelectedItem(plotter); |
|
repaint(); |
|
} |
|
} |
|
|
|
public String getToolTipText(MouseEvent e) { |
|
Plotter plotter = getPlotter(e); |
|
|
|
return (plotter != null) ? plotter.toString() : null; |
|
} |
|
|
|
private Plotter getPlotter(MouseEvent e) { |
|
Point p = e.getPoint(); |
|
Plotter plotter = null; |
|
|
|
if (heapRect.contains(p)) { |
|
plotter = heapPlotter; |
|
} else if (nonHeapRect.contains(p)) { |
|
plotter = nonHeapPlotter; |
|
} else { |
|
for (PoolPlotter poolPlotter : poolPlotters) { |
|
if (getBarRect(poolPlotter).contains(p)) { |
|
plotter = poolPlotter; |
|
break; |
|
} |
|
} |
|
} |
|
return plotter; |
|
} |
|
|
|
public void mousePressed(MouseEvent e) {} |
|
public void mouseReleased(MouseEvent e) {} |
|
public void mouseEntered(MouseEvent e) {} |
|
public void mouseExited(MouseEvent e) {} |
|
|
|
|
|
public AccessibleContext getAccessibleContext() { |
|
if (accessibleContext == null) { |
|
accessibleContext = new AccessiblePoolChart(); |
|
} |
|
return accessibleContext; |
|
} |
|
|
|
protected class AccessiblePoolChart extends AccessibleJPanel { |
|
public String getAccessibleName() { |
|
String name = Messages.MEMORY_TAB_POOL_CHART_ACCESSIBLE_NAME; |
|
|
|
String keyValueList = ""; |
|
for (PoolPlotter poolPlotter : poolPlotters) { |
|
String value = (poolPlotter.value * 100 / poolPlotter.max) + "%"; |
|
|
|
keyValueList += |
|
Resources.format(Messages.PLOTTER_ACCESSIBLE_NAME_KEY_AND_VALUE, |
|
poolPlotter.toString(), value); |
|
if (poolPlotter.threshold > 0L) { |
|
String threshold = |
|
(poolPlotter.threshold * 100 / poolPlotter.max) + "%"; |
|
if (poolPlotter.value > poolPlotter.threshold) { |
|
keyValueList += |
|
Resources.format(Messages.MEMORY_TAB_POOL_CHART_ABOVE_THRESHOLD, |
|
threshold); |
|
} else { |
|
keyValueList += |
|
Resources.format(Messages.MEMORY_TAB_POOL_CHART_BELOW_THRESHOLD, |
|
threshold); |
|
} |
|
} |
|
} |
|
|
|
return name + "\n" + keyValueList + "."; |
|
} |
|
} |
|
} |
|
|
|
|
|
OverviewPanel[] getOverviewPanels() { |
|
if (overviewPanel == null) { |
|
overviewPanel = new MemoryOverviewPanel(); |
|
} |
|
return new OverviewPanel[] { overviewPanel }; |
|
} |
|
|
|
private static class MemoryOverviewPanel extends OverviewPanel { |
|
MemoryOverviewPanel() { |
|
super(Messages.HEAP_MEMORY_USAGE, usedKey, Messages.USED, Plotter.Unit.BYTES); |
|
} |
|
|
|
private void updateMemoryInfo(long used, long committed, long max) { |
|
getInfoLabel().setText(Resources.format(Messages.MEMORY_TAB_INFO_LABEL_FORMAT, |
|
formatBytes(used, true), |
|
formatBytes(committed, true), |
|
formatBytes(max, true))); |
|
} |
|
} |
|
} |