1 /* 2 * Copyright (c) 2018, Red Hat, Inc. All rights reserved. 3 * Copyright Amazon.com Inc. or its affiliates. All Rights Reserved. 4 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. 5 * 6 * This code is free software; you can redistribute it and/or modify it 7 * under the terms of the GNU General Public License version 2 only, as 8 * published by the Free Software Foundation. 9 * 10 * This code is distributed in the hope that it will be useful, but WITHOUT 11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or 12 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License 13 * version 2 for more details (a copy is included in the LICENSE file that 14 * accompanied this code). 15 * 16 * You should have received a copy of the GNU General Public License version 17 * 2 along with this work; if not, write to the Free Software Foundation, 18 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. 19 * 20 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA 21 * or visit www.oracle.com if you need additional information or have any 22 * questions. 23 * 24 */ 25 26 /** 27 * @test 28 * @summary Test OOME in separate thread is recoverable 29 * @requires vm.gc.Shenandoah 30 * @library /test/lib 31 * @run driver TestThreadFailure 32 */ 33 34 import java.util.*; 35 36 import jdk.test.lib.process.OutputAnalyzer; 37 import jdk.test.lib.process.ProcessTools; 38 39 public class TestThreadFailure { 40 41 static final int SIZE = 1024; 42 static final int COUNT = 16; 43 44 static class NastyThread extends Thread { 45 @Override 46 public void run() { 47 List<Object> root = new ArrayList<Object>(); 48 while (true) { 49 root.add(new Object[SIZE]); 50 } 51 } 52 } 53 54 public static void main(String[] args) throws Exception { 55 if (args.length > 0) { 56 for (int t = 0; t < COUNT; t++) { 57 // If we experience OutOfMemoryError during our attempt to instantiate NastyThread, we'll abort 58 // main and will not print "All good". We'll also report a non-zero termination code. In the 59 // case that the previously instantiated NastyThread accumulated more than SheanndoahNoProgressThreshold 60 // unproductive GC cycles before failing, the main thread may not try a Full GC before it experiences 61 // OutOfMemoryError exception. 62 Thread thread = new NastyThread(); 63 thread.start(); 64 thread.join(); 65 // Having joined thread, we know the memory consumed by thread is now garbage, and will eventually be 66 // collected. Some or all of that memory may have been promoted, so we may need to perform a Full GC 67 // in order to reclaim it quickly. 68 } 69 System.out.println("All good"); 70 return; 71 } 72 73 { 74 OutputAnalyzer analyzer = ProcessTools.executeLimitedTestJava( 75 "-Xmx32m", 76 "-XX:+UnlockExperimentalVMOptions", 77 "-XX:+UseShenandoahGC", 78 TestThreadFailure.class.getName(), 79 "test"); 80 81 analyzer.shouldHaveExitValue(0); 82 analyzer.shouldContain("java.lang.OutOfMemoryError"); 83 analyzer.shouldContain("All good"); 84 } 85 86 { 87 ProcessBuilder pb = ProcessTools.createLimitedTestJavaProcessBuilder( 88 "-Xmx32m", 89 "-XX:+UnlockExperimentalVMOptions", 90 "-XX:+UseShenandoahGC", "-XX:ShenandoahGCMode=generational", 91 "-XX:ShenandoahNoProgressThreshold=16", 92 TestThreadFailure.class.getName(), 93 "test"); 94 95 OutputAnalyzer analyzer = new OutputAnalyzer(pb.start()); 96 analyzer.shouldHaveExitValue(0); 97 analyzer.shouldContain("java.lang.OutOfMemoryError"); 98 analyzer.shouldContain("All good"); 99 } 100 } 101 }