/* * Copyright 2003-2004 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. * * 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. */ /* * @test * @bug 4530538 * @summary Basic unit test of memory management testing: * 1) setUsatgeThreshold() and getUsageThreshold() * 2) test low memory detection on the old generation. * * @author Mandy Chung * * @build MemoryManagement MemoryUtil * @run main/timeout=600 MemoryManagement */ import java.lang.management.*; import java.util.*; import javax.management.*; import javax.management.openmbean.CompositeData; public class MemoryManagement { private static MemoryMXBean mm = ManagementFactory.getMemoryMXBean(); private static List pools = ManagementFactory.getMemoryPoolMXBeans(); private static List managers = ManagementFactory.getMemoryManagerMXBeans(); private static MemoryPoolMXBean mpool = null; private static boolean trace = false; private static boolean testFailed = false; private static final int NUM_CHUNKS = 2; private static long chunkSize; private static int listenerInvoked = 0; static class SensorListener implements NotificationListener { public void handleNotification(Notification notif, Object handback) { String type = notif.getType(); if (type.equals(MemoryNotificationInfo.MEMORY_THRESHOLD_EXCEEDED) || type.equals(MemoryNotificationInfo. MEMORY_COLLECTION_THRESHOLD_EXCEEDED)) { MemoryNotificationInfo minfo = MemoryNotificationInfo. from((CompositeData) notif.getUserData()); MemoryUtil.printMemoryNotificationInfo(minfo, type); listenerInvoked++; } } } private static long newThreshold; public static void main(String args[]) throws Exception { if (args.length > 0 && args[0].equals("trace")) { trace = true; } if (trace) { MemoryUtil.printMemoryPools(pools); MemoryUtil.printMemoryManagers(managers); } // Find the Old generation which supports low memory detection ListIterator iter = pools.listIterator(); while (iter.hasNext()) { MemoryPoolMXBean p = (MemoryPoolMXBean) iter.next(); if (p.getType() == MemoryType.HEAP && p.isUsageThresholdSupported()) { mpool = p; if (trace) { System.out.println("Selected memory pool for low memory " + "detection."); MemoryUtil.printMemoryPool(mpool); } break; } } SensorListener listener = new SensorListener(); NotificationEmitter emitter = (NotificationEmitter) mm; emitter.addNotificationListener(listener, null, null); Thread allocator = new AllocatorThread(); // Now set threshold MemoryUsage mu = mpool.getUsage(); chunkSize = (mu.getMax() - mu.getUsed()) / 20; newThreshold = mu.getUsed() + (chunkSize * NUM_CHUNKS); System.out.println("Setting threshold for " + mpool.getName() + " from " + mpool.getUsageThreshold() + " to " + newThreshold + ". Current used = " + mu.getUsed()); mpool.setUsageThreshold(newThreshold); if (mpool.getUsageThreshold() != newThreshold) { throw new RuntimeException("TEST FAILED: " + "Threshold for Memory pool " + mpool.getName() + "is " + mpool.getUsageThreshold() + " but expected to be" + newThreshold); } // Start the AllocatorThread to continously allocate memory System.out.println("Starting an AllocatorThread to allocate memory."); allocator.start(); try { allocator.join(); } catch (InterruptedException e) { e.printStackTrace(); System.out.println("Unexpected exception."); testFailed = true; } if (listenerInvoked == 0) { throw new RuntimeException("No listener is invoked"); } if (testFailed) throw new RuntimeException("TEST FAILED."); System.out.println("Test passed."); } static class AllocatorThread extends Thread { private List objectPool = new ArrayList(); public void run() { int iterations = 0; int numElements = (int) (chunkSize / 4); // minimal object size while (listenerInvoked == 0) { iterations++; if (trace) { System.out.println(" Iteration " + iterations + ": before allocation " + mpool.getUsage().getUsed()); } Object[] o = new Object[numElements]; if (iterations <= NUM_CHUNKS) { // only hold a reference to the first NUM_CHUNKS // allocated objects objectPool.add(o); } if (trace) { System.out.println(" " + " after allocation " + mpool.getUsage().getUsed()); } try { Thread.sleep(100); } catch (InterruptedException e) { e.printStackTrace(); System.out.println("Unexpected exception."); testFailed = true; } } System.out.println("AllocatedThread finished memory allocation " + " num_iteration = " + iterations); } } }