1 /* 2 * Copyright (c) 2014, 2015, Oracle and/or its affiliates. All rights reserved. 3 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 4 * 5 * This code is free software; you can redistribute it and/or modify it 6 * under the terms of the GNU General Public License version 2 only, as 7 * published by the Free Software Foundation. 8 * 9 * This code is distributed in the hope that it will be useful, but WITHOUT 10 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 11 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 12 * version 2 for more details (a copy is included in the LICENSE file that 13 * accompanied this code). 14 * 15 * You should have received a copy of the GNU General Public License version 16 * 2 along with this work; if not, write to the Free Software Foundation, 17 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 18 * 19 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 20 * or visit www.oracle.com if you need additional information or have any 21 * questions. 22 */ 23 24 package compiler.codecache.cli.codeheapsize; 25 26 import compiler.codecache.cli.common.CodeCacheCLITestCase; 27 import compiler.codecache.cli.common.CodeCacheOptions; 28 import jdk.test.lib.ExitCode; 29 import jdk.test.lib.Utils; 30 import jdk.test.lib.cli.CommandLineOptionTest; 31 import sun.hotspot.code.BlobType; 32 33 import java.util.Random; 34 35 /** 36 * Test case runner aimed to verify option's consistency. 37 */ 38 public class JVMStartupRunner implements CodeCacheCLITestCase.Runner { 39 private static final String INCONSISTENT_CH_SIZES_ERROR 40 = "Invalid code heap sizes.*"; 41 42 @Override 43 public void run(CodeCacheCLITestCase.Description testCaseDescription, 44 CodeCacheOptions options) throws Throwable { 45 // Everything should be fine when 46 // sum(all code heap sizes) == reserved CC size 47 CommandLineOptionTest.verifySameJVMStartup(/* expected messages */ null, 48 new String[]{ INCONSISTENT_CH_SIZES_ERROR }, 49 "JVM startup should not fail with consistent code heap sizes", 50 "JVM output should not contain warning about inconsistent code " 51 + "heap sizes", ExitCode.OK, options.prepareOptions()); 52 53 verifySingleInconsistentValue(options); 54 verifyAllInconsistentValues(options); 55 } 56 57 /** 58 * Verifies that if at least one of three options will have value, such 59 * that sum of all three values will be inconsistent, then JVM startup will 60 * fail. 61 */ 62 private static void verifySingleInconsistentValue(CodeCacheOptions options) 63 throws Throwable { 64 verifyHeapSizesSum(options.reserved, 65 scaleCodeHeapSize(options.profiled), options.nonProfiled, 66 options.nonNmethods); 67 verifyHeapSizesSum(options.reserved, options.profiled, 68 scaleCodeHeapSize(options.nonProfiled), options.nonNmethods); 69 verifyHeapSizesSum(options.reserved, options.profiled, 70 options.nonProfiled, scaleCodeHeapSize(options.nonNmethods)); 71 } 72 73 /** 74 * Verifies that if all three options will have values such that their sum 75 * is inconsistent with ReservedCodeCacheSize value, then JVM startup will 76 * fail. 77 */ 78 private static void verifyAllInconsistentValues(CodeCacheOptions options) 79 throws Throwable { 80 long profiled = options.profiled; 81 long nonProfiled = options.nonProfiled; 82 long nonNMethods = options.nonNmethods; 83 84 while (options.reserved == profiled + nonProfiled + nonNMethods) { 85 profiled = scaleCodeHeapSize(profiled); 86 nonProfiled = scaleCodeHeapSize(nonProfiled); 87 nonNMethods = scaleCodeHeapSize(nonNMethods); 88 } 89 90 verifyHeapSizesSum(options.reserved, profiled, nonProfiled, 91 nonNMethods); 92 } 93 94 private static void verifyHeapSizesSum(long reserved, long profiled, 95 long nonProfiled, long nonNmethods) throws Throwable { 96 // JVM startup expected to fail when 97 // sum(all code heap sizes) != reserved CC size 98 CommandLineOptionTest.verifySameJVMStartup( 99 new String[]{ INCONSISTENT_CH_SIZES_ERROR }, 100 /* unexpected messages */ null, 101 "JVM startup should fail with inconsistent code heap size.", 102 "JVM output should contain appropriate error message of code " 103 + "heap sizes are inconsistent", 104 ExitCode.FAIL, 105 CommandLineOptionTest.prepareBooleanFlag( 106 CodeCacheOptions.SEGMENTED_CODE_CACHE, true), 107 CommandLineOptionTest.prepareNumericFlag( 108 BlobType.All.sizeOptionName, reserved), 109 CommandLineOptionTest.prepareNumericFlag( 110 BlobType.MethodProfiled.sizeOptionName, profiled), 111 CommandLineOptionTest.prepareNumericFlag( 112 BlobType.MethodNonProfiled.sizeOptionName, nonProfiled), 113 CommandLineOptionTest.prepareNumericFlag( 114 BlobType.NonNMethod.sizeOptionName, nonNmethods)); 115 } 116 117 /** 118 * Returns {@code unscaledSize} value scaled by a random factor from 119 * range (1, 2). If {@code unscaledSize} is not 0, then this 120 * method will return value that won't be equal to {@code unscaledSize}. 121 * 122 * @param unscaledSize The value to be scaled. 123 * @return {@code unscaledSize} value scaled by a factor from range (1, 2). 124 */ 125 private static long scaleCodeHeapSize(long unscaledSize) { 126 Random random = Utils.getRandomInstance(); 127 128 long scaledSize = unscaledSize; 129 while (scaledSize == unscaledSize && unscaledSize != 0) { 130 float scale = 1.0f + random.nextFloat(); 131 scaledSize = (long) Math.ceil(scale * unscaledSize); 132 } 133 return scaledSize; 134 } 135 }