src/jdk.jconsole/share/classes/sun/tools/jconsole/MemoryTab.java
changeset 47216 71c04702a3d5
parent 25859 3317bb8137f4
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/src/jdk.jconsole/share/classes/sun/tools/jconsole/MemoryTab.java	Tue Sep 12 19:03:39 2017 +0200
@@ -0,0 +1,768 @@
+/*
+ * Copyright (c) 2004, 2013, 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.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);
+
+        // Plotter choice
+        plotterChoice = new JComboBox<Plotter>();
+        plotterChoice.addItemListener(this);
+        controlPanel.add(new LabeledComponent(Messages.CHART_COLON,
+                                              Resources.getMnemonicInt(Messages.CHART_COLON),
+                                              plotterChoice));
+
+        // Range control
+        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);
+
+        // Now add memory pools
+        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"));
+                // Heap or non-heap?
+                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 {
+                    // Will be added to plotterList below
+                    nonHeapPlotters.add(poolPlotter);
+                }
+            }
+        }
+        // Add non-heap plotters last
+        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) {
+                    // Add Memory Pools
+                    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 = plotter.getLastTimeStamp();
+        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);
+
+                // Paint green background
+                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);
+
+                    // Paint red background
+                    g.setColor(redBarBackground);
+                    g.fillRect(1, 1, barRect.width - 1, barRect.height - redHeight);
+
+                    if (value > threshold) {
+                        // Over threshold, paint red bar
+                        g.setColor(thresholdColor);
+                        g.fillRect(1, barRect.height - greenHeight,
+                                   barRect.width - 1, greenHeight - redHeight);
+                        greenHeight = redHeight;
+                    }
+                }
+
+                // Paint green bar
+                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);
+            }
+
+            // Paint background for chart area
+            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();
+
+            // Paint percentage axis
+            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);
+
+            // Highlight current plotter
+            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) + "%";
+                    // Assume format string ends with newline
+                    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)));
+        }
+    }
+}