1 /* 2 * Copyright (C) 2016 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 public class Main { main(String[] args)18 public static void main(String[] args) { 19 System.loadLibrary(args[0]); 20 new SubMain(); 21 if ($noinline$returnInt() != 53) { 22 throw new Error("Unexpected return value"); 23 } 24 if ($noinline$returnFloat() != 42.2f) { 25 throw new Error("Unexpected return value"); 26 } 27 if ($noinline$returnDouble() != Double.longBitsToDouble(0xF000000000001111L)) { 28 throw new Error("Unexpected return value "); 29 } 30 if ($noinline$returnLong() != 0xFFFF000000001111L) { 31 throw new Error("Unexpected return value"); 32 } 33 34 try { 35 $noinline$deopt(); 36 } catch (Exception e) {} 37 DeoptimizationController.stopDeoptimization(); 38 39 $noinline$inlineCache(new Main(), /* isSecondInvocation */ false); 40 if ($noinline$inlineCache(new SubMain(), /* isSecondInvocation */ true) != SubMain.class) { 41 throw new Error("Unexpected return value"); 42 } 43 44 $noinline$inlineCache2(new Main(), /* isSecondInvocation */ false); 45 if ($noinline$inlineCache2(new SubMain(), /* isSecondInvocation */ true) != SubMain.class) { 46 throw new Error("Unexpected return value"); 47 } 48 49 // Test polymorphic inline cache to the same target (inlineCache3). 50 $noinline$inlineCache3(new Main(), /* isSecondInvocation */ false); 51 $noinline$inlineCache3(new SubMain(), /* isSecondInvocation */ false); 52 if ($noinline$inlineCache3(new SubMain(), /* isSecondInvocation */ true) != null) { 53 throw new Error("Unexpected return value"); 54 } 55 56 $noinline$stackOverflow(new Main(), /* isSecondInvocation */ false); 57 $noinline$stackOverflow(new SubMain(), /* isSecondInvocation */ true); 58 59 $opt$noinline$testOsrInlineLoop(null); 60 System.out.println("b28210356 passed."); 61 } 62 $noinline$returnInt()63 public static int $noinline$returnInt() { 64 // If we are running in non-JIT mode, or were unlucky enough to get this method 65 // already JITted, skip the wait for OSR code. 66 boolean interpreting = isInInterpreter("$noinline$returnInt"); 67 int i = 0; 68 for (; i < 100000; ++i) { 69 } 70 if (interpreting) { 71 while (!isInOsrCode("$noinline$returnInt")) {} 72 } 73 System.out.println(i); 74 return 53; 75 } 76 $noinline$returnFloat()77 public static float $noinline$returnFloat() { 78 // If we are running in non-JIT mode, or were unlucky enough to get this method 79 // already JITted, skip the wait for OSR code. 80 boolean interpreting = isInInterpreter("$noinline$returnFloat"); 81 int i = 0; 82 for (; i < 200000; ++i) { 83 } 84 if (interpreting) { 85 while (!isInOsrCode("$noinline$returnFloat")) {} 86 } 87 System.out.println(i); 88 return 42.2f; 89 } 90 $noinline$returnDouble()91 public static double $noinline$returnDouble() { 92 // If we are running in non-JIT mode, or were unlucky enough to get this method 93 // already JITted, skip the wait for OSR code. 94 boolean interpreting = isInInterpreter("$noinline$returnDouble"); 95 int i = 0; 96 for (; i < 300000; ++i) { 97 } 98 if (interpreting) { 99 while (!isInOsrCode("$noinline$returnDouble")) {} 100 } 101 System.out.println(i); 102 return Double.longBitsToDouble(0xF000000000001111L); 103 } 104 $noinline$returnLong()105 public static long $noinline$returnLong() { 106 // If we are running in non-JIT mode, or were unlucky enough to get this method 107 // already JITted, skip the wait for OSR code. 108 boolean interpreting = isInInterpreter("$noinline$returnLong"); 109 int i = 0; 110 for (; i < 400000; ++i) { 111 } 112 if (interpreting) { 113 while (!isInOsrCode("$noinline$returnLong")) {} 114 } 115 System.out.println(i); 116 return 0xFFFF000000001111L; 117 } 118 $noinline$deopt()119 public static void $noinline$deopt() { 120 // If we are running in non-JIT mode, or were unlucky enough to get this method 121 // already JITted, skip the wait for OSR code. 122 boolean interpreting = isInInterpreter("$noinline$deopt"); 123 int i = 0; 124 for (; i < 100000; ++i) { 125 } 126 if (interpreting) { 127 while (!isInOsrCode("$noinline$deopt")) {} 128 } 129 DeoptimizationController.startDeoptimization(); 130 } 131 $noinline$inlineCache(Main m, boolean isSecondInvocation)132 public static Class<?> $noinline$inlineCache(Main m, boolean isSecondInvocation) { 133 // If we are running in non-JIT mode, or were unlucky enough to get this method 134 // already JITted, just return the expected value. 135 if (!isInInterpreter("$noinline$inlineCache")) { 136 return SubMain.class; 137 } 138 139 ensureHasProfilingInfo("$noinline$inlineCache"); 140 141 // Ensure that we have OSR code to jump to. 142 if (isSecondInvocation) { 143 ensureHasOsrCode("$noinline$inlineCache"); 144 } 145 146 // This call will be optimized in the OSR compiled code 147 // to check and deoptimize if m is not of type 'Main'. 148 Main other = m.inlineCache(); 149 150 // Jump to OSR compiled code. The second run 151 // of this method will have 'm' as a SubMain, and the compiled 152 // code we are jumping to will have wrongly optimize other as being a 153 // 'Main'. 154 if (isSecondInvocation) { 155 while (!isInOsrCode("$noinline$inlineCache")) {} 156 } 157 158 // We used to wrongly optimize this call and assume 'other' was a 'Main'. 159 return other.returnClass(); 160 } 161 $noinline$inlineCache2(Main m, boolean isSecondInvocation)162 public static Class<?> $noinline$inlineCache2(Main m, boolean isSecondInvocation) { 163 // If we are running in non-JIT mode, or were unlucky enough to get this method 164 // already JITted, just return the expected value. 165 if (!isInInterpreter("$noinline$inlineCache2")) { 166 return SubMain.class; 167 } 168 169 ensureHasProfilingInfo("$noinline$inlineCache2"); 170 171 // Ensure that we have OSR code to jump to. 172 if (isSecondInvocation) { 173 ensureHasOsrCode("$noinline$inlineCache2"); 174 } 175 176 // This call will be optimized in the OSR compiled code 177 // to check and deoptimize if m is not of type 'Main'. 178 Main other = m.inlineCache2(); 179 180 // Jump to OSR compiled code. The second run 181 // of this method will have 'm' as a SubMain, and the compiled 182 // code we are jumping to will have wrongly optimize other as being null. 183 if (isSecondInvocation) { 184 while (!isInOsrCode("$noinline$inlineCache2")) {} 185 } 186 187 // We used to wrongly optimize this code and assume 'other' was always null. 188 return (other == null) ? null : other.returnClass(); 189 } 190 $noinline$inlineCache3(Main m, boolean isSecondInvocation)191 public static Class<?> $noinline$inlineCache3(Main m, boolean isSecondInvocation) { 192 // If we are running in non-JIT mode, or were unlucky enough to get this method 193 // already JITted, just return the expected value. 194 if (!isInInterpreter("$noinline$inlineCache3")) { 195 return null; 196 } 197 198 ensureHasProfilingInfo("$noinline$inlineCache3"); 199 200 // Ensure that we have OSR code to jump to. 201 if (isSecondInvocation) { 202 ensureHasOsrCode("$noinline$inlineCache3"); 203 } 204 205 // This call will be optimized in the OSR compiled code 206 // to check and deoptimize if m is not of type 'Main'. 207 Main other = m.inlineCache3(); 208 209 // Jump to OSR compiled code. The second run 210 // of this method will have 'm' as a SubMain, and the compiled 211 // code we are jumping to will have wrongly optimize other as being null. 212 if (isSecondInvocation) { 213 while (!isInOsrCode("$noinline$inlineCache3")) {} 214 } 215 216 // We used to wrongly optimize this code and assume 'other' was always null. 217 return (other == null) ? null : other.returnClass(); 218 } 219 inlineCache()220 public Main inlineCache() { 221 return new Main(); 222 } 223 inlineCache2()224 public Main inlineCache2() { 225 return null; 226 } 227 inlineCache3()228 public Main inlineCache3() { 229 return null; 230 } 231 returnClass()232 public Class<?> returnClass() { 233 return Main.class; 234 } 235 otherInlineCache()236 public void otherInlineCache() { 237 return; 238 } 239 $noinline$stackOverflow(Main m, boolean isSecondInvocation)240 public static void $noinline$stackOverflow(Main m, boolean isSecondInvocation) { 241 // If we are running in non-JIT mode, or were unlucky enough to get this method 242 // already JITted, just return the expected value. 243 if (!isInInterpreter("$noinline$stackOverflow")) { 244 return; 245 } 246 247 // We need a ProfilingInfo object to populate the 'otherInlineCache' call. 248 ensureHasProfilingInfo("$noinline$stackOverflow"); 249 250 if (isSecondInvocation) { 251 // Ensure we have an OSR code and we jump to it. 252 while (!isInOsrCode("$noinline$stackOverflow")) {} 253 } 254 255 for (int i = 0; i < (isSecondInvocation ? 10000000 : 1); ++i) { 256 // The first invocation of $noinline$stackOverflow will populate the inline 257 // cache with Main. The second invocation of the method, will see a SubMain 258 // and will therefore trigger deoptimization. 259 m.otherInlineCache(); 260 } 261 } 262 $opt$noinline$testOsrInlineLoop(String[] args)263 public static void $opt$noinline$testOsrInlineLoop(String[] args) { 264 // Regression test for inlining a method with a loop to a method without a loop in OSR mode. 265 assertIntEquals(12, $opt$inline$testRemoveSuspendCheck(12, 5)); 266 // Since we cannot have a loop directly in this method, we need to force the OSR 267 // compilation from native code. 268 ensureHasProfilingInfo("$opt$noinline$testOsrInlineLoop"); 269 ensureHasOsrCode("$opt$noinline$testOsrInlineLoop"); 270 } 271 $opt$inline$testRemoveSuspendCheck(int x, int y)272 public static int $opt$inline$testRemoveSuspendCheck(int x, int y) { 273 // For this test we need an inlined loop and have DCE re-run loop analysis 274 // after inlining. 275 while (y > 0) { 276 while ($opt$inline$inlineFalse() || !$opt$inline$inlineTrue()) { 277 x++; 278 } 279 y--; 280 } 281 return x; 282 } 283 $opt$inline$inlineTrue()284 public static boolean $opt$inline$inlineTrue() { 285 return true; 286 } 287 $opt$inline$inlineFalse()288 public static boolean $opt$inline$inlineFalse() { 289 return false; 290 } 291 assertIntEquals(int expected, int result)292 public static void assertIntEquals(int expected, int result) { 293 if (expected != result) { 294 throw new Error("Expected: " + expected + ", found: " + result); 295 } 296 } 297 isInOsrCode(String methodName)298 public static native boolean isInOsrCode(String methodName); isInInterpreter(String methodName)299 public static native boolean isInInterpreter(String methodName); ensureHasProfilingInfo(String methodName)300 public static void ensureHasProfilingInfo(String methodName) { 301 ensureJitBaselineCompiled(Main.class, methodName); 302 } ensureHasOsrCode(String methodName)303 public static native void ensureHasOsrCode(String methodName); ensureJitBaselineCompiled(Class<?> cls, String methodName)304 public static native void ensureJitBaselineCompiled(Class<?> cls, String methodName); 305 } 306 307 class SubMain extends Main { returnClass()308 public Class<?> returnClass() { 309 return SubMain.class; 310 } 311 inlineCache()312 public Main inlineCache() { 313 return new SubMain(); 314 } 315 inlineCache2()316 public Main inlineCache2() { 317 return new SubMain(); 318 } 319 otherInlineCache()320 public void otherInlineCache() { 321 return; 322 } 323 } 324