// MemoryControl.java // ------------------------------------------- // (C) 2005 by Michael Peter Christen; mc@yacy.net, Frankfurt a. M., Germany // first published 22.09.2005 on http://yacy.net // // $LastChangedDate$ // $LastChangedRevision$ // $LastChangedBy$ // // LICENSE // // This program is free software; you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation; either version 2 of the License, or // (at your option) any later version. // // This program 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 for more details. // // You should have received a copy of the GNU General Public License // along with this program; if not, write to the Free Software // Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA package net.yacy.kelondro.util; import java.util.concurrent.atomic.AtomicBoolean; /** * Use this to get information about memory usage or try to free some memory */ public class MemoryControl { private static AtomicBoolean shortStatus = new AtomicBoolean(false); private static boolean simulatedShortStatus = false, usingStandardStrategy = true; private static MemoryStrategy strategy; private static MemoryStrategy getStrategy() { if (strategy == null || MemoryStrategy.hasError()) { if (!usingStandardStrategy) { strategy = new GenerationMemoryStrategy(); // if (strategy.hasError()) { // perhaps we do have a G1 // strategy = new G1MemoryStrategy(); // } // fall back if error detected if (MemoryStrategy.hasError()) { usingStandardStrategy = true; strategy = new StandardMemoryStrategy(); } } else { strategy = new StandardMemoryStrategy(); } } return strategy; } public final static void setStandardStrategy(final boolean std) { if (usingStandardStrategy != std) { usingStandardStrategy = std; strategy = null; } } /** * @return the name of the used strategy */ public final static String getStrategyName() { getStrategy(); return MemoryStrategy.getName(); } /** * Runs the garbage collector if last garbage collection is more than last millis ago * @param last time which must be passed since lased gc * @param info additional info for log */ public final synchronized static boolean gc(final int last, final String info) { // thq return getStrategy().gc(last, info); } /** * memory that is free without increasing of total memory taken from os * @return bytes */ public static final long free() { return getStrategy().free(); } /** * memory that is available including increasing total memory up to maximum * @return bytes */ public static final long available() { long available = getStrategy().available(); return available; } /** * maximum memory the Java virtual will allocate machine; may vary over time in some cases * @return bytes */ public static final long maxMemory() { return getStrategy().maxMemory(); } /** * currently allocated memory in the Java virtual machine; may vary over time * @return bytes */ public static final long total() { return getStrategy().total(); } /** * check for a specified amount of bytes * * @param size the requested amount of free memory in bytes * @param force specifies whether risk an expensive GC * @return whether enough memory could be freed (or is free) or not */ public static boolean request(final long size, final boolean force) { if (size < 1024) return true; // to speed up things. If this would fail, it would be much too late to check this. return getStrategy().request(size, force, shortStatus); } /** * the simulated short status can be set to find out if the short status has effects to the system * @param status */ public static void setSimulatedShortStatus(final boolean status) { simulatedShortStatus = status; } /** * the simulated short status can be retrieved to show that option in online interfaces * @return */ public static boolean getSimulatedShortStatus() { return simulatedShortStatus; } /** * @return if last request failed */ public static boolean shortStatus() { //if (shortStatus) System.out.println("**** SHORT MEMORY ****"); return simulatedShortStatus || shortStatus.get(); } /** * memory that is currently bound in objects * @return used bytes */ public static long used() { return getStrategy().used(); } /** * @return if Memory seams to be in a proper state */ public static boolean properState() { return getStrategy().properState(); } /** * forced enable properState - StandardMemoryStrategy only */ public static void resetProperState() { getStrategy().resetProperState(); } /** * set the memory to be available for properState - StandardMemoryStrategy only */ public static void setProperMbyte(final long mbyte) { getStrategy().setProperMbyte(mbyte); } /** * main * @param args use 'force' to request by force, use 'std' / 'gen' to specify strategy */ public static void main(final String[] args) { // try this with different strategy and compare results final int mb = 1024 * 1024; boolean force = false; for (final String arg : args) { if (arg.equals("force")) force = true; if (arg.equalsIgnoreCase("gen")) usingStandardStrategy = false; if (arg.equalsIgnoreCase("std")) usingStandardStrategy = true; } System.out.println("vm: " + System.getProperty("java.vm.version")); System.out.println("computed max = " + (maxMemory() / mb) + " mb"); System.out.println("using " + getStrategyName()); final byte[][] x = new byte[100000][]; for (int i = 0; i < 100000; i++) { if (request(mb, force)) { x[i] = new byte[mb]; System.out.println("used = " + i + " / " + (used() /mb) + ", total = " + (total() / mb) + ", free = " + (free() / mb) + ", max = " + (maxMemory() / mb) + ", avail = " + (available() / mb) + (usingStandardStrategy? ", averageGC = " + ((StandardMemoryStrategy)getStrategy()).getAverageGCFree() : "")); } else System.exit(0); } } }